1*4882a593Smuzhiyun#!/bin/bash 2*4882a593Smuzhiyun# SPDX-License-Identifier: GPL-2.0 3*4882a593Smuzhiyun# (c) 2014, Sasha Levin <sasha.levin@oracle.com> 4*4882a593Smuzhiyun#set -x 5*4882a593Smuzhiyun 6*4882a593Smuzhiyunif [[ $# < 1 ]]; then 7*4882a593Smuzhiyun echo "Usage:" 8*4882a593Smuzhiyun echo " $0 -r <release> | <vmlinux> [base path] [modules path]" 9*4882a593Smuzhiyun exit 1 10*4882a593Smuzhiyunfi 11*4882a593Smuzhiyun 12*4882a593Smuzhiyunif [[ $1 == "-r" ]] ; then 13*4882a593Smuzhiyun vmlinux="" 14*4882a593Smuzhiyun basepath="auto" 15*4882a593Smuzhiyun modpath="" 16*4882a593Smuzhiyun release=$2 17*4882a593Smuzhiyun 18*4882a593Smuzhiyun for fn in {,/usr/lib/debug}/boot/vmlinux-$release{,.debug} /lib/modules/$release{,/build}/vmlinux ; do 19*4882a593Smuzhiyun if [ -e "$fn" ] ; then 20*4882a593Smuzhiyun vmlinux=$fn 21*4882a593Smuzhiyun break 22*4882a593Smuzhiyun fi 23*4882a593Smuzhiyun done 24*4882a593Smuzhiyun 25*4882a593Smuzhiyun if [[ $vmlinux == "" ]] ; then 26*4882a593Smuzhiyun echo "ERROR! vmlinux image for release $release is not found" >&2 27*4882a593Smuzhiyun exit 2 28*4882a593Smuzhiyun fi 29*4882a593Smuzhiyunelse 30*4882a593Smuzhiyun vmlinux=$1 31*4882a593Smuzhiyun basepath=${2-auto} 32*4882a593Smuzhiyun modpath=$3 33*4882a593Smuzhiyun release="" 34*4882a593Smuzhiyunfi 35*4882a593Smuzhiyun 36*4882a593Smuzhiyundeclare -A cache 37*4882a593Smuzhiyundeclare -A modcache 38*4882a593Smuzhiyun 39*4882a593Smuzhiyunfind_module() { 40*4882a593Smuzhiyun if [[ "$modpath" != "" ]] ; then 41*4882a593Smuzhiyun for fn in $(find "$modpath" -name "${module//_/[-_]}.ko*") ; do 42*4882a593Smuzhiyun if readelf -WS "$fn" | grep -qwF .debug_line ; then 43*4882a593Smuzhiyun echo $fn 44*4882a593Smuzhiyun return 45*4882a593Smuzhiyun fi 46*4882a593Smuzhiyun done 47*4882a593Smuzhiyun return 1 48*4882a593Smuzhiyun fi 49*4882a593Smuzhiyun 50*4882a593Smuzhiyun modpath=$(dirname "$vmlinux") 51*4882a593Smuzhiyun find_module && return 52*4882a593Smuzhiyun 53*4882a593Smuzhiyun if [[ $release == "" ]] ; then 54*4882a593Smuzhiyun release=$(gdb -ex 'print init_uts_ns.name.release' -ex 'quit' -quiet -batch "$vmlinux" | sed -n 's/\$1 = "\(.*\)".*/\1/p') 55*4882a593Smuzhiyun fi 56*4882a593Smuzhiyun 57*4882a593Smuzhiyun for dn in {/usr/lib/debug,}/lib/modules/$release ; do 58*4882a593Smuzhiyun if [ -e "$dn" ] ; then 59*4882a593Smuzhiyun modpath="$dn" 60*4882a593Smuzhiyun find_module && return 61*4882a593Smuzhiyun fi 62*4882a593Smuzhiyun done 63*4882a593Smuzhiyun 64*4882a593Smuzhiyun modpath="" 65*4882a593Smuzhiyun return 1 66*4882a593Smuzhiyun} 67*4882a593Smuzhiyun 68*4882a593Smuzhiyunparse_symbol() { 69*4882a593Smuzhiyun # The structure of symbol at this point is: 70*4882a593Smuzhiyun # ([name]+[offset]/[total length]) 71*4882a593Smuzhiyun # 72*4882a593Smuzhiyun # For example: 73*4882a593Smuzhiyun # do_basic_setup+0x9c/0xbf 74*4882a593Smuzhiyun 75*4882a593Smuzhiyun if [[ $module == "" ]] ; then 76*4882a593Smuzhiyun local objfile=$vmlinux 77*4882a593Smuzhiyun elif [[ "${modcache[$module]+isset}" == "isset" ]]; then 78*4882a593Smuzhiyun local objfile=${modcache[$module]} 79*4882a593Smuzhiyun else 80*4882a593Smuzhiyun local objfile=$(find_module) 81*4882a593Smuzhiyun if [[ $objfile == "" ]] ; then 82*4882a593Smuzhiyun echo "WARNING! Modules path isn't set, but is needed to parse this symbol" >&2 83*4882a593Smuzhiyun return 84*4882a593Smuzhiyun fi 85*4882a593Smuzhiyun modcache[$module]=$objfile 86*4882a593Smuzhiyun fi 87*4882a593Smuzhiyun 88*4882a593Smuzhiyun # Remove the englobing parenthesis 89*4882a593Smuzhiyun symbol=${symbol#\(} 90*4882a593Smuzhiyun symbol=${symbol%\)} 91*4882a593Smuzhiyun 92*4882a593Smuzhiyun # Strip segment 93*4882a593Smuzhiyun local segment 94*4882a593Smuzhiyun if [[ $symbol == *:* ]] ; then 95*4882a593Smuzhiyun segment=${symbol%%:*}: 96*4882a593Smuzhiyun symbol=${symbol#*:} 97*4882a593Smuzhiyun fi 98*4882a593Smuzhiyun 99*4882a593Smuzhiyun # Strip the symbol name so that we could look it up 100*4882a593Smuzhiyun local name=${symbol%+*} 101*4882a593Smuzhiyun 102*4882a593Smuzhiyun # Use 'nm vmlinux' to figure out the base address of said symbol. 103*4882a593Smuzhiyun # It's actually faster to call it every time than to load it 104*4882a593Smuzhiyun # all into bash. 105*4882a593Smuzhiyun if [[ "${cache[$module,$name]+isset}" == "isset" ]]; then 106*4882a593Smuzhiyun local base_addr=${cache[$module,$name]} 107*4882a593Smuzhiyun else 108*4882a593Smuzhiyun local base_addr=$(nm "$objfile" | awk '$3 == "'$name'" && ($2 == "t" || $2 == "T") {print $1; exit}') 109*4882a593Smuzhiyun if [[ $base_addr == "" ]] ; then 110*4882a593Smuzhiyun # address not found 111*4882a593Smuzhiyun return 112*4882a593Smuzhiyun fi 113*4882a593Smuzhiyun cache[$module,$name]="$base_addr" 114*4882a593Smuzhiyun fi 115*4882a593Smuzhiyun # Let's start doing the math to get the exact address into the 116*4882a593Smuzhiyun # symbol. First, strip out the symbol total length. 117*4882a593Smuzhiyun local expr=${symbol%/*} 118*4882a593Smuzhiyun 119*4882a593Smuzhiyun # Now, replace the symbol name with the base address we found 120*4882a593Smuzhiyun # before. 121*4882a593Smuzhiyun expr=${expr/$name/0x$base_addr} 122*4882a593Smuzhiyun 123*4882a593Smuzhiyun # Evaluate it to find the actual address 124*4882a593Smuzhiyun expr=$((expr)) 125*4882a593Smuzhiyun local address=$(printf "%x\n" "$expr") 126*4882a593Smuzhiyun 127*4882a593Smuzhiyun # Pass it to addr2line to get filename and line number 128*4882a593Smuzhiyun # Could get more than one result 129*4882a593Smuzhiyun if [[ "${cache[$module,$address]+isset}" == "isset" ]]; then 130*4882a593Smuzhiyun local code=${cache[$module,$address]} 131*4882a593Smuzhiyun else 132*4882a593Smuzhiyun local code=$(${CROSS_COMPILE}addr2line -i -e "$objfile" "$address") 133*4882a593Smuzhiyun cache[$module,$address]=$code 134*4882a593Smuzhiyun fi 135*4882a593Smuzhiyun 136*4882a593Smuzhiyun # addr2line doesn't return a proper error code if it fails, so 137*4882a593Smuzhiyun # we detect it using the value it prints so that we could preserve 138*4882a593Smuzhiyun # the offset/size into the function and bail out 139*4882a593Smuzhiyun if [[ $code == "??:0" ]]; then 140*4882a593Smuzhiyun return 141*4882a593Smuzhiyun fi 142*4882a593Smuzhiyun 143*4882a593Smuzhiyun # Strip out the base of the path on each line 144*4882a593Smuzhiyun code=$(while read -r line; do echo "${line#$basepath/}"; done <<< "$code") 145*4882a593Smuzhiyun 146*4882a593Smuzhiyun # In the case of inlines, move everything to same line 147*4882a593Smuzhiyun code=${code//$'\n'/' '} 148*4882a593Smuzhiyun 149*4882a593Smuzhiyun # Replace old address with pretty line numbers 150*4882a593Smuzhiyun symbol="$segment$name ($code)" 151*4882a593Smuzhiyun} 152*4882a593Smuzhiyun 153*4882a593Smuzhiyundecode_code() { 154*4882a593Smuzhiyun local scripts=`dirname "${BASH_SOURCE[0]}"` 155*4882a593Smuzhiyun 156*4882a593Smuzhiyun echo "$1" | $scripts/decodecode 157*4882a593Smuzhiyun} 158*4882a593Smuzhiyun 159*4882a593Smuzhiyunhandle_line() { 160*4882a593Smuzhiyun local words 161*4882a593Smuzhiyun 162*4882a593Smuzhiyun # Tokenize 163*4882a593Smuzhiyun read -a words <<<"$1" 164*4882a593Smuzhiyun 165*4882a593Smuzhiyun # Remove hex numbers. Do it ourselves until it happens in the 166*4882a593Smuzhiyun # kernel 167*4882a593Smuzhiyun 168*4882a593Smuzhiyun # We need to know the index of the last element before we 169*4882a593Smuzhiyun # remove elements because arrays are sparse 170*4882a593Smuzhiyun local last=$(( ${#words[@]} - 1 )) 171*4882a593Smuzhiyun 172*4882a593Smuzhiyun for i in "${!words[@]}"; do 173*4882a593Smuzhiyun # Remove the address 174*4882a593Smuzhiyun if [[ ${words[$i]} =~ \[\<([^]]+)\>\] ]]; then 175*4882a593Smuzhiyun unset words[$i] 176*4882a593Smuzhiyun fi 177*4882a593Smuzhiyun 178*4882a593Smuzhiyun # Format timestamps with tabs 179*4882a593Smuzhiyun if [[ ${words[$i]} == \[ && ${words[$i+1]} == *\] ]]; then 180*4882a593Smuzhiyun unset words[$i] 181*4882a593Smuzhiyun words[$i+1]=$(printf "[%13s\n" "${words[$i+1]}") 182*4882a593Smuzhiyun fi 183*4882a593Smuzhiyun done 184*4882a593Smuzhiyun 185*4882a593Smuzhiyun if [[ ${words[$last]} =~ \[([^]]+)\] ]]; then 186*4882a593Smuzhiyun module=${words[$last]} 187*4882a593Smuzhiyun module=${module#\[} 188*4882a593Smuzhiyun module=${module%\]} 189*4882a593Smuzhiyun symbol=${words[$last-1]} 190*4882a593Smuzhiyun unset words[$last-1] 191*4882a593Smuzhiyun else 192*4882a593Smuzhiyun # The symbol is the last element, process it 193*4882a593Smuzhiyun symbol=${words[$last]} 194*4882a593Smuzhiyun module= 195*4882a593Smuzhiyun fi 196*4882a593Smuzhiyun 197*4882a593Smuzhiyun unset words[$last] 198*4882a593Smuzhiyun parse_symbol # modifies $symbol 199*4882a593Smuzhiyun 200*4882a593Smuzhiyun # Add up the line number to the symbol 201*4882a593Smuzhiyun echo "${words[@]}" "$symbol $module" 202*4882a593Smuzhiyun} 203*4882a593Smuzhiyun 204*4882a593Smuzhiyunif [[ $basepath == "auto" ]] ; then 205*4882a593Smuzhiyun module="" 206*4882a593Smuzhiyun symbol="kernel_init+0x0/0x0" 207*4882a593Smuzhiyun parse_symbol 208*4882a593Smuzhiyun basepath=${symbol#kernel_init (} 209*4882a593Smuzhiyun basepath=${basepath%/init/main.c:*)} 210*4882a593Smuzhiyunfi 211*4882a593Smuzhiyun 212*4882a593Smuzhiyunwhile read line; do 213*4882a593Smuzhiyun # Let's see if we have an address in the line 214*4882a593Smuzhiyun if [[ $line =~ \[\<([^]]+)\>\] ]] || 215*4882a593Smuzhiyun [[ $line =~ [^+\ ]+\+0x[0-9a-f]+/0x[0-9a-f]+ ]]; then 216*4882a593Smuzhiyun # Translate address to line numbers 217*4882a593Smuzhiyun handle_line "$line" 218*4882a593Smuzhiyun # Is it a code line? 219*4882a593Smuzhiyun elif [[ $line == *Code:* ]]; then 220*4882a593Smuzhiyun decode_code "$line" 221*4882a593Smuzhiyun else 222*4882a593Smuzhiyun # Nothing special in this line, show it as is 223*4882a593Smuzhiyun echo "$line" 224*4882a593Smuzhiyun fi 225*4882a593Smuzhiyundone 226