1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
|
#!/usr/bin/env python3
#
# //===----------------------------------------------------------------------===//
# //
# // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
# // See https://llvm.org/LICENSE.txt for license information.
# // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
# //
# //===----------------------------------------------------------------------===//
#
import argparse
import os
import platform
import re
import sys
from libomputils import (
ScriptError,
error,
execute_command,
print_info_line,
print_error_line,
)
def get_deps_readelf(filename):
"""Get list of dependencies from readelf"""
deps = []
# Force readelf call to be in English
os.environ["LANG"] = "C"
r = execute_command(["readelf", "-d", filename])
if r.returncode != 0:
error("readelf -d {} failed".format(filename))
neededRegex = re.compile(r"\(NEEDED\)\s+Shared library: \[([a-zA-Z0-9_.-]+)\]")
for line in r.stdout.split(os.linesep):
match = neededRegex.search(line)
if match:
deps.append(match.group(1))
return deps
def get_deps_otool(filename):
"""Get list of dependencies from otool"""
deps = []
r = execute_command(["otool", "-L", filename])
if r.returncode != 0:
error("otool -L {} failed".format(filename))
libRegex = re.compile(r"([^ \t]+)\s+\(compatibility version ")
thisLibRegex = re.compile(r"@rpath/{}".format(os.path.basename(filename)))
for line in r.stdout.split(os.linesep):
match = thisLibRegex.search(line)
if match:
# Don't include the library itself as a needed dependency
continue
match = libRegex.search(line)
if match:
deps.append(match.group(1))
continue
return deps
def get_deps_link(filename):
"""Get list of dependecies from link (Windows OS)"""
depsSet = set([])
f = filename.lower()
args = ["link", "/DUMP"]
if f.endswith(".lib"):
args.append("/DIRECTIVES")
elif f.endswith(".dll") or f.endswith(".exe"):
args.append("/DEPENDENTS")
else:
error("unrecognized file extension: {}".format(filename))
args.append(filename)
r = execute_command(args)
if r.returncode != 0:
error("{} failed".format(args.command))
if f.endswith(".lib"):
regex = re.compile(r"\s*[-/]defaultlib:(.*)\s*$")
for line in r.stdout.split(os.linesep):
line = line.lower()
match = regex.search(line)
if match:
depsSet.add(match.group(1))
else:
started = False
markerStart = re.compile(r"Image has the following depend")
markerEnd = re.compile(r"Summary")
markerEnd2 = re.compile(r"Image has the following delay load depend")
for line in r.stdout.split(os.linesep):
if not started:
if markerStart.search(line):
started = True
continue
else: # Started parsing the libs
line = line.strip()
if not line:
continue
if markerEnd.search(line) or markerEnd2.search(line):
break
depsSet.add(line.lower())
return list(depsSet)
def main():
parser = argparse.ArgumentParser(description="Check library dependencies")
parser.add_argument(
"--bare",
action="store_true",
help="Produce plain, bare output: just a list"
" of libraries, a library per line",
)
parser.add_argument(
"--expected",
metavar="CSV_LIST",
help="CSV_LIST is a comma-separated list of expected"
' dependencies (or "none"). checks the specified'
" library has only expected dependencies.",
)
parser.add_argument("library", help="The library file to check")
commandArgs = parser.parse_args()
# Get dependencies
deps = []
system = platform.system()
if system == "Windows":
deps = get_deps_link(commandArgs.library)
elif system == "Darwin":
deps = get_deps_otool(commandArgs.library)
else:
deps = get_deps_readelf(commandArgs.library)
deps = sorted(deps)
# If bare output specified, then just print the dependencies one per line
if commandArgs.bare:
print(os.linesep.join(deps))
return
# Calculate unexpected dependencies if expected list specified
unexpected = []
if commandArgs.expected:
# none => any dependency is unexpected
if commandArgs.expected == "none":
unexpected = list(deps)
else:
expected = [d.strip() for d in commandArgs.expected.split(",")]
unexpected = [d for d in deps if d not in expected]
# Regular output
print_info_line("Dependencies:")
for dep in deps:
print_info_line(" {}".format(dep))
if unexpected:
print_error_line("Unexpected Dependencies:")
for dep in unexpected:
print_error_line(" {}".format(dep))
error("found unexpected dependencies")
if __name__ == "__main__":
try:
main()
except ScriptError as e:
print_error_line(str(e))
sys.exit(1)
# end of file
|