124ebfcd6SJosh Poimboeuf#!/bin/bash 224ebfcd6SJosh Poimboeuf# SPDX-License-Identifier: GPL-2.0 324ebfcd6SJosh Poimboeuf# 424ebfcd6SJosh Poimboeuf# Build a livepatch module 524ebfcd6SJosh Poimboeuf 624ebfcd6SJosh Poimboeuf# shellcheck disable=SC1090,SC2155 724ebfcd6SJosh Poimboeuf 824ebfcd6SJosh Poimboeufif (( BASH_VERSINFO[0] < 4 || \ 924ebfcd6SJosh Poimboeuf (BASH_VERSINFO[0] == 4 && BASH_VERSINFO[1] < 4) )); then 1024ebfcd6SJosh Poimboeuf echo "error: this script requires bash 4.4+" >&2 1124ebfcd6SJosh Poimboeuf exit 1 1224ebfcd6SJosh Poimboeuffi 1324ebfcd6SJosh Poimboeuf 1424ebfcd6SJosh Poimboeufset -o errexit 1524ebfcd6SJosh Poimboeufset -o errtrace 1624ebfcd6SJosh Poimboeufset -o pipefail 1724ebfcd6SJosh Poimboeufset -o nounset 1824ebfcd6SJosh Poimboeuf 1924ebfcd6SJosh Poimboeuf# Allow doing 'cmd | mapfile -t array' instead of 'mapfile -t array < <(cmd)'. 2024ebfcd6SJosh Poimboeuf# This helps keep execution in pipes so pipefail+errexit can catch errors. 2124ebfcd6SJosh Poimboeufshopt -s lastpipe 2224ebfcd6SJosh Poimboeuf 2378be9facSJosh Poimboeufunset DEBUG_CLONE DIFF_CHECKSUM SKIP_CLEANUP XTRACE 2424ebfcd6SJosh Poimboeuf 2524ebfcd6SJosh PoimboeufREPLACE=1 2624ebfcd6SJosh PoimboeufSHORT_CIRCUIT=0 2724ebfcd6SJosh PoimboeufJOBS="$(getconf _NPROCESSORS_ONLN)" 2824ebfcd6SJosh PoimboeufVERBOSE="-s" 2924ebfcd6SJosh Poimboeufshopt -o xtrace | grep -q 'on' && XTRACE=1 3024ebfcd6SJosh Poimboeuf 3124ebfcd6SJosh Poimboeuf# Avoid removing the previous $TMP_DIR until args have been fully processed. 3224ebfcd6SJosh PoimboeufKEEP_TMP=1 3324ebfcd6SJosh Poimboeuf 3424ebfcd6SJosh PoimboeufSCRIPT="$(basename "$0")" 3524ebfcd6SJosh PoimboeufSCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" 3624ebfcd6SJosh PoimboeufFIX_PATCH_LINES="$SCRIPT_DIR/fix-patch-lines" 3724ebfcd6SJosh Poimboeuf 3824ebfcd6SJosh PoimboeufSRC="$(pwd)" 3924ebfcd6SJosh PoimboeufOBJ="$(pwd)" 4024ebfcd6SJosh Poimboeuf 4124ebfcd6SJosh PoimboeufCONFIG="$OBJ/.config" 4224ebfcd6SJosh PoimboeufTMP_DIR="$OBJ/klp-tmp" 4324ebfcd6SJosh Poimboeuf 4424ebfcd6SJosh PoimboeufORIG_DIR="$TMP_DIR/orig" 4524ebfcd6SJosh PoimboeufPATCHED_DIR="$TMP_DIR/patched" 4624ebfcd6SJosh PoimboeufDIFF_DIR="$TMP_DIR/diff" 4724ebfcd6SJosh PoimboeufKMOD_DIR="$TMP_DIR/kmod" 4824ebfcd6SJosh Poimboeuf 4924ebfcd6SJosh PoimboeufSTASH_DIR="$TMP_DIR/stash" 5024ebfcd6SJosh PoimboeufTIMESTAMP="$TMP_DIR/timestamp" 5124ebfcd6SJosh PoimboeufPATCH_TMP_DIR="$TMP_DIR/tmp" 5224ebfcd6SJosh Poimboeuf 5324ebfcd6SJosh PoimboeufKLP_DIFF_LOG="$DIFF_DIR/diff.log" 5424ebfcd6SJosh Poimboeuf 5524ebfcd6SJosh Poimboeuf# Terminal output colors 5624ebfcd6SJosh Poimboeufread -r COLOR_RESET COLOR_BOLD COLOR_ERROR COLOR_WARN <<< "" 5724ebfcd6SJosh Poimboeufif [[ -t 1 && -t 2 ]]; then 5824ebfcd6SJosh Poimboeuf COLOR_RESET="\033[0m" 5924ebfcd6SJosh Poimboeuf COLOR_BOLD="\033[1m" 6024ebfcd6SJosh Poimboeuf COLOR_ERROR="\033[0;31m" 6124ebfcd6SJosh Poimboeuf COLOR_WARN="\033[0;33m" 6224ebfcd6SJosh Poimboeuffi 6324ebfcd6SJosh Poimboeuf 6424ebfcd6SJosh Poimboeufgrep0() { 6524ebfcd6SJosh Poimboeuf # shellcheck disable=SC2317 6624ebfcd6SJosh Poimboeuf command grep "$@" || true 6724ebfcd6SJosh Poimboeuf} 6824ebfcd6SJosh Poimboeuf 6924ebfcd6SJosh Poimboeuf# Because pipefail is enabled, the grep0 helper should be used instead of 7024ebfcd6SJosh Poimboeuf# grep, otherwise a failed match can propagate to an error. 7124ebfcd6SJosh Poimboeufgrep() { 7224ebfcd6SJosh Poimboeuf echo "error: $SCRIPT: use grep0 or 'command grep' instead of bare grep" >&2 7324ebfcd6SJosh Poimboeuf exit 1 7424ebfcd6SJosh Poimboeuf} 7524ebfcd6SJosh Poimboeuf 7624ebfcd6SJosh Poimboeufstatus() { 7724ebfcd6SJosh Poimboeuf echo -e "${COLOR_BOLD}$*${COLOR_RESET}" 7824ebfcd6SJosh Poimboeuf} 7924ebfcd6SJosh Poimboeuf 8024ebfcd6SJosh Poimboeufwarn() { 8124ebfcd6SJosh Poimboeuf echo -e "${COLOR_WARN}warning${COLOR_RESET}: $SCRIPT: $*" >&2 8224ebfcd6SJosh Poimboeuf} 8324ebfcd6SJosh Poimboeuf 8424ebfcd6SJosh Poimboeufdie() { 8524ebfcd6SJosh Poimboeuf echo -e "${COLOR_ERROR}error${COLOR_RESET}: $SCRIPT: $*" >&2 8624ebfcd6SJosh Poimboeuf exit 1 8724ebfcd6SJosh Poimboeuf} 8824ebfcd6SJosh Poimboeuf 8924ebfcd6SJosh Poimboeufdeclare -a STASHED_FILES 9024ebfcd6SJosh Poimboeuf 9124ebfcd6SJosh Poimboeufstash_file() { 9224ebfcd6SJosh Poimboeuf local file="$1" 9324ebfcd6SJosh Poimboeuf local rel_file="${file#"$SRC"/}" 9424ebfcd6SJosh Poimboeuf 9524ebfcd6SJosh Poimboeuf [[ ! -e "$file" ]] && die "no file to stash: $file" 9624ebfcd6SJosh Poimboeuf 9724ebfcd6SJosh Poimboeuf mkdir -p "$STASH_DIR/$(dirname "$rel_file")" 9824ebfcd6SJosh Poimboeuf cp -f "$file" "$STASH_DIR/$rel_file" 9924ebfcd6SJosh Poimboeuf 10024ebfcd6SJosh Poimboeuf STASHED_FILES+=("$rel_file") 10124ebfcd6SJosh Poimboeuf} 10224ebfcd6SJosh Poimboeuf 10324ebfcd6SJosh Poimboeufrestore_files() { 10424ebfcd6SJosh Poimboeuf local file 10524ebfcd6SJosh Poimboeuf 10624ebfcd6SJosh Poimboeuf for file in "${STASHED_FILES[@]}"; do 10724ebfcd6SJosh Poimboeuf mv -f "$STASH_DIR/$file" "$SRC/$file" || warn "can't restore file: $file" 10824ebfcd6SJosh Poimboeuf done 10924ebfcd6SJosh Poimboeuf 11024ebfcd6SJosh Poimboeuf STASHED_FILES=() 11124ebfcd6SJosh Poimboeuf} 11224ebfcd6SJosh Poimboeuf 11324ebfcd6SJosh Poimboeufcleanup() { 11424ebfcd6SJosh Poimboeuf set +o nounset 11524ebfcd6SJosh Poimboeuf revert_patches 11624ebfcd6SJosh Poimboeuf restore_files 11778be9facSJosh Poimboeuf [[ "$KEEP_TMP" -eq 0 ]] && rm -rf "$TMP_DIR" 11824ebfcd6SJosh Poimboeuf return 0 11924ebfcd6SJosh Poimboeuf} 12024ebfcd6SJosh Poimboeuf 12124ebfcd6SJosh Poimboeuftrap_err() { 12224ebfcd6SJosh Poimboeuf die "line ${BASH_LINENO[0]}: '$BASH_COMMAND'" 12324ebfcd6SJosh Poimboeuf} 1242c2f0b86SJosh Poimboeuf 12524ebfcd6SJosh Poimboeuftrap cleanup EXIT INT TERM HUP 12624ebfcd6SJosh Poimboeuftrap trap_err ERR 12724ebfcd6SJosh Poimboeuf 12824ebfcd6SJosh Poimboeuf__usage() { 12924ebfcd6SJosh Poimboeuf cat <<EOF 13024ebfcd6SJosh PoimboeufUsage: $SCRIPT [OPTIONS] PATCH_FILE(s) 13124ebfcd6SJosh PoimboeufGenerate a livepatch module. 13224ebfcd6SJosh Poimboeuf 13324ebfcd6SJosh PoimboeufOptions: 13424ebfcd6SJosh Poimboeuf -f, --show-first-changed Show address of first changed instruction 13524ebfcd6SJosh Poimboeuf -j, --jobs=<jobs> Build jobs to run simultaneously [default: $JOBS] 13624ebfcd6SJosh Poimboeuf -o, --output=<file.ko> Output file [default: livepatch-<patch-name>.ko] 13724ebfcd6SJosh Poimboeuf --no-replace Disable livepatch atomic replace 13824ebfcd6SJosh Poimboeuf -v, --verbose Pass V=1 to kernel/module builds 13924ebfcd6SJosh Poimboeuf 14024ebfcd6SJosh PoimboeufAdvanced Options: 14124ebfcd6SJosh Poimboeuf -d, --debug Show symbol/reloc cloning decisions 14224ebfcd6SJosh Poimboeuf -S, --short-circuit=STEP Start at build step (requires prior --keep-tmp) 14324ebfcd6SJosh Poimboeuf 1|orig Build original kernel (default) 14424ebfcd6SJosh Poimboeuf 2|patched Build patched kernel 14578be9facSJosh Poimboeuf 3|diff Diff objects 14678be9facSJosh Poimboeuf 4|kmod Build patch module 14724ebfcd6SJosh Poimboeuf -T, --keep-tmp Preserve tmp dir on exit 14824ebfcd6SJosh Poimboeuf 14924ebfcd6SJosh PoimboeufEOF 15024ebfcd6SJosh Poimboeuf} 15124ebfcd6SJosh Poimboeuf 15224ebfcd6SJosh Poimboeufusage() { 15324ebfcd6SJosh Poimboeuf __usage >&2 15424ebfcd6SJosh Poimboeuf} 15524ebfcd6SJosh Poimboeuf 15624ebfcd6SJosh Poimboeufprocess_args() { 15724ebfcd6SJosh Poimboeuf local keep_tmp=0 15824ebfcd6SJosh Poimboeuf local short 15978be9facSJosh Poimboeuf local long 16078be9facSJosh Poimboeuf local args 16178be9facSJosh Poimboeuf 16278be9facSJosh Poimboeuf short="hfj:o:vdS:T" 16324ebfcd6SJosh Poimboeuf long="help,show-first-changed,jobs:,output:,no-replace,verbose,debug,short-circuit:,keep-tmp" 16424ebfcd6SJosh Poimboeuf 16524ebfcd6SJosh Poimboeuf args=$(getopt --options "$short" --longoptions "$long" -- "$@") || { 16624ebfcd6SJosh Poimboeuf echo; usage; exit 16724ebfcd6SJosh Poimboeuf } 16824ebfcd6SJosh Poimboeuf eval set -- "$args" 16924ebfcd6SJosh Poimboeuf 17024ebfcd6SJosh Poimboeuf while true; do 17124ebfcd6SJosh Poimboeuf case "$1" in 17224ebfcd6SJosh Poimboeuf -h | --help) 17324ebfcd6SJosh Poimboeuf usage 17424ebfcd6SJosh Poimboeuf exit 0 17524ebfcd6SJosh Poimboeuf ;; 17624ebfcd6SJosh Poimboeuf -f | --show-first-changed) 17724ebfcd6SJosh Poimboeuf DIFF_CHECKSUM=1 17824ebfcd6SJosh Poimboeuf shift 17924ebfcd6SJosh Poimboeuf ;; 18024ebfcd6SJosh Poimboeuf -j | --jobs) 18124ebfcd6SJosh Poimboeuf JOBS="$2" 18224ebfcd6SJosh Poimboeuf shift 2 1832c2f0b86SJosh Poimboeuf ;; 1842c2f0b86SJosh Poimboeuf -o | --output) 1852c2f0b86SJosh Poimboeuf [[ "$2" != *.ko ]] && die "output filename should end with .ko" 1862c2f0b86SJosh Poimboeuf OUTFILE="$2" 1872c2f0b86SJosh Poimboeuf NAME="$(basename "$OUTFILE")" 18824ebfcd6SJosh Poimboeuf NAME="${NAME%.ko}" 18924ebfcd6SJosh Poimboeuf NAME="$(module_name_string "$NAME")" 19024ebfcd6SJosh Poimboeuf shift 2 19124ebfcd6SJosh Poimboeuf ;; 19224ebfcd6SJosh Poimboeuf --no-replace) 19324ebfcd6SJosh Poimboeuf REPLACE=0 19424ebfcd6SJosh Poimboeuf shift 19524ebfcd6SJosh Poimboeuf ;; 19624ebfcd6SJosh Poimboeuf -v | --verbose) 19724ebfcd6SJosh Poimboeuf VERBOSE="V=1" 19824ebfcd6SJosh Poimboeuf shift 19924ebfcd6SJosh Poimboeuf ;; 20024ebfcd6SJosh Poimboeuf -d | --debug) 20124ebfcd6SJosh Poimboeuf DEBUG_CLONE=1 20224ebfcd6SJosh Poimboeuf keep_tmp=1 20324ebfcd6SJosh Poimboeuf shift 20424ebfcd6SJosh Poimboeuf ;; 20524ebfcd6SJosh Poimboeuf -S | --short-circuit) 20624ebfcd6SJosh Poimboeuf [[ ! -d "$TMP_DIR" ]] && die "--short-circuit requires preserved klp-tmp dir" 20724ebfcd6SJosh Poimboeuf keep_tmp=1 20824ebfcd6SJosh Poimboeuf case "$2" in 20924ebfcd6SJosh Poimboeuf 1 | orig) SHORT_CIRCUIT=1; ;; 21024ebfcd6SJosh Poimboeuf 2 | patched) SHORT_CIRCUIT=2; ;; 21124ebfcd6SJosh Poimboeuf 3 | diff) SHORT_CIRCUIT=3; ;; 21224ebfcd6SJosh Poimboeuf 4 | mod) SHORT_CIRCUIT=4; ;; 21324ebfcd6SJosh Poimboeuf *) die "invalid short-circuit step '$2'" ;; 21424ebfcd6SJosh Poimboeuf esac 21524ebfcd6SJosh Poimboeuf shift 2 21624ebfcd6SJosh Poimboeuf ;; 21724ebfcd6SJosh Poimboeuf -T | --keep-tmp) 21824ebfcd6SJosh Poimboeuf keep_tmp=1 21924ebfcd6SJosh Poimboeuf shift 22024ebfcd6SJosh Poimboeuf ;; 22124ebfcd6SJosh Poimboeuf --) 22224ebfcd6SJosh Poimboeuf shift 22324ebfcd6SJosh Poimboeuf break 22424ebfcd6SJosh Poimboeuf ;; 22524ebfcd6SJosh Poimboeuf *) 22624ebfcd6SJosh Poimboeuf usage 22724ebfcd6SJosh Poimboeuf exit 1 22824ebfcd6SJosh Poimboeuf ;; 22924ebfcd6SJosh Poimboeuf esac 23024ebfcd6SJosh Poimboeuf done 23124ebfcd6SJosh Poimboeuf 23224ebfcd6SJosh Poimboeuf if [[ $# -eq 0 ]] && (( SHORT_CIRCUIT <= 2 )); then 23324ebfcd6SJosh Poimboeuf usage 23424ebfcd6SJosh Poimboeuf exit 1 23524ebfcd6SJosh Poimboeuf fi 23624ebfcd6SJosh Poimboeuf 23724ebfcd6SJosh Poimboeuf KEEP_TMP="$keep_tmp" 23824ebfcd6SJosh Poimboeuf PATCHES=("$@") 23924ebfcd6SJosh Poimboeuf} 24024ebfcd6SJosh Poimboeuf 24124ebfcd6SJosh Poimboeuf# temporarily disable xtrace for especially verbose code 24224ebfcd6SJosh Poimboeufxtrace_save() { 24324ebfcd6SJosh Poimboeuf [[ -v XTRACE ]] && set +x 24424ebfcd6SJosh Poimboeuf return 0 24524ebfcd6SJosh Poimboeuf} 24624ebfcd6SJosh Poimboeuf 24724ebfcd6SJosh Poimboeufxtrace_restore() { 24824ebfcd6SJosh Poimboeuf [[ -v XTRACE ]] && set -x 24924ebfcd6SJosh Poimboeuf return 0 25024ebfcd6SJosh Poimboeuf} 25124ebfcd6SJosh Poimboeuf 252a8ff29f0SJosh Poimboeufvalidate_config() { 253a8ff29f0SJosh Poimboeuf xtrace_save "reading .config" 254a8ff29f0SJosh Poimboeuf source "$CONFIG" || die "no .config file in $(dirname "$CONFIG")" 255a8ff29f0SJosh Poimboeuf xtrace_restore 25624ebfcd6SJosh Poimboeuf 25724ebfcd6SJosh Poimboeuf [[ -v CONFIG_LIVEPATCH ]] || \ 25824ebfcd6SJosh Poimboeuf die "CONFIG_LIVEPATCH not enabled" 25924ebfcd6SJosh Poimboeuf 26024ebfcd6SJosh Poimboeuf [[ -v CONFIG_KLP_BUILD ]] || \ 26124ebfcd6SJosh Poimboeuf die "CONFIG_KLP_BUILD not enabled" 26224ebfcd6SJosh Poimboeuf 26324ebfcd6SJosh Poimboeuf [[ -v CONFIG_GCC_PLUGIN_LATENT_ENTROPY ]] && \ 26424ebfcd6SJosh Poimboeuf die "kernel option 'CONFIG_GCC_PLUGIN_LATENT_ENTROPY' not supported" 26524ebfcd6SJosh Poimboeuf 26624ebfcd6SJosh Poimboeuf [[ -v CONFIG_GCC_PLUGIN_RANDSTRUCT ]] && \ 26724ebfcd6SJosh Poimboeuf die "kernel option 'CONFIG_GCC_PLUGIN_RANDSTRUCT' not supported" 26824ebfcd6SJosh Poimboeuf 26924ebfcd6SJosh Poimboeuf [[ -v CONFIG_AS_IS_LLVM ]] && \ 27024ebfcd6SJosh Poimboeuf [[ "$CONFIG_AS_VERSION" -lt 200000 ]] && \ 27124ebfcd6SJosh Poimboeuf die "Clang assembler version < 20 not supported" 27224ebfcd6SJosh Poimboeuf 27324ebfcd6SJosh Poimboeuf return 0 27424ebfcd6SJosh Poimboeuf} 27524ebfcd6SJosh Poimboeuf 27624ebfcd6SJosh Poimboeuf# Only allow alphanumerics and '_' and '-' in the module name. Everything else 27724ebfcd6SJosh Poimboeuf# is replaced with '-'. Also truncate to 55 chars so the full name + NUL 27824ebfcd6SJosh Poimboeuf# terminator fits in the kernel's 56-byte module name array. 27924ebfcd6SJosh Poimboeufmodule_name_string() { 28024ebfcd6SJosh Poimboeuf echo "${1//[^a-zA-Z0-9_-]/-}" | cut -c 1-55 28124ebfcd6SJosh Poimboeuf} 28224ebfcd6SJosh Poimboeuf 28324ebfcd6SJosh Poimboeuf# If the module name wasn't specified on the cmdline with --output, give it a 28424ebfcd6SJosh Poimboeuf# name based on the patch name. 28524ebfcd6SJosh Poimboeufset_module_name() { 28624ebfcd6SJosh Poimboeuf [[ -v NAME ]] && return 0 28724ebfcd6SJosh Poimboeuf 288*6f93f7b0SJosh Poimboeuf if [[ "${#PATCHES[@]}" -eq 1 ]]; then 28924ebfcd6SJosh Poimboeuf NAME="$(basename "${PATCHES[0]}")" 29024ebfcd6SJosh Poimboeuf NAME="${NAME%.*}" 29124ebfcd6SJosh Poimboeuf else 292*6f93f7b0SJosh Poimboeuf NAME="patch" 293*6f93f7b0SJosh Poimboeuf fi 29424ebfcd6SJosh Poimboeuf 295*6f93f7b0SJosh Poimboeuf NAME="livepatch-$NAME" 29624ebfcd6SJosh Poimboeuf NAME="$(module_name_string "$NAME")" 29724ebfcd6SJosh Poimboeuf 29824ebfcd6SJosh Poimboeuf OUTFILE="$NAME.ko" 29924ebfcd6SJosh Poimboeuf} 30024ebfcd6SJosh Poimboeuf 30124ebfcd6SJosh Poimboeuf# Hardcode the value printed by the localversion script to prevent patch 30224ebfcd6SJosh Poimboeuf# application from appending it with '+' due to a dirty working tree. 30324ebfcd6SJosh Poimboeufset_kernelversion() { 30424ebfcd6SJosh Poimboeuf local file="$SRC/scripts/setlocalversion" 30524ebfcd6SJosh Poimboeuf local kernelrelease 30624ebfcd6SJosh Poimboeuf 30724ebfcd6SJosh Poimboeuf stash_file "$file" 30824ebfcd6SJosh Poimboeuf 30924ebfcd6SJosh Poimboeuf kernelrelease="$(cd "$SRC" && make syncconfig &>/dev/null && make -s kernelrelease)" 31024ebfcd6SJosh Poimboeuf [[ -z "$kernelrelease" ]] && die "failed to get kernel version" 31124ebfcd6SJosh Poimboeuf 31224ebfcd6SJosh Poimboeuf sed -i "2i echo $kernelrelease; exit 0" scripts/setlocalversion 31324ebfcd6SJosh Poimboeuf} 31424ebfcd6SJosh Poimboeuf 31524ebfcd6SJosh Poimboeufget_patch_input_files() { 31624ebfcd6SJosh Poimboeuf local patch="$1" 31724ebfcd6SJosh Poimboeuf 31824ebfcd6SJosh Poimboeuf grep0 -E '^--- ' "$patch" \ 31924ebfcd6SJosh Poimboeuf | grep0 -v -e '/dev/null' -e '1969-12-31' -e '1970-01-01' \ 32024ebfcd6SJosh Poimboeuf | gawk '{print $2}' \ 32124ebfcd6SJosh Poimboeuf | sed 's|^[^/]*/||' \ 32224ebfcd6SJosh Poimboeuf | sort -u 32324ebfcd6SJosh Poimboeuf} 32424ebfcd6SJosh Poimboeuf 32524ebfcd6SJosh Poimboeufget_patch_output_files() { 32624ebfcd6SJosh Poimboeuf local patch="$1" 32724ebfcd6SJosh Poimboeuf 32824ebfcd6SJosh Poimboeuf grep0 -E '^\+\+\+ ' "$patch" \ 32924ebfcd6SJosh Poimboeuf | grep0 -v -e '/dev/null' -e '1969-12-31' -e '1970-01-01' \ 33024ebfcd6SJosh Poimboeuf | gawk '{print $2}' \ 33124ebfcd6SJosh Poimboeuf | sed 's|^[^/]*/||' \ 33224ebfcd6SJosh Poimboeuf | sort -u 33324ebfcd6SJosh Poimboeuf} 33424ebfcd6SJosh Poimboeuf 33524ebfcd6SJosh Poimboeufget_patch_files() { 33624ebfcd6SJosh Poimboeuf local patch="$1" 33724ebfcd6SJosh Poimboeuf 33824ebfcd6SJosh Poimboeuf { get_patch_input_files "$patch"; get_patch_output_files "$patch"; } \ 33924ebfcd6SJosh Poimboeuf | sort -u 34024ebfcd6SJosh Poimboeuf} 34124ebfcd6SJosh Poimboeuf 34224ebfcd6SJosh Poimboeufcheck_unsupported_patches() { 34324ebfcd6SJosh Poimboeuf local patch 34424ebfcd6SJosh Poimboeuf 34524ebfcd6SJosh Poimboeuf for patch in "${PATCHES[@]}"; do 34624ebfcd6SJosh Poimboeuf local files=() 34724ebfcd6SJosh Poimboeuf 34824ebfcd6SJosh Poimboeuf get_patch_files "$patch" | mapfile -t files 34924ebfcd6SJosh Poimboeuf 35024ebfcd6SJosh Poimboeuf for file in "${files[@]}"; do 35124ebfcd6SJosh Poimboeuf case "$file" in 35224ebfcd6SJosh Poimboeuf lib/*|*.S) 35324ebfcd6SJosh Poimboeuf die "${patch}: unsupported patch to $file" 35424ebfcd6SJosh Poimboeuf ;; 35524ebfcd6SJosh Poimboeuf esac 35624ebfcd6SJosh Poimboeuf done 35724ebfcd6SJosh Poimboeuf done 35824ebfcd6SJosh Poimboeuf} 35924ebfcd6SJosh Poimboeuf 36024ebfcd6SJosh Poimboeufapply_patch() { 36124ebfcd6SJosh Poimboeuf local patch="$1" 36224ebfcd6SJosh Poimboeuf shift 36324ebfcd6SJosh Poimboeuf local extra_args=("$@") 36424ebfcd6SJosh Poimboeuf local drift_regex="with fuzz|offset [0-9]+ line" 36524ebfcd6SJosh Poimboeuf local output 36624ebfcd6SJosh Poimboeuf local status 36724ebfcd6SJosh Poimboeuf 36824ebfcd6SJosh Poimboeuf [[ ! -f "$patch" ]] && die "$patch doesn't exist" 36924ebfcd6SJosh Poimboeuf status=0 37024ebfcd6SJosh Poimboeuf output=$(patch -d "$SRC" -p1 --dry-run --no-backup-if-mismatch -r /dev/null "${extra_args[@]}" < "$patch" 2>&1) || status=$? 37124ebfcd6SJosh Poimboeuf if [[ "$status" -ne 0 ]]; then 37224ebfcd6SJosh Poimboeuf echo "$output" >&2 37324ebfcd6SJosh Poimboeuf die "$patch did not apply" 37424ebfcd6SJosh Poimboeuf elif [[ "$output" =~ $drift_regex ]]; then 37524ebfcd6SJosh Poimboeuf echo "$output" >&2 37624ebfcd6SJosh Poimboeuf warn "${patch} applied with fuzz" 37724ebfcd6SJosh Poimboeuf fi 37824ebfcd6SJosh Poimboeuf 37924ebfcd6SJosh Poimboeuf patch -d "$SRC" -p1 --no-backup-if-mismatch -r /dev/null "${extra_args[@]}" --silent < "$patch" 38024ebfcd6SJosh Poimboeuf APPLIED_PATCHES+=("$patch") 38124ebfcd6SJosh Poimboeuf} 38224ebfcd6SJosh Poimboeuf 38324ebfcd6SJosh Poimboeufrevert_patch() { 38424ebfcd6SJosh Poimboeuf local patch="$1" 38524ebfcd6SJosh Poimboeuf local tmp=() 38624ebfcd6SJosh Poimboeuf 38724ebfcd6SJosh Poimboeuf patch -d "$SRC" -p1 -R --silent --no-backup-if-mismatch -r /dev/null < "$patch" 38824ebfcd6SJosh Poimboeuf 38924ebfcd6SJosh Poimboeuf for p in "${APPLIED_PATCHES[@]}"; do 39024ebfcd6SJosh Poimboeuf [[ "$p" == "$patch" ]] && continue 39124ebfcd6SJosh Poimboeuf tmp+=("$p") 39224ebfcd6SJosh Poimboeuf done 39324ebfcd6SJosh Poimboeuf 39424ebfcd6SJosh Poimboeuf APPLIED_PATCHES=("${tmp[@]}") 39524ebfcd6SJosh Poimboeuf} 39624ebfcd6SJosh Poimboeuf 39724ebfcd6SJosh Poimboeufapply_patches() { 39824ebfcd6SJosh Poimboeuf local extra_args=("$@") 39924ebfcd6SJosh Poimboeuf local patch 40024ebfcd6SJosh Poimboeuf 40124ebfcd6SJosh Poimboeuf for patch in "${PATCHES[@]}"; do 40224ebfcd6SJosh Poimboeuf apply_patch "$patch" "${extra_args[@]}" 40324ebfcd6SJosh Poimboeuf done 40424ebfcd6SJosh Poimboeuf} 40524ebfcd6SJosh Poimboeuf 40624ebfcd6SJosh Poimboeufrevert_patches() { 40724ebfcd6SJosh Poimboeuf local patches=("${APPLIED_PATCHES[@]}") 40824ebfcd6SJosh Poimboeuf 40924ebfcd6SJosh Poimboeuf for (( i=${#patches[@]}-1 ; i>=0 ; i-- )) ; do 41024ebfcd6SJosh Poimboeuf revert_patch "${patches[$i]}" 41124ebfcd6SJosh Poimboeuf done 41224ebfcd6SJosh Poimboeuf 41324ebfcd6SJosh Poimboeuf APPLIED_PATCHES=() 41424ebfcd6SJosh Poimboeuf} 41524ebfcd6SJosh Poimboeuf 41624ebfcd6SJosh Poimboeufvalidate_patches() { 41724ebfcd6SJosh Poimboeuf check_unsupported_patches 41824ebfcd6SJosh Poimboeuf apply_patches 41924ebfcd6SJosh Poimboeuf revert_patches 42024ebfcd6SJosh Poimboeuf} 42124ebfcd6SJosh Poimboeuf 42224ebfcd6SJosh Poimboeufdo_init() { 42324ebfcd6SJosh Poimboeuf # We're not yet smart enough to handle anything other than in-tree 42424ebfcd6SJosh Poimboeuf # builds in pwd. 42524ebfcd6SJosh Poimboeuf [[ ! "$SRC" -ef "$SCRIPT_DIR/../.." ]] && die "please run from the kernel root directory" 42624ebfcd6SJosh Poimboeuf [[ ! "$OBJ" -ef "$SCRIPT_DIR/../.." ]] && die "please run from the kernel root directory" 42724ebfcd6SJosh Poimboeuf 42824ebfcd6SJosh Poimboeuf (( SHORT_CIRCUIT <= 1 )) && rm -rf "$TMP_DIR" 42924ebfcd6SJosh Poimboeuf mkdir -p "$TMP_DIR" 43024ebfcd6SJosh Poimboeuf 43124ebfcd6SJosh Poimboeuf APPLIED_PATCHES=() 43224ebfcd6SJosh Poimboeuf 43324ebfcd6SJosh Poimboeuf [[ -x "$FIX_PATCH_LINES" ]] || die "can't find fix-patch-lines" 43424ebfcd6SJosh Poimboeuf command -v recountdiff &>/dev/null || die "recountdiff not found (install patchutils)" 43524ebfcd6SJosh Poimboeuf 43624ebfcd6SJosh Poimboeuf validate_config 43724ebfcd6SJosh Poimboeuf set_module_name 43824ebfcd6SJosh Poimboeuf set_kernelversion 43924ebfcd6SJosh Poimboeuf} 44024ebfcd6SJosh Poimboeuf 44124ebfcd6SJosh Poimboeuf# Refresh the patch hunk headers, specifically the line numbers and counts. 44224ebfcd6SJosh Poimboeufrefresh_patch() { 44324ebfcd6SJosh Poimboeuf local patch="$1" 44424ebfcd6SJosh Poimboeuf local tmpdir="$PATCH_TMP_DIR" 44524ebfcd6SJosh Poimboeuf local input_files=() 44624ebfcd6SJosh Poimboeuf local output_files=() 44724ebfcd6SJosh Poimboeuf 44824ebfcd6SJosh Poimboeuf rm -rf "$tmpdir" 44924ebfcd6SJosh Poimboeuf mkdir -p "$tmpdir/a" 45024ebfcd6SJosh Poimboeuf mkdir -p "$tmpdir/b" 45124ebfcd6SJosh Poimboeuf 45224ebfcd6SJosh Poimboeuf # Get all source files affected by the patch 45324ebfcd6SJosh Poimboeuf get_patch_input_files "$patch" | mapfile -t input_files 45424ebfcd6SJosh Poimboeuf get_patch_output_files "$patch" | mapfile -t output_files 45524ebfcd6SJosh Poimboeuf 45624ebfcd6SJosh Poimboeuf # Copy orig source files to 'a' 45724ebfcd6SJosh Poimboeuf ( cd "$SRC" && echo "${input_files[@]}" | xargs cp --parents --target-directory="$tmpdir/a" ) 45824ebfcd6SJosh Poimboeuf 45924ebfcd6SJosh Poimboeuf # Copy patched source files to 'b' 46024ebfcd6SJosh Poimboeuf apply_patch "$patch" "--silent" 46124ebfcd6SJosh Poimboeuf ( cd "$SRC" && echo "${output_files[@]}" | xargs cp --parents --target-directory="$tmpdir/b" ) 46224ebfcd6SJosh Poimboeuf revert_patch "$patch" 46324ebfcd6SJosh Poimboeuf 46424ebfcd6SJosh Poimboeuf # Diff 'a' and 'b' to make a clean patch 46524ebfcd6SJosh Poimboeuf ( cd "$tmpdir" && diff -Nupr a b > "$patch" ) || true 46624ebfcd6SJosh Poimboeuf} 46724ebfcd6SJosh Poimboeuf 46824ebfcd6SJosh Poimboeuf# Copy the patches to a temporary directory, fix their lines so as not to 46924ebfcd6SJosh Poimboeuf# affect the __LINE__ macro for otherwise unchanged functions further down the 47024ebfcd6SJosh Poimboeuf# file, and update $PATCHES to point to the fixed patches. 47124ebfcd6SJosh Poimboeuffix_patches() { 47224ebfcd6SJosh Poimboeuf local idx 47324ebfcd6SJosh Poimboeuf local i 47424ebfcd6SJosh Poimboeuf 47524ebfcd6SJosh Poimboeuf rm -f "$TMP_DIR"/*.patch 47624ebfcd6SJosh Poimboeuf 47724ebfcd6SJosh Poimboeuf idx=0001 47824ebfcd6SJosh Poimboeuf for i in "${!PATCHES[@]}"; do 47924ebfcd6SJosh Poimboeuf local old_patch="${PATCHES[$i]}" 48024ebfcd6SJosh Poimboeuf local tmp_patch="$TMP_DIR/tmp.patch" 48124ebfcd6SJosh Poimboeuf local patch="${PATCHES[$i]}" 48224ebfcd6SJosh Poimboeuf local new_patch 48324ebfcd6SJosh Poimboeuf 48424ebfcd6SJosh Poimboeuf new_patch="$TMP_DIR/$idx-fixed-$(basename "$patch")" 48524ebfcd6SJosh Poimboeuf 48624ebfcd6SJosh Poimboeuf cp -f "$old_patch" "$tmp_patch" 48724ebfcd6SJosh Poimboeuf refresh_patch "$tmp_patch" 48824ebfcd6SJosh Poimboeuf "$FIX_PATCH_LINES" "$tmp_patch" | recountdiff > "$new_patch" 48924ebfcd6SJosh Poimboeuf 49024ebfcd6SJosh Poimboeuf PATCHES[i]="$new_patch" 49124ebfcd6SJosh Poimboeuf 49224ebfcd6SJosh Poimboeuf rm -f "$tmp_patch" 49324ebfcd6SJosh Poimboeuf idx=$(printf "%04d" $(( 10#$idx + 1 ))) 49424ebfcd6SJosh Poimboeuf done 4952092007aSJosh Poimboeuf} 49624ebfcd6SJosh Poimboeuf 49724ebfcd6SJosh Poimboeufclean_kernel() { 4982092007aSJosh Poimboeuf local cmd=() 4992092007aSJosh Poimboeuf 50024ebfcd6SJosh Poimboeuf cmd=("make") 50124ebfcd6SJosh Poimboeuf cmd+=("--silent") 50224ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 50324ebfcd6SJosh Poimboeuf cmd+=("clean") 50424ebfcd6SJosh Poimboeuf 50524ebfcd6SJosh Poimboeuf ( 50624ebfcd6SJosh Poimboeuf cd "$SRC" 50724ebfcd6SJosh Poimboeuf "${cmd[@]}" 50824ebfcd6SJosh Poimboeuf ) 50924ebfcd6SJosh Poimboeuf} 51024ebfcd6SJosh Poimboeuf 51124ebfcd6SJosh Poimboeufbuild_kernel() { 51224ebfcd6SJosh Poimboeuf local build="$1" 51324ebfcd6SJosh Poimboeuf local log="$TMP_DIR/build.log" 51424ebfcd6SJosh Poimboeuf local objtool_args=() 51524ebfcd6SJosh Poimboeuf local cmd=() 51624ebfcd6SJosh Poimboeuf 51724ebfcd6SJosh Poimboeuf objtool_args=("--checksum") 51824ebfcd6SJosh Poimboeuf 51924ebfcd6SJosh Poimboeuf cmd=("make") 52024ebfcd6SJosh Poimboeuf 52124ebfcd6SJosh Poimboeuf # When a patch to a kernel module references a newly created unexported 5222092007aSJosh Poimboeuf # symbol which lives in vmlinux or another kernel module, the patched 52324ebfcd6SJosh Poimboeuf # kernel build fails with the following error: 52424ebfcd6SJosh Poimboeuf # 52524ebfcd6SJosh Poimboeuf # ERROR: modpost: "klp_string" [fs/xfs/xfs.ko] undefined! 52624ebfcd6SJosh Poimboeuf # 52724ebfcd6SJosh Poimboeuf # The undefined symbols are working as designed in that case. They get 52824ebfcd6SJosh Poimboeuf # resolved later when the livepatch module build link pulls all the 52924ebfcd6SJosh Poimboeuf # disparate objects together into the same kernel module. 53024ebfcd6SJosh Poimboeuf # 53124ebfcd6SJosh Poimboeuf # It would be good to have a way to tell modpost to skip checking for 53224ebfcd6SJosh Poimboeuf # undefined symbols altogether. For now, just convert the error to a 53324ebfcd6SJosh Poimboeuf # warning with KBUILD_MODPOST_WARN, and grep out the warning to avoid 53424ebfcd6SJosh Poimboeuf # confusing the user. 53524ebfcd6SJosh Poimboeuf # 53624ebfcd6SJosh Poimboeuf cmd+=("KBUILD_MODPOST_WARN=1") 53724ebfcd6SJosh Poimboeuf 53824ebfcd6SJosh Poimboeuf cmd+=("$VERBOSE") 53924ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 54024ebfcd6SJosh Poimboeuf cmd+=("KCFLAGS=-ffunction-sections -fdata-sections") 54124ebfcd6SJosh Poimboeuf cmd+=("OBJTOOL_ARGS=${objtool_args[*]}") 54224ebfcd6SJosh Poimboeuf cmd+=("vmlinux") 54324ebfcd6SJosh Poimboeuf cmd+=("modules") 54424ebfcd6SJosh Poimboeuf 54524ebfcd6SJosh Poimboeuf ( 54624ebfcd6SJosh Poimboeuf cd "$SRC" 54724ebfcd6SJosh Poimboeuf "${cmd[@]}" \ 54824ebfcd6SJosh Poimboeuf 1> >(tee -a "$log") \ 54924ebfcd6SJosh Poimboeuf 2> >(tee -a "$log" | grep0 -v "modpost.*undefined!" >&2) 55024ebfcd6SJosh Poimboeuf ) || die "$build kernel build failed" 55124ebfcd6SJosh Poimboeuf} 55224ebfcd6SJosh Poimboeuf 55324ebfcd6SJosh Poimboeuffind_objects() { 55424ebfcd6SJosh Poimboeuf local opts=("$@") 55524ebfcd6SJosh Poimboeuf 55624ebfcd6SJosh Poimboeuf # Find root-level vmlinux.o and non-root-level .ko files, 55724ebfcd6SJosh Poimboeuf # excluding klp-tmp/ and .git/ 55824ebfcd6SJosh Poimboeuf find "$OBJ" \( -path "$TMP_DIR" -o -path "$OBJ/.git" -o -regex "$OBJ/[^/][^/]*\.ko" \) -prune -o \ 55924ebfcd6SJosh Poimboeuf -type f "${opts[@]}" \ 56024ebfcd6SJosh Poimboeuf \( -name "*.ko" -o -path "$OBJ/vmlinux.o" \) \ 56124ebfcd6SJosh Poimboeuf -printf '%P\n' 56224ebfcd6SJosh Poimboeuf} 56324ebfcd6SJosh Poimboeuf 56424ebfcd6SJosh Poimboeuf# Copy all .o archives to $ORIG_DIR 56524ebfcd6SJosh Poimboeufcopy_orig_objects() { 56624ebfcd6SJosh Poimboeuf local files=() 56724ebfcd6SJosh Poimboeuf 56824ebfcd6SJosh Poimboeuf rm -rf "$ORIG_DIR" 56924ebfcd6SJosh Poimboeuf mkdir -p "$ORIG_DIR" 57024ebfcd6SJosh Poimboeuf 57124ebfcd6SJosh Poimboeuf find_objects | mapfile -t files 57224ebfcd6SJosh Poimboeuf 57324ebfcd6SJosh Poimboeuf xtrace_save "copying orig objects" 57424ebfcd6SJosh Poimboeuf for _file in "${files[@]}"; do 57524ebfcd6SJosh Poimboeuf local rel_file="${_file/.ko/.o}" 57624ebfcd6SJosh Poimboeuf local file="$OBJ/$rel_file" 57724ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 57824ebfcd6SJosh Poimboeuf local orig_dir="$(dirname "$orig_file")" 57924ebfcd6SJosh Poimboeuf 58024ebfcd6SJosh Poimboeuf [[ ! -f "$file" ]] && die "missing $(basename "$file") for $_file" 58124ebfcd6SJosh Poimboeuf 58224ebfcd6SJosh Poimboeuf mkdir -p "$orig_dir" 58324ebfcd6SJosh Poimboeuf cp -f "$file" "$orig_dir" 58424ebfcd6SJosh Poimboeuf done 58524ebfcd6SJosh Poimboeuf xtrace_restore 58624ebfcd6SJosh Poimboeuf 58724ebfcd6SJosh Poimboeuf mv -f "$TMP_DIR/build.log" "$ORIG_DIR" 58824ebfcd6SJosh Poimboeuf touch "$TIMESTAMP" 58924ebfcd6SJosh Poimboeuf} 59024ebfcd6SJosh Poimboeuf 59124ebfcd6SJosh Poimboeuf# Copy all changed objects to $PATCHED_DIR 59224ebfcd6SJosh Poimboeufcopy_patched_objects() { 59324ebfcd6SJosh Poimboeuf local files=() 59424ebfcd6SJosh Poimboeuf local opts=() 59524ebfcd6SJosh Poimboeuf local found=0 59624ebfcd6SJosh Poimboeuf 59724ebfcd6SJosh Poimboeuf rm -rf "$PATCHED_DIR" 59824ebfcd6SJosh Poimboeuf mkdir -p "$PATCHED_DIR" 59924ebfcd6SJosh Poimboeuf 60024ebfcd6SJosh Poimboeuf # Note this doesn't work with some configs, thus the 'cmp' below. 60124ebfcd6SJosh Poimboeuf opts=("-newer") 60224ebfcd6SJosh Poimboeuf opts+=("$TIMESTAMP") 60324ebfcd6SJosh Poimboeuf 60424ebfcd6SJosh Poimboeuf find_objects "${opts[@]}" | mapfile -t files 60524ebfcd6SJosh Poimboeuf 60624ebfcd6SJosh Poimboeuf xtrace_save "copying changed objects" 60724ebfcd6SJosh Poimboeuf for _file in "${files[@]}"; do 60824ebfcd6SJosh Poimboeuf local rel_file="${_file/.ko/.o}" 60924ebfcd6SJosh Poimboeuf local file="$OBJ/$rel_file" 61024ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 61124ebfcd6SJosh Poimboeuf local patched_file="$PATCHED_DIR/$rel_file" 61224ebfcd6SJosh Poimboeuf local patched_dir="$(dirname "$patched_file")" 61324ebfcd6SJosh Poimboeuf 61424ebfcd6SJosh Poimboeuf [[ ! -f "$file" ]] && die "missing $(basename "$file") for $_file" 6152c2f0b86SJosh Poimboeuf 61624ebfcd6SJosh Poimboeuf cmp -s "$orig_file" "$file" && continue 61724ebfcd6SJosh Poimboeuf 61824ebfcd6SJosh Poimboeuf mkdir -p "$patched_dir" 61924ebfcd6SJosh Poimboeuf cp -f "$file" "$patched_dir" 62024ebfcd6SJosh Poimboeuf found=1 62124ebfcd6SJosh Poimboeuf done 62224ebfcd6SJosh Poimboeuf xtrace_restore 6232c2f0b86SJosh Poimboeuf 6242c2f0b86SJosh Poimboeuf (( found == 0 )) && die "no changes detected" 62524ebfcd6SJosh Poimboeuf 62624ebfcd6SJosh Poimboeuf mv -f "$TMP_DIR/build.log" "$PATCHED_DIR" 62724ebfcd6SJosh Poimboeuf} 62824ebfcd6SJosh Poimboeuf 62924ebfcd6SJosh Poimboeuf# Diff changed objects, writing output object to $DIFF_DIR 63024ebfcd6SJosh Poimboeufdiff_objects() { 63178be9facSJosh Poimboeuf local log="$KLP_DIFF_LOG" 63224ebfcd6SJosh Poimboeuf local files=() 63324ebfcd6SJosh Poimboeuf local opts=() 63424ebfcd6SJosh Poimboeuf 63524ebfcd6SJosh Poimboeuf rm -rf "$DIFF_DIR" 63624ebfcd6SJosh Poimboeuf mkdir -p "$DIFF_DIR" 63724ebfcd6SJosh Poimboeuf 63824ebfcd6SJosh Poimboeuf find "$PATCHED_DIR" -type f -name "*.o" | mapfile -t files 6392c2f0b86SJosh Poimboeuf [[ ${#files[@]} -eq 0 ]] && die "no changes detected" 64024ebfcd6SJosh Poimboeuf 64124ebfcd6SJosh Poimboeuf [[ -v DEBUG_CLONE ]] && opts=("--debug") 64224ebfcd6SJosh Poimboeuf 64324ebfcd6SJosh Poimboeuf # Diff all changed objects 64478be9facSJosh Poimboeuf for file in "${files[@]}"; do 64578be9facSJosh Poimboeuf local rel_file="${file#"$PATCHED_DIR"/}" 64678be9facSJosh Poimboeuf local orig_file="$rel_file" 64778be9facSJosh Poimboeuf local patched_file="$PATCHED_DIR/$rel_file" 64878be9facSJosh Poimboeuf local out_file="$DIFF_DIR/$rel_file" 64978be9facSJosh Poimboeuf local filter=() 65078be9facSJosh Poimboeuf local cmd=() 65178be9facSJosh Poimboeuf 65224ebfcd6SJosh Poimboeuf mkdir -p "$(dirname "$out_file")" 65324ebfcd6SJosh Poimboeuf 65424ebfcd6SJosh Poimboeuf cmd=("$SRC/tools/objtool/objtool") 65524ebfcd6SJosh Poimboeuf cmd+=("klp") 65678be9facSJosh Poimboeuf cmd+=("diff") 65724ebfcd6SJosh Poimboeuf (( ${#opts[@]} > 0 )) && cmd+=("${opts[@]}") 65824ebfcd6SJosh Poimboeuf cmd+=("$orig_file") 65924ebfcd6SJosh Poimboeuf cmd+=("$patched_file") 66024ebfcd6SJosh Poimboeuf cmd+=("$out_file") 66124ebfcd6SJosh Poimboeuf 66278be9facSJosh Poimboeuf if [[ -v DIFF_CHECKSUM ]]; then 66378be9facSJosh Poimboeuf filter=("grep0") 66478be9facSJosh Poimboeuf filter+=("-Ev") 66578be9facSJosh Poimboeuf filter+=("DEBUG: .*checksum: ") 66678be9facSJosh Poimboeuf else 66778be9facSJosh Poimboeuf filter=("cat") 66878be9facSJosh Poimboeuf fi 66978be9facSJosh Poimboeuf 67078be9facSJosh Poimboeuf ( 67178be9facSJosh Poimboeuf cd "$ORIG_DIR" 67278be9facSJosh Poimboeuf "${cmd[@]}" \ 67378be9facSJosh Poimboeuf 1> >(tee -a "$log") \ 67478be9facSJosh Poimboeuf 2> >(tee -a "$log" | "${filter[@]}" >&2) || \ 67578be9facSJosh Poimboeuf die "objtool klp diff failed" 67678be9facSJosh Poimboeuf ) 67778be9facSJosh Poimboeuf done 67878be9facSJosh Poimboeuf} 67978be9facSJosh Poimboeuf 68078be9facSJosh Poimboeuf# For each changed object, run objtool with --debug-checksum to get the 68178be9facSJosh Poimboeuf# per-instruction checksums, and then diff those to find the first changed 68278be9facSJosh Poimboeuf# instruction for each function. 68378be9facSJosh Poimboeufdiff_checksums() { 68478be9facSJosh Poimboeuf local orig_log="$ORIG_DIR/checksum.log" 68578be9facSJosh Poimboeuf local patched_log="$PATCHED_DIR/checksum.log" 68678be9facSJosh Poimboeuf local -A funcs 68778be9facSJosh Poimboeuf local cmd=() 68878be9facSJosh Poimboeuf local line 68978be9facSJosh Poimboeuf local file 69078be9facSJosh Poimboeuf local func 69178be9facSJosh Poimboeuf 69278be9facSJosh Poimboeuf gawk '/\.o: changed function: / { 69378be9facSJosh Poimboeuf sub(/:$/, "", $1) 69478be9facSJosh Poimboeuf print $1, $NF 69578be9facSJosh Poimboeuf }' "$KLP_DIFF_LOG" | mapfile -t lines 69678be9facSJosh Poimboeuf 69778be9facSJosh Poimboeuf for line in "${lines[@]}"; do 69878be9facSJosh Poimboeuf read -r file func <<< "$line" 69978be9facSJosh Poimboeuf if [[ ! -v funcs["$file"] ]]; then 70078be9facSJosh Poimboeuf funcs["$file"]="$func" 70178be9facSJosh Poimboeuf else 70278be9facSJosh Poimboeuf funcs["$file"]+=" $func" 70378be9facSJosh Poimboeuf fi 70478be9facSJosh Poimboeuf done 70578be9facSJosh Poimboeuf 70678be9facSJosh Poimboeuf cmd=("$SRC/tools/objtool/objtool") 70778be9facSJosh Poimboeuf cmd+=("--checksum") 70878be9facSJosh Poimboeuf cmd+=("--link") 70978be9facSJosh Poimboeuf cmd+=("--dry-run") 71078be9facSJosh Poimboeuf 71178be9facSJosh Poimboeuf for file in "${!funcs[@]}"; do 71278be9facSJosh Poimboeuf local opt="--debug-checksum=${funcs[$file]// /,}" 71378be9facSJosh Poimboeuf 71478be9facSJosh Poimboeuf ( 71578be9facSJosh Poimboeuf cd "$ORIG_DIR" 71678be9facSJosh Poimboeuf "${cmd[@]}" "$opt" "$file" &> "$orig_log" || \ 71778be9facSJosh Poimboeuf ( cat "$orig_log" >&2; die "objtool --debug-checksum failed" ) 71824ebfcd6SJosh Poimboeuf 71924ebfcd6SJosh Poimboeuf cd "$PATCHED_DIR" 72024ebfcd6SJosh Poimboeuf "${cmd[@]}" "$opt" "$file" &> "$patched_log" || \ 72124ebfcd6SJosh Poimboeuf ( cat "$patched_log" >&2; die "objtool --debug-checksum failed" ) 72224ebfcd6SJosh Poimboeuf ) 72324ebfcd6SJosh Poimboeuf 72424ebfcd6SJosh Poimboeuf for func in ${funcs[$file]}; do 72524ebfcd6SJosh Poimboeuf diff <( grep0 -E "^DEBUG: .*checksum: $func " "$orig_log" | sed "s|$ORIG_DIR/||") \ 72624ebfcd6SJosh Poimboeuf <( grep0 -E "^DEBUG: .*checksum: $func " "$patched_log" | sed "s|$PATCHED_DIR/||") \ 72724ebfcd6SJosh Poimboeuf | gawk '/^< DEBUG: / { 72824ebfcd6SJosh Poimboeuf gsub(/:/, "") 72924ebfcd6SJosh Poimboeuf printf "%s: %s: %s\n", $3, $5, $6 73024ebfcd6SJosh Poimboeuf exit 73124ebfcd6SJosh Poimboeuf }' || true 73224ebfcd6SJosh Poimboeuf done 73324ebfcd6SJosh Poimboeuf done 73424ebfcd6SJosh Poimboeuf} 73524ebfcd6SJosh Poimboeuf 73624ebfcd6SJosh Poimboeuf# Build and post-process livepatch module in $KMOD_DIR 73724ebfcd6SJosh Poimboeufbuild_patch_module() { 73824ebfcd6SJosh Poimboeuf local makefile="$KMOD_DIR/Kbuild" 73924ebfcd6SJosh Poimboeuf local log="$KMOD_DIR/build.log" 74024ebfcd6SJosh Poimboeuf local kmod_file 74124ebfcd6SJosh Poimboeuf local cflags=() 74224ebfcd6SJosh Poimboeuf local files=() 74324ebfcd6SJosh Poimboeuf local cmd=() 74478c268f3SJosh Poimboeuf 74524ebfcd6SJosh Poimboeuf rm -rf "$KMOD_DIR" 74624ebfcd6SJosh Poimboeuf mkdir -p "$KMOD_DIR" 74724ebfcd6SJosh Poimboeuf 74824ebfcd6SJosh Poimboeuf cp -f "$SRC/scripts/livepatch/init.c" "$KMOD_DIR" 74924ebfcd6SJosh Poimboeuf 75024ebfcd6SJosh Poimboeuf echo "obj-m := $NAME.o" > "$makefile" 75124ebfcd6SJosh Poimboeuf echo -n "$NAME-y := init.o" >> "$makefile" 75278c268f3SJosh Poimboeuf 75378c268f3SJosh Poimboeuf find "$DIFF_DIR" -type f -name "*.o" | mapfile -t files 75478c268f3SJosh Poimboeuf [[ ${#files[@]} -eq 0 ]] && die "no changes detected" 75524ebfcd6SJosh Poimboeuf 75624ebfcd6SJosh Poimboeuf for file in "${files[@]}"; do 75724ebfcd6SJosh Poimboeuf local rel_file="${file#"$DIFF_DIR"/}" 75824ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 75924ebfcd6SJosh Poimboeuf local orig_dir="$(dirname "$orig_file")" 76024ebfcd6SJosh Poimboeuf local kmod_file="$KMOD_DIR/$rel_file" 76124ebfcd6SJosh Poimboeuf local kmod_dir="$(dirname "$kmod_file")" 76224ebfcd6SJosh Poimboeuf local cmd_file="$kmod_dir/.$(basename "$file").cmd" 76324ebfcd6SJosh Poimboeuf 76424ebfcd6SJosh Poimboeuf mkdir -p "$kmod_dir" 76524ebfcd6SJosh Poimboeuf cp -f "$file" "$kmod_dir" 76624ebfcd6SJosh Poimboeuf 76724ebfcd6SJosh Poimboeuf # Tell kbuild this is a prebuilt object 76824ebfcd6SJosh Poimboeuf cp -f "$file" "${kmod_file}_shipped" 76924ebfcd6SJosh Poimboeuf 77024ebfcd6SJosh Poimboeuf # Make modpost happy 77124ebfcd6SJosh Poimboeuf touch "$cmd_file" 77224ebfcd6SJosh Poimboeuf 77324ebfcd6SJosh Poimboeuf echo -n " $rel_file" >> "$makefile" 77424ebfcd6SJosh Poimboeuf done 77524ebfcd6SJosh Poimboeuf 77624ebfcd6SJosh Poimboeuf echo >> "$makefile" 77724ebfcd6SJosh Poimboeuf 77824ebfcd6SJosh Poimboeuf cflags=("-ffunction-sections") 77924ebfcd6SJosh Poimboeuf cflags+=("-fdata-sections") 78024ebfcd6SJosh Poimboeuf [[ $REPLACE -eq 0 ]] && cflags+=("-DKLP_NO_REPLACE") 78124ebfcd6SJosh Poimboeuf 78224ebfcd6SJosh Poimboeuf cmd=("make") 78324ebfcd6SJosh Poimboeuf cmd+=("$VERBOSE") 78424ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 78524ebfcd6SJosh Poimboeuf cmd+=("--directory=.") 78624ebfcd6SJosh Poimboeuf cmd+=("M=$KMOD_DIR") 78724ebfcd6SJosh Poimboeuf cmd+=("KCFLAGS=${cflags[*]}") 78824ebfcd6SJosh Poimboeuf 78924ebfcd6SJosh Poimboeuf # Build a "normal" kernel module with init.c and the diffed objects 79024ebfcd6SJosh Poimboeuf ( 79124ebfcd6SJosh Poimboeuf cd "$SRC" 79224ebfcd6SJosh Poimboeuf "${cmd[@]}" \ 79324ebfcd6SJosh Poimboeuf 1> >(tee -a "$log") \ 79424ebfcd6SJosh Poimboeuf 2> >(tee -a "$log" >&2) 79524ebfcd6SJosh Poimboeuf ) 79624ebfcd6SJosh Poimboeuf 79724ebfcd6SJosh Poimboeuf kmod_file="$KMOD_DIR/$NAME.ko" 79824ebfcd6SJosh Poimboeuf 79924ebfcd6SJosh Poimboeuf # Save off the intermediate binary for debugging 80024ebfcd6SJosh Poimboeuf cp -f "$kmod_file" "$kmod_file.orig" 80124ebfcd6SJosh Poimboeuf 80224ebfcd6SJosh Poimboeuf # Work around issue where slight .config change makes corrupt BTF 80324ebfcd6SJosh Poimboeuf objcopy --remove-section=.BTF "$kmod_file" 80424ebfcd6SJosh Poimboeuf 80524ebfcd6SJosh Poimboeuf # Fix (and work around) linker wreckage for klp syms / relocs 80624ebfcd6SJosh Poimboeuf "$SRC/tools/objtool/objtool" klp post-link "$kmod_file" || die "objtool klp post-link failed" 80724ebfcd6SJosh Poimboeuf 80824ebfcd6SJosh Poimboeuf cp -f "$kmod_file" "$OUTFILE" 80924ebfcd6SJosh Poimboeuf} 81024ebfcd6SJosh Poimboeuf 81124ebfcd6SJosh Poimboeuf 81224ebfcd6SJosh Poimboeuf################################################################################ 81324ebfcd6SJosh Poimboeuf 81424ebfcd6SJosh Poimboeufprocess_args "$@" 81524ebfcd6SJosh Poimboeufdo_init 81624ebfcd6SJosh Poimboeuf 81724ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 2 )); then 81824ebfcd6SJosh Poimboeuf status "Validating patch(es)" 81924ebfcd6SJosh Poimboeuf validate_patches 82024ebfcd6SJosh Poimboeuffi 82124ebfcd6SJosh Poimboeuf 82224ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 1 )); then 82378be9facSJosh Poimboeuf status "Building original kernel" 82478be9facSJosh Poimboeuf clean_kernel 82578be9facSJosh Poimboeuf build_kernel "original" 82678be9facSJosh Poimboeuf status "Copying original object files" 82724ebfcd6SJosh Poimboeuf copy_orig_objects 82824ebfcd6SJosh Poimboeuffi 82924ebfcd6SJosh Poimboeuf 83024ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 2 )); then 83124ebfcd6SJosh Poimboeuf status "Fixing patch(es)" 83224ebfcd6SJosh Poimboeuf fix_patches 83324ebfcd6SJosh Poimboeuf apply_patches "--silent" 83424ebfcd6SJosh Poimboeuf status "Building patched kernel" 835 build_kernel "patched" 836 revert_patches 837 status "Copying patched object files" 838 copy_patched_objects 839fi 840 841if (( SHORT_CIRCUIT <= 3 )); then 842 status "Diffing objects" 843 diff_objects 844 if [[ -v DIFF_CHECKSUM ]]; then 845 status "Finding first changed instructions" 846 diff_checksums 847 fi 848fi 849 850if (( SHORT_CIRCUIT <= 4 )); then 851 status "Building patch module: $OUTFILE" 852 build_patch_module 853fi 854 855status "SUCCESS" 856