#!/bin/sh -efu # # Copyright (C) 2000-2006 Dmitry V. Levin # Copyright (C) 2007-2010 Alexey Tourbin # # 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 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, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA # . @RPMCONFIGDIR@/functions dump_ld_config='@RPMCONFIGDIR@/dump_ld_config' shlib_req='@RPMCONFIGDIR@/shlib.req.awk' elf_ldd='@RPMCONFIGDIR@/ldd' [ -n "${RPM_LIBDIR-}" ] || RPM_LIBDIR=`rpm --eval %_libdir` [ -n "${RPM_LIB-}" ] || RPM_LIB=`rpm --eval %_lib` [ -n "${RPM_ARCH-}" ] || RPM_ARCH=`rpm --eval %_arch` RPM_FINDREQ_RPATH="/$RPM_LIB $RPM_LIBDIR $("$dump_ld_config")" [ -z "${RPM_BUILD_ROOT-}" ] || RPM_FINDREQ_RPATH="$("$dump_ld_config" '' "$RPM_BUILD_ROOT") $RPM_FINDREQ_RPATH" Debug "RPM_FINDREQ_RPATH=$RPM_FINDREQ_RPATH" Cleanup() { rm -rf "$tmpdir" exit $1 } tmpdir=$(mktemp -dt "$PROG.XXXXXXXX") trap 'Cleanup $?' EXIT trap 'Cleanup 143' HUP INT QUIT PIPE TERM >"$tmpdir"/lib2dep >"$tmpdir"/sym2lib >"$tmpdir"/versioned LibReq() { local f="$1" local fname="${f#${RPM_BUILD_ROOT-}}" # Obtain objdump info. local dump if ! dump=$(objdump -p "$f"); then Warning "$f: objdump failed" return 0 fi # Obtain ELF segments. local segments if ! segments=$(readelf -Wl "$f"); then Warning "$f: readelf failed" return 0 fi # Interp. local interp interp="$(printf '%s\n' "$segments" | sed -ne 's,^[[:space:]]*\[Requesting program interpreter: \(/[^]]\+\)\]$,\1,p')" [ -z "$interp" ] || printf '%s\n' "$interp" # GNU_HASH. if printf '%s\n' "$segments" | egrep -qs '[[:space:]]\.gnu\.hash([[:space:]]|$)' && ! printf '%s\n' "$segments" | egrep -qs '[[:space:]]\.hash([[:space:]]|$)'; then echo 'rtld(GNU_HASH)' fi # GNU_UNIQUE. if readelf -Ws "$f" |grep -qs '^[[:space:]]*[0-9]\+:[[:space:]]\+[0-9a-f]\+[[:space:]]\+[0-9a-f]\+[[:space:]]\+[^[:space:]]\+[[:space:]]\+UNIQUE[[:space:]]'; then echo 'rtld(GNU_UNIQUE)' fi # That could be "statically linked (uses shared libs)". printf '%s\n' "$dump" |grep -qs '^Dynamic Section:$' || return 0 # The rest is soname stuff. local braces canon_prefix deps dir name pathname prefix rpath suffix v vers suffix="$(printf '%s\n' "$dump" |sed -ne 's/^.*file format \(elf64\).*$/(64bit)/p')" [ -z "$suffix" ] && braces= || braces='()' rpath="$(printf %s "$dump" | awk '($1=="RPATH"||$1=="RUNPATH"){print $2}' | tr -s : ' ' | sed -e "s|\$ORIGIN|${fname%/*}|g")" if [ -n "$rpath" ]; then rpath="$rpath $RPM_FINDREQ_RPATH" else rpath="$RPM_FINDREQ_RPATH" fi rpath="$(printf %s "$rpath" | tr -s '[:space:]' '\n' | grep -v '^$' | LANG=C uniq | sed -e "s|^|${RPM_BUILD_ROOT-}&|" | tr -s '[:space:]' : | sed -e 's/^:\+//; s/:\+$//')" deps="$("$elf_ldd" -- "$f" "$rpath")" || return 1 # Shared library dependencies, version references. rm -rf "$tmpdir"/a mkdir "$tmpdir"/a for vers in `printf '%s\n' "$dump" |"$shlib_req"`; do name="$(printf %s "$vers" |cut -d: -f1)" vers="$(printf %s "$vers" |cut -d: -f2-)" pathname="$(printf %s "$deps" |awk "-vname=$name" ' function basename(f) { sub("^.*/","",f); return f; } NF>=3 && ($1==name || basename($1)==name) && $2=="=>" && $3~"^/" {print $3} NF==2 && ($1==name || basename($1)==name) && $1~"^/" && $2~"^[(]0x" {print $1} ')" if [ -z "$pathname" ]; then Warning "$fname: library $name not found" continue fi local orig_pathname="$pathname" pathname=$(CanonPath "$pathname") Verbose "$fname: $name -> $pathname" local under_buildroot= if [ -n "${RPM_BUILD_ROOT-}" ] && [ -z "${pathname##$RPM_BUILD_ROOT*}" ]; then pathname=${pathname#$RPM_BUILD_ROOT} under_buildroot=1 fi prefix="${pathname%/*}" canon_prefix="$(printf %s "$prefix/" | sed -e 's|/tls/|/|' -e 's|/sse2/|/|' -e "s|/$RPM_ARCH/|/|" -e 's|/i[3-9]86/|/|' -e 's|/\+$||')" if [ -z "$canon_prefix" -o -n "${canon_prefix##/*}" ]; then Warning "$fname: library $name not found" continue fi for dir in $RPM_FINDREQ_RPATH; do if [ "$canon_prefix" = "$dir" ]; then prefix= break fi done [ -z "$prefix" ] || prefix="$prefix/" local filedep= if [ -n "$prefix" ]; then if [ -n "$under_buildroot" ]; then filedep=1 elif [ -z "$vers" ]; then filedep=1 fi fi if [ -n "$filedep" ]; then printf '%s\n' "$pathname" printf '%s\n' "$orig_pathname" >>"$tmpdir"/a/flib continue fi # Postpone main requires entry. local dep="$(printf '%s%s%s%s\n' "$prefix" "$name" "$braces" "$suffix")" printf '%s\t%s\n' "$orig_pathname" "$dep" >>"$tmpdir"/a/lib2dep # Print versioned references. v= for v in `printf %s "$vers" |tr : ' '`; do printf '%s%s(%s)%s\n' "$prefix" "$name" "$v" "$suffix" done # Remember versioned sonames. if [ -n "$v" ]; then printf '%s\n' "$dep" | LC_ALL=C sort -u -m -o "$tmpdir"/versioned{,} - fi done [ -s "$tmpdir"/a/lib2dep ] || return 0 # Deal with symbols. $elf_ldd --bindings -- "$f" "$rpath" 2>&1 | LC_ALL=C fgrep "binding file $f [" | awk '($4!=$7){print$11"\t"$7}' | sed -e "s/[\`']//g" >"$tmpdir"/a/sym2lib if ! [ -s "$tmpdir"/a/sym2lib ]; then Warning "$f: no symbol bindings" return 0 fi # Diagnose overlinking. cut -f1 "$tmpdir"/a/lib2dep >"$tmpdir"/a/lib1 if [ -s "$tmpdir"/a/flib ]; then cat "$tmpdir"/a/flib >>"$tmpdir"/a/lib1 fi LC_ALL=C sort -u -o "$tmpdir"/a/lib1{,} cut -f2 "$tmpdir"/a/sym2lib >"$tmpdir"/a/lib2 LC_ALL=C sort -u -o "$tmpdir"/a/lib2{,} local libs libs=$(LC_ALL=C comm -23 "$tmpdir"/a/lib{1,2} ) if [ -n "$libs" ]; then # extra libraries from objdump Warning "$f: overlinked libraries:" $libs fi # Exclude weak undefined symbols. nm -D "$f" |awk 'NF==2&&($1=="w"||$1=="v"){print$2}' >"$tmpdir"/a/weak if [ -s "$tmpdir"/a/weak ]; then LC_ALL=C sort -u -o "$tmpdir"/a/weak{,} LC_ALL=C sort -u -o "$tmpdir"/a/sym2lib{,} LC_ALL=C join -t$'\t' -v1 -o 1.1,1.2 "$tmpdir"/a/{sym2lib,weak} >"$tmpdir"/a/sym2lib+ mv -f "$tmpdir"/a/sym2lib{+,} cut -f2 "$tmpdir"/a/sym2lib >"$tmpdir"/a/lib2 LC_ALL=C sort -u -o "$tmpdir"/a/lib2{,} fi # Diagnose underlinking. libs=$(LC_ALL=C comm -13 "$tmpdir"/a/lib{1,2} ) if [ -n "$libs" ]; then # extra libraries from ldd Warning "$f: underlinked libraries:" $libs fi # Append. LC_ALL=C sort -t$'\t' -u -k1,1 -k2,2 -o "$tmpdir"/a/lib2dep{,} LC_ALL=C sort -t$'\t' -u -k2,2 -k1,1 -o "$tmpdir"/a/sym2lib{,} LC_ALL=C sort -t$'\t' -m -u -k1,1 -k2,2 -o "$tmpdir"/lib2dep{,} "$tmpdir"/a/lib2dep LC_ALL=C sort -t$'\t' -m -u -k2,2 -k1,1 -o "$tmpdir"/sym2lib{,} "$tmpdir"/a/sym2lib } ProvidedSymbols() { readelf --wide --dyn-syms "$1" | LC_ALL=C egrep '^ *[0-9]+: ' | awk 'NF>=8 && # dl-lookup.c: /st_value ($2!="00000000" && $2!="0000000000000000" || $4=="TLS") && # dl-lookup.c: /st_shndx ($7!="UND") && # dl-lookup.c: /ALLOWED_STT ($4=="NOTYPE" || $4=="OBJECT" || $4 == "FUNC" || $4=="COMMON" || $4=="TLS" || $4=="IFUNC") && # dl-lookup.c: /hidden ($6=="DEFAULT" || $6=="PROTECTED") && # dl-lookup.c: /switch.*ST_BIND ($5=="GLOBAL" || $5=="WEAK" || $5 == "UNIQUE") && # Ignore special symbols found in each library: ($8!="__bss_start" && $8!="_edata" && $8!="_end" && $8!="_fini" && $8!="_init") { sym = $8 # No symbol versioning yet. ix = index(sym, "@") if (ix > 1) sym = substr(sym, 0, ix-1) print sym }' | LC_ALL=C sort -u } SuggestBPP() { # For the number of symbols in the range 2^{m-1}..2^m-1, # use m+10 bits per symbol. Upper bound on the error rate # is 2^{-10} (about 0.1%). perl -le 'print int log(shift) / log(2) + 11' "$1" } ArgvFileAction LibReq "$@" [ -s "$tmpdir"/lib2dep ] || exit 0 # Some standard libraries which use symbol versioning. cat >"$tmpdir"/stddep <"$tmpdir"/nonstdsym <= %s\n' "$dep" "$set" done <"$tmpdir"/lib2dep