#!/bin/bash # SPDX-License-Identifier: GPL-2.0-only # Script to check commits for UAPI backwards compatibility set -o errexit set -o pipefail print_usage() { name=$(basename "$0") cat << EOF $name - check for UAPI header stability across Git commits By default, the script will check to make sure the latest commit (or current dirty changes) did not introduce ABI changes when compared to HEAD^1. You can check against additional commit ranges with the -b and -p options. The script will not check UAPI headers for architectures other than the one defined in ARCH. Usage: $name [-b BASE_REF] [-p PAST_REF] [-j N] [-l ERROR_LOG] [-i] [-q] [-v] Options: -b BASE_REF Base git reference to use for comparison. If unspecified or empty, will use any dirty changes in tree to UAPI files. If there are no dirty changes, HEAD will be used. -p PAST_REF Compare BASE_REF to PAST_REF (e.g. -p v6.1). If unspecified or empty, will use BASE_REF^1. Must be an ancestor of BASE_REF. Only headers that exist on PAST_REF will be checked for compatibility. -j JOBS Number of checks to run in parallel (default: number of CPU cores). -l ERROR_LOG Write error log to file (default: no error log is generated). -i Ignore ambiguous changes that may or may not break UAPI compatibility. -q Quiet operation. -v Verbose operation (print more information about each header being checked). Environmental args: ABIDIFF Custom path to abidiff binary CC C compiler (default is "gcc") ARCH Target architecture for the UAPI check (default is host arch) Exit codes: $SUCCESS) Success $FAIL_ABI) ABI difference detected $FAIL_PREREQ) Prerequisite not met EOF } readonly SUCCESS=0 readonly FAIL_ABI=1 readonly FAIL_PREREQ=2 # Print to stderr eprintf() { # shellcheck disable=SC2059 printf "$@" >&2 } # Expand an array with a specific character (similar to Python string.join()) join() { local IFS="$1" shift printf "%s" "$*" } # Create abidiff suppressions gen_suppressions() { # Common enum variant names which we don't want to worry about # being shifted when new variants are added. local -a enum_regex=( ".*_AFTER_LAST$" ".*_CNT$" ".*_COUNT$" ".*_END$" ".*_LAST$" ".*_MASK$" ".*_MAX$" ".*_MAX_BIT$" ".*_MAX_BPF_ATTACH_TYPE$" ".*_MAX_ID$" ".*_MAX_SHIFT$" ".*_NBITS$" ".*_NETDEV_NUMHOOKS$" ".*_NFT_META_IIFTYPE$" ".*_NL80211_ATTR$" ".*_NLDEV_NUM_OPS$" ".*_NUM$" ".*_NUM_ELEMS$" ".*_NUM_IRQS$" ".*_SIZE$" ".*_TLSMAX$" "^MAX_.*" "^NUM_.*" ) # Common padding field names which can be expanded into # without worrying about users. local -a padding_regex=( ".*end$" ".*pad$" ".*pad[0-9]?$" ".*pad_[0-9]?$" ".*padding$" ".*padding[0-9]?$" ".*padding_[0-9]?$" ".*res$" ".*resv$" ".*resv[0-9]?$" ".*resv_[0-9]?$" ".*reserved$" ".*reserved[0-9]?$" ".*reserved_[0-9]?$" ".*rsvd[0-9]?$" ".*unused$" ) cat << EOF [suppress_type] type_kind = enum changed_enumerators_regexp = $(join , "${enum_regex[@]}") EOF for p in "${padding_regex[@]}"; do cat << EOF [suppress_type] type_kind = struct has_data_member_inserted_at = offset_of_first_data_member_regexp(${p}) EOF done if [ "$IGNORE_AMBIGUOUS_CHANGES" = "true" ]; then cat << EOF [suppress_type] type_kind = struct has_data_member_inserted_at = end has_size_change = yes EOF fi } # Check if git tree is dirty tree_is_dirty() { ! git diff --quiet } # Get list of files installed in $ref get_file_list() { local -r ref="$1" local -r tree="$(get_header_tree "$ref")" # Print all installed headers, filtering out ones that can't be compiled find "$tree" -type f -name '*.h' -printf '%P\n' | grep -v -f "$INCOMPAT_LIST" } # Add to the list of incompatible headers add_to_incompat_list() { local -r ref="$1" # Start with the usr/include/Makefile to get a list of the headers # that don't compile using this method. if [ ! -f usr/include/Makefile ]; then eprintf "error - no usr/include/Makefile present at %s\n" "$ref" eprintf "Note: usr/include/Makefile was added in the v5.3 kernel release\n" exit "$FAIL_PREREQ" fi { # shellcheck disable=SC2016 printf 'all: ; @echo $(no-header-test)\n' cat usr/include/Makefile } | SRCARCH="$ARCH" make --always-make -f - | tr " " "\n" \ | grep -v "asm-generic" >> "$INCOMPAT_LIST" # The makefile also skips all asm-generic files, but prints "asm-generic/%" # which won't work for our grep match. Instead, print something grep will match. printf "asm-generic/.*\.h\n" >> "$INCOMPAT_LIST" } # Compile the simple test app do_compile() { local -r inc_dir="$1" local -r header="$2" local -r out="$3" printf "int main(void) { return 0; }\n" | \ "$CC" -c \ -o "$out" \ -x c \ -O0 \ -std=c90 \ -fno-eliminate-unused-debug-types \ -g \ "-I${inc_dir}" \ -include "$header" \ - } # Run make headers_install run_make_headers_install() { local -r ref="$1" local -r install_dir="$(get_header_tree "$ref")" make -j "$MAX_THREADS" ARCH="$ARCH" INSTALL_HDR_PATH="$install_dir" \ headers_install > /dev/null } # Install headers for both git refs install_headers() { local -r base_ref="$1" local -r past_ref="$2" for ref in "$base_ref" "$past_ref"; do printf "Installing user-facing UAPI headers from %s... " "${ref:-dirty tree}" if [ -n "$ref" ]; then git archive --format=tar --prefix="${ref}-archive/" "$ref" \ | (cd "$TMP_DIR" && tar xf -) ( cd "${TMP_DIR}/${ref}-archive" run_make_headers_install "$ref" add_to_incompat_list "$ref" "$INCOMPAT_LIST" ) else run_make_headers_install "$ref" add_to_incompat_list "$ref" "$INCOMPAT_LIST" fi printf "OK\n" done sort -u -o "$INCOMPAT_LIST" "$INCOMPAT_LIST" sed -i -e '/^$/d' "$INCOMPAT_LIST" } # Print the path to the headers_install tree for a given ref get_header_tree() { local -r ref="$1" printf "%s" "${TMP_DIR}/${ref}/usr" } # Check file list for UAPI compatibility check_uapi_files() { local -r base_ref="$1" local -r past_ref="$2" local -r abi_error_log="$3" local passed=0; local failed=0; local -a threads=() set -o errexit printf "Checking changes to UAPI headers between %s and %s...\n" "$past_ref" "${base_ref:-dirty tree}" # Loop over all UAPI headers that were installed by $past_ref (if they only exist on $base_ref, # there's no way they're broken and no way to compare anyway) while read -r file; do if [ "${#threads[@]}" -ge "$MAX_THREADS" ]; then if wait "${threads[0]}"; then passed=$((passed + 1)) else failed=$((failed + 1)) fi threads=("${threads[@]:1}") fi check_individual_file "$base_ref" "$past_ref" "$file" & threads+=("$!") done < <(get_file_list "$past_ref") for t in "${threads[@]}"; do if wait "$t"; then passed=$((passed + 1)) else failed=$((failed + 1)) fi done if [ -n "$abi_error_log" ]; then printf 'Generated by "%s %s" from git ref %s\n\n' \ "$0" "$*" "$(git rev-parse HEAD)" > "$abi_error_log" fi while read -r error_file; do { cat "$error_file" printf "\n\n" } | tee -a "${abi_error_log:-/dev/null}" >&2 done < <(find "$TMP_DIR" -type f -name '*.error' | sort) total="$((passed + failed))" if [ "$failed" -gt 0 ]; then eprintf "error - %d/%d UAPI headers compatible with %s appear _not_ to be backwards compatible\n" \ "$failed" "$total" "$ARCH" if [ -n "$abi_error_log" ]; then eprintf "Failure summary saved to %s\n" "$abi_error_log" fi else printf "All %d UAPI headers compatible with %s appear to be backwards compatible\n" \ "$total" "$ARCH" fi return "$failed" } # Check an individual file for UAPI compatibility check_individual_file() { local -r base_ref="$1" local -r past_ref="$2" local -r file="$3" local -r base_header="$(get_header_tree "$base_ref")/${file}" local -r past_header="$(get_header_tree "$past_ref")/${file}" if [ ! -f "$base_header" ]; then mkdir -p "$(dirname "$base_header")" printf "==== UAPI header %s was removed between %s and %s ====" \ "$file" "$past_ref" "$base_ref" \ > "${base_header}.error" return 1 fi compare_abi "$file" "$base_header" "$past_header" "$base_ref" "$past_ref" } # Perform the A/B compilation and compare output ABI compare_abi() { local -r file="$1" local -r base_header="$2" local -r past_header="$3" local -r base_ref="$4" local -r past_ref="$5" local -r log="${TMP_DIR}/log/${file}.log" local -r error_log="${TMP_DIR}/log/${file}.error" mkdir -p "$(dirname "$log")" if ! do_compile "$(get_header_tree "$base_ref")/include" "$base_header" "${base_header}.bin" 2> "$log"; then { warn_str=$(printf "==== Could not compile version of UAPI header %s at %s ====\n" \ "$file" "$base_ref") printf "%s\n" "$warn_str" cat "$log" printf -- "=%.0s" $(seq 0 ${#warn_str}) } > "$error_log" return 1 fi if ! do_compile "$(get_header_tree "$past_ref")/include" "$past_header" "${past_header}.bin" 2> "$log"; then { warn_str=$(printf "==== Could not compile version of UAPI header %s at %s ====\n" \ "$file" "$past_ref") printf "%s\n" "$warn_str" cat "$log" printf -- "=%.0s" $(seq 0 ${#warn_str}) } > "$error_log" return 1 fi local ret=0 "$ABIDIFF" --non-reachable-types \ --suppressions "$SUPPRESSIONS" \ "${past_header}.bin" "${base_header}.bin" > "$log" || ret="$?" if [ "$ret" -eq 0 ]; then if [ "$VERBOSE" = "true" ]; then printf "No ABI differences detected in %s from %s -> %s\n" \ "$file" "$past_ref" "$base_ref" fi else # Bits in abidiff's return code can be used to determine the type of error if [ $((ret & 0x2)) -gt 0 ]; then eprintf "error - abidiff did not run properly\n" exit 1 fi if [ "$IGNORE_AMBIGUOUS_CHANGES" = "true" ] && [ "$ret" -eq 4 ]; then return 0 fi # If the only changes were additions (not modifications to existing APIs), then # there's no problem. Ignore these diffs. if grep "Unreachable types summary" "$log" | grep -q "0 removed" && grep "Unreachable types summary" "$log" | grep -q "0 changed"; then return 0 fi { warn_str=$(printf "==== ABI differences detected in %s from %s -> %s ====" \ "$file" "$past_ref" "$base_ref") printf "%s\n" "$warn_str" sed -e '/summary:/d' -e '/changed type/d' -e '/^$/d' -e 's/^/ /g' "$log" printf -- "=%.0s" $(seq 0 ${#warn_str}) if cmp "$past_header" "$base_header" > /dev/null 2>&1; then printf "\n%s did not change between %s and %s...\n" "$file" "$past_ref" "${base_ref:-dirty tree}" printf "It's possible a change to one of the headers it includes caused this error:\n" grep '^#include' "$base_header" printf "\n" fi } > "$error_log" return 1 fi } # Check that a minimum software version number is satisfied min_version_is_satisfied() { local -r min_version="$1" local -r version_installed="$2" printf "%s\n%s\n" "$min_version" "$version_installed" \ | sort -Vc > /dev/null 2>&1 } # Make sure we have the tools we need and the arguments make sense check_deps() { ABIDIFF="${ABIDIFF:-abidiff}" CC="${CC:-gcc}" ARCH="${ARCH:-$(uname -m)}" if [ "$ARCH" = "x86_64" ]; then ARCH="x86" fi local -r abidiff_min_version="2.4" local -r libdw_min_version_if_clang="0.171" if ! command -v "$ABIDIFF" > /dev/null 2>&1; then eprintf "error - abidiff not found!\n" eprintf "Please install abigail-tools version %s or greater\n" "$abidiff_min_version" eprintf "See: https://sourceware.org/libabigail/manual/libabigail-overview.html\n" return 1 fi local -r abidiff_version="$("$ABIDIFF" --version | cut -d ' ' -f 2)" if ! min_version_is_satisfied "$abidiff_min_version" "$abidiff_version"; then eprintf "error - abidiff version too old: %s\n" "$abidiff_version" eprintf "Please install abigail-tools version %s or greater\n" "$abidiff_min_version" eprintf "See: https://sourceware.org/libabigail/manual/libabigail-overview.html\n" return 1 fi if ! command -v "$CC" > /dev/null 2>&1; then eprintf 'error - %s not found\n' "$CC" return 1 fi if "$CC" --version | grep -q clang; then local -r libdw_version="$(ldconfig -v 2>/dev/null | grep -v SKIPPED | grep -m 1 -o 'libdw-[0-9]\+.[0-9]\+' | cut -c 7-)" if ! min_version_is_satisfied "$libdw_min_version_if_clang" "$libdw_version"; then eprintf "error - libdw version too old for use with clang: %s\n" "$libdw_version" eprintf "Please install libdw from elfutils version %s or greater\n" "$libdw_min_version_if_clang" eprintf "See: https://sourceware.org/elfutils/\n" return 1 fi fi if [ ! -d "arch/${ARCH}" ]; then eprintf 'error - ARCH "%s" is not a subdirectory under arch/\n' "$ARCH" eprintf "Please set ARCH to one of:\n%s\n" "$(find arch -maxdepth 1 -mindepth 1 -type d -printf '%f ' | fmt)" return 1 fi if ! git rev-parse --is-inside-work-tree > /dev/null 2>&1; then eprintf "error - this script requires the kernel tree to be initialized with Git\n" return 1 fi if ! git rev-parse --verify "$past_ref" > /dev/null 2>&1; then printf 'error - invalid git reference "%s"\n' "$past_ref" return 1 fi if [ -n "$base_ref" ]; then if ! git merge-base --is-ancestor "$past_ref" "$base_ref" > /dev/null 2>&1; then printf 'error - "%s" is not an ancestor of base ref "%s"\n' "$past_ref" "$base_ref" return 1 fi if [ "$(git rev-parse "$base_ref")" = "$(git rev-parse "$past_ref")" ]; then printf 'error - "%s" and "%s" are the same reference\n' "$past_ref" "$base_ref" return 1 fi fi } run() { local base_ref="$1" local past_ref="$2" local abi_error_log="$3" shift 3 if [ -z "$KERNEL_SRC" ]; then KERNEL_SRC="$(realpath "$(dirname "$0")"/..)" fi cd "$KERNEL_SRC" if [ -z "$base_ref" ] && ! tree_is_dirty; then base_ref=HEAD fi if [ -z "$past_ref" ]; then if [ -n "$base_ref" ]; then past_ref="${base_ref}^1" else past_ref=HEAD fi fi if ! check_deps; then exit "$FAIL_PREREQ" fi TMP_DIR=$(mktemp -d) readonly TMP_DIR trap 'rm -rf "$TMP_DIR"' EXIT readonly INCOMPAT_LIST="${TMP_DIR}/incompat_list.txt" touch "$INCOMPAT_LIST" readonly SUPPRESSIONS="${TMP_DIR}/suppressions.txt" gen_suppressions > "$SUPPRESSIONS" # Run make install_headers for both refs install_headers "$base_ref" "$past_ref" # Check for any differences in the installed header trees if diff -r -q "$(get_header_tree "$base_ref")" "$(get_header_tree "$past_ref")" > /dev/null 2>&1; then printf "No changes to UAPI headers were applied between %s and %s\n" "$past_ref" "${base_ref:-dirty tree}" exit "$SUCCESS" fi if ! check_uapi_files "$base_ref" "$past_ref" "$abi_error_log"; then exit "$FAIL_ABI" fi } main() { MAX_THREADS=$(nproc) VERBOSE="false" IGNORE_AMBIGUOUS_CHANGES="false" quiet="false" local base_ref="" while getopts "hb:p:j:l:iqv" opt; do case $opt in h) print_usage exit "$SUCCESS" ;; b) base_ref="$OPTARG" ;; p) past_ref="$OPTARG" ;; j) MAX_THREADS="$OPTARG" ;; l) abi_error_log="$OPTARG" ;; i) IGNORE_AMBIGUOUS_CHANGES="true" ;; q) quiet="true" VERBOSE="false" ;; v) VERBOSE="true" quiet="false" ;; *) exit "$FAIL_PREREQ" esac done if [ "$quiet" = "true" ]; then exec > /dev/null 2>&1 fi run "$base_ref" "$past_ref" "$abi_error_log" "$@" } main "$@"