Mike Frysinger 564dcfbde6 check_deps: handle "include" directives in ld.so.conf
The ld.so.conf file supports an "include" directive where it'll include
all the files matched by the following glob.  Since Gentoo has started
using this, we need to support parsing of it too.  So move the parsing
code into a dedicated function and recursively call ourselves when we
hit an include directive.

BUG=chromium-os:20636
TEST=`./check_deps /build/amd64-generic /bin/bash` now finds libs
TEST=`./build_image --board=x86-alex` still works

Change-Id: I8894ca42358d91d8f2ee6e95b47faf9334ccdd26
Reviewed-on: http://gerrit.chromium.org/gerrit/8494
Reviewed-by: David James <davidjames@chromium.org>
Commit-Ready: Mike Frysinger <vapier@chromium.org>
Tested-by: Mike Frysinger <vapier@chromium.org>
2011-09-30 22:29:41 -07:00

173 lines
4.5 KiB
Python
Executable File

#!/usr/bin/python
# Copyright (c) 2010 The Chromium OS Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
import os
import re
import sys
import glob
_SHARED_RE = re.compile(r"Shared library: \[([^\]]+)\]")
_RPATH_RE = re.compile(r"Library rpath: \[([^\]]+)\]")
class CheckDependencies(object):
"""Check that dependencies for binaries can be found in the specified dir."""
def _ReadLdSoConf(self, path):
"""Parse ld.so.conf files.
Starting with the file at PATH (searched relative to self._root), return
all the valid libdirs found. Include directives are handled recursively.
Args:
path: the path to the ld.so.conf file (inside of the root).
Returns:
A list of valid libdirs.
"""
libdirs = []
ld_so_conf = self._root + path
if os.path.exists(ld_so_conf):
f = file(ld_so_conf)
for line in f:
line = line.rstrip()
if line.startswith("/"):
libpath = self._root + line
if os.path.exists(libpath):
libdirs.append(libpath)
elif line.startswith("include "):
# Includes are absolute or relative to the file itself.
line = os.path.join(os.path.dirname(path), line[8:])
for p in glob.glob(self._root + line):
libdirs.extend(self._ReadLdSoConf(os.path.relpath(p, self._root)))
f.close()
return libdirs
def __init__(self, root, verbose=False):
"""Initializer.
Args:
root: The sysroot (e.g. "/")
verbose: Print helpful messages.
"""
self._root = root
self._libcache = set()
self._verbose = verbose
# Insert some default directories into our library cache.
libdirs = [
"%s/lib" % root,
"%s/usr/lib" % root,
"%s/opt/google/o3d/lib" % root,
"%s/usr/lib/opengl/xorg-x11/lib" % root,
"%s/usr/local/lib/icedtea6/jre/lib/i386/client" % root,
"%s/usr/local/lib/icedtea6/jre/lib/i386/headless" % root
]
# Read more directories from ld.so.conf.
libdirs.extend(self._ReadLdSoConf("/etc/ld.so.conf"))
self._ReadLibs(libdirs, self._libcache)
def _ReadLibs(self, paths, libcache):
for path in paths:
if os.path.exists(path):
for lib in os.listdir(path):
libcache.add(lib)
def _ReadDependencies(self, binary):
"""Run readelf -d on BINARY, returning (deps, rpaths)."""
deps = set()
rpaths = set()
# Read list of dynamic libraries, ignoring error messages that occur
# when we look at files that aren't actually libraries
f = os.popen("readelf -d '%s' 2>/dev/null" % binary)
for line in f:
# Grab dependencies
m = _SHARED_RE.search(line)
if m:
deps.add(m.group(1))
# Add RPATHs in our search path
m = _RPATH_RE.search(line)
if m:
for path in m.group(1).split(":"):
if path.startswith("$ORIGIN"):
rpaths.add(path.replace("$ORIGIN", os.path.dirname(binary)))
else:
rpaths.add(os.path.join(self._root, path[1:]))
f.close()
return (deps, rpaths)
def CheckDependencies(self, binary):
"""Check whether the libs for BINARY can be found in our sysroot."""
good = True
deps, rpaths = self._ReadDependencies(binary)
if self._verbose:
for lib in self._libcache & deps:
print "Found %s" % lib
for lib in deps - self._libcache:
if lib[0] != "/":
for path in rpaths:
if os.path.exists(os.path.join(path, lib)):
if self._verbose:
print "Found %s" % lib
break
else:
print >>sys.stderr, "Problem with %s: Can't find %s" % (binary, lib)
good = False
else:
full_path = os.path.join(self._root, lib[1:])
if os.path.exists(full_path):
if self._verbose: print "Found %s" % lib
else:
print >>sys.stderr, "Problem with %s: Can't find %s" % (binary, lib)
good = False
return good
def main():
if len(sys.argv) < 3:
print "Usage: %s [-v] sysroot binary [ binary ... ]" % sys.argv[0]
sys.exit(1)
verbose = False
if sys.argv[1] == "-v":
verbose = True
sys.argv = sys.argv[0:1] + sys.argv[2:]
checker = CheckDependencies(sys.argv[1], verbose)
errors = False
for binary in sys.argv[2:]:
if verbose: print "Checking %s" % binary
if not checker.CheckDependencies(binary):
errors = True
if errors:
sys.exit(1)
else:
sys.exit(0)
if __name__ == "__main__":
main()