#! /bin/sh # Add a .gdb_index section to a file. # Copyright (C) 2010-2021 Free Software Foundation, Inc. # 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 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . # This program assumes gdb and objcopy are in $PATH. # If not, or you want others, pass the following in the environment GDB=${GDB:=gdb} OBJCOPY=${OBJCOPY:=objcopy} READELF=${READELF:=readelf} myname="${0##*/}" dwarf5="" if [ "$1" = "-dwarf-5" ]; then dwarf5="$1" shift fi if test $# != 1; then echo "usage: $myname [-dwarf-5] FILE" 1>&2 exit 1 fi file="$1" if test -L "$file"; then if ! command -v readlink >/dev/null 2>&1; then echo "$myname: 'readlink' missing. Failed to follow symlink $1." 1>&2 exit 1 fi # Count number of links followed in order to detect loops. count=0 while test -L "$file"; do target=$(readlink "$file") case "$target" in /*) file="$target" ;; *) file="$(dirname "$file")/$target" ;; esac count="$((count + 1))" if test "$count" -gt 10; then echo "$myname: Detected loop while following link $file" exit 1 fi done fi if test ! -r "$file"; then echo "$myname: unable to access: $file" 1>&2 exit 1 fi dir="${file%/*}" test "$dir" = "$file" && dir="." dwz_file="" if $READELF -S "$file" | grep -q " \.gnu_debugaltlink "; then dwz_file=$($READELF --string-dump=.gnu_debugaltlink "$file" \ | grep -A1 "'\.gnu_debugaltlink':" \ | tail -n +2 \ | sed 's/.*]//') dwz_file=$(echo $dwz_file) if $READELF -S "$dwz_file" | grep -E -q " \.(gdb_index|debug_names) "; then # Already has an index, skip it. dwz_file="" fi fi set_files () { fpath="$1" index4="${fpath}.gdb-index" index5="${fpath}.debug_names" debugstr="${fpath}.debug_str" debugstrmerge="${fpath}.debug_str.merge" debugstrerr="${fpath}.debug_str.err" } tmp_files= for f in "$file" "$dwz_file"; do if [ "$f" = "" ]; then continue fi set_files "$f" tmp_files="$tmp_files $index4 $index5 $debugstr $debugstrmerge $debugstrerr" done rm -f $tmp_files # Ensure intermediate index file is removed when we exit. trap "rm -f $tmp_files" 0 $GDB --batch -nx -iex 'set auto-load no' \ -ex "file $file" -ex "save gdb-index $dwarf5 $dir" || { # Just in case. status=$? echo "$myname: gdb error generating index for $file" 1>&2 exit $status } # In some situations gdb can exit without creating an index. This is # not an error. # E.g., if $file is stripped. This behaviour is akin to stripping an # already stripped binary, it's a no-op. status=0 handle_file () { fpath="$1" set_files "$fpath" if test -f "$index4" -a -f "$index5"; then echo "$myname: Both index types were created for $fpath" 1>&2 status=1 elif test -f "$index4" -o -f "$index5"; then if test -f "$index4"; then index="$index4" section=".gdb_index" else index="$index5" section=".debug_names" fi debugstradd=false debugstrupdate=false if test -s "$debugstr"; then if ! $OBJCOPY --dump-section .debug_str="$debugstrmerge" "$fpath" \ /dev/null 2>$debugstrerr; then cat >&2 $debugstrerr exit 1 fi if grep -q "can't dump section '.debug_str' - it does not exist" \ $debugstrerr; then debugstradd=true else debugstrupdate=true cat >&2 $debugstrerr fi cat "$debugstr" >>"$debugstrmerge" fi $OBJCOPY --add-section $section="$index" \ --set-section-flags $section=readonly \ $(if $debugstradd; then \ echo --add-section .debug_str="$debugstrmerge"; \ echo --set-section-flags .debug_str=readonly; \ fi; \ if $debugstrupdate; then \ echo --update-section .debug_str="$debugstrmerge"; \ fi) \ "$fpath" "$fpath" status=$? else echo "$myname: No index was created for $fpath" 1>&2 echo "$myname: [Was there no debuginfo? Was there already an index?]" \ 1>&2 fi } handle_file "$file" if [ "$dwz_file" != "" ]; then handle_file "$dwz_file" fi exit $status