2020-08-11 22:04:28 +08:00
|
|
|
#!/bin/bash
|
2017-05-12 01:40:38 +08:00
|
|
|
# Copyright © 2016 IBM Corporation
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU General Public License
|
|
|
|
# as published by the Free Software Foundation; either version
|
|
|
|
# 2 of the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This script checks the relocations of a vmlinux for "suspicious"
|
|
|
|
# branches from unrelocated code (head_64.S code).
|
|
|
|
|
|
|
|
# Turn this on if you want more debug output:
|
|
|
|
# set -x
|
|
|
|
|
|
|
|
# Have Kbuild supply the path to objdump so we handle cross compilation.
|
|
|
|
objdump="$1"
|
|
|
|
vmlinux="$2"
|
|
|
|
|
|
|
|
#__end_interrupts should be located within the first 64K
|
2020-08-11 22:04:29 +08:00
|
|
|
kstart=0xc000000000000000
|
|
|
|
printf -v kend '0x%x' $(( kstart + 0x10000 ))
|
2017-05-12 01:40:38 +08:00
|
|
|
|
|
|
|
end_intr=0x$(
|
2020-08-11 22:04:29 +08:00
|
|
|
$objdump -R -d --start-address="$kstart" --stop-address="$kend" "$vmlinux" |
|
|
|
|
awk '$2 == "<__end_interrupts>:" { print $1 }'
|
2017-05-12 01:40:38 +08:00
|
|
|
)
|
|
|
|
|
2020-08-11 22:04:30 +08:00
|
|
|
$objdump -R -D --no-show-raw-insn --start-address="$kstart" --stop-address="$end_intr" "$vmlinux" |
|
2020-08-11 22:04:31 +08:00
|
|
|
sed -E -n '
|
|
|
|
# match lines that start with a kernel address
|
|
|
|
/^c[0-9a-f]*:\s*b/ {
|
|
|
|
# drop a target that we do not care about
|
|
|
|
/\<__start_initialization_multiplatform>/d
|
|
|
|
# drop branches via ctr or lr
|
|
|
|
/\<b.?.?(ct|l)r/d
|
|
|
|
# cope with some differences between Clang and GNU objdumps
|
|
|
|
s/\<bt.?\s*[[:digit:]]+,/beq/
|
|
|
|
s/\<bf.?\s*[[:digit:]]+,/bne/
|
|
|
|
# tidy up
|
|
|
|
s/\s0x/ /
|
|
|
|
s/://
|
|
|
|
# format for the loop below
|
|
|
|
s/^(\S+)\s+(\S+)\s+(\S+)\s*(\S*).*$/\1:\2:0x\3:\4/
|
|
|
|
# strip out condition registers
|
|
|
|
s/:0xcr[0-7],/:0x/
|
|
|
|
p
|
2020-08-11 22:04:32 +08:00
|
|
|
}' | {
|
2017-05-12 01:40:38 +08:00
|
|
|
|
2020-08-11 22:04:32 +08:00
|
|
|
all_good=true
|
|
|
|
while IFS=: read -r from branch to sym; do
|
2020-08-11 22:04:29 +08:00
|
|
|
if (( to > end_intr )); then
|
2020-08-11 22:04:32 +08:00
|
|
|
if $all_good; then
|
|
|
|
printf '%s\n' 'WARNING: Unrelocated relative branches'
|
|
|
|
all_good=false
|
2017-05-12 01:40:38 +08:00
|
|
|
fi
|
2020-08-11 22:04:32 +08:00
|
|
|
printf '%s %s-> %s %s\n' "$from" "$branch" "$to" "$sym"
|
2017-05-12 01:40:38 +08:00
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
2020-08-11 22:04:32 +08:00
|
|
|
$all_good
|
|
|
|
|
|
|
|
}
|