#!/bin/bash
#
# Copyright (C) 2013 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
### Usage: generate_uapi_headers.sh [<options>]
###
### This script is used to get a copy of the uapi kernel headers
### from an android kernel tree and copies them into an android source
### tree without any processing. The script also creates all of the
### generated headers and copies them into the android source tree.
###
### Options:
### --skip-generation
### Skip the step that generates all of the include files.
### --download-kernel <VERSION>
### Automatically create a temporary git repository and check out the
### linux kernel source code for the given version.
### --use-kernel-dir <DIR>
### Do not check out the kernel source, use the kernel directory
### pointed to by <DIR>.
### --verify-modified-headers-only <DIR>
### Do not build anything, simply verify that the set of modified
### kernel headers have not changed.
# Terminate the script if any command fails.
set -eE
TMPDIR=""
ANDROID_DIR=""
KERNEL_VERSION=""
KERNEL_REPO="git://git.kernel.org/pub/scm/linux/kernel/git/stable/linux-stable.git"
KERNEL_DIR=""
KERNEL_DOWNLOAD=0
ARCH_LIST=("arm" "arm64" "mips" "x86")
ANDROID_KERNEL_DIR="external/kernel-headers/original"
SKIP_GENERATION=0
VERIFY_HEADERS_ONLY=0
function cleanup () {
if [[ "${TMPDIR}" =~ /tmp ]] && [[ -d "${TMPDIR}" ]]; then
echo "Removing temporary directory ${TMPDIR}"
rm -rf "${TMPDIR}"
TMPDIR=""
fi
}
function usage () {
grep '^###' $0 | sed -e 's/^###//'
}
function copy_hdrs () {
local src_dir=$1
local tgt_dir=$2
local dont_copy_dirs=$3
mkdir -p ${tgt_dir}
local search_dirs=()
# This only works if none of the filenames have spaces.
for file in $(ls -d ${src_dir}/* 2> /dev/null); do
if [[ -d "${file}" ]]; then
search_dirs+=("${file}")
elif [[ -f "${file}" ]] && [[ "${file}" =~ .h$ ]]; then
cp ${file} ${tgt_dir}
fi
done
if [[ "${dont_copy_dirs}" == "" ]]; then
for dir in "${search_dirs[@]}"; do
copy_hdrs "${dir}" ${tgt_dir}/$(basename ${dir})
done
fi
}
function copy_if_exists () {
local check_dir=$1
local src_dir=$2
local tgt_dir=$3
mkdir -p ${tgt_dir}
# This only works if none of the filenames have spaces.
for file in $(ls -d ${src_dir}/* 2> /dev/null); do
if [[ -f "${file}" ]] && [[ "${file}" =~ .h$ ]]; then
# Check that this file exists in check_dir.
header=$(basename ${file})
if [[ -f "${check_dir}/${header}" ]]; then
cp ${file} ${tgt_dir}
fi
fi
done
}
function verify_modified_hdrs () {
local src_dir=$1
local tgt_dir=$2
local kernel_dir=$3
local search_dirs=()
# This only works if none of the filenames have spaces.
for file in $(ls -d ${src_dir}/* 2> /dev/null); do
if [[ -d "${file}" ]]; then
search_dirs+=("${file}")
elif [[ -f "${file}" ]] && [[ "${file}" =~ .h$ ]]; then
tgt_file=${tgt_dir}/$(basename ${file})
if [[ -e ${tgt_file} ]] && ! diff "${file}" "${tgt_file}" > /dev/null; then
if [[ ${file} =~ ${kernel_dir}/*(.+) ]]; then
echo "New version of ${BASH_REMATCH[1]} found in kernel headers."
else
echo "New version of ${file} found in kernel headers."
fi
echo "This file needs to be updated manually."
fi
fi
done
for dir in "${search_dirs[@]}"; do
verify_modified_hdrs "${dir}" ${tgt_dir}/$(basename ${dir}) "${kernel_dir}"
done
}
trap cleanup EXIT
# This automatically triggers a call to cleanup.
trap "exit 1" HUP INT TERM TSTP
while [ $# -gt 0 ]; do
case "$1" in
"--skip-generation")
SKIP_GENERATION=1
;;
"--download-kernel")
if [[ $# -lt 2 ]]; then
echo "--download-kernel requires an argument."
exit 1
fi
shift
KERNEL_VERSION="$1"
KERNEL_DOWNLOAD=1
;;
"--use-kernel-dir")
if [[ $# -lt 2 ]]; then
echo "--use-kernel-dir requires an argument."
exit 1
fi
shift
KERNEL_DIR="$1"
KERNEL_DOWNLOAD=0
;;
"--verify-modified-headers-only")
if [[ $# -lt 2 ]]; then
echo "--verify-modified-headers-only requires an argument."
exit 1
fi
shift
KERNEL_DIR="$1"
KERNEL_DOWNLOAD=0
VERIFY_HEADERS_ONLY=1
;;
"-h" | "--help")
usage
exit 1
;;
"-"*)
echo "Error: Unrecognized option $1"
usage
exit 1
;;
*)
echo "Error: Extra arguments on the command-line."
usage
exit 1
;;
esac
shift
done
ANDROID_KERNEL_DIR="${ANDROID_BUILD_TOP}/${ANDROID_KERNEL_DIR}"
if [[ "${ANDROID_BUILD_TOP}" == "" ]]; then
echo "ANDROID_BUILD_TOP is not set, did you run lunch?"
exit 1
elif [[ ! -d "${ANDROID_KERNEL_DIR}" ]]; then
echo "${ANDROID_BUILD_TOP} doesn't appear to be the root of an android tree."
echo " ${ANDROID_KERNEL_DIR} is not a directory."
exit 1
fi
src_dir="linux-stable"
if [[ ${KERNEL_DOWNLOAD} -eq 1 ]]; then
TMPDIR=$(mktemp -d /tmp/android_kernelXXXXXXXX)
cd "${TMPDIR}"
echo "Fetching linux kernel source ${KERNEL_VERSION}"
git clone ${KERNEL_REPO}
cd ${src_dir}
git checkout tags/"${KERNEL_VERSION}"
KERNEL_DIR="${TMPDIR}"
elif [[ "${KERNEL_DIR}" == "" ]]; then
echo "Must specify one of --use-kernel-dir or --download-kernel."
exit 1
elif [[ ! -d "${KERNEL_DIR}" ]] || [[ ! -d "${KERNEL_DIR}/${src_dir}" ]]; then
echo "The kernel directory $KERNEL_DIR or $KERNEL_DIR/${src_dir} does not exist."
exit 1
else
cd "${KERNEL_DIR}/${src_dir}"
fi
if [[ ${VERIFY_HEADERS_ONLY} -eq 1 ]]; then
# Verify if modified headers have changed.
verify_modified_hdrs "${KERNEL_DIR}/${src_dir}/include/scsi" \
"${ANDROID_KERNEL_DIR}/scsi" \
"${KERNEL_DIR}/${src_dir}"
exit 0
fi
if [[ ${SKIP_GENERATION} -eq 0 ]]; then
# Clean up any leftover headers.
make distclean
# Build all of the generated headers.
for arch in "${ARCH_LIST[@]}"; do
echo "Generating headers for arch ${arch}"
make ARCH=${arch} headers_install
done
fi
# Completely delete the old original headers so that any deleted/moved
# headers are also removed.
rm -rf "${ANDROID_KERNEL_DIR}/uapi"
mkdir -p "${ANDROID_KERNEL_DIR}/uapi"
cd ${ANDROID_BUILD_TOP}
# Copy all of the include/uapi files to the kernel headers uapi directory.
copy_hdrs "${KERNEL_DIR}/${src_dir}/include/uapi" "${ANDROID_KERNEL_DIR}/uapi"
# Copy the staging files to uapi/linux.
copy_hdrs "${KERNEL_DIR}/${src_dir}/drivers/staging/android/uapi" \
"${ANDROID_KERNEL_DIR}/uapi/linux" "no-copy-dirs"
# Remove ion.h, it's not fully supported by the upstream kernel (see b/77976082).
rm -f "${ANDROID_KERNEL_DIR}/uapi/linux/ion.h"
# Copy the generated headers.
copy_hdrs "${KERNEL_DIR}/${src_dir}/include/generated/uapi" \
"${ANDROID_KERNEL_DIR}/uapi"
for arch in "${ARCH_LIST[@]}"; do
# Copy arch headers.
copy_hdrs "${KERNEL_DIR}/${src_dir}/arch/${arch}/include/uapi" \
"${ANDROID_KERNEL_DIR}/uapi/asm-${arch}"
# Copy the generated arch headers.
copy_hdrs "${KERNEL_DIR}/${src_dir}/arch/${arch}/include/generated/uapi" \
"${ANDROID_KERNEL_DIR}/uapi/asm-${arch}"
# Special copy of generated header files from arch/<ARCH>/generated/asm that
# also exist in uapi/asm-generic.
copy_if_exists "${KERNEL_DIR}/${src_dir}/include/uapi/asm-generic" \
"${KERNEL_DIR}/${src_dir}/arch/${arch}/include/generated/asm" \
"${ANDROID_KERNEL_DIR}/uapi/asm-${arch}/asm"
done
# The arm types.h uapi header is not properly being generated, so copy it
# directly.
cp "${KERNEL_DIR}/${src_dir}/include/uapi/asm-generic/types.h" \
"${ANDROID_KERNEL_DIR}/uapi/asm-arm/asm"
# Verify if modified headers have changed.
verify_modified_hdrs "${KERNEL_DIR}/${src_dir}/include/scsi" \
"${ANDROID_KERNEL_DIR}/scsi" \
"${KERNEL_DIR}/${src_dir}"
echo "Headers updated."