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 23*78be9facSJosh 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 Poimboeufgrep0() { 5624ebfcd6SJosh Poimboeuf command grep "$@" || true 5724ebfcd6SJosh Poimboeuf} 5824ebfcd6SJosh Poimboeuf 5924ebfcd6SJosh Poimboeufstatus() { 6024ebfcd6SJosh Poimboeuf echo "$*" 6124ebfcd6SJosh Poimboeuf} 6224ebfcd6SJosh Poimboeuf 6324ebfcd6SJosh Poimboeufwarn() { 6424ebfcd6SJosh Poimboeuf echo "error: $SCRIPT: $*" >&2 6524ebfcd6SJosh Poimboeuf} 6624ebfcd6SJosh Poimboeuf 6724ebfcd6SJosh Poimboeufdie() { 6824ebfcd6SJosh Poimboeuf warn "$@" 6924ebfcd6SJosh Poimboeuf exit 1 7024ebfcd6SJosh Poimboeuf} 7124ebfcd6SJosh Poimboeuf 7224ebfcd6SJosh Poimboeufdeclare -a STASHED_FILES 7324ebfcd6SJosh Poimboeuf 7424ebfcd6SJosh Poimboeufstash_file() { 7524ebfcd6SJosh Poimboeuf local file="$1" 7624ebfcd6SJosh Poimboeuf local rel_file="${file#"$SRC"/}" 7724ebfcd6SJosh Poimboeuf 7824ebfcd6SJosh Poimboeuf [[ ! -e "$file" ]] && die "no file to stash: $file" 7924ebfcd6SJosh Poimboeuf 8024ebfcd6SJosh Poimboeuf mkdir -p "$STASH_DIR/$(dirname "$rel_file")" 8124ebfcd6SJosh Poimboeuf cp -f "$file" "$STASH_DIR/$rel_file" 8224ebfcd6SJosh Poimboeuf 8324ebfcd6SJosh Poimboeuf STASHED_FILES+=("$rel_file") 8424ebfcd6SJosh Poimboeuf} 8524ebfcd6SJosh Poimboeuf 8624ebfcd6SJosh Poimboeufrestore_files() { 8724ebfcd6SJosh Poimboeuf local file 8824ebfcd6SJosh Poimboeuf 8924ebfcd6SJosh Poimboeuf for file in "${STASHED_FILES[@]}"; do 9024ebfcd6SJosh Poimboeuf mv -f "$STASH_DIR/$file" "$SRC/$file" || warn "can't restore file: $file" 9124ebfcd6SJosh Poimboeuf done 9224ebfcd6SJosh Poimboeuf 9324ebfcd6SJosh Poimboeuf STASHED_FILES=() 9424ebfcd6SJosh Poimboeuf} 9524ebfcd6SJosh Poimboeuf 9624ebfcd6SJosh Poimboeufcleanup() { 9724ebfcd6SJosh Poimboeuf set +o nounset 9824ebfcd6SJosh Poimboeuf revert_patches "--recount" 9924ebfcd6SJosh Poimboeuf restore_files 10024ebfcd6SJosh Poimboeuf [[ "$KEEP_TMP" -eq 0 ]] && rm -rf "$TMP_DIR" 10124ebfcd6SJosh Poimboeuf return 0 10224ebfcd6SJosh Poimboeuf} 10324ebfcd6SJosh Poimboeuf 10424ebfcd6SJosh Poimboeuftrap_err() { 10524ebfcd6SJosh Poimboeuf warn "line ${BASH_LINENO[0]}: '$BASH_COMMAND'" 10624ebfcd6SJosh Poimboeuf} 10724ebfcd6SJosh Poimboeuf 10824ebfcd6SJosh Poimboeuftrap cleanup EXIT INT TERM HUP 10924ebfcd6SJosh Poimboeuftrap trap_err ERR 11024ebfcd6SJosh Poimboeuf 11124ebfcd6SJosh Poimboeuf__usage() { 11224ebfcd6SJosh Poimboeuf cat <<EOF 11324ebfcd6SJosh PoimboeufUsage: $SCRIPT [OPTIONS] PATCH_FILE(s) 11424ebfcd6SJosh PoimboeufGenerate a livepatch module. 11524ebfcd6SJosh Poimboeuf 11624ebfcd6SJosh PoimboeufOptions: 117*78be9facSJosh Poimboeuf -f, --show-first-changed Show address of first changed instruction 11824ebfcd6SJosh Poimboeuf -j, --jobs=<jobs> Build jobs to run simultaneously [default: $JOBS] 11924ebfcd6SJosh Poimboeuf -o, --output=<file.ko> Output file [default: livepatch-<patch-name>.ko] 12024ebfcd6SJosh Poimboeuf --no-replace Disable livepatch atomic replace 12124ebfcd6SJosh Poimboeuf -v, --verbose Pass V=1 to kernel/module builds 12224ebfcd6SJosh Poimboeuf 12324ebfcd6SJosh PoimboeufAdvanced Options: 1242c2f0b86SJosh Poimboeuf -d, --debug Show symbol/reloc cloning decisions 12524ebfcd6SJosh Poimboeuf -S, --short-circuit=STEP Start at build step (requires prior --keep-tmp) 12624ebfcd6SJosh Poimboeuf 1|orig Build original kernel (default) 12724ebfcd6SJosh Poimboeuf 2|patched Build patched kernel 12824ebfcd6SJosh Poimboeuf 3|diff Diff objects 12924ebfcd6SJosh Poimboeuf 4|kmod Build patch module 13024ebfcd6SJosh Poimboeuf -T, --keep-tmp Preserve tmp dir on exit 13124ebfcd6SJosh Poimboeuf 13224ebfcd6SJosh PoimboeufEOF 13324ebfcd6SJosh Poimboeuf} 13424ebfcd6SJosh Poimboeuf 13524ebfcd6SJosh Poimboeufusage() { 13624ebfcd6SJosh Poimboeuf __usage >&2 13724ebfcd6SJosh Poimboeuf} 13824ebfcd6SJosh Poimboeuf 13924ebfcd6SJosh Poimboeufprocess_args() { 14024ebfcd6SJosh Poimboeuf local keep_tmp=0 14124ebfcd6SJosh Poimboeuf local short 14224ebfcd6SJosh Poimboeuf local long 14324ebfcd6SJosh Poimboeuf local args 14424ebfcd6SJosh Poimboeuf 145*78be9facSJosh Poimboeuf short="hfj:o:vdS:T" 146*78be9facSJosh Poimboeuf long="help,show-first-changed,jobs:,output:,no-replace,verbose,debug,short-circuit:,keep-tmp" 14724ebfcd6SJosh Poimboeuf 14824ebfcd6SJosh Poimboeuf args=$(getopt --options "$short" --longoptions "$long" -- "$@") || { 14924ebfcd6SJosh Poimboeuf echo; usage; exit 15024ebfcd6SJosh Poimboeuf } 15124ebfcd6SJosh Poimboeuf eval set -- "$args" 15224ebfcd6SJosh Poimboeuf 15324ebfcd6SJosh Poimboeuf while true; do 15424ebfcd6SJosh Poimboeuf case "$1" in 15524ebfcd6SJosh Poimboeuf -h | --help) 15624ebfcd6SJosh Poimboeuf usage 15724ebfcd6SJosh Poimboeuf exit 0 15824ebfcd6SJosh Poimboeuf ;; 159*78be9facSJosh Poimboeuf -f | --show-first-changed) 160*78be9facSJosh Poimboeuf DIFF_CHECKSUM=1 161*78be9facSJosh Poimboeuf shift 162*78be9facSJosh Poimboeuf ;; 16324ebfcd6SJosh Poimboeuf -j | --jobs) 16424ebfcd6SJosh Poimboeuf JOBS="$2" 16524ebfcd6SJosh Poimboeuf shift 2 16624ebfcd6SJosh Poimboeuf ;; 16724ebfcd6SJosh Poimboeuf -o | --output) 16824ebfcd6SJosh Poimboeuf [[ "$2" != *.ko ]] && die "output filename should end with .ko" 16924ebfcd6SJosh Poimboeuf OUTFILE="$2" 17024ebfcd6SJosh Poimboeuf NAME="$(basename "$OUTFILE")" 17124ebfcd6SJosh Poimboeuf NAME="${NAME%.ko}" 17224ebfcd6SJosh Poimboeuf NAME="$(module_name_string "$NAME")" 17324ebfcd6SJosh Poimboeuf shift 2 17424ebfcd6SJosh Poimboeuf ;; 17524ebfcd6SJosh Poimboeuf --no-replace) 17624ebfcd6SJosh Poimboeuf REPLACE=0 17724ebfcd6SJosh Poimboeuf shift 17824ebfcd6SJosh Poimboeuf ;; 17924ebfcd6SJosh Poimboeuf -v | --verbose) 18024ebfcd6SJosh Poimboeuf VERBOSE="V=1" 18124ebfcd6SJosh Poimboeuf shift 18224ebfcd6SJosh Poimboeuf ;; 1832c2f0b86SJosh Poimboeuf -d | --debug) 1842c2f0b86SJosh Poimboeuf DEBUG_CLONE=1 1852c2f0b86SJosh Poimboeuf keep_tmp=1 1862c2f0b86SJosh Poimboeuf shift 1872c2f0b86SJosh Poimboeuf ;; 18824ebfcd6SJosh Poimboeuf -S | --short-circuit) 18924ebfcd6SJosh Poimboeuf [[ ! -d "$TMP_DIR" ]] && die "--short-circuit requires preserved klp-tmp dir" 19024ebfcd6SJosh Poimboeuf keep_tmp=1 19124ebfcd6SJosh Poimboeuf case "$2" in 19224ebfcd6SJosh Poimboeuf 1 | orig) SHORT_CIRCUIT=1; ;; 19324ebfcd6SJosh Poimboeuf 2 | patched) SHORT_CIRCUIT=2; ;; 19424ebfcd6SJosh Poimboeuf 3 | diff) SHORT_CIRCUIT=3; ;; 19524ebfcd6SJosh Poimboeuf 4 | mod) SHORT_CIRCUIT=4; ;; 19624ebfcd6SJosh Poimboeuf *) die "invalid short-circuit step '$2'" ;; 19724ebfcd6SJosh Poimboeuf esac 19824ebfcd6SJosh Poimboeuf shift 2 19924ebfcd6SJosh Poimboeuf ;; 20024ebfcd6SJosh Poimboeuf -T | --keep-tmp) 20124ebfcd6SJosh Poimboeuf keep_tmp=1 20224ebfcd6SJosh Poimboeuf shift 20324ebfcd6SJosh Poimboeuf ;; 20424ebfcd6SJosh Poimboeuf --) 20524ebfcd6SJosh Poimboeuf shift 20624ebfcd6SJosh Poimboeuf break 20724ebfcd6SJosh Poimboeuf ;; 20824ebfcd6SJosh Poimboeuf *) 20924ebfcd6SJosh Poimboeuf usage 21024ebfcd6SJosh Poimboeuf exit 1 21124ebfcd6SJosh Poimboeuf ;; 21224ebfcd6SJosh Poimboeuf esac 21324ebfcd6SJosh Poimboeuf done 21424ebfcd6SJosh Poimboeuf 21524ebfcd6SJosh Poimboeuf if [[ $# -eq 0 ]]; then 21624ebfcd6SJosh Poimboeuf usage 21724ebfcd6SJosh Poimboeuf exit 1 21824ebfcd6SJosh Poimboeuf fi 21924ebfcd6SJosh Poimboeuf 22024ebfcd6SJosh Poimboeuf KEEP_TMP="$keep_tmp" 22124ebfcd6SJosh Poimboeuf PATCHES=("$@") 22224ebfcd6SJosh Poimboeuf} 22324ebfcd6SJosh Poimboeuf 22424ebfcd6SJosh Poimboeuf# temporarily disable xtrace for especially verbose code 22524ebfcd6SJosh Poimboeufxtrace_save() { 22624ebfcd6SJosh Poimboeuf [[ -v XTRACE ]] && set +x 22724ebfcd6SJosh Poimboeuf return 0 22824ebfcd6SJosh Poimboeuf} 22924ebfcd6SJosh Poimboeuf 23024ebfcd6SJosh Poimboeufxtrace_restore() { 23124ebfcd6SJosh Poimboeuf [[ -v XTRACE ]] && set -x 23224ebfcd6SJosh Poimboeuf return 0 23324ebfcd6SJosh Poimboeuf} 23424ebfcd6SJosh Poimboeuf 23524ebfcd6SJosh Poimboeufvalidate_config() { 23624ebfcd6SJosh Poimboeuf xtrace_save "reading .config" 23724ebfcd6SJosh Poimboeuf source "$CONFIG" || die "no .config file in $(dirname "$CONFIG")" 23824ebfcd6SJosh Poimboeuf xtrace_restore 23924ebfcd6SJosh Poimboeuf 24024ebfcd6SJosh Poimboeuf [[ -v CONFIG_LIVEPATCH ]] || \ 24124ebfcd6SJosh Poimboeuf die "CONFIG_LIVEPATCH not enabled" 24224ebfcd6SJosh Poimboeuf 24324ebfcd6SJosh Poimboeuf [[ -v CONFIG_KLP_BUILD ]] || \ 24424ebfcd6SJosh Poimboeuf die "CONFIG_KLP_BUILD not enabled" 24524ebfcd6SJosh Poimboeuf 24624ebfcd6SJosh Poimboeuf [[ -v CONFIG_GCC_PLUGIN_LATENT_ENTROPY ]] && \ 24724ebfcd6SJosh Poimboeuf die "kernel option 'CONFIG_GCC_PLUGIN_LATENT_ENTROPY' not supported" 24824ebfcd6SJosh Poimboeuf 24924ebfcd6SJosh Poimboeuf [[ -v CONFIG_GCC_PLUGIN_RANDSTRUCT ]] && \ 25024ebfcd6SJosh Poimboeuf die "kernel option 'CONFIG_GCC_PLUGIN_RANDSTRUCT' not supported" 25124ebfcd6SJosh Poimboeuf 25224ebfcd6SJosh Poimboeuf return 0 25324ebfcd6SJosh Poimboeuf} 25424ebfcd6SJosh Poimboeuf 25524ebfcd6SJosh Poimboeuf# Only allow alphanumerics and '_' and '-' in the module name. Everything else 25624ebfcd6SJosh Poimboeuf# is replaced with '-'. Also truncate to 55 chars so the full name + NUL 25724ebfcd6SJosh Poimboeuf# terminator fits in the kernel's 56-byte module name array. 25824ebfcd6SJosh Poimboeufmodule_name_string() { 25924ebfcd6SJosh Poimboeuf echo "${1//[^a-zA-Z0-9_-]/-}" | cut -c 1-55 26024ebfcd6SJosh Poimboeuf} 26124ebfcd6SJosh Poimboeuf 26224ebfcd6SJosh Poimboeuf# If the module name wasn't specified on the cmdline with --output, give it a 26324ebfcd6SJosh Poimboeuf# name based on the patch name. 26424ebfcd6SJosh Poimboeufset_module_name() { 26524ebfcd6SJosh Poimboeuf [[ -v NAME ]] && return 0 26624ebfcd6SJosh Poimboeuf 26724ebfcd6SJosh Poimboeuf if [[ "${#PATCHES[@]}" -eq 1 ]]; then 26824ebfcd6SJosh Poimboeuf NAME="$(basename "${PATCHES[0]}")" 26924ebfcd6SJosh Poimboeuf NAME="${NAME%.*}" 27024ebfcd6SJosh Poimboeuf else 27124ebfcd6SJosh Poimboeuf NAME="patch" 27224ebfcd6SJosh Poimboeuf fi 27324ebfcd6SJosh Poimboeuf 27424ebfcd6SJosh Poimboeuf NAME="livepatch-$NAME" 27524ebfcd6SJosh Poimboeuf NAME="$(module_name_string "$NAME")" 27624ebfcd6SJosh Poimboeuf 27724ebfcd6SJosh Poimboeuf OUTFILE="$NAME.ko" 27824ebfcd6SJosh Poimboeuf} 27924ebfcd6SJosh Poimboeuf 28024ebfcd6SJosh Poimboeuf# Hardcode the value printed by the localversion script to prevent patch 28124ebfcd6SJosh Poimboeuf# application from appending it with '+' due to a dirty git working tree. 28224ebfcd6SJosh Poimboeufset_kernelversion() { 28324ebfcd6SJosh Poimboeuf local file="$SRC/scripts/setlocalversion" 28424ebfcd6SJosh Poimboeuf local localversion 28524ebfcd6SJosh Poimboeuf 28624ebfcd6SJosh Poimboeuf stash_file "$file" 28724ebfcd6SJosh Poimboeuf 28824ebfcd6SJosh Poimboeuf localversion="$(cd "$SRC" && make --no-print-directory kernelversion)" 28924ebfcd6SJosh Poimboeuf localversion="$(cd "$SRC" && KERNELVERSION="$localversion" ./scripts/setlocalversion)" 29024ebfcd6SJosh Poimboeuf [[ -z "$localversion" ]] && die "setlocalversion failed" 29124ebfcd6SJosh Poimboeuf 29224ebfcd6SJosh Poimboeuf sed -i "2i echo $localversion; exit 0" scripts/setlocalversion 29324ebfcd6SJosh Poimboeuf} 29424ebfcd6SJosh Poimboeuf 29524ebfcd6SJosh Poimboeufget_patch_files() { 29624ebfcd6SJosh Poimboeuf local patch="$1" 29724ebfcd6SJosh Poimboeuf 29824ebfcd6SJosh Poimboeuf grep0 -E '^(--- |\+\+\+ )' "$patch" \ 29924ebfcd6SJosh Poimboeuf | gawk '{print $2}' \ 30024ebfcd6SJosh Poimboeuf | sed 's|^[^/]*/||' \ 30124ebfcd6SJosh Poimboeuf | sort -u 30224ebfcd6SJosh Poimboeuf} 30324ebfcd6SJosh Poimboeuf 30424ebfcd6SJosh Poimboeuf# Make sure git re-stats the changed files 30524ebfcd6SJosh Poimboeufgit_refresh() { 30624ebfcd6SJosh Poimboeuf local patch="$1" 30724ebfcd6SJosh Poimboeuf local files=() 30824ebfcd6SJosh Poimboeuf 30924ebfcd6SJosh Poimboeuf [[ ! -e "$SRC/.git" ]] && return 31024ebfcd6SJosh Poimboeuf 31124ebfcd6SJosh Poimboeuf get_patch_files "$patch" | mapfile -t files 31224ebfcd6SJosh Poimboeuf 31324ebfcd6SJosh Poimboeuf ( 31424ebfcd6SJosh Poimboeuf cd "$SRC" 31524ebfcd6SJosh Poimboeuf git update-index -q --refresh -- "${files[@]}" 31624ebfcd6SJosh Poimboeuf ) 31724ebfcd6SJosh Poimboeuf} 31824ebfcd6SJosh Poimboeuf 31924ebfcd6SJosh Poimboeufcheck_unsupported_patches() { 32024ebfcd6SJosh Poimboeuf local patch 32124ebfcd6SJosh Poimboeuf 32224ebfcd6SJosh Poimboeuf for patch in "${PATCHES[@]}"; do 32324ebfcd6SJosh Poimboeuf local files=() 32424ebfcd6SJosh Poimboeuf 32524ebfcd6SJosh Poimboeuf get_patch_files "$patch" | mapfile -t files 32624ebfcd6SJosh Poimboeuf 32724ebfcd6SJosh Poimboeuf for file in "${files[@]}"; do 32824ebfcd6SJosh Poimboeuf case "$file" in 32924ebfcd6SJosh Poimboeuf lib/*|*.S) 33024ebfcd6SJosh Poimboeuf die "unsupported patch to $file" 33124ebfcd6SJosh Poimboeuf ;; 33224ebfcd6SJosh Poimboeuf esac 33324ebfcd6SJosh Poimboeuf done 33424ebfcd6SJosh Poimboeuf done 33524ebfcd6SJosh Poimboeuf} 33624ebfcd6SJosh Poimboeuf 33724ebfcd6SJosh Poimboeufapply_patch() { 33824ebfcd6SJosh Poimboeuf local patch="$1" 33924ebfcd6SJosh Poimboeuf shift 34024ebfcd6SJosh Poimboeuf local extra_args=("$@") 34124ebfcd6SJosh Poimboeuf 34224ebfcd6SJosh Poimboeuf [[ ! -f "$patch" ]] && die "$patch doesn't exist" 34324ebfcd6SJosh Poimboeuf 34424ebfcd6SJosh Poimboeuf ( 34524ebfcd6SJosh Poimboeuf cd "$SRC" 34624ebfcd6SJosh Poimboeuf 34724ebfcd6SJosh Poimboeuf # The sed strips the version signature from 'git format-patch', 34824ebfcd6SJosh Poimboeuf # otherwise 'git apply --recount' warns. 34924ebfcd6SJosh Poimboeuf sed -n '/^-- /q;p' "$patch" | 35024ebfcd6SJosh Poimboeuf git apply "${extra_args[@]}" 35124ebfcd6SJosh Poimboeuf ) 35224ebfcd6SJosh Poimboeuf 35324ebfcd6SJosh Poimboeuf APPLIED_PATCHES+=("$patch") 35424ebfcd6SJosh Poimboeuf} 35524ebfcd6SJosh Poimboeuf 35624ebfcd6SJosh Poimboeufrevert_patch() { 35724ebfcd6SJosh Poimboeuf local patch="$1" 35824ebfcd6SJosh Poimboeuf shift 35924ebfcd6SJosh Poimboeuf local extra_args=("$@") 36024ebfcd6SJosh Poimboeuf local tmp=() 36124ebfcd6SJosh Poimboeuf 36224ebfcd6SJosh Poimboeuf ( 36324ebfcd6SJosh Poimboeuf cd "$SRC" 36424ebfcd6SJosh Poimboeuf 36524ebfcd6SJosh Poimboeuf sed -n '/^-- /q;p' "$patch" | 36624ebfcd6SJosh Poimboeuf git apply --reverse "${extra_args[@]}" 36724ebfcd6SJosh Poimboeuf ) 36824ebfcd6SJosh Poimboeuf git_refresh "$patch" 36924ebfcd6SJosh Poimboeuf 37024ebfcd6SJosh Poimboeuf for p in "${APPLIED_PATCHES[@]}"; do 37124ebfcd6SJosh Poimboeuf [[ "$p" == "$patch" ]] && continue 37224ebfcd6SJosh Poimboeuf tmp+=("$p") 37324ebfcd6SJosh Poimboeuf done 37424ebfcd6SJosh Poimboeuf 37524ebfcd6SJosh Poimboeuf APPLIED_PATCHES=("${tmp[@]}") 37624ebfcd6SJosh Poimboeuf} 37724ebfcd6SJosh Poimboeuf 37824ebfcd6SJosh Poimboeufapply_patches() { 37924ebfcd6SJosh Poimboeuf local patch 38024ebfcd6SJosh Poimboeuf 38124ebfcd6SJosh Poimboeuf for patch in "${PATCHES[@]}"; do 38224ebfcd6SJosh Poimboeuf apply_patch "$patch" 38324ebfcd6SJosh Poimboeuf done 38424ebfcd6SJosh Poimboeuf} 38524ebfcd6SJosh Poimboeuf 38624ebfcd6SJosh Poimboeufrevert_patches() { 38724ebfcd6SJosh Poimboeuf local extra_args=("$@") 38824ebfcd6SJosh Poimboeuf local patches=("${APPLIED_PATCHES[@]}") 38924ebfcd6SJosh Poimboeuf 39024ebfcd6SJosh Poimboeuf for (( i=${#patches[@]}-1 ; i>=0 ; i-- )) ; do 39124ebfcd6SJosh Poimboeuf revert_patch "${patches[$i]}" "${extra_args[@]}" 39224ebfcd6SJosh Poimboeuf done 39324ebfcd6SJosh Poimboeuf 39424ebfcd6SJosh Poimboeuf APPLIED_PATCHES=() 39524ebfcd6SJosh Poimboeuf} 39624ebfcd6SJosh Poimboeuf 39724ebfcd6SJosh Poimboeufvalidate_patches() { 39824ebfcd6SJosh Poimboeuf check_unsupported_patches 39924ebfcd6SJosh Poimboeuf apply_patches 40024ebfcd6SJosh Poimboeuf revert_patches 40124ebfcd6SJosh Poimboeuf} 40224ebfcd6SJosh Poimboeuf 40324ebfcd6SJosh Poimboeufdo_init() { 40424ebfcd6SJosh Poimboeuf # We're not yet smart enough to handle anything other than in-tree 40524ebfcd6SJosh Poimboeuf # builds in pwd. 40624ebfcd6SJosh Poimboeuf [[ ! "$SRC" -ef "$SCRIPT_DIR/../.." ]] && die "please run from the kernel root directory" 40724ebfcd6SJosh Poimboeuf [[ ! "$OBJ" -ef "$SCRIPT_DIR/../.." ]] && die "please run from the kernel root directory" 40824ebfcd6SJosh Poimboeuf 40924ebfcd6SJosh Poimboeuf (( SHORT_CIRCUIT <= 1 )) && rm -rf "$TMP_DIR" 41024ebfcd6SJosh Poimboeuf mkdir -p "$TMP_DIR" 41124ebfcd6SJosh Poimboeuf 41224ebfcd6SJosh Poimboeuf APPLIED_PATCHES=() 41324ebfcd6SJosh Poimboeuf 41424ebfcd6SJosh Poimboeuf [[ -x "$FIX_PATCH_LINES" ]] || die "can't find fix-patch-lines" 41524ebfcd6SJosh Poimboeuf 41624ebfcd6SJosh Poimboeuf validate_config 41724ebfcd6SJosh Poimboeuf set_module_name 41824ebfcd6SJosh Poimboeuf set_kernelversion 41924ebfcd6SJosh Poimboeuf} 42024ebfcd6SJosh Poimboeuf 42124ebfcd6SJosh Poimboeuf# Refresh the patch hunk headers, specifically the line numbers and counts. 42224ebfcd6SJosh Poimboeufrefresh_patch() { 42324ebfcd6SJosh Poimboeuf local patch="$1" 42424ebfcd6SJosh Poimboeuf local tmpdir="$PATCH_TMP_DIR" 42524ebfcd6SJosh Poimboeuf local files=() 42624ebfcd6SJosh Poimboeuf 42724ebfcd6SJosh Poimboeuf rm -rf "$tmpdir" 42824ebfcd6SJosh Poimboeuf mkdir -p "$tmpdir/a" 42924ebfcd6SJosh Poimboeuf mkdir -p "$tmpdir/b" 43024ebfcd6SJosh Poimboeuf 43124ebfcd6SJosh Poimboeuf # Get all source files affected by the patch 43224ebfcd6SJosh Poimboeuf get_patch_files "$patch" | mapfile -t files 43324ebfcd6SJosh Poimboeuf 43424ebfcd6SJosh Poimboeuf # Copy orig source files to 'a' 43524ebfcd6SJosh Poimboeuf ( cd "$SRC" && echo "${files[@]}" | xargs cp --parents --target-directory="$tmpdir/a" ) 43624ebfcd6SJosh Poimboeuf 43724ebfcd6SJosh Poimboeuf # Copy patched source files to 'b' 43824ebfcd6SJosh Poimboeuf apply_patch "$patch" --recount 43924ebfcd6SJosh Poimboeuf ( cd "$SRC" && echo "${files[@]}" | xargs cp --parents --target-directory="$tmpdir/b" ) 44024ebfcd6SJosh Poimboeuf revert_patch "$patch" --recount 44124ebfcd6SJosh Poimboeuf 44224ebfcd6SJosh Poimboeuf # Diff 'a' and 'b' to make a clean patch 44324ebfcd6SJosh Poimboeuf ( cd "$tmpdir" && git diff --no-index --no-prefix a b > "$patch" ) || true 44424ebfcd6SJosh Poimboeuf} 44524ebfcd6SJosh Poimboeuf 44624ebfcd6SJosh Poimboeuf# Copy the patches to a temporary directory, fix their lines so as not to 44724ebfcd6SJosh Poimboeuf# affect the __LINE__ macro for otherwise unchanged functions further down the 44824ebfcd6SJosh Poimboeuf# file, and update $PATCHES to point to the fixed patches. 44924ebfcd6SJosh Poimboeuffix_patches() { 45024ebfcd6SJosh Poimboeuf local idx 45124ebfcd6SJosh Poimboeuf local i 45224ebfcd6SJosh Poimboeuf 45324ebfcd6SJosh Poimboeuf rm -f "$TMP_DIR"/*.patch 45424ebfcd6SJosh Poimboeuf 45524ebfcd6SJosh Poimboeuf idx=0001 45624ebfcd6SJosh Poimboeuf for i in "${!PATCHES[@]}"; do 45724ebfcd6SJosh Poimboeuf local old_patch="${PATCHES[$i]}" 45824ebfcd6SJosh Poimboeuf local tmp_patch="$TMP_DIR/tmp.patch" 45924ebfcd6SJosh Poimboeuf local patch="${PATCHES[$i]}" 46024ebfcd6SJosh Poimboeuf local new_patch 46124ebfcd6SJosh Poimboeuf 46224ebfcd6SJosh Poimboeuf new_patch="$TMP_DIR/$idx-fixed-$(basename "$patch")" 46324ebfcd6SJosh Poimboeuf 46424ebfcd6SJosh Poimboeuf cp -f "$old_patch" "$tmp_patch" 46524ebfcd6SJosh Poimboeuf refresh_patch "$tmp_patch" 46624ebfcd6SJosh Poimboeuf "$FIX_PATCH_LINES" "$tmp_patch" > "$new_patch" 46724ebfcd6SJosh Poimboeuf refresh_patch "$new_patch" 46824ebfcd6SJosh Poimboeuf 46924ebfcd6SJosh Poimboeuf PATCHES[i]="$new_patch" 47024ebfcd6SJosh Poimboeuf 47124ebfcd6SJosh Poimboeuf rm -f "$tmp_patch" 47224ebfcd6SJosh Poimboeuf idx=$(printf "%04d" $(( 10#$idx + 1 ))) 47324ebfcd6SJosh Poimboeuf done 47424ebfcd6SJosh Poimboeuf} 47524ebfcd6SJosh Poimboeuf 47624ebfcd6SJosh Poimboeufclean_kernel() { 47724ebfcd6SJosh Poimboeuf local cmd=() 47824ebfcd6SJosh Poimboeuf 47924ebfcd6SJosh Poimboeuf cmd=("make") 48024ebfcd6SJosh Poimboeuf cmd+=("--silent") 48124ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 48224ebfcd6SJosh Poimboeuf cmd+=("clean") 48324ebfcd6SJosh Poimboeuf 48424ebfcd6SJosh Poimboeuf ( 48524ebfcd6SJosh Poimboeuf cd "$SRC" 48624ebfcd6SJosh Poimboeuf "${cmd[@]}" 48724ebfcd6SJosh Poimboeuf ) 48824ebfcd6SJosh Poimboeuf} 48924ebfcd6SJosh Poimboeuf 49024ebfcd6SJosh Poimboeufbuild_kernel() { 49124ebfcd6SJosh Poimboeuf local log="$TMP_DIR/build.log" 49224ebfcd6SJosh Poimboeuf local cmd=() 49324ebfcd6SJosh Poimboeuf 49424ebfcd6SJosh Poimboeuf cmd=("make") 49524ebfcd6SJosh Poimboeuf 49624ebfcd6SJosh Poimboeuf # When a patch to a kernel module references a newly created unexported 49724ebfcd6SJosh Poimboeuf # symbol which lives in vmlinux or another kernel module, the patched 49824ebfcd6SJosh Poimboeuf # kernel build fails with the following error: 49924ebfcd6SJosh Poimboeuf # 50024ebfcd6SJosh Poimboeuf # ERROR: modpost: "klp_string" [fs/xfs/xfs.ko] undefined! 50124ebfcd6SJosh Poimboeuf # 50224ebfcd6SJosh Poimboeuf # The undefined symbols are working as designed in that case. They get 50324ebfcd6SJosh Poimboeuf # resolved later when the livepatch module build link pulls all the 50424ebfcd6SJosh Poimboeuf # disparate objects together into the same kernel module. 50524ebfcd6SJosh Poimboeuf # 50624ebfcd6SJosh Poimboeuf # It would be good to have a way to tell modpost to skip checking for 50724ebfcd6SJosh Poimboeuf # undefined symbols altogether. For now, just convert the error to a 50824ebfcd6SJosh Poimboeuf # warning with KBUILD_MODPOST_WARN, and grep out the warning to avoid 50924ebfcd6SJosh Poimboeuf # confusing the user. 51024ebfcd6SJosh Poimboeuf # 51124ebfcd6SJosh Poimboeuf cmd+=("KBUILD_MODPOST_WARN=1") 51224ebfcd6SJosh Poimboeuf 51324ebfcd6SJosh Poimboeuf cmd+=("$VERBOSE") 51424ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 51524ebfcd6SJosh Poimboeuf cmd+=("KCFLAGS=-ffunction-sections -fdata-sections") 51624ebfcd6SJosh Poimboeuf cmd+=("vmlinux") 51724ebfcd6SJosh Poimboeuf cmd+=("modules") 51824ebfcd6SJosh Poimboeuf 51924ebfcd6SJosh Poimboeuf ( 52024ebfcd6SJosh Poimboeuf cd "$SRC" 52124ebfcd6SJosh Poimboeuf "${cmd[@]}" \ 52224ebfcd6SJosh Poimboeuf 1> >(tee -a "$log") \ 52324ebfcd6SJosh Poimboeuf 2> >(tee -a "$log" | grep0 -v "modpost.*undefined!" >&2) 52424ebfcd6SJosh Poimboeuf ) 52524ebfcd6SJosh Poimboeuf} 52624ebfcd6SJosh Poimboeuf 52724ebfcd6SJosh Poimboeuffind_objects() { 52824ebfcd6SJosh Poimboeuf local opts=("$@") 52924ebfcd6SJosh Poimboeuf 53024ebfcd6SJosh Poimboeuf # Find root-level vmlinux.o and non-root-level .ko files, 53124ebfcd6SJosh Poimboeuf # excluding klp-tmp/ and .git/ 53224ebfcd6SJosh Poimboeuf find "$OBJ" \( -path "$TMP_DIR" -o -path "$OBJ/.git" -o -regex "$OBJ/[^/][^/]*\.ko" \) -prune -o \ 53324ebfcd6SJosh Poimboeuf -type f "${opts[@]}" \ 53424ebfcd6SJosh Poimboeuf \( -name "*.ko" -o -path "$OBJ/vmlinux.o" \) \ 53524ebfcd6SJosh Poimboeuf -printf '%P\n' 53624ebfcd6SJosh Poimboeuf} 53724ebfcd6SJosh Poimboeuf 53824ebfcd6SJosh Poimboeuf# Copy all .o archives to $ORIG_DIR 53924ebfcd6SJosh Poimboeufcopy_orig_objects() { 54024ebfcd6SJosh Poimboeuf local files=() 54124ebfcd6SJosh Poimboeuf 54224ebfcd6SJosh Poimboeuf rm -rf "$ORIG_DIR" 54324ebfcd6SJosh Poimboeuf mkdir -p "$ORIG_DIR" 54424ebfcd6SJosh Poimboeuf 54524ebfcd6SJosh Poimboeuf find_objects | mapfile -t files 54624ebfcd6SJosh Poimboeuf 54724ebfcd6SJosh Poimboeuf xtrace_save "copying orig objects" 54824ebfcd6SJosh Poimboeuf for _file in "${files[@]}"; do 54924ebfcd6SJosh Poimboeuf local rel_file="${_file/.ko/.o}" 55024ebfcd6SJosh Poimboeuf local file="$OBJ/$rel_file" 55124ebfcd6SJosh Poimboeuf local file_dir="$(dirname "$file")" 55224ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 55324ebfcd6SJosh Poimboeuf local orig_dir="$(dirname "$orig_file")" 55424ebfcd6SJosh Poimboeuf local cmd_file="$file_dir/.$(basename "$file").cmd" 55524ebfcd6SJosh Poimboeuf 55624ebfcd6SJosh Poimboeuf [[ ! -f "$file" ]] && die "missing $(basename "$file") for $_file" 55724ebfcd6SJosh Poimboeuf 55824ebfcd6SJosh Poimboeuf mkdir -p "$orig_dir" 55924ebfcd6SJosh Poimboeuf cp -f "$file" "$orig_dir" 56024ebfcd6SJosh Poimboeuf [[ -e "$cmd_file" ]] && cp -f "$cmd_file" "$orig_dir" 56124ebfcd6SJosh Poimboeuf done 56224ebfcd6SJosh Poimboeuf xtrace_restore 56324ebfcd6SJosh Poimboeuf 56424ebfcd6SJosh Poimboeuf mv -f "$TMP_DIR/build.log" "$ORIG_DIR" 56524ebfcd6SJosh Poimboeuf touch "$TIMESTAMP" 56624ebfcd6SJosh Poimboeuf} 56724ebfcd6SJosh Poimboeuf 56824ebfcd6SJosh Poimboeuf# Copy all changed objects to $PATCHED_DIR 56924ebfcd6SJosh Poimboeufcopy_patched_objects() { 57024ebfcd6SJosh Poimboeuf local files=() 57124ebfcd6SJosh Poimboeuf local opts=() 57224ebfcd6SJosh Poimboeuf local found=0 57324ebfcd6SJosh Poimboeuf 57424ebfcd6SJosh Poimboeuf rm -rf "$PATCHED_DIR" 57524ebfcd6SJosh Poimboeuf mkdir -p "$PATCHED_DIR" 57624ebfcd6SJosh Poimboeuf 57724ebfcd6SJosh Poimboeuf # Note this doesn't work with some configs, thus the 'cmp' below. 57824ebfcd6SJosh Poimboeuf opts=("-newer") 57924ebfcd6SJosh Poimboeuf opts+=("$TIMESTAMP") 58024ebfcd6SJosh Poimboeuf 58124ebfcd6SJosh Poimboeuf find_objects "${opts[@]}" | mapfile -t files 58224ebfcd6SJosh Poimboeuf 58324ebfcd6SJosh Poimboeuf xtrace_save "copying changed objects" 58424ebfcd6SJosh Poimboeuf for _file in "${files[@]}"; do 58524ebfcd6SJosh Poimboeuf local rel_file="${_file/.ko/.o}" 58624ebfcd6SJosh Poimboeuf local file="$OBJ/$rel_file" 58724ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 58824ebfcd6SJosh Poimboeuf local patched_file="$PATCHED_DIR/$rel_file" 58924ebfcd6SJosh Poimboeuf local patched_dir="$(dirname "$patched_file")" 59024ebfcd6SJosh Poimboeuf 59124ebfcd6SJosh Poimboeuf [[ ! -f "$file" ]] && die "missing $(basename "$file") for $_file" 59224ebfcd6SJosh Poimboeuf 59324ebfcd6SJosh Poimboeuf cmp -s "$orig_file" "$file" && continue 59424ebfcd6SJosh Poimboeuf 59524ebfcd6SJosh Poimboeuf mkdir -p "$patched_dir" 59624ebfcd6SJosh Poimboeuf cp -f "$file" "$patched_dir" 59724ebfcd6SJosh Poimboeuf found=1 59824ebfcd6SJosh Poimboeuf done 59924ebfcd6SJosh Poimboeuf xtrace_restore 60024ebfcd6SJosh Poimboeuf 60124ebfcd6SJosh Poimboeuf (( found == 0 )) && die "no changes detected" 60224ebfcd6SJosh Poimboeuf 60324ebfcd6SJosh Poimboeuf mv -f "$TMP_DIR/build.log" "$PATCHED_DIR" 60424ebfcd6SJosh Poimboeuf} 60524ebfcd6SJosh Poimboeuf 60624ebfcd6SJosh Poimboeuf# Diff changed objects, writing output object to $DIFF_DIR 60724ebfcd6SJosh Poimboeufdiff_objects() { 60824ebfcd6SJosh Poimboeuf local log="$KLP_DIFF_LOG" 60924ebfcd6SJosh Poimboeuf local files=() 6102c2f0b86SJosh Poimboeuf local opts=() 61124ebfcd6SJosh Poimboeuf 61224ebfcd6SJosh Poimboeuf rm -rf "$DIFF_DIR" 61324ebfcd6SJosh Poimboeuf mkdir -p "$DIFF_DIR" 61424ebfcd6SJosh Poimboeuf 61524ebfcd6SJosh Poimboeuf find "$PATCHED_DIR" -type f -name "*.o" | mapfile -t files 61624ebfcd6SJosh Poimboeuf [[ ${#files[@]} -eq 0 ]] && die "no changes detected" 61724ebfcd6SJosh Poimboeuf 6182c2f0b86SJosh Poimboeuf [[ -v DEBUG_CLONE ]] && opts=("--debug") 6192c2f0b86SJosh Poimboeuf 62024ebfcd6SJosh Poimboeuf # Diff all changed objects 62124ebfcd6SJosh Poimboeuf for file in "${files[@]}"; do 62224ebfcd6SJosh Poimboeuf local rel_file="${file#"$PATCHED_DIR"/}" 62324ebfcd6SJosh Poimboeuf local orig_file="$rel_file" 62424ebfcd6SJosh Poimboeuf local patched_file="$PATCHED_DIR/$rel_file" 62524ebfcd6SJosh Poimboeuf local out_file="$DIFF_DIR/$rel_file" 626*78be9facSJosh Poimboeuf local filter=() 62724ebfcd6SJosh Poimboeuf local cmd=() 62824ebfcd6SJosh Poimboeuf 62924ebfcd6SJosh Poimboeuf mkdir -p "$(dirname "$out_file")" 63024ebfcd6SJosh Poimboeuf 63124ebfcd6SJosh Poimboeuf cmd=("$SRC/tools/objtool/objtool") 63224ebfcd6SJosh Poimboeuf cmd+=("klp") 63324ebfcd6SJosh Poimboeuf cmd+=("diff") 6342c2f0b86SJosh Poimboeuf (( ${#opts[@]} > 0 )) && cmd+=("${opts[@]}") 63524ebfcd6SJosh Poimboeuf cmd+=("$orig_file") 63624ebfcd6SJosh Poimboeuf cmd+=("$patched_file") 63724ebfcd6SJosh Poimboeuf cmd+=("$out_file") 63824ebfcd6SJosh Poimboeuf 639*78be9facSJosh Poimboeuf if [[ -v DIFF_CHECKSUM ]]; then 640*78be9facSJosh Poimboeuf filter=("grep0") 641*78be9facSJosh Poimboeuf filter+=("-Ev") 642*78be9facSJosh Poimboeuf filter+=("DEBUG: .*checksum: ") 643*78be9facSJosh Poimboeuf else 644*78be9facSJosh Poimboeuf filter=("cat") 645*78be9facSJosh Poimboeuf fi 646*78be9facSJosh Poimboeuf 64724ebfcd6SJosh Poimboeuf ( 64824ebfcd6SJosh Poimboeuf cd "$ORIG_DIR" 64924ebfcd6SJosh Poimboeuf "${cmd[@]}" \ 65024ebfcd6SJosh Poimboeuf 1> >(tee -a "$log") \ 651*78be9facSJosh Poimboeuf 2> >(tee -a "$log" | "${filter[@]}" >&2) || \ 65224ebfcd6SJosh Poimboeuf die "objtool klp diff failed" 65324ebfcd6SJosh Poimboeuf ) 65424ebfcd6SJosh Poimboeuf done 65524ebfcd6SJosh Poimboeuf} 65624ebfcd6SJosh Poimboeuf 657*78be9facSJosh Poimboeuf# For each changed object, run objtool with --debug-checksum to get the 658*78be9facSJosh Poimboeuf# per-instruction checksums, and then diff those to find the first changed 659*78be9facSJosh Poimboeuf# instruction for each function. 660*78be9facSJosh Poimboeufdiff_checksums() { 661*78be9facSJosh Poimboeuf local orig_log="$ORIG_DIR/checksum.log" 662*78be9facSJosh Poimboeuf local patched_log="$PATCHED_DIR/checksum.log" 663*78be9facSJosh Poimboeuf local -A funcs 664*78be9facSJosh Poimboeuf local cmd=() 665*78be9facSJosh Poimboeuf local line 666*78be9facSJosh Poimboeuf local file 667*78be9facSJosh Poimboeuf local func 668*78be9facSJosh Poimboeuf 669*78be9facSJosh Poimboeuf gawk '/\.o: changed function: / { 670*78be9facSJosh Poimboeuf sub(/:$/, "", $1) 671*78be9facSJosh Poimboeuf print $1, $NF 672*78be9facSJosh Poimboeuf }' "$KLP_DIFF_LOG" | mapfile -t lines 673*78be9facSJosh Poimboeuf 674*78be9facSJosh Poimboeuf for line in "${lines[@]}"; do 675*78be9facSJosh Poimboeuf read -r file func <<< "$line" 676*78be9facSJosh Poimboeuf if [[ ! -v funcs["$file"] ]]; then 677*78be9facSJosh Poimboeuf funcs["$file"]="$func" 678*78be9facSJosh Poimboeuf else 679*78be9facSJosh Poimboeuf funcs["$file"]+=" $func" 680*78be9facSJosh Poimboeuf fi 681*78be9facSJosh Poimboeuf done 682*78be9facSJosh Poimboeuf 683*78be9facSJosh Poimboeuf cmd=("$SRC/tools/objtool/objtool") 684*78be9facSJosh Poimboeuf cmd+=("--checksum") 685*78be9facSJosh Poimboeuf cmd+=("--link") 686*78be9facSJosh Poimboeuf cmd+=("--dry-run") 687*78be9facSJosh Poimboeuf 688*78be9facSJosh Poimboeuf for file in "${!funcs[@]}"; do 689*78be9facSJosh Poimboeuf local opt="--debug-checksum=${funcs[$file]// /,}" 690*78be9facSJosh Poimboeuf 691*78be9facSJosh Poimboeuf ( 692*78be9facSJosh Poimboeuf cd "$ORIG_DIR" 693*78be9facSJosh Poimboeuf "${cmd[@]}" "$opt" "$file" &> "$orig_log" || \ 694*78be9facSJosh Poimboeuf ( cat "$orig_log" >&2; die "objtool --debug-checksum failed" ) 695*78be9facSJosh Poimboeuf 696*78be9facSJosh Poimboeuf cd "$PATCHED_DIR" 697*78be9facSJosh Poimboeuf "${cmd[@]}" "$opt" "$file" &> "$patched_log" || \ 698*78be9facSJosh Poimboeuf ( cat "$patched_log" >&2; die "objtool --debug-checksum failed" ) 699*78be9facSJosh Poimboeuf ) 700*78be9facSJosh Poimboeuf 701*78be9facSJosh Poimboeuf for func in ${funcs[$file]}; do 702*78be9facSJosh Poimboeuf diff <( grep0 -E "^DEBUG: .*checksum: $func " "$orig_log" | sed "s|$ORIG_DIR/||") \ 703*78be9facSJosh Poimboeuf <( grep0 -E "^DEBUG: .*checksum: $func " "$patched_log" | sed "s|$PATCHED_DIR/||") \ 704*78be9facSJosh Poimboeuf | gawk '/^< DEBUG: / { 705*78be9facSJosh Poimboeuf gsub(/:/, "") 706*78be9facSJosh Poimboeuf printf "%s: %s: %s\n", $3, $5, $6 707*78be9facSJosh Poimboeuf exit 708*78be9facSJosh Poimboeuf }' || true 709*78be9facSJosh Poimboeuf done 710*78be9facSJosh Poimboeuf done 711*78be9facSJosh Poimboeuf} 712*78be9facSJosh Poimboeuf 71324ebfcd6SJosh Poimboeuf# Build and post-process livepatch module in $KMOD_DIR 71424ebfcd6SJosh Poimboeufbuild_patch_module() { 71524ebfcd6SJosh Poimboeuf local makefile="$KMOD_DIR/Kbuild" 71624ebfcd6SJosh Poimboeuf local log="$KMOD_DIR/build.log" 71724ebfcd6SJosh Poimboeuf local kmod_file 71824ebfcd6SJosh Poimboeuf local cflags=() 71924ebfcd6SJosh Poimboeuf local files=() 72024ebfcd6SJosh Poimboeuf local cmd=() 72124ebfcd6SJosh Poimboeuf 72224ebfcd6SJosh Poimboeuf rm -rf "$KMOD_DIR" 72324ebfcd6SJosh Poimboeuf mkdir -p "$KMOD_DIR" 72424ebfcd6SJosh Poimboeuf 72524ebfcd6SJosh Poimboeuf cp -f "$SRC/scripts/livepatch/init.c" "$KMOD_DIR" 72624ebfcd6SJosh Poimboeuf 72724ebfcd6SJosh Poimboeuf echo "obj-m := $NAME.o" > "$makefile" 72824ebfcd6SJosh Poimboeuf echo -n "$NAME-y := init.o" >> "$makefile" 72924ebfcd6SJosh Poimboeuf 73024ebfcd6SJosh Poimboeuf find "$DIFF_DIR" -type f -name "*.o" | mapfile -t files 73124ebfcd6SJosh Poimboeuf [[ ${#files[@]} -eq 0 ]] && die "no changes detected" 73224ebfcd6SJosh Poimboeuf 73324ebfcd6SJosh Poimboeuf for file in "${files[@]}"; do 73424ebfcd6SJosh Poimboeuf local rel_file="${file#"$DIFF_DIR"/}" 73524ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 73624ebfcd6SJosh Poimboeuf local orig_dir="$(dirname "$orig_file")" 73724ebfcd6SJosh Poimboeuf local kmod_file="$KMOD_DIR/$rel_file" 73824ebfcd6SJosh Poimboeuf local kmod_dir="$(dirname "$kmod_file")" 73924ebfcd6SJosh Poimboeuf local cmd_file="$orig_dir/.$(basename "$file").cmd" 74024ebfcd6SJosh Poimboeuf 74124ebfcd6SJosh Poimboeuf mkdir -p "$kmod_dir" 74224ebfcd6SJosh Poimboeuf cp -f "$file" "$kmod_dir" 74324ebfcd6SJosh Poimboeuf [[ -e "$cmd_file" ]] && cp -f "$cmd_file" "$kmod_dir" 74424ebfcd6SJosh Poimboeuf 74524ebfcd6SJosh Poimboeuf # Tell kbuild this is a prebuilt object 74624ebfcd6SJosh Poimboeuf cp -f "$file" "${kmod_file}_shipped" 74724ebfcd6SJosh Poimboeuf 74824ebfcd6SJosh Poimboeuf echo -n " $rel_file" >> "$makefile" 74924ebfcd6SJosh Poimboeuf done 75024ebfcd6SJosh Poimboeuf 75124ebfcd6SJosh Poimboeuf echo >> "$makefile" 75224ebfcd6SJosh Poimboeuf 75324ebfcd6SJosh Poimboeuf cflags=("-ffunction-sections") 75424ebfcd6SJosh Poimboeuf cflags+=("-fdata-sections") 75524ebfcd6SJosh Poimboeuf [[ $REPLACE -eq 0 ]] && cflags+=("-DKLP_NO_REPLACE") 75624ebfcd6SJosh Poimboeuf 75724ebfcd6SJosh Poimboeuf cmd=("make") 75824ebfcd6SJosh Poimboeuf cmd+=("$VERBOSE") 75924ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 76024ebfcd6SJosh Poimboeuf cmd+=("--directory=.") 76124ebfcd6SJosh Poimboeuf cmd+=("M=$KMOD_DIR") 76224ebfcd6SJosh Poimboeuf cmd+=("KCFLAGS=${cflags[*]}") 76324ebfcd6SJosh Poimboeuf 76424ebfcd6SJosh Poimboeuf # Build a "normal" kernel module with init.c and the diffed objects 76524ebfcd6SJosh Poimboeuf ( 76624ebfcd6SJosh Poimboeuf cd "$SRC" 76724ebfcd6SJosh Poimboeuf "${cmd[@]}" \ 76824ebfcd6SJosh Poimboeuf 1> >(tee -a "$log") \ 76924ebfcd6SJosh Poimboeuf 2> >(tee -a "$log" >&2) 77024ebfcd6SJosh Poimboeuf ) 77124ebfcd6SJosh Poimboeuf 77224ebfcd6SJosh Poimboeuf kmod_file="$KMOD_DIR/$NAME.ko" 77324ebfcd6SJosh Poimboeuf 77424ebfcd6SJosh Poimboeuf # Save off the intermediate binary for debugging 77524ebfcd6SJosh Poimboeuf cp -f "$kmod_file" "$kmod_file.orig" 77624ebfcd6SJosh Poimboeuf 77724ebfcd6SJosh Poimboeuf # Work around issue where slight .config change makes corrupt BTF 77824ebfcd6SJosh Poimboeuf objcopy --remove-section=.BTF "$kmod_file" 77924ebfcd6SJosh Poimboeuf 78024ebfcd6SJosh Poimboeuf # Fix (and work around) linker wreckage for klp syms / relocs 78124ebfcd6SJosh Poimboeuf "$SRC/tools/objtool/objtool" klp post-link "$kmod_file" || die "objtool klp post-link failed" 78224ebfcd6SJosh Poimboeuf 78324ebfcd6SJosh Poimboeuf cp -f "$kmod_file" "$OUTFILE" 78424ebfcd6SJosh Poimboeuf} 78524ebfcd6SJosh Poimboeuf 78624ebfcd6SJosh Poimboeuf 78724ebfcd6SJosh Poimboeuf################################################################################ 78824ebfcd6SJosh Poimboeuf 78924ebfcd6SJosh Poimboeufprocess_args "$@" 79024ebfcd6SJosh Poimboeufdo_init 79124ebfcd6SJosh Poimboeuf 79224ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 1 )); then 79324ebfcd6SJosh Poimboeuf status "Validating patch(es)" 79424ebfcd6SJosh Poimboeuf validate_patches 79524ebfcd6SJosh Poimboeuf status "Building original kernel" 79624ebfcd6SJosh Poimboeuf clean_kernel 79724ebfcd6SJosh Poimboeuf build_kernel 79824ebfcd6SJosh Poimboeuf status "Copying original object files" 79924ebfcd6SJosh Poimboeuf copy_orig_objects 80024ebfcd6SJosh Poimboeuffi 80124ebfcd6SJosh Poimboeuf 80224ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 2 )); then 80324ebfcd6SJosh Poimboeuf status "Fixing patch(es)" 80424ebfcd6SJosh Poimboeuf fix_patches 80524ebfcd6SJosh Poimboeuf apply_patches 80624ebfcd6SJosh Poimboeuf status "Building patched kernel" 80724ebfcd6SJosh Poimboeuf build_kernel 80824ebfcd6SJosh Poimboeuf revert_patches 80924ebfcd6SJosh Poimboeuf status "Copying patched object files" 81024ebfcd6SJosh Poimboeuf copy_patched_objects 81124ebfcd6SJosh Poimboeuffi 81224ebfcd6SJosh Poimboeuf 81324ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 3 )); then 81424ebfcd6SJosh Poimboeuf status "Diffing objects" 81524ebfcd6SJosh Poimboeuf diff_objects 816*78be9facSJosh Poimboeuf if [[ -v DIFF_CHECKSUM ]]; then 817*78be9facSJosh Poimboeuf status "Finding first changed instructions" 818*78be9facSJosh Poimboeuf diff_checksums 819*78be9facSJosh Poimboeuf fi 82024ebfcd6SJosh Poimboeuffi 82124ebfcd6SJosh Poimboeuf 82224ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 4 )); then 82324ebfcd6SJosh Poimboeuf status "Building patch module: $OUTFILE" 82424ebfcd6SJosh Poimboeuf build_patch_module 82524ebfcd6SJosh Poimboeuffi 82624ebfcd6SJosh Poimboeuf 82724ebfcd6SJosh Poimboeufstatus "SUCCESS" 828