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*2c2f0b86SJosh Poimboeufunset DEBUG_CLONE 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: 11724ebfcd6SJosh Poimboeuf -j, --jobs=<jobs> Build jobs to run simultaneously [default: $JOBS] 11824ebfcd6SJosh Poimboeuf -o, --output=<file.ko> Output file [default: livepatch-<patch-name>.ko] 11924ebfcd6SJosh Poimboeuf --no-replace Disable livepatch atomic replace 12024ebfcd6SJosh Poimboeuf -v, --verbose Pass V=1 to kernel/module builds 12124ebfcd6SJosh Poimboeuf 12224ebfcd6SJosh PoimboeufAdvanced Options: 123*2c2f0b86SJosh Poimboeuf -d, --debug Show symbol/reloc cloning decisions 12424ebfcd6SJosh Poimboeuf -S, --short-circuit=STEP Start at build step (requires prior --keep-tmp) 12524ebfcd6SJosh Poimboeuf 1|orig Build original kernel (default) 12624ebfcd6SJosh Poimboeuf 2|patched Build patched kernel 12724ebfcd6SJosh Poimboeuf 3|diff Diff objects 12824ebfcd6SJosh Poimboeuf 4|kmod Build patch module 12924ebfcd6SJosh Poimboeuf -T, --keep-tmp Preserve tmp dir on exit 13024ebfcd6SJosh Poimboeuf 13124ebfcd6SJosh PoimboeufEOF 13224ebfcd6SJosh Poimboeuf} 13324ebfcd6SJosh Poimboeuf 13424ebfcd6SJosh Poimboeufusage() { 13524ebfcd6SJosh Poimboeuf __usage >&2 13624ebfcd6SJosh Poimboeuf} 13724ebfcd6SJosh Poimboeuf 13824ebfcd6SJosh Poimboeufprocess_args() { 13924ebfcd6SJosh Poimboeuf local keep_tmp=0 14024ebfcd6SJosh Poimboeuf local short 14124ebfcd6SJosh Poimboeuf local long 14224ebfcd6SJosh Poimboeuf local args 14324ebfcd6SJosh Poimboeuf 144*2c2f0b86SJosh Poimboeuf short="hj:o:vdS:T" 145*2c2f0b86SJosh Poimboeuf long="help,jobs:,output:,no-replace,verbose,debug,short-circuit:,keep-tmp" 14624ebfcd6SJosh Poimboeuf 14724ebfcd6SJosh Poimboeuf args=$(getopt --options "$short" --longoptions "$long" -- "$@") || { 14824ebfcd6SJosh Poimboeuf echo; usage; exit 14924ebfcd6SJosh Poimboeuf } 15024ebfcd6SJosh Poimboeuf eval set -- "$args" 15124ebfcd6SJosh Poimboeuf 15224ebfcd6SJosh Poimboeuf while true; do 15324ebfcd6SJosh Poimboeuf case "$1" in 15424ebfcd6SJosh Poimboeuf -h | --help) 15524ebfcd6SJosh Poimboeuf usage 15624ebfcd6SJosh Poimboeuf exit 0 15724ebfcd6SJosh Poimboeuf ;; 15824ebfcd6SJosh Poimboeuf -j | --jobs) 15924ebfcd6SJosh Poimboeuf JOBS="$2" 16024ebfcd6SJosh Poimboeuf shift 2 16124ebfcd6SJosh Poimboeuf ;; 16224ebfcd6SJosh Poimboeuf -o | --output) 16324ebfcd6SJosh Poimboeuf [[ "$2" != *.ko ]] && die "output filename should end with .ko" 16424ebfcd6SJosh Poimboeuf OUTFILE="$2" 16524ebfcd6SJosh Poimboeuf NAME="$(basename "$OUTFILE")" 16624ebfcd6SJosh Poimboeuf NAME="${NAME%.ko}" 16724ebfcd6SJosh Poimboeuf NAME="$(module_name_string "$NAME")" 16824ebfcd6SJosh Poimboeuf shift 2 16924ebfcd6SJosh Poimboeuf ;; 17024ebfcd6SJosh Poimboeuf --no-replace) 17124ebfcd6SJosh Poimboeuf REPLACE=0 17224ebfcd6SJosh Poimboeuf shift 17324ebfcd6SJosh Poimboeuf ;; 17424ebfcd6SJosh Poimboeuf -v | --verbose) 17524ebfcd6SJosh Poimboeuf VERBOSE="V=1" 17624ebfcd6SJosh Poimboeuf shift 17724ebfcd6SJosh Poimboeuf ;; 178*2c2f0b86SJosh Poimboeuf -d | --debug) 179*2c2f0b86SJosh Poimboeuf DEBUG_CLONE=1 180*2c2f0b86SJosh Poimboeuf keep_tmp=1 181*2c2f0b86SJosh Poimboeuf shift 182*2c2f0b86SJosh Poimboeuf ;; 18324ebfcd6SJosh Poimboeuf -S | --short-circuit) 18424ebfcd6SJosh Poimboeuf [[ ! -d "$TMP_DIR" ]] && die "--short-circuit requires preserved klp-tmp dir" 18524ebfcd6SJosh Poimboeuf keep_tmp=1 18624ebfcd6SJosh Poimboeuf case "$2" in 18724ebfcd6SJosh Poimboeuf 1 | orig) SHORT_CIRCUIT=1; ;; 18824ebfcd6SJosh Poimboeuf 2 | patched) SHORT_CIRCUIT=2; ;; 18924ebfcd6SJosh Poimboeuf 3 | diff) SHORT_CIRCUIT=3; ;; 19024ebfcd6SJosh Poimboeuf 4 | mod) SHORT_CIRCUIT=4; ;; 19124ebfcd6SJosh Poimboeuf *) die "invalid short-circuit step '$2'" ;; 19224ebfcd6SJosh Poimboeuf esac 19324ebfcd6SJosh Poimboeuf shift 2 19424ebfcd6SJosh Poimboeuf ;; 19524ebfcd6SJosh Poimboeuf -T | --keep-tmp) 19624ebfcd6SJosh Poimboeuf keep_tmp=1 19724ebfcd6SJosh Poimboeuf shift 19824ebfcd6SJosh Poimboeuf ;; 19924ebfcd6SJosh Poimboeuf --) 20024ebfcd6SJosh Poimboeuf shift 20124ebfcd6SJosh Poimboeuf break 20224ebfcd6SJosh Poimboeuf ;; 20324ebfcd6SJosh Poimboeuf *) 20424ebfcd6SJosh Poimboeuf usage 20524ebfcd6SJosh Poimboeuf exit 1 20624ebfcd6SJosh Poimboeuf ;; 20724ebfcd6SJosh Poimboeuf esac 20824ebfcd6SJosh Poimboeuf done 20924ebfcd6SJosh Poimboeuf 21024ebfcd6SJosh Poimboeuf if [[ $# -eq 0 ]]; then 21124ebfcd6SJosh Poimboeuf usage 21224ebfcd6SJosh Poimboeuf exit 1 21324ebfcd6SJosh Poimboeuf fi 21424ebfcd6SJosh Poimboeuf 21524ebfcd6SJosh Poimboeuf KEEP_TMP="$keep_tmp" 21624ebfcd6SJosh Poimboeuf PATCHES=("$@") 21724ebfcd6SJosh Poimboeuf} 21824ebfcd6SJosh Poimboeuf 21924ebfcd6SJosh Poimboeuf# temporarily disable xtrace for especially verbose code 22024ebfcd6SJosh Poimboeufxtrace_save() { 22124ebfcd6SJosh Poimboeuf [[ -v XTRACE ]] && set +x 22224ebfcd6SJosh Poimboeuf return 0 22324ebfcd6SJosh Poimboeuf} 22424ebfcd6SJosh Poimboeuf 22524ebfcd6SJosh Poimboeufxtrace_restore() { 22624ebfcd6SJosh Poimboeuf [[ -v XTRACE ]] && set -x 22724ebfcd6SJosh Poimboeuf return 0 22824ebfcd6SJosh Poimboeuf} 22924ebfcd6SJosh Poimboeuf 23024ebfcd6SJosh Poimboeufvalidate_config() { 23124ebfcd6SJosh Poimboeuf xtrace_save "reading .config" 23224ebfcd6SJosh Poimboeuf source "$CONFIG" || die "no .config file in $(dirname "$CONFIG")" 23324ebfcd6SJosh Poimboeuf xtrace_restore 23424ebfcd6SJosh Poimboeuf 23524ebfcd6SJosh Poimboeuf [[ -v CONFIG_LIVEPATCH ]] || \ 23624ebfcd6SJosh Poimboeuf die "CONFIG_LIVEPATCH not enabled" 23724ebfcd6SJosh Poimboeuf 23824ebfcd6SJosh Poimboeuf [[ -v CONFIG_KLP_BUILD ]] || \ 23924ebfcd6SJosh Poimboeuf die "CONFIG_KLP_BUILD not enabled" 24024ebfcd6SJosh Poimboeuf 24124ebfcd6SJosh Poimboeuf [[ -v CONFIG_GCC_PLUGIN_LATENT_ENTROPY ]] && \ 24224ebfcd6SJosh Poimboeuf die "kernel option 'CONFIG_GCC_PLUGIN_LATENT_ENTROPY' not supported" 24324ebfcd6SJosh Poimboeuf 24424ebfcd6SJosh Poimboeuf [[ -v CONFIG_GCC_PLUGIN_RANDSTRUCT ]] && \ 24524ebfcd6SJosh Poimboeuf die "kernel option 'CONFIG_GCC_PLUGIN_RANDSTRUCT' not supported" 24624ebfcd6SJosh Poimboeuf 24724ebfcd6SJosh Poimboeuf return 0 24824ebfcd6SJosh Poimboeuf} 24924ebfcd6SJosh Poimboeuf 25024ebfcd6SJosh Poimboeuf# Only allow alphanumerics and '_' and '-' in the module name. Everything else 25124ebfcd6SJosh Poimboeuf# is replaced with '-'. Also truncate to 55 chars so the full name + NUL 25224ebfcd6SJosh Poimboeuf# terminator fits in the kernel's 56-byte module name array. 25324ebfcd6SJosh Poimboeufmodule_name_string() { 25424ebfcd6SJosh Poimboeuf echo "${1//[^a-zA-Z0-9_-]/-}" | cut -c 1-55 25524ebfcd6SJosh Poimboeuf} 25624ebfcd6SJosh Poimboeuf 25724ebfcd6SJosh Poimboeuf# If the module name wasn't specified on the cmdline with --output, give it a 25824ebfcd6SJosh Poimboeuf# name based on the patch name. 25924ebfcd6SJosh Poimboeufset_module_name() { 26024ebfcd6SJosh Poimboeuf [[ -v NAME ]] && return 0 26124ebfcd6SJosh Poimboeuf 26224ebfcd6SJosh Poimboeuf if [[ "${#PATCHES[@]}" -eq 1 ]]; then 26324ebfcd6SJosh Poimboeuf NAME="$(basename "${PATCHES[0]}")" 26424ebfcd6SJosh Poimboeuf NAME="${NAME%.*}" 26524ebfcd6SJosh Poimboeuf else 26624ebfcd6SJosh Poimboeuf NAME="patch" 26724ebfcd6SJosh Poimboeuf fi 26824ebfcd6SJosh Poimboeuf 26924ebfcd6SJosh Poimboeuf NAME="livepatch-$NAME" 27024ebfcd6SJosh Poimboeuf NAME="$(module_name_string "$NAME")" 27124ebfcd6SJosh Poimboeuf 27224ebfcd6SJosh Poimboeuf OUTFILE="$NAME.ko" 27324ebfcd6SJosh Poimboeuf} 27424ebfcd6SJosh Poimboeuf 27524ebfcd6SJosh Poimboeuf# Hardcode the value printed by the localversion script to prevent patch 27624ebfcd6SJosh Poimboeuf# application from appending it with '+' due to a dirty git working tree. 27724ebfcd6SJosh Poimboeufset_kernelversion() { 27824ebfcd6SJosh Poimboeuf local file="$SRC/scripts/setlocalversion" 27924ebfcd6SJosh Poimboeuf local localversion 28024ebfcd6SJosh Poimboeuf 28124ebfcd6SJosh Poimboeuf stash_file "$file" 28224ebfcd6SJosh Poimboeuf 28324ebfcd6SJosh Poimboeuf localversion="$(cd "$SRC" && make --no-print-directory kernelversion)" 28424ebfcd6SJosh Poimboeuf localversion="$(cd "$SRC" && KERNELVERSION="$localversion" ./scripts/setlocalversion)" 28524ebfcd6SJosh Poimboeuf [[ -z "$localversion" ]] && die "setlocalversion failed" 28624ebfcd6SJosh Poimboeuf 28724ebfcd6SJosh Poimboeuf sed -i "2i echo $localversion; exit 0" scripts/setlocalversion 28824ebfcd6SJosh Poimboeuf} 28924ebfcd6SJosh Poimboeuf 29024ebfcd6SJosh Poimboeufget_patch_files() { 29124ebfcd6SJosh Poimboeuf local patch="$1" 29224ebfcd6SJosh Poimboeuf 29324ebfcd6SJosh Poimboeuf grep0 -E '^(--- |\+\+\+ )' "$patch" \ 29424ebfcd6SJosh Poimboeuf | gawk '{print $2}' \ 29524ebfcd6SJosh Poimboeuf | sed 's|^[^/]*/||' \ 29624ebfcd6SJosh Poimboeuf | sort -u 29724ebfcd6SJosh Poimboeuf} 29824ebfcd6SJosh Poimboeuf 29924ebfcd6SJosh Poimboeuf# Make sure git re-stats the changed files 30024ebfcd6SJosh Poimboeufgit_refresh() { 30124ebfcd6SJosh Poimboeuf local patch="$1" 30224ebfcd6SJosh Poimboeuf local files=() 30324ebfcd6SJosh Poimboeuf 30424ebfcd6SJosh Poimboeuf [[ ! -e "$SRC/.git" ]] && return 30524ebfcd6SJosh Poimboeuf 30624ebfcd6SJosh Poimboeuf get_patch_files "$patch" | mapfile -t files 30724ebfcd6SJosh Poimboeuf 30824ebfcd6SJosh Poimboeuf ( 30924ebfcd6SJosh Poimboeuf cd "$SRC" 31024ebfcd6SJosh Poimboeuf git update-index -q --refresh -- "${files[@]}" 31124ebfcd6SJosh Poimboeuf ) 31224ebfcd6SJosh Poimboeuf} 31324ebfcd6SJosh Poimboeuf 31424ebfcd6SJosh Poimboeufcheck_unsupported_patches() { 31524ebfcd6SJosh Poimboeuf local patch 31624ebfcd6SJosh Poimboeuf 31724ebfcd6SJosh Poimboeuf for patch in "${PATCHES[@]}"; do 31824ebfcd6SJosh Poimboeuf local files=() 31924ebfcd6SJosh Poimboeuf 32024ebfcd6SJosh Poimboeuf get_patch_files "$patch" | mapfile -t files 32124ebfcd6SJosh Poimboeuf 32224ebfcd6SJosh Poimboeuf for file in "${files[@]}"; do 32324ebfcd6SJosh Poimboeuf case "$file" in 32424ebfcd6SJosh Poimboeuf lib/*|*.S) 32524ebfcd6SJosh Poimboeuf die "unsupported patch to $file" 32624ebfcd6SJosh Poimboeuf ;; 32724ebfcd6SJosh Poimboeuf esac 32824ebfcd6SJosh Poimboeuf done 32924ebfcd6SJosh Poimboeuf done 33024ebfcd6SJosh Poimboeuf} 33124ebfcd6SJosh Poimboeuf 33224ebfcd6SJosh Poimboeufapply_patch() { 33324ebfcd6SJosh Poimboeuf local patch="$1" 33424ebfcd6SJosh Poimboeuf shift 33524ebfcd6SJosh Poimboeuf local extra_args=("$@") 33624ebfcd6SJosh Poimboeuf 33724ebfcd6SJosh Poimboeuf [[ ! -f "$patch" ]] && die "$patch doesn't exist" 33824ebfcd6SJosh Poimboeuf 33924ebfcd6SJosh Poimboeuf ( 34024ebfcd6SJosh Poimboeuf cd "$SRC" 34124ebfcd6SJosh Poimboeuf 34224ebfcd6SJosh Poimboeuf # The sed strips the version signature from 'git format-patch', 34324ebfcd6SJosh Poimboeuf # otherwise 'git apply --recount' warns. 34424ebfcd6SJosh Poimboeuf sed -n '/^-- /q;p' "$patch" | 34524ebfcd6SJosh Poimboeuf git apply "${extra_args[@]}" 34624ebfcd6SJosh Poimboeuf ) 34724ebfcd6SJosh Poimboeuf 34824ebfcd6SJosh Poimboeuf APPLIED_PATCHES+=("$patch") 34924ebfcd6SJosh Poimboeuf} 35024ebfcd6SJosh Poimboeuf 35124ebfcd6SJosh Poimboeufrevert_patch() { 35224ebfcd6SJosh Poimboeuf local patch="$1" 35324ebfcd6SJosh Poimboeuf shift 35424ebfcd6SJosh Poimboeuf local extra_args=("$@") 35524ebfcd6SJosh Poimboeuf local tmp=() 35624ebfcd6SJosh Poimboeuf 35724ebfcd6SJosh Poimboeuf ( 35824ebfcd6SJosh Poimboeuf cd "$SRC" 35924ebfcd6SJosh Poimboeuf 36024ebfcd6SJosh Poimboeuf sed -n '/^-- /q;p' "$patch" | 36124ebfcd6SJosh Poimboeuf git apply --reverse "${extra_args[@]}" 36224ebfcd6SJosh Poimboeuf ) 36324ebfcd6SJosh Poimboeuf git_refresh "$patch" 36424ebfcd6SJosh Poimboeuf 36524ebfcd6SJosh Poimboeuf for p in "${APPLIED_PATCHES[@]}"; do 36624ebfcd6SJosh Poimboeuf [[ "$p" == "$patch" ]] && continue 36724ebfcd6SJosh Poimboeuf tmp+=("$p") 36824ebfcd6SJosh Poimboeuf done 36924ebfcd6SJosh Poimboeuf 37024ebfcd6SJosh Poimboeuf APPLIED_PATCHES=("${tmp[@]}") 37124ebfcd6SJosh Poimboeuf} 37224ebfcd6SJosh Poimboeuf 37324ebfcd6SJosh Poimboeufapply_patches() { 37424ebfcd6SJosh Poimboeuf local patch 37524ebfcd6SJosh Poimboeuf 37624ebfcd6SJosh Poimboeuf for patch in "${PATCHES[@]}"; do 37724ebfcd6SJosh Poimboeuf apply_patch "$patch" 37824ebfcd6SJosh Poimboeuf done 37924ebfcd6SJosh Poimboeuf} 38024ebfcd6SJosh Poimboeuf 38124ebfcd6SJosh Poimboeufrevert_patches() { 38224ebfcd6SJosh Poimboeuf local extra_args=("$@") 38324ebfcd6SJosh Poimboeuf local patches=("${APPLIED_PATCHES[@]}") 38424ebfcd6SJosh Poimboeuf 38524ebfcd6SJosh Poimboeuf for (( i=${#patches[@]}-1 ; i>=0 ; i-- )) ; do 38624ebfcd6SJosh Poimboeuf revert_patch "${patches[$i]}" "${extra_args[@]}" 38724ebfcd6SJosh Poimboeuf done 38824ebfcd6SJosh Poimboeuf 38924ebfcd6SJosh Poimboeuf APPLIED_PATCHES=() 39024ebfcd6SJosh Poimboeuf} 39124ebfcd6SJosh Poimboeuf 39224ebfcd6SJosh Poimboeufvalidate_patches() { 39324ebfcd6SJosh Poimboeuf check_unsupported_patches 39424ebfcd6SJosh Poimboeuf apply_patches 39524ebfcd6SJosh Poimboeuf revert_patches 39624ebfcd6SJosh Poimboeuf} 39724ebfcd6SJosh Poimboeuf 39824ebfcd6SJosh Poimboeufdo_init() { 39924ebfcd6SJosh Poimboeuf # We're not yet smart enough to handle anything other than in-tree 40024ebfcd6SJosh Poimboeuf # builds in pwd. 40124ebfcd6SJosh Poimboeuf [[ ! "$SRC" -ef "$SCRIPT_DIR/../.." ]] && die "please run from the kernel root directory" 40224ebfcd6SJosh Poimboeuf [[ ! "$OBJ" -ef "$SCRIPT_DIR/../.." ]] && die "please run from the kernel root directory" 40324ebfcd6SJosh Poimboeuf 40424ebfcd6SJosh Poimboeuf (( SHORT_CIRCUIT <= 1 )) && rm -rf "$TMP_DIR" 40524ebfcd6SJosh Poimboeuf mkdir -p "$TMP_DIR" 40624ebfcd6SJosh Poimboeuf 40724ebfcd6SJosh Poimboeuf APPLIED_PATCHES=() 40824ebfcd6SJosh Poimboeuf 40924ebfcd6SJosh Poimboeuf [[ -x "$FIX_PATCH_LINES" ]] || die "can't find fix-patch-lines" 41024ebfcd6SJosh Poimboeuf 41124ebfcd6SJosh Poimboeuf validate_config 41224ebfcd6SJosh Poimboeuf set_module_name 41324ebfcd6SJosh Poimboeuf set_kernelversion 41424ebfcd6SJosh Poimboeuf} 41524ebfcd6SJosh Poimboeuf 41624ebfcd6SJosh Poimboeuf# Refresh the patch hunk headers, specifically the line numbers and counts. 41724ebfcd6SJosh Poimboeufrefresh_patch() { 41824ebfcd6SJosh Poimboeuf local patch="$1" 41924ebfcd6SJosh Poimboeuf local tmpdir="$PATCH_TMP_DIR" 42024ebfcd6SJosh Poimboeuf local files=() 42124ebfcd6SJosh Poimboeuf 42224ebfcd6SJosh Poimboeuf rm -rf "$tmpdir" 42324ebfcd6SJosh Poimboeuf mkdir -p "$tmpdir/a" 42424ebfcd6SJosh Poimboeuf mkdir -p "$tmpdir/b" 42524ebfcd6SJosh Poimboeuf 42624ebfcd6SJosh Poimboeuf # Get all source files affected by the patch 42724ebfcd6SJosh Poimboeuf get_patch_files "$patch" | mapfile -t files 42824ebfcd6SJosh Poimboeuf 42924ebfcd6SJosh Poimboeuf # Copy orig source files to 'a' 43024ebfcd6SJosh Poimboeuf ( cd "$SRC" && echo "${files[@]}" | xargs cp --parents --target-directory="$tmpdir/a" ) 43124ebfcd6SJosh Poimboeuf 43224ebfcd6SJosh Poimboeuf # Copy patched source files to 'b' 43324ebfcd6SJosh Poimboeuf apply_patch "$patch" --recount 43424ebfcd6SJosh Poimboeuf ( cd "$SRC" && echo "${files[@]}" | xargs cp --parents --target-directory="$tmpdir/b" ) 43524ebfcd6SJosh Poimboeuf revert_patch "$patch" --recount 43624ebfcd6SJosh Poimboeuf 43724ebfcd6SJosh Poimboeuf # Diff 'a' and 'b' to make a clean patch 43824ebfcd6SJosh Poimboeuf ( cd "$tmpdir" && git diff --no-index --no-prefix a b > "$patch" ) || true 43924ebfcd6SJosh Poimboeuf} 44024ebfcd6SJosh Poimboeuf 44124ebfcd6SJosh Poimboeuf# Copy the patches to a temporary directory, fix their lines so as not to 44224ebfcd6SJosh Poimboeuf# affect the __LINE__ macro for otherwise unchanged functions further down the 44324ebfcd6SJosh Poimboeuf# file, and update $PATCHES to point to the fixed patches. 44424ebfcd6SJosh Poimboeuffix_patches() { 44524ebfcd6SJosh Poimboeuf local idx 44624ebfcd6SJosh Poimboeuf local i 44724ebfcd6SJosh Poimboeuf 44824ebfcd6SJosh Poimboeuf rm -f "$TMP_DIR"/*.patch 44924ebfcd6SJosh Poimboeuf 45024ebfcd6SJosh Poimboeuf idx=0001 45124ebfcd6SJosh Poimboeuf for i in "${!PATCHES[@]}"; do 45224ebfcd6SJosh Poimboeuf local old_patch="${PATCHES[$i]}" 45324ebfcd6SJosh Poimboeuf local tmp_patch="$TMP_DIR/tmp.patch" 45424ebfcd6SJosh Poimboeuf local patch="${PATCHES[$i]}" 45524ebfcd6SJosh Poimboeuf local new_patch 45624ebfcd6SJosh Poimboeuf 45724ebfcd6SJosh Poimboeuf new_patch="$TMP_DIR/$idx-fixed-$(basename "$patch")" 45824ebfcd6SJosh Poimboeuf 45924ebfcd6SJosh Poimboeuf cp -f "$old_patch" "$tmp_patch" 46024ebfcd6SJosh Poimboeuf refresh_patch "$tmp_patch" 46124ebfcd6SJosh Poimboeuf "$FIX_PATCH_LINES" "$tmp_patch" > "$new_patch" 46224ebfcd6SJosh Poimboeuf refresh_patch "$new_patch" 46324ebfcd6SJosh Poimboeuf 46424ebfcd6SJosh Poimboeuf PATCHES[i]="$new_patch" 46524ebfcd6SJosh Poimboeuf 46624ebfcd6SJosh Poimboeuf rm -f "$tmp_patch" 46724ebfcd6SJosh Poimboeuf idx=$(printf "%04d" $(( 10#$idx + 1 ))) 46824ebfcd6SJosh Poimboeuf done 46924ebfcd6SJosh Poimboeuf} 47024ebfcd6SJosh Poimboeuf 47124ebfcd6SJosh Poimboeufclean_kernel() { 47224ebfcd6SJosh Poimboeuf local cmd=() 47324ebfcd6SJosh Poimboeuf 47424ebfcd6SJosh Poimboeuf cmd=("make") 47524ebfcd6SJosh Poimboeuf cmd+=("--silent") 47624ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 47724ebfcd6SJosh Poimboeuf cmd+=("clean") 47824ebfcd6SJosh Poimboeuf 47924ebfcd6SJosh Poimboeuf ( 48024ebfcd6SJosh Poimboeuf cd "$SRC" 48124ebfcd6SJosh Poimboeuf "${cmd[@]}" 48224ebfcd6SJosh Poimboeuf ) 48324ebfcd6SJosh Poimboeuf} 48424ebfcd6SJosh Poimboeuf 48524ebfcd6SJosh Poimboeufbuild_kernel() { 48624ebfcd6SJosh Poimboeuf local log="$TMP_DIR/build.log" 48724ebfcd6SJosh Poimboeuf local cmd=() 48824ebfcd6SJosh Poimboeuf 48924ebfcd6SJosh Poimboeuf cmd=("make") 49024ebfcd6SJosh Poimboeuf 49124ebfcd6SJosh Poimboeuf # When a patch to a kernel module references a newly created unexported 49224ebfcd6SJosh Poimboeuf # symbol which lives in vmlinux or another kernel module, the patched 49324ebfcd6SJosh Poimboeuf # kernel build fails with the following error: 49424ebfcd6SJosh Poimboeuf # 49524ebfcd6SJosh Poimboeuf # ERROR: modpost: "klp_string" [fs/xfs/xfs.ko] undefined! 49624ebfcd6SJosh Poimboeuf # 49724ebfcd6SJosh Poimboeuf # The undefined symbols are working as designed in that case. They get 49824ebfcd6SJosh Poimboeuf # resolved later when the livepatch module build link pulls all the 49924ebfcd6SJosh Poimboeuf # disparate objects together into the same kernel module. 50024ebfcd6SJosh Poimboeuf # 50124ebfcd6SJosh Poimboeuf # It would be good to have a way to tell modpost to skip checking for 50224ebfcd6SJosh Poimboeuf # undefined symbols altogether. For now, just convert the error to a 50324ebfcd6SJosh Poimboeuf # warning with KBUILD_MODPOST_WARN, and grep out the warning to avoid 50424ebfcd6SJosh Poimboeuf # confusing the user. 50524ebfcd6SJosh Poimboeuf # 50624ebfcd6SJosh Poimboeuf cmd+=("KBUILD_MODPOST_WARN=1") 50724ebfcd6SJosh Poimboeuf 50824ebfcd6SJosh Poimboeuf cmd+=("$VERBOSE") 50924ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 51024ebfcd6SJosh Poimboeuf cmd+=("KCFLAGS=-ffunction-sections -fdata-sections") 51124ebfcd6SJosh Poimboeuf cmd+=("vmlinux") 51224ebfcd6SJosh Poimboeuf cmd+=("modules") 51324ebfcd6SJosh Poimboeuf 51424ebfcd6SJosh Poimboeuf ( 51524ebfcd6SJosh Poimboeuf cd "$SRC" 51624ebfcd6SJosh Poimboeuf "${cmd[@]}" \ 51724ebfcd6SJosh Poimboeuf 1> >(tee -a "$log") \ 51824ebfcd6SJosh Poimboeuf 2> >(tee -a "$log" | grep0 -v "modpost.*undefined!" >&2) 51924ebfcd6SJosh Poimboeuf ) 52024ebfcd6SJosh Poimboeuf} 52124ebfcd6SJosh Poimboeuf 52224ebfcd6SJosh Poimboeuffind_objects() { 52324ebfcd6SJosh Poimboeuf local opts=("$@") 52424ebfcd6SJosh Poimboeuf 52524ebfcd6SJosh Poimboeuf # Find root-level vmlinux.o and non-root-level .ko files, 52624ebfcd6SJosh Poimboeuf # excluding klp-tmp/ and .git/ 52724ebfcd6SJosh Poimboeuf find "$OBJ" \( -path "$TMP_DIR" -o -path "$OBJ/.git" -o -regex "$OBJ/[^/][^/]*\.ko" \) -prune -o \ 52824ebfcd6SJosh Poimboeuf -type f "${opts[@]}" \ 52924ebfcd6SJosh Poimboeuf \( -name "*.ko" -o -path "$OBJ/vmlinux.o" \) \ 53024ebfcd6SJosh Poimboeuf -printf '%P\n' 53124ebfcd6SJosh Poimboeuf} 53224ebfcd6SJosh Poimboeuf 53324ebfcd6SJosh Poimboeuf# Copy all .o archives to $ORIG_DIR 53424ebfcd6SJosh Poimboeufcopy_orig_objects() { 53524ebfcd6SJosh Poimboeuf local files=() 53624ebfcd6SJosh Poimboeuf 53724ebfcd6SJosh Poimboeuf rm -rf "$ORIG_DIR" 53824ebfcd6SJosh Poimboeuf mkdir -p "$ORIG_DIR" 53924ebfcd6SJosh Poimboeuf 54024ebfcd6SJosh Poimboeuf find_objects | mapfile -t files 54124ebfcd6SJosh Poimboeuf 54224ebfcd6SJosh Poimboeuf xtrace_save "copying orig objects" 54324ebfcd6SJosh Poimboeuf for _file in "${files[@]}"; do 54424ebfcd6SJosh Poimboeuf local rel_file="${_file/.ko/.o}" 54524ebfcd6SJosh Poimboeuf local file="$OBJ/$rel_file" 54624ebfcd6SJosh Poimboeuf local file_dir="$(dirname "$file")" 54724ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 54824ebfcd6SJosh Poimboeuf local orig_dir="$(dirname "$orig_file")" 54924ebfcd6SJosh Poimboeuf local cmd_file="$file_dir/.$(basename "$file").cmd" 55024ebfcd6SJosh Poimboeuf 55124ebfcd6SJosh Poimboeuf [[ ! -f "$file" ]] && die "missing $(basename "$file") for $_file" 55224ebfcd6SJosh Poimboeuf 55324ebfcd6SJosh Poimboeuf mkdir -p "$orig_dir" 55424ebfcd6SJosh Poimboeuf cp -f "$file" "$orig_dir" 55524ebfcd6SJosh Poimboeuf [[ -e "$cmd_file" ]] && cp -f "$cmd_file" "$orig_dir" 55624ebfcd6SJosh Poimboeuf done 55724ebfcd6SJosh Poimboeuf xtrace_restore 55824ebfcd6SJosh Poimboeuf 55924ebfcd6SJosh Poimboeuf mv -f "$TMP_DIR/build.log" "$ORIG_DIR" 56024ebfcd6SJosh Poimboeuf touch "$TIMESTAMP" 56124ebfcd6SJosh Poimboeuf} 56224ebfcd6SJosh Poimboeuf 56324ebfcd6SJosh Poimboeuf# Copy all changed objects to $PATCHED_DIR 56424ebfcd6SJosh Poimboeufcopy_patched_objects() { 56524ebfcd6SJosh Poimboeuf local files=() 56624ebfcd6SJosh Poimboeuf local opts=() 56724ebfcd6SJosh Poimboeuf local found=0 56824ebfcd6SJosh Poimboeuf 56924ebfcd6SJosh Poimboeuf rm -rf "$PATCHED_DIR" 57024ebfcd6SJosh Poimboeuf mkdir -p "$PATCHED_DIR" 57124ebfcd6SJosh Poimboeuf 57224ebfcd6SJosh Poimboeuf # Note this doesn't work with some configs, thus the 'cmp' below. 57324ebfcd6SJosh Poimboeuf opts=("-newer") 57424ebfcd6SJosh Poimboeuf opts+=("$TIMESTAMP") 57524ebfcd6SJosh Poimboeuf 57624ebfcd6SJosh Poimboeuf find_objects "${opts[@]}" | mapfile -t files 57724ebfcd6SJosh Poimboeuf 57824ebfcd6SJosh Poimboeuf xtrace_save "copying changed objects" 57924ebfcd6SJosh Poimboeuf for _file in "${files[@]}"; do 58024ebfcd6SJosh Poimboeuf local rel_file="${_file/.ko/.o}" 58124ebfcd6SJosh Poimboeuf local file="$OBJ/$rel_file" 58224ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 58324ebfcd6SJosh Poimboeuf local patched_file="$PATCHED_DIR/$rel_file" 58424ebfcd6SJosh Poimboeuf local patched_dir="$(dirname "$patched_file")" 58524ebfcd6SJosh Poimboeuf 58624ebfcd6SJosh Poimboeuf [[ ! -f "$file" ]] && die "missing $(basename "$file") for $_file" 58724ebfcd6SJosh Poimboeuf 58824ebfcd6SJosh Poimboeuf cmp -s "$orig_file" "$file" && continue 58924ebfcd6SJosh Poimboeuf 59024ebfcd6SJosh Poimboeuf mkdir -p "$patched_dir" 59124ebfcd6SJosh Poimboeuf cp -f "$file" "$patched_dir" 59224ebfcd6SJosh Poimboeuf found=1 59324ebfcd6SJosh Poimboeuf done 59424ebfcd6SJosh Poimboeuf xtrace_restore 59524ebfcd6SJosh Poimboeuf 59624ebfcd6SJosh Poimboeuf (( found == 0 )) && die "no changes detected" 59724ebfcd6SJosh Poimboeuf 59824ebfcd6SJosh Poimboeuf mv -f "$TMP_DIR/build.log" "$PATCHED_DIR" 59924ebfcd6SJosh Poimboeuf} 60024ebfcd6SJosh Poimboeuf 60124ebfcd6SJosh Poimboeuf# Diff changed objects, writing output object to $DIFF_DIR 60224ebfcd6SJosh Poimboeufdiff_objects() { 60324ebfcd6SJosh Poimboeuf local log="$KLP_DIFF_LOG" 60424ebfcd6SJosh Poimboeuf local files=() 605*2c2f0b86SJosh Poimboeuf local opts=() 60624ebfcd6SJosh Poimboeuf 60724ebfcd6SJosh Poimboeuf rm -rf "$DIFF_DIR" 60824ebfcd6SJosh Poimboeuf mkdir -p "$DIFF_DIR" 60924ebfcd6SJosh Poimboeuf 61024ebfcd6SJosh Poimboeuf find "$PATCHED_DIR" -type f -name "*.o" | mapfile -t files 61124ebfcd6SJosh Poimboeuf [[ ${#files[@]} -eq 0 ]] && die "no changes detected" 61224ebfcd6SJosh Poimboeuf 613*2c2f0b86SJosh Poimboeuf [[ -v DEBUG_CLONE ]] && opts=("--debug") 614*2c2f0b86SJosh Poimboeuf 61524ebfcd6SJosh Poimboeuf # Diff all changed objects 61624ebfcd6SJosh Poimboeuf for file in "${files[@]}"; do 61724ebfcd6SJosh Poimboeuf local rel_file="${file#"$PATCHED_DIR"/}" 61824ebfcd6SJosh Poimboeuf local orig_file="$rel_file" 61924ebfcd6SJosh Poimboeuf local patched_file="$PATCHED_DIR/$rel_file" 62024ebfcd6SJosh Poimboeuf local out_file="$DIFF_DIR/$rel_file" 62124ebfcd6SJosh Poimboeuf local cmd=() 62224ebfcd6SJosh Poimboeuf 62324ebfcd6SJosh Poimboeuf mkdir -p "$(dirname "$out_file")" 62424ebfcd6SJosh Poimboeuf 62524ebfcd6SJosh Poimboeuf cmd=("$SRC/tools/objtool/objtool") 62624ebfcd6SJosh Poimboeuf cmd+=("klp") 62724ebfcd6SJosh Poimboeuf cmd+=("diff") 628*2c2f0b86SJosh Poimboeuf (( ${#opts[@]} > 0 )) && cmd+=("${opts[@]}") 62924ebfcd6SJosh Poimboeuf cmd+=("$orig_file") 63024ebfcd6SJosh Poimboeuf cmd+=("$patched_file") 63124ebfcd6SJosh Poimboeuf cmd+=("$out_file") 63224ebfcd6SJosh Poimboeuf 63324ebfcd6SJosh Poimboeuf ( 63424ebfcd6SJosh Poimboeuf cd "$ORIG_DIR" 63524ebfcd6SJosh Poimboeuf "${cmd[@]}" \ 63624ebfcd6SJosh Poimboeuf 1> >(tee -a "$log") \ 63724ebfcd6SJosh Poimboeuf 2> >(tee -a "$log" >&2) || \ 63824ebfcd6SJosh Poimboeuf die "objtool klp diff failed" 63924ebfcd6SJosh Poimboeuf ) 64024ebfcd6SJosh Poimboeuf done 64124ebfcd6SJosh Poimboeuf} 64224ebfcd6SJosh Poimboeuf 64324ebfcd6SJosh Poimboeuf# Build and post-process livepatch module in $KMOD_DIR 64424ebfcd6SJosh Poimboeufbuild_patch_module() { 64524ebfcd6SJosh Poimboeuf local makefile="$KMOD_DIR/Kbuild" 64624ebfcd6SJosh Poimboeuf local log="$KMOD_DIR/build.log" 64724ebfcd6SJosh Poimboeuf local kmod_file 64824ebfcd6SJosh Poimboeuf local cflags=() 64924ebfcd6SJosh Poimboeuf local files=() 65024ebfcd6SJosh Poimboeuf local cmd=() 65124ebfcd6SJosh Poimboeuf 65224ebfcd6SJosh Poimboeuf rm -rf "$KMOD_DIR" 65324ebfcd6SJosh Poimboeuf mkdir -p "$KMOD_DIR" 65424ebfcd6SJosh Poimboeuf 65524ebfcd6SJosh Poimboeuf cp -f "$SRC/scripts/livepatch/init.c" "$KMOD_DIR" 65624ebfcd6SJosh Poimboeuf 65724ebfcd6SJosh Poimboeuf echo "obj-m := $NAME.o" > "$makefile" 65824ebfcd6SJosh Poimboeuf echo -n "$NAME-y := init.o" >> "$makefile" 65924ebfcd6SJosh Poimboeuf 66024ebfcd6SJosh Poimboeuf find "$DIFF_DIR" -type f -name "*.o" | mapfile -t files 66124ebfcd6SJosh Poimboeuf [[ ${#files[@]} -eq 0 ]] && die "no changes detected" 66224ebfcd6SJosh Poimboeuf 66324ebfcd6SJosh Poimboeuf for file in "${files[@]}"; do 66424ebfcd6SJosh Poimboeuf local rel_file="${file#"$DIFF_DIR"/}" 66524ebfcd6SJosh Poimboeuf local orig_file="$ORIG_DIR/$rel_file" 66624ebfcd6SJosh Poimboeuf local orig_dir="$(dirname "$orig_file")" 66724ebfcd6SJosh Poimboeuf local kmod_file="$KMOD_DIR/$rel_file" 66824ebfcd6SJosh Poimboeuf local kmod_dir="$(dirname "$kmod_file")" 66924ebfcd6SJosh Poimboeuf local cmd_file="$orig_dir/.$(basename "$file").cmd" 67024ebfcd6SJosh Poimboeuf 67124ebfcd6SJosh Poimboeuf mkdir -p "$kmod_dir" 67224ebfcd6SJosh Poimboeuf cp -f "$file" "$kmod_dir" 67324ebfcd6SJosh Poimboeuf [[ -e "$cmd_file" ]] && cp -f "$cmd_file" "$kmod_dir" 67424ebfcd6SJosh Poimboeuf 67524ebfcd6SJosh Poimboeuf # Tell kbuild this is a prebuilt object 67624ebfcd6SJosh Poimboeuf cp -f "$file" "${kmod_file}_shipped" 67724ebfcd6SJosh Poimboeuf 67824ebfcd6SJosh Poimboeuf echo -n " $rel_file" >> "$makefile" 67924ebfcd6SJosh Poimboeuf done 68024ebfcd6SJosh Poimboeuf 68124ebfcd6SJosh Poimboeuf echo >> "$makefile" 68224ebfcd6SJosh Poimboeuf 68324ebfcd6SJosh Poimboeuf cflags=("-ffunction-sections") 68424ebfcd6SJosh Poimboeuf cflags+=("-fdata-sections") 68524ebfcd6SJosh Poimboeuf [[ $REPLACE -eq 0 ]] && cflags+=("-DKLP_NO_REPLACE") 68624ebfcd6SJosh Poimboeuf 68724ebfcd6SJosh Poimboeuf cmd=("make") 68824ebfcd6SJosh Poimboeuf cmd+=("$VERBOSE") 68924ebfcd6SJosh Poimboeuf cmd+=("-j$JOBS") 69024ebfcd6SJosh Poimboeuf cmd+=("--directory=.") 69124ebfcd6SJosh Poimboeuf cmd+=("M=$KMOD_DIR") 69224ebfcd6SJosh Poimboeuf cmd+=("KCFLAGS=${cflags[*]}") 69324ebfcd6SJosh Poimboeuf 69424ebfcd6SJosh Poimboeuf # Build a "normal" kernel module with init.c and the diffed objects 69524ebfcd6SJosh Poimboeuf ( 69624ebfcd6SJosh Poimboeuf cd "$SRC" 69724ebfcd6SJosh Poimboeuf "${cmd[@]}" \ 69824ebfcd6SJosh Poimboeuf 1> >(tee -a "$log") \ 69924ebfcd6SJosh Poimboeuf 2> >(tee -a "$log" >&2) 70024ebfcd6SJosh Poimboeuf ) 70124ebfcd6SJosh Poimboeuf 70224ebfcd6SJosh Poimboeuf kmod_file="$KMOD_DIR/$NAME.ko" 70324ebfcd6SJosh Poimboeuf 70424ebfcd6SJosh Poimboeuf # Save off the intermediate binary for debugging 70524ebfcd6SJosh Poimboeuf cp -f "$kmod_file" "$kmod_file.orig" 70624ebfcd6SJosh Poimboeuf 70724ebfcd6SJosh Poimboeuf # Work around issue where slight .config change makes corrupt BTF 70824ebfcd6SJosh Poimboeuf objcopy --remove-section=.BTF "$kmod_file" 70924ebfcd6SJosh Poimboeuf 71024ebfcd6SJosh Poimboeuf # Fix (and work around) linker wreckage for klp syms / relocs 71124ebfcd6SJosh Poimboeuf "$SRC/tools/objtool/objtool" klp post-link "$kmod_file" || die "objtool klp post-link failed" 71224ebfcd6SJosh Poimboeuf 71324ebfcd6SJosh Poimboeuf cp -f "$kmod_file" "$OUTFILE" 71424ebfcd6SJosh Poimboeuf} 71524ebfcd6SJosh Poimboeuf 71624ebfcd6SJosh Poimboeuf 71724ebfcd6SJosh Poimboeuf################################################################################ 71824ebfcd6SJosh Poimboeuf 71924ebfcd6SJosh Poimboeufprocess_args "$@" 72024ebfcd6SJosh Poimboeufdo_init 72124ebfcd6SJosh Poimboeuf 72224ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 1 )); then 72324ebfcd6SJosh Poimboeuf status "Validating patch(es)" 72424ebfcd6SJosh Poimboeuf validate_patches 72524ebfcd6SJosh Poimboeuf status "Building original kernel" 72624ebfcd6SJosh Poimboeuf clean_kernel 72724ebfcd6SJosh Poimboeuf build_kernel 72824ebfcd6SJosh Poimboeuf status "Copying original object files" 72924ebfcd6SJosh Poimboeuf copy_orig_objects 73024ebfcd6SJosh Poimboeuffi 73124ebfcd6SJosh Poimboeuf 73224ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 2 )); then 73324ebfcd6SJosh Poimboeuf status "Fixing patch(es)" 73424ebfcd6SJosh Poimboeuf fix_patches 73524ebfcd6SJosh Poimboeuf apply_patches 73624ebfcd6SJosh Poimboeuf status "Building patched kernel" 73724ebfcd6SJosh Poimboeuf build_kernel 73824ebfcd6SJosh Poimboeuf revert_patches 73924ebfcd6SJosh Poimboeuf status "Copying patched object files" 74024ebfcd6SJosh Poimboeuf copy_patched_objects 74124ebfcd6SJosh Poimboeuffi 74224ebfcd6SJosh Poimboeuf 74324ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 3 )); then 74424ebfcd6SJosh Poimboeuf status "Diffing objects" 74524ebfcd6SJosh Poimboeuf diff_objects 74624ebfcd6SJosh Poimboeuffi 74724ebfcd6SJosh Poimboeuf 74824ebfcd6SJosh Poimboeufif (( SHORT_CIRCUIT <= 4 )); then 74924ebfcd6SJosh Poimboeuf status "Building patch module: $OUTFILE" 75024ebfcd6SJosh Poimboeuf build_patch_module 75124ebfcd6SJosh Poimboeuffi 75224ebfcd6SJosh Poimboeuf 75324ebfcd6SJosh Poimboeufstatus "SUCCESS" 754