2007-03-06 23:32:51 +03:00
#!/bin/sh -efu
#
# Copyright (C) 2000-2006 Dmitry V. Levin <ldv@altlinux.org>
2010-10-01 08:32:46 +04:00
# Copyright (C) 2007-2010 Alexey Tourbin <at@altlinux.org>
2007-03-06 23:32:51 +03:00
#
# 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"
2010-10-01 08:32:46 +04:00
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
2010-09-30 16:45:40 +04:00
LibReq()
2007-03-06 23:32:51 +03:00
{
2010-09-30 16:45:40 +04:00
local f="$1"
2007-03-06 23:32:51 +03:00
local fname="${f#${RPM_BUILD_ROOT-}}"
2010-09-30 16:45:40 +04:00
# Obtain objdump info.
local dump
if ! dump=$(objdump -p "$f"); then
Warning "$f: objdump failed"
return 0
fi
2008-02-18 22:31:29 +03:00
2010-09-30 16:45:40 +04:00
# Obtain ELF segments.
local segments
if ! segments=$(readelf -Wl "$f"); then
Warning "$f: readelf failed"
return 0
fi
2007-03-06 23:32:51 +03:00
2010-09-30 16:45:40 +04:00
# Interp.
local interp
interp="$(printf '%s\n' "$segments" |
sed -ne 's,^[[:space:]]*\[Requesting program interpreter: \(/[^]]\+\)\]$,\1,p')"
[ -z "$interp" ] ||
printf '%s\n' "$interp"
2008-02-19 00:29:32 +03:00
2010-09-30 16:45:40 +04:00
# 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
2007-03-06 23:32:51 +03:00
2010-09-30 16:45:40 +04:00
# 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.
2010-10-01 08:32:46 +04:00
rm -rf "$tmpdir"/a
mkdir "$tmpdir"/a
2010-09-30 16:45:40 +04:00
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" '
2008-01-16 01:59:30 +03:00
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}
2010-09-30 16:45:40 +04:00
')"
if [ -z "$pathname" ]; then
Warning "$fname: library $name not found"
continue
fi
2010-10-01 08:32:46 +04:00
local orig_pathname="$pathname"
2010-09-30 16:45:40 +04:00
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
2007-11-25 18:34:41 +03:00
fi
2010-09-30 16:45:40 +04:00
done
[ -z "$prefix" ] || prefix="$prefix/"
local filedep=
if [ -n "$prefix" ]; then
if [ -n "$under_buildroot" ]; then
filedep=1
elif [ -z "$vers" ]; then
filedep=1
2007-11-25 18:34:41 +03:00
fi
2010-09-30 16:45:40 +04:00
fi
if [ -n "$filedep" ]; then
printf '%s\n' "$pathname"
continue
fi
2010-10-01 08:32:46 +04:00
# 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.
2010-09-30 16:45:40 +04:00
v=
for v in `printf %s "$vers" |tr : ' '`; do
printf '%s%s(%s)%s\n' "$prefix" "$name" "$v" "$suffix"
2007-03-06 23:32:51 +03:00
done
2010-10-01 08:32:46 +04:00
# Remember versioned sonames.
if [ -n "$v" ]; then
printf '%s\n' "$dep" |
LC_ALL=C sort -u -m -o "$tmpdir"/versioned{,} -
fi
2010-09-30 16:45:40 +04:00
done
2010-10-01 08:32:46 +04:00
[ -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.
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{,}
local libs
libs=$(LC_ALL=C join -t$'\t' -11 -22 -v1 -o 1.1 "$tmpdir"/a/{lib2dep,sym2lib} )
if [ -n "$libs" ]; then
# extra libraries from objdump
libs=$(printf '%s\n' "$libs" |LC_ALL=C sort -u)
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{+,}
LC_ALL=C sort -t$'\t' -u -k2,2 -k1,1 -o "$tmpdir"/a/sym2lib{,}
fi
# Diagnose underlinking.
libs=$(LC_ALL=C join -t$'\t' -11 -22 -v2 -o 2.2 "$tmpdir"/a/{lib2dep,sym2lib} )
if [ -n "$libs" ]; then
# extra libraries from ldd
libs=$(printf '%s\n' "$libs" |LC_ALL=C sort -u)
Warning "$f: underlinked libraries:" $libs
fi
# Append.
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"
2007-03-06 23:32:51 +03:00
}
2010-09-30 16:45:40 +04:00
ArgvFileAction LibReq "$@"
2010-10-01 08:32:46 +04:00
[ -s "$tmpdir"/lib2dep ] || exit 0
# Some standard libraries which use symbol versioning.
cat >"$tmpdir"/stddep <<EOF
libc.so.6
libc.so.6()(64bit)
libpthread.so.0
libpthread.so.0()(64bit)
libm.so.6
libm.so.6()(64bit)
libdl.so.2
libdl.so.2()(64bit)
libgcc_s.so.1
libgcc_s.so.1()(64bit)
libstdc++.so.6
libstdc++.so.6()(64bit)
EOF
# Some non-standard symbols found in standard libraries.
cat >"$tmpdir"/nonstdsym <<EOF
strlcat
strlcpy
EOF
while IFS=$'\t' read -r lib dep; do
# Deal with bpp.
provsym=$(ProvidedSymbols "$lib")
if [ -n "$provsym" ]; then
provn=$(printf '%s\n' "$provsym" |wc -l)
bpp=$(SuggestBPP "$provn")
else
Warning "$lib provides no symbols"
bpp=10
fi
# Deal with required symbols.
reqsym=$(printf '%s\n' "$lib" |LC_ALL=C join -t$'\t' -12 -o 1.1 "$tmpdir"/sym2lib -)
if [ -z "$reqsym" ]; then
LC_ALL=C fgrep -qs -x -e "$dep" "$tmpdir"/versioned ||
printf '%s\n' "$dep"
continue
fi
# Handle standard libraries.
if LC_ALL=C fgrep -qs -x -e "$dep" "$tmpdir"/stddep; then
if ! reqsym=$(printf '%s\n' "$reqsym" |fgrep -x -f "$tmpdir"/nonstdsym); then
LC_ALL=C fgrep -qs -x -e "$dep" "$tmpdir"/versioned ||
printf '%s\n' "$dep"
continue
fi
fi
# See if this dep is provided.
if [ -z "${lib##${RPM_BUILD_ROOT:-foo}/*}" ] ||
rpmquery --whatprovides --provides -- "$dep" |fgrep -qs -e "$dep = set:"; then
: good
else
Warning "$dep is not yet set-versioned"
LC_ALL=C fgrep -qs -x -e "$dep" "$tmpdir"/versioned ||
printf '%s\n' "$dep"
continue
fi
# Make set-versioned soname dependency.
#printf '%s\n' "$reqsym" |LC_ALL=C sort -c -u
set=$(printf '%s\n' "$reqsym" |@RPMCONFIGDIR@/mkset "$bpp")
printf '%s >= %s\n' "$dep" "$set"
done <"$tmpdir"/lib2dep