mirror of
https://sourceware.org/git/glibc.git
synced 2025-01-15 05:20:05 +00:00
94bf958a94
It was noted in 2005 (BZ #832), 2006 (BZ #3266), and 2007 [1] that ldd fails on shells other than Bash >= 3.0 because of the pipefail option around try_trace (added on 2004-12-08). EGLIBC was patched in 2008 [2] (r6912) to make the pipefail check run only on shells that support it, but RTLD output would still be lost on other shells with certain SELinux policies. This patch rewrites try_trace to work on any POSIX-conformant shell in such a way as to also work with such SELinux policies. It also obviates one difference between glibc and EGLIBC. URL: https://sourceware.org/ml/libc-alpha/2007-01/msg00041.html URL: http://www.eglibc.org/archives/patches/msg00526.html 2013-09-11 P. J. McDermott <pj@pehjota.net> [BZ #832] * elf/ldd.bash.in (try_trace): More robustly and portably work around SELinux terminal write permissions by using a command substitution instead of a pipeline and pipefail option.
204 lines
5.5 KiB
Bash
204 lines
5.5 KiB
Bash
#! @BASH@
|
|
# Copyright (C) 1996-2013 Free Software Foundation, Inc.
|
|
# This file is part of the GNU C Library.
|
|
|
|
# The GNU C Library is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU Lesser General Public
|
|
# License as published by the Free Software Foundation; either
|
|
# version 2.1 of the License, or (at your option) any later version.
|
|
|
|
# The GNU C Library 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
|
|
# Lesser General Public License for more details.
|
|
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
# License along with the GNU C Library; if not, see
|
|
# <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
# This is the `ldd' command, which lists what shared libraries are
|
|
# used by given dynamically-linked executables. It works by invoking the
|
|
# run-time dynamic linker as a command and setting the environment
|
|
# variable LD_TRACE_LOADED_OBJECTS to a non-empty value.
|
|
|
|
# We should be able to find the translation right at the beginning.
|
|
TEXTDOMAIN=libc
|
|
TEXTDOMAINDIR=@TEXTDOMAINDIR@
|
|
|
|
RTLDLIST=@RTLD@
|
|
warn=
|
|
bind_now=
|
|
verbose=
|
|
|
|
while test $# -gt 0; do
|
|
case "$1" in
|
|
--vers | --versi | --versio | --version)
|
|
echo 'ldd @PKGVERSION@@VERSION@'
|
|
printf $"Copyright (C) %s Free Software Foundation, Inc.
|
|
This is free software; see the source for copying conditions. There is NO
|
|
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
" "2013"
|
|
printf $"Written by %s and %s.
|
|
" "Roland McGrath" "Ulrich Drepper"
|
|
exit 0
|
|
;;
|
|
--h | --he | --hel | --help)
|
|
echo $"Usage: ldd [OPTION]... FILE...
|
|
--help print this help and exit
|
|
--version print version information and exit
|
|
-d, --data-relocs process data relocations
|
|
-r, --function-relocs process data and function relocations
|
|
-u, --unused print unused direct dependencies
|
|
-v, --verbose print all information
|
|
"
|
|
printf $"For bug reporting instructions, please see:\\n%s.\\n" \
|
|
"@REPORT_BUGS_TO@"
|
|
exit 0
|
|
;;
|
|
-d | --d | --da | --dat | --data | --data- | --data-r | --data-re | \
|
|
--data-rel | --data-relo | --data-reloc | --data-relocs)
|
|
warn=yes
|
|
shift
|
|
;;
|
|
-r | --f | --fu | --fun | --func | --funct | --functi | --functio | \
|
|
--function | --function- | --function-r | --function-re | --function-rel | \
|
|
--function-relo | --function-reloc | --function-relocs)
|
|
warn=yes
|
|
bind_now=yes
|
|
shift
|
|
;;
|
|
-v | --verb | --verbo | --verbos | --verbose)
|
|
verbose=yes
|
|
shift
|
|
;;
|
|
-u | --u | --un | --unu | --unus | --unuse | --unused)
|
|
unused=yes
|
|
shift
|
|
;;
|
|
--v | --ve | --ver)
|
|
echo >&2 $"ldd: option \`$1' is ambiguous"
|
|
exit 1
|
|
;;
|
|
--) # Stop option processing.
|
|
shift; break
|
|
;;
|
|
-*)
|
|
echo >&2 'ldd:' $"unrecognized option" "\`$1'"
|
|
echo >&2 $"Try \`ldd --help' for more information."
|
|
exit 1
|
|
;;
|
|
*)
|
|
break
|
|
;;
|
|
esac
|
|
done
|
|
|
|
nonelf ()
|
|
{
|
|
# Maybe extra code for non-ELF binaries.
|
|
return 1;
|
|
}
|
|
|
|
add_env="LD_TRACE_LOADED_OBJECTS=1 LD_WARN=$warn LD_BIND_NOW=$bind_now"
|
|
add_env="$add_env LD_VERBOSE=$verbose"
|
|
if test "$unused" = yes; then
|
|
add_env="$add_env LD_DEBUG=\"$LD_DEBUG${LD_DEBUG:+,}unused\""
|
|
fi
|
|
|
|
# The following command substitution is needed to make ldd work in SELinux
|
|
# environments where the RTLD might not have permission to write to the
|
|
# terminal. The extra "x" character prevents the shell from trimming trailing
|
|
# newlines from command substitution results. This function is defined as a
|
|
# subshell compound list (using "(...)") to prevent parameter assignments from
|
|
# affecting the calling shell execution environment.
|
|
try_trace() (
|
|
output=$(eval $add_env '"$@"' 2>&1; rc=$?; printf 'x'; exit $rc)
|
|
rc=$?
|
|
printf '%s' "${output%x}"
|
|
return $rc
|
|
)
|
|
|
|
case $# in
|
|
0)
|
|
echo >&2 'ldd:' $"missing file arguments"
|
|
echo >&2 $"Try \`ldd --help' for more information."
|
|
exit 1
|
|
;;
|
|
1)
|
|
single_file=t
|
|
;;
|
|
*)
|
|
single_file=f
|
|
;;
|
|
esac
|
|
|
|
result=0
|
|
for file do
|
|
# We don't list the file name when there is only one.
|
|
test $single_file = t || echo "${file}:"
|
|
case $file in
|
|
*/*) :
|
|
;;
|
|
*) file=./$file
|
|
;;
|
|
esac
|
|
if test ! -e "$file"; then
|
|
echo "ldd: ${file}:" $"No such file or directory" >&2
|
|
result=1
|
|
elif test ! -f "$file"; then
|
|
echo "ldd: ${file}:" $"not regular file" >&2
|
|
result=1
|
|
elif test -r "$file"; then
|
|
test -x "$file" || echo 'ldd:' $"\
|
|
warning: you do not have execution permission for" "\`$file'" >&2
|
|
RTLD=
|
|
ret=1
|
|
for rtld in ${RTLDLIST}; do
|
|
if test -x $rtld; then
|
|
verify_out=`${rtld} --verify "$file"`
|
|
ret=$?
|
|
case $ret in
|
|
[02]) RTLD=${rtld}; break;;
|
|
esac
|
|
fi
|
|
done
|
|
case $ret in
|
|
0)
|
|
# If the program exits with exit code 5, it means the process has been
|
|
# invoked with __libc_enable_secure. Fall back to running it through
|
|
# the dynamic linker.
|
|
try_trace "$file"
|
|
rc=$?
|
|
if [ $rc = 5 ]; then
|
|
try_trace "$RTLD" "$file"
|
|
rc=$?
|
|
fi
|
|
[ $rc = 0 ] || result=1
|
|
;;
|
|
1)
|
|
# This can be a non-ELF binary or no binary at all.
|
|
nonelf "$file" || {
|
|
echo $" not a dynamic executable"
|
|
result=1
|
|
}
|
|
;;
|
|
2)
|
|
try_trace "$RTLD" "$file" || result=1
|
|
;;
|
|
*)
|
|
echo 'ldd:' ${RTLD} $"exited with unknown exit code" "($ret)" >&2
|
|
exit 1
|
|
;;
|
|
esac
|
|
else
|
|
echo 'ldd:' $"error: you do not have read permission for" "\`$file'" >&2
|
|
result=1
|
|
fi
|
|
done
|
|
|
|
exit $result
|
|
# Local Variables:
|
|
# mode:ksh
|
|
# End:
|