Android build system and create_reference_dumps.py support: - Creating dumps in vndk/ for the libraries that are LLNDK but not NDK. - Creating dumps in platform/ for opt-in libraries. Test: ./create_reference_dumps.py -libs libc -libs libcgrouprc Bug: 133176785 Bug: 138219814 Change-Id: I827d8bfae2c4b6aff7e6cb835d0bf05ce1483ddf
337 lines
12 KiB
Python
337 lines
12 KiB
Python
#!/usr/bin/env python3
|
|
|
|
import gzip
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
import tempfile
|
|
import collections
|
|
|
|
|
|
SCRIPT_DIR = os.path.abspath(os.path.dirname(__file__))
|
|
|
|
try:
|
|
AOSP_DIR = os.environ['ANDROID_BUILD_TOP']
|
|
except KeyError:
|
|
print('error: ANDROID_BUILD_TOP environment variable is not set.',
|
|
file=sys.stderr)
|
|
sys.exit(1)
|
|
|
|
BUILTIN_HEADERS_DIR = (
|
|
os.path.join(AOSP_DIR, 'bionic', 'libc', 'include'),
|
|
os.path.join(AOSP_DIR, 'external', 'libcxx', 'include'),
|
|
os.path.join(AOSP_DIR, 'prebuilts', 'clang-tools', 'linux-x86',
|
|
'clang-headers'),
|
|
)
|
|
|
|
EXPORTED_HEADERS_DIR = (
|
|
os.path.join(AOSP_DIR, 'development', 'vndk', 'tools', 'header-checker',
|
|
'tests'),
|
|
)
|
|
|
|
SO_EXT = '.so'
|
|
SOURCE_ABI_DUMP_EXT_END = '.lsdump'
|
|
SOURCE_ABI_DUMP_EXT = SO_EXT + SOURCE_ABI_DUMP_EXT_END
|
|
COMPRESSED_SOURCE_ABI_DUMP_EXT = SOURCE_ABI_DUMP_EXT + '.gz'
|
|
VENDOR_SUFFIX = '.vendor'
|
|
|
|
DEFAULT_CPPFLAGS = ['-x', 'c++', '-std=c++11']
|
|
DEFAULT_CFLAGS = ['-std=gnu99']
|
|
DEFAULT_HEADER_FLAGS = ["-dump-function-declarations"]
|
|
DEFAULT_FORMAT = 'ProtobufTextFormat'
|
|
|
|
|
|
class Target(object):
|
|
def __init__(self, is_2nd, product):
|
|
extra = '_2ND' if is_2nd else ''
|
|
build_vars_to_fetch = ['TARGET_ARCH',
|
|
'TARGET{}_ARCH'.format(extra),
|
|
'TARGET{}_ARCH_VARIANT'.format(extra),
|
|
'TARGET{}_CPU_VARIANT'.format(extra)]
|
|
build_vars = get_build_vars_for_product(build_vars_to_fetch, product)
|
|
self.primary_arch = build_vars[0]
|
|
assert self.primary_arch != ''
|
|
self.arch = build_vars[1]
|
|
self.arch_variant = build_vars[2]
|
|
self.cpu_variant = build_vars[3]
|
|
|
|
def get_arch_cpu_str(self):
|
|
"""Return a string that represents the architecture, the architecture
|
|
variant, and the CPU variant.
|
|
|
|
If TARGET_ARCH == TARGET_ARCH_VARIANT, soong makes targetArchVariant
|
|
empty. This is the case for aosp_x86_64 and aosp_x86_ab.
|
|
"""
|
|
if not self.arch_variant or self.arch_variant == self.arch:
|
|
arch_variant = ''
|
|
else:
|
|
arch_variant = '_' + self.arch_variant
|
|
|
|
if not self.cpu_variant or self.cpu_variant == 'generic':
|
|
cpu_variant = ''
|
|
else:
|
|
cpu_variant = '_' + self.cpu_variant
|
|
|
|
return self.arch + arch_variant + cpu_variant
|
|
|
|
def get_module_variant_dir_name(self, variant_suffix):
|
|
"""Create module variant directory name from the architecture, the
|
|
architecture variant, the CPU variant, and a variant suffix
|
|
(e.g. `_core_shared`, `_vendor_shared`, etc)."""
|
|
return 'android_' + self.get_arch_cpu_str() + variant_suffix
|
|
|
|
|
|
def copy_reference_dump(lib_path, reference_dump_dir, compress):
|
|
reference_dump_path = os.path.join(
|
|
reference_dump_dir, os.path.basename(lib_path))
|
|
if compress:
|
|
reference_dump_path += '.gz'
|
|
os.makedirs(os.path.dirname(reference_dump_path), exist_ok=True)
|
|
output_content = read_output_content(lib_path, AOSP_DIR)
|
|
if compress:
|
|
with gzip.open(reference_dump_path, 'wb') as f:
|
|
f.write(bytes(output_content, 'utf-8'))
|
|
else:
|
|
with open(reference_dump_path, 'wb') as f:
|
|
f.write(bytes(output_content, 'utf-8'))
|
|
print('Created abi dump at', reference_dump_path)
|
|
return reference_dump_path
|
|
|
|
|
|
def read_output_content(output_path, replace_str):
|
|
with open(output_path, 'r') as f:
|
|
return f.read().replace(replace_str, '')
|
|
|
|
|
|
def run_header_abi_dumper(input_path, cflags=tuple(),
|
|
export_include_dirs=EXPORTED_HEADERS_DIR,
|
|
flags=tuple()):
|
|
"""Run header-abi-dumper to dump ABI from `input_path` and return the
|
|
output."""
|
|
with tempfile.TemporaryDirectory() as tmp:
|
|
output_path = os.path.join(tmp, os.path.basename(input_path)) + '.dump'
|
|
run_header_abi_dumper_on_file(input_path, output_path,
|
|
export_include_dirs, cflags, flags)
|
|
return read_output_content(output_path, AOSP_DIR)
|
|
|
|
|
|
def run_header_abi_dumper_on_file(input_path, output_path,
|
|
export_include_dirs=tuple(), cflags=tuple(),
|
|
flags=tuple()):
|
|
"""Run header-abi-dumper to dump ABI from `input_path` and the output is
|
|
written to `output_path`."""
|
|
input_ext = os.path.splitext(input_path)[1]
|
|
cmd = ['header-abi-dumper', '-o', output_path, input_path]
|
|
for dir in export_include_dirs:
|
|
cmd += ['-I', dir]
|
|
cmd += flags
|
|
if '-output-format' not in flags:
|
|
cmd += ['-output-format', DEFAULT_FORMAT]
|
|
if input_ext == ".h":
|
|
cmd += DEFAULT_HEADER_FLAGS
|
|
cmd += ['--']
|
|
cmd += cflags
|
|
if input_ext in ('.cpp', '.cc', '.h'):
|
|
cmd += DEFAULT_CPPFLAGS
|
|
else:
|
|
cmd += DEFAULT_CFLAGS
|
|
|
|
for dir in BUILTIN_HEADERS_DIR:
|
|
cmd += ['-isystem', dir]
|
|
# The export include dirs imply local include dirs.
|
|
for dir in export_include_dirs:
|
|
cmd += ['-I', dir]
|
|
subprocess.check_call(cmd)
|
|
|
|
|
|
def run_header_abi_linker(output_path, inputs, version_script, api, arch,
|
|
flags=tuple()):
|
|
"""Link inputs, taking version_script into account"""
|
|
cmd = ['header-abi-linker', '-o', output_path, '-v', version_script,
|
|
'-api', api, '-arch', arch]
|
|
cmd += flags
|
|
if '-input-format' not in flags:
|
|
cmd += ['-input-format', DEFAULT_FORMAT]
|
|
if '-output-format' not in flags:
|
|
cmd += ['-output-format', DEFAULT_FORMAT]
|
|
cmd += inputs
|
|
subprocess.check_call(cmd)
|
|
return read_output_content(output_path, AOSP_DIR)
|
|
|
|
|
|
def make_targets(product, variant, targets):
|
|
make_cmd = ['build/soong/soong_ui.bash', '--make-mode', '-j',
|
|
'TARGET_PRODUCT=' + product, 'TARGET_BUILD_VARIANT=' + variant]
|
|
make_cmd += targets
|
|
subprocess.check_call(make_cmd, cwd=AOSP_DIR)
|
|
|
|
|
|
def make_tree(product, variant):
|
|
"""Build all lsdump files."""
|
|
return make_targets(product, variant, ['findlsdumps'])
|
|
|
|
|
|
def make_libraries(product, variant, targets, libs):
|
|
"""Build lsdump files for specific libs."""
|
|
lsdump_paths = read_lsdump_paths(product, variant, targets, build=True)
|
|
make_target_paths = []
|
|
for name in libs:
|
|
make_target_paths.extend(path for tag, path in
|
|
lsdump_paths[name].values())
|
|
make_targets(product, variant, make_target_paths)
|
|
|
|
|
|
def get_lsdump_paths_file_path(product, variant):
|
|
"""Get the path to lsdump_paths.txt."""
|
|
product_out = get_build_vars_for_product(
|
|
['PRODUCT_OUT'], product, variant)[0]
|
|
return os.path.join(product_out, 'lsdump_paths.txt')
|
|
|
|
|
|
def _is_sanitizer_variation(variation):
|
|
"""Check whether the variation is introduced by a sanitizer."""
|
|
return variation in {'asan', 'hwasan', 'tsan', 'intOverflow', 'cfi', 'scs'}
|
|
|
|
|
|
def _tag_to_variant_suffix(tag):
|
|
"""Map a tag to a variant suffix."""
|
|
if tag in ('LLNDK', 'NDK', 'PLATFORM'):
|
|
return '_core_shared'
|
|
if tag.startswith('VNDK'):
|
|
return '_vendor_shared'
|
|
raise ValueError(tag + ' is not a known tag.')
|
|
|
|
|
|
def _read_lsdump_paths(lsdump_paths_file_path, targets):
|
|
"""Read lsdump paths from lsdump_paths.txt for each libname and variant.
|
|
|
|
This function returns a dictionary, {lib_name: {arch_cpu: (tag, path)}}.
|
|
For example,
|
|
{
|
|
"libc": {
|
|
"x86_x86_64": (
|
|
"NDK",
|
|
"path/to/libc.so.lsdump"
|
|
)
|
|
}
|
|
}
|
|
"""
|
|
lsdump_paths = collections.defaultdict(dict)
|
|
suffixes = collections.defaultdict(dict)
|
|
|
|
with open(lsdump_paths_file_path, 'r') as lsdump_paths_file:
|
|
for line in lsdump_paths_file:
|
|
tag, path = (x.strip() for x in line.split(':', 1))
|
|
variant_suffix = _tag_to_variant_suffix(tag)
|
|
if not path:
|
|
continue
|
|
dirname, filename = os.path.split(path)
|
|
if not filename.endswith(SOURCE_ABI_DUMP_EXT):
|
|
continue
|
|
libname = filename[:-len(SOURCE_ABI_DUMP_EXT)]
|
|
if not libname:
|
|
continue
|
|
variant = os.path.basename(dirname)
|
|
if not variant:
|
|
continue
|
|
for target in targets:
|
|
prefix = target.get_module_variant_dir_name(variant_suffix)
|
|
if not variant.startswith(prefix):
|
|
continue
|
|
new_suffix = variant[len(prefix):]
|
|
# Skip if the suffix contains APEX variations.
|
|
new_variations = [x for x in new_suffix.split('_') if x]
|
|
if new_variations and not all(_is_sanitizer_variation(x)
|
|
for x in new_variations):
|
|
continue
|
|
arch_cpu = target.get_arch_cpu_str()
|
|
old_suffix = suffixes[libname].get(arch_cpu)
|
|
if not old_suffix or new_suffix > old_suffix:
|
|
lsdump_paths[libname][arch_cpu] = (tag, path)
|
|
suffixes[libname][arch_cpu] = new_suffix
|
|
return lsdump_paths
|
|
|
|
|
|
def read_lsdump_paths(product, variant, targets, build=True):
|
|
"""Build lsdump_paths.txt and read the paths."""
|
|
lsdump_paths_file_path = get_lsdump_paths_file_path(product, variant)
|
|
if build:
|
|
make_targets(product, variant, [lsdump_paths_file_path])
|
|
lsdump_paths_file_abspath = os.path.join(AOSP_DIR, lsdump_paths_file_path)
|
|
return _read_lsdump_paths(lsdump_paths_file_abspath, targets)
|
|
|
|
|
|
def find_lib_lsdumps(lsdump_paths, libs, target):
|
|
"""Find the lsdump corresponding to libs for the given target.
|
|
|
|
This function returns a list of (tag, absolute_path).
|
|
For example,
|
|
[
|
|
(
|
|
"NDK",
|
|
"/path/to/libc.so.lsdump"
|
|
)
|
|
]
|
|
"""
|
|
arch_cpu = target.get_arch_cpu_str()
|
|
result = []
|
|
if libs:
|
|
for lib_name in libs:
|
|
result.append(lsdump_paths[lib_name][arch_cpu])
|
|
else:
|
|
result.extend(paths[arch_cpu] for paths in lsdump_paths.values())
|
|
return [(tag, os.path.join(AOSP_DIR, path)) for tag, path in result]
|
|
|
|
|
|
def run_abi_diff(old_test_dump_path, new_test_dump_path, arch, lib_name,
|
|
flags=tuple()):
|
|
abi_diff_cmd = ['header-abi-diff', '-new', new_test_dump_path, '-old',
|
|
old_test_dump_path, '-arch', arch, '-lib', lib_name]
|
|
with tempfile.TemporaryDirectory() as tmp:
|
|
output_name = os.path.join(tmp, lib_name) + '.abidiff'
|
|
abi_diff_cmd += ['-o', output_name]
|
|
abi_diff_cmd += flags
|
|
if '-input-format-old' not in flags:
|
|
abi_diff_cmd += ['-input-format-old', DEFAULT_FORMAT]
|
|
if '-input-format-new' not in flags:
|
|
abi_diff_cmd += ['-input-format-new', DEFAULT_FORMAT]
|
|
try:
|
|
subprocess.check_call(abi_diff_cmd)
|
|
except subprocess.CalledProcessError as err:
|
|
return err.returncode
|
|
|
|
return 0
|
|
|
|
|
|
def get_build_vars_for_product(names, product=None, variant=None):
|
|
""" Get build system variable for the launched target."""
|
|
|
|
if product is None and 'ANDROID_PRODUCT_OUT' not in os.environ:
|
|
return None
|
|
|
|
env = os.environ.copy()
|
|
if product:
|
|
env['TARGET_PRODUCT'] = product
|
|
if variant:
|
|
env['TARGET_BUILD_VARIANT'] = variant
|
|
cmd = [
|
|
os.path.join('build', 'soong', 'soong_ui.bash'),
|
|
'--dumpvars-mode', '-vars', ' '.join(names),
|
|
]
|
|
|
|
proc = subprocess.Popen(cmd, stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE, cwd=AOSP_DIR, env=env)
|
|
out, err = proc.communicate()
|
|
|
|
if proc.returncode != 0:
|
|
print("error: %s" % err.decode('utf-8'), file=sys.stderr)
|
|
return None
|
|
|
|
build_vars = out.decode('utf-8').strip().splitlines()
|
|
|
|
build_vars_list = []
|
|
for build_var in build_vars:
|
|
value = build_var.partition('=')[2]
|
|
build_vars_list.append(value.replace('\'', ''))
|
|
return build_vars_list
|