blob: f3d41729e4f0808dce2f82498679a633eaf6c1d6 [file] [log] [blame]
#!/bin/bash
# Copyright (C) 2019 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:
# build/build_abi.sh
#
# The following environment variables are considered during execution:
#
# ABI_OUT_TAG
# Customize the output file name for the abi dump. If undefined, the tag is
# derived from `git describe`.
#
# ABI_DEFINITION
# Specify an expected Kernel ABI representation. If defined, this script
# will, in addition to extracting the ABI representation from the currently
# built kernel, compare the extracted ABI to the expected one. In case of
# any significant differences, it will exit with the return code of
# diff_abi and optionally (-r) print a report.
# ABI_DEFINITION is supposed to be defined relative to $KERNEL_DIR/
#
# KMI_WHITELIST
# Define a Kernel Module Interface white list description. If defined, it
# will be taken into account when extracting Kernel ABI information from
# vmlinux and kernel modules.
# KMI_WHITELIST is supposed to be defined relative to $KERNEL_DIR/
#
export ROOT_DIR=$(readlink -f $(dirname $0)/..)
function show_help {
echo "USAGE: $0 [-u|--update] [-n|--nodiff]"
echo
echo " -u | --update Update the abi.xml in the source directory"
echo " -n | --nodiff Do not generate a ABI report with abidiff"
echo " -r | --print-report Print ABI report in case of differences"
}
UPDATE=0
DIFF=1
PRINT_REPORT=0
ARGS=()
for i in "$@"
do
case $i in
-u|--update)
UPDATE=1
shift # past argument=value
;;
-n|--nodiff)
DIFF=0
shift # past argument=value
;;
-r|--print-report)
PRINT_REPORT=1
shift # past argument=value
;;
-h|--help)
show_help
exit 0
;;
*)
ARGS+=("$1")
shift
;;
esac
done
set -- "${ARGS[@]}"
set -e
set -a
# if we are using the default OUT_DIR, add a suffix so we are free to wipe it
# before building to ensure a clean build/analysis. That is the default case.
if [[ -z "$OUT_DIR" ]]; then
export OUT_DIR_SUFFIX="_abi"
wipe_out_dir=1
fi
source "${ROOT_DIR}/build/_setup_env.sh"
# Now actually do the wipe out as above.
if [[ $wipe_out_dir -eq 1 ]]; then
rm -rf "${COMMON_OUT_DIR}"
fi
# inject CONFIG_DEBUG_INFO=y
export POST_DEFCONFIG_CMDS="${POST_DEFCONFIG_CMDS} : && update_config_for_abi_dump"
function update_config_for_abi_dump() {
${KERNEL_DIR}/scripts/config --file ${OUT_DIR}/.config \
-e CONFIG_DEBUG_INFO
(cd ${OUT_DIR} && \
make O=${OUT_DIR} "${TOOL_ARGS[@]}" $archsubarch CROSS_COMPILE=${CROSS_COMPILE} olddefconfig)
}
export -f check_defconfig
export -f update_config_for_abi_dump
function version_greater_than() {
test "$(printf '%s\n' "$@" | sort -V | head -n 1)" != "$1";
}
# ensure that abigail is present in path
if ! ( hash abidiff 2>/dev/null); then
echo "ERROR: libabigail is not found in \$PATH at all!"
echo "Have you run build/abi/bootstrap and followed the instructions?"
exit 1
fi
# ensure we have a "new enough" version of abigail present before continuing
if ! ( version_greater_than "$(abidiff --version | awk '{print $2}')" \
"1.6.0" ); then
echo "ERROR: no suitable libabigail (>= 1.6.0) in \$PATH."
echo "Have you run build/abi/bootstrap and followed the instructions?"
exit 1
fi
# For now we require a specific versions of libabigail identified by a commit
# hash. That is a bit inconvenient, but we do not have another reliable
# identifier at this time.
required_abigail_version="1.7.0-$(cat ${ROOT_DIR}/build/abi/bootstrap| grep 'ABIGAIL_VERSION=' | cut -d= -f2)"
if [[ ! $(abidiff --version) =~ $required_abigail_version ]]; then
echo "ERROR: required libabigail version is $required_abigail_version"
echo "Have you run build/abi/bootstrap and followed the instructions?"
exit 1
fi
# delegate the actual build to build.sh.
# suppress possible values of ABI_DEFINITION when invoking build.sh to avoid
# the generated abi.xml to be copied to <DIST_DIR>/abi.out.
ABI_DEFINITION= ${ROOT_DIR}/build/build.sh $*
# define a common KMI whitelist flag for the abi tools
KMI_WHITELIST_FLAG=
if [ -n "$KMI_WHITELIST" ]; then
if [ $UPDATE -eq 1 ]; then
echo "========================================================"
echo " Updating the ABI whitelist"
${ROOT_DIR}/build/abi/extract_symbols \
--whitelist $KERNEL_DIR/$KMI_WHITELIST \
${DIST_DIR}
fi
KMI_WHITELIST_FLAG="--kmi-whitelist ${DIST_DIR}/abi_whitelist"
fi
echo "========================================================"
echo " Creating ABI dump"
# create abi dump
COMMON_OUT_DIR=$(readlink -m ${OUT_DIR:-${ROOT_DIR}/out/${BRANCH}})
id=${ABI_OUT_TAG:-$(git -C $KERNEL_DIR describe --dirty --always)}
abi_out_file=abi-${id}.xml
${ROOT_DIR}/build/abi/dump_abi \
--linux-tree ${DIST_DIR} \
--out-file ${DIST_DIR}/${abi_out_file} \
$KMI_WHITELIST_FLAG
# sanitize the abi.xml by removing any occurences of the kernel path
sed -i "s#${ROOT_DIR}/${KERNEL_DIR}/##g" ${DIST_DIR}/${abi_out_file}
# now also do that with any left over paths sneaking in
# (e.g. from the prebuilts)
sed -i "s#${ROOT_DIR}/##g" ${DIST_DIR}/${abi_out_file}
# Append debug information to abi file
echo "
<!--
libabigail: $(abidw --version)
built with: $CC: $($CC --version | head -n1)
-->" >> ${DIST_DIR}/${abi_out_file}
ln -sf ${abi_out_file} ${DIST_DIR}/abi.xml
echo "========================================================"
echo " ABI dump has been created at ${DIST_DIR}/${abi_out_file}"
rc=0
if [ -n "$ABI_DEFINITION" ]; then
if [ $DIFF -eq 1 ]; then
echo "========================================================"
echo " Comparing ABI against expected definition ($ABI_DEFINITION)"
abi_report=${DIST_DIR}/abi.report
set +e
${ROOT_DIR}/build/abi/diff_abi --baseline $KERNEL_DIR/$ABI_DEFINITION \
--new ${DIST_DIR}/${abi_out_file} \
--report ${abi_report} \
--short-report ${abi_report}.short \
$KMI_WHITELIST_FLAG
rc=$?
set -e
echo "========================================================"
echo " ABI report has been created at ${abi_report}"
if [ $rc -ne 0 ]; then
echo " ABI DIFFERENCES HAVE BEEN DETECTED! (RC=$rc)"
fi
if [ $PRINT_REPORT -eq 1 ] && [ $rc -ne 0 ] ; then
echo "========================================================"
cat ${abi_report}.short
fi
fi
if [ $UPDATE -eq 1 ] ; then
echo "========================================================"
echo " Updating expected ABI definition ($ABI_DEFINITION)"
cp -v ${DIST_DIR}/${abi_out_file} $KERNEL_DIR/$ABI_DEFINITION
fi
fi
exit $rc