| #!/usr/bin/env python3 |
| # Copyright (c) 2020, the R8 project authors. Please see the AUTHORS file |
| # for details. All rights reserved. Use of this source code is governed by a |
| # BSD-style license that can be found in the LICENSE file. |
| |
| import argparse |
| import os |
| import shutil |
| import subprocess |
| import sys |
| import zipfile |
| |
| import archive |
| import jdk |
| import retrace |
| import utils |
| |
| |
| def make_parser(): |
| parser = argparse.ArgumentParser(description = 'Compile a dump artifact.') |
| parser.add_argument( |
| '--summary', |
| help='List a summary of the contents of the dumps.', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '-d', |
| '--dump', |
| help='Dump file or directory to compile', |
| default=None) |
| parser.add_argument( |
| '--temp', |
| help='Temp directory to extract the dump to, allows you to rerun the command' |
| ' more easily in the terminal with changes', |
| default=None) |
| parser.add_argument( |
| '-c', |
| '--compiler', |
| help='Compiler to use', |
| default=None) |
| parser.add_argument( |
| '--minify', |
| help='Force enable/disable minification' |
| ' (defaults to app proguard config)', |
| choices=['default', 'force-enable', 'force-disable'], |
| default='default') |
| parser.add_argument( |
| '--optimize', |
| help='Force enable/disable optimizations' |
| ' (defaults to app proguard config)', |
| choices=['default', 'force-enable', 'force-disable'], |
| default='default') |
| parser.add_argument( |
| '--shrink', |
| help='Force enable/disable shrinking' |
| ' (defaults to app proguard config)', |
| choices=['default', 'force-enable', 'force-disable'], |
| default='default') |
| parser.add_argument( |
| '-v', |
| '--version', |
| help='Compiler version to use (default read from dump version file).' |
| 'Valid arguments are:' |
| ' "main" to run from your own tree,' |
| ' "source" to run from build classes directly,' |
| ' "X.Y.Z" to run a specific version, or' |
| ' <hash> to run that hash from main.', |
| default=None) |
| parser.add_argument( |
| '--r8-jar', |
| help='Path to an R8 jar.', |
| default=None) |
| parser.add_argument( |
| '--r8-flags', '--r8_flags', |
| help='Additional option(s) for the compiler.') |
| parser.add_argument( |
| '--pg-conf', '--pg_conf', |
| help='Keep rule file(s).', |
| action='append') |
| parser.add_argument( |
| '--override', |
| help='Do not override any extracted dump in temp-dir', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--nolib', |
| help='Use the non-lib distribution (default uses the lib distribution)', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--print-times', |
| help='Print timing information from r8', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--ea', |
| help='Enable Java assertions when running the compiler (default disabled)', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--classfile', |
| help='Run with classfile output', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--debug-agent', |
| help='Enable Java debug agent and suspend compilation (default disabled)', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--xmx', |
| help='Set JVM max heap size (-Xmx)', |
| default=None) |
| parser.add_argument( |
| '--threads', |
| help='Set the number of threads to use', |
| default=None) |
| parser.add_argument( |
| '--min-api', |
| help='Set min-api (default read from dump properties file)', |
| default=None) |
| parser.add_argument( |
| '--desugared-lib', |
| help='Set desugared-library (default set from dump)', |
| default=None) |
| parser.add_argument( |
| '--disable-desugared-lib', |
| help='Disable desugared-libary if it will be set from dump', |
| default=False, |
| action='store_true' |
| ) |
| parser.add_argument( |
| '--loop', |
| help='Run the compilation in a loop', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--enable-missing-library-api-modeling', |
| help='Run with api modeling', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--android-platform-build', |
| help='Run as a platform build', |
| default=False, |
| action='store_true') |
| parser.add_argument( |
| '--compilation-mode', '--compilation_mode', |
| help='Run compilation in specified mode', |
| choices=['debug', 'release'], |
| default=None) |
| return parser |
| |
| def error(msg): |
| print(msg) |
| sys.exit(1) |
| |
| class Dump(object): |
| |
| def __init__(self, directory): |
| self.directory = directory |
| |
| def if_exists(self, name): |
| f = os.path.join(self.directory, name) |
| if os.path.exists(f): |
| return f |
| return None |
| |
| def program_jar(self): |
| return self.if_exists('program.jar') |
| |
| def feature_jars(self): |
| feature_jars = [] |
| i = 1 |
| while True: |
| feature_jar = self.if_exists('feature-%s.jar' % i) |
| if feature_jar: |
| feature_jars.append(feature_jar) |
| i = i + 1 |
| else: |
| return feature_jars |
| |
| def library_jar(self): |
| return self.if_exists('library.jar') |
| |
| def classpath_jar(self): |
| return self.if_exists('classpath.jar') |
| |
| def desugared_library_json(self): |
| return self.if_exists('desugared-library.json') |
| |
| def proguard_input_map(self): |
| if self.if_exists('proguard_input.config'): |
| print("Unimplemented: proguard_input configuration.") |
| |
| def main_dex_list_resource(self): |
| return self.if_exists('main-dex-list.txt') |
| |
| def main_dex_rules_resource(self): |
| return self.if_exists('main-dex-rules.txt') |
| |
| def art_profile_resources(self): |
| art_profile_resources = [] |
| while True: |
| current_art_profile_index = len(art_profile_resources) + 1 |
| art_profile_resource = self.if_exists( |
| 'art-profile-%s.txt' % current_art_profile_index) |
| if art_profile_resource is None: |
| return art_profile_resources |
| art_profile_resources.append(art_profile_resource) |
| |
| def startup_profile_resources(self): |
| startup_profile_resources = [] |
| while True: |
| current_startup_profile_index = len(startup_profile_resources) + 1 |
| startup_profile_resource = self.if_exists( |
| 'startup-profile-%s.txt' % current_startup_profile_index) |
| if startup_profile_resource is None: |
| return startup_profile_resources |
| startup_profile_resources.append(startup_profile_resource) |
| |
| def build_properties_file(self): |
| return self.if_exists('build.properties') |
| |
| def config_file(self): |
| return self.if_exists('proguard.config') |
| |
| def version_file(self): |
| return self.if_exists('r8-version') |
| |
| def version(self): |
| f = self.version_file() |
| if f: |
| return open(f).read().split(' ')[0] |
| return None |
| |
| def read_dump_from_args(args, temp): |
| if args.dump is None: |
| error("A dump file or directory must be specified") |
| return read_dump(args.dump, temp, args.override) |
| |
| def read_dump(dump, temp, override=False): |
| if os.path.isdir(dump): |
| return Dump(dump) |
| dump_file = zipfile.ZipFile(os.path.abspath(dump), 'r') |
| with utils.ChangedWorkingDirectory(temp, quiet=True): |
| if override or not os.path.isfile('r8-version'): |
| dump_file.extractall() |
| if not os.path.isfile('r8-version'): |
| error("Did not extract into %s. Either the zip file is invalid or the " |
| "dump is missing files" % temp) |
| return Dump(temp) |
| |
| def determine_build_properties(args, dump): |
| build_properties = {} |
| build_properties_file = dump.build_properties_file() |
| if build_properties_file: |
| with open(build_properties_file) as f: |
| build_properties_contents = f.readlines() |
| for line in build_properties_contents: |
| stripped = line.strip() |
| if stripped: |
| pair = stripped.split('=') |
| build_properties[pair[0]] = pair[1] |
| if 'mode' not in build_properties: |
| build_properties['mode'] = 'release' |
| return build_properties |
| |
| def determine_version(args, dump): |
| if args.version is None: |
| return dump.version() |
| return args.version |
| |
| def determine_compiler(args, build_properties): |
| compilers = ['d8', 'r8', 'r8full', 'l8', 'l8d8', 'tracereferences'] |
| compiler = args.compiler |
| if not compiler and 'tool' in build_properties: |
| compiler = build_properties.get('tool').lower() |
| if compiler == 'r8': |
| if not 'force-proguard-compatibility' in build_properties: |
| error("Unable to determine R8 compiler variant from build.properties." |
| " No value for 'force-proguard-compatibility'.") |
| if build_properties.get('force-proguard-compatibility').lower() == 'false': |
| compiler = compiler + 'full' |
| if compiler == 'TraceReferences': |
| compiler = build_properties.get('tool').lower() |
| if compiler not in compilers: |
| error("Unable to determine a compiler to use. Specified %s," |
| " Valid options: %s" % (args.compiler, ', '.join(compilers))) |
| return compiler |
| |
| def determine_trace_references_commands(build_properties, output): |
| trace_ref_consumer = build_properties.get('trace_references_consumer') |
| if trace_ref_consumer == 'com.android.tools.r8.tracereferences.TraceReferencesCheckConsumer': |
| return ["--check"] |
| else: |
| assert trace_ref_consumer == 'com.android.tools.r8.tracereferences.TraceReferencesKeepRules' |
| args = ['--allowobfuscation'] if build_properties.get('minification') == 'true' else [] |
| args.extend(['--keep-rules', '--output', output]) |
| return args |
| |
| def is_l8_compiler(compiler): |
| return compiler.startswith('l8') |
| |
| def is_r8_compiler(compiler): |
| return compiler.startswith('r8') |
| |
| def determine_config_files(args, dump, temp): |
| if args.pg_conf: |
| config_files = [] |
| for config_file in args.pg_conf: |
| dst = os.path.join(temp, 'proguard-%s.config' % len(config_files)) |
| shutil.copyfile(config_file, dst) |
| config_files.append(dst) |
| return config_files |
| dump_config_file = dump.config_file() |
| if dump_config_file: |
| return [dump_config_file] |
| return [] |
| |
| def determine_output(args, temp): |
| return os.path.join(temp, 'out.jar') |
| |
| def determine_min_api(args, build_properties): |
| if args.min_api: |
| return args.min_api |
| if 'min-api' in build_properties: |
| return build_properties.get('min-api') |
| return None |
| |
| def determine_residual_art_profile_output(art_profile, temp): |
| return os.path.join(temp, os.path.basename(art_profile)[:-4] + ".out.txt") |
| |
| def determine_desugared_lib_pg_conf_output(temp): |
| return os.path.join(temp, 'desugared-library-keep-rules.config') |
| |
| def determine_feature_output(feature_jar, temp): |
| return os.path.join(temp, os.path.basename(feature_jar)[:-4] + ".out.jar") |
| |
| def determine_program_jar(args, dump): |
| if hasattr(args, 'program_jar') and args.program_jar: |
| return args.program_jar |
| return dump.program_jar() |
| |
| def determine_class_file(args, build_properties): |
| return args.classfile \ |
| or build_properties.get('backend', 'dex').lower() == 'cf' |
| |
| def determine_android_platform_build(args, build_properties): |
| if args.android_platform_build: |
| return True |
| return build_properties.get('android-platform-build') == 'true' |
| |
| def determine_enable_missing_library_api_modeling(args, build_properties): |
| if args.enable_missing_library_api_modeling: |
| return True |
| return build_properties.get('enable-missing-library-api-modeling') == 'true' |
| |
| def determine_compilation_mode(args, build_properties): |
| if args.compilation_mode: |
| return args.compilation_mode |
| return build_properties.get('mode') |
| |
| def determine_properties(build_properties): |
| args = [] |
| for key, value in build_properties.items(): |
| # When writing dumps all system properties starting with com.android.tools.r8 |
| # are written to the build.properties file in the format |
| # system-property-com.android.tools.r8.XXX=<value> |
| if key.startswith('system-property-'): |
| name = key[len('system-property-'):] |
| if name.endswith('dumpinputtofile') or name.endswith('dumpinputtodirectory'): |
| continue |
| if len(value) == 0: |
| args.append('-D' + name) |
| else: |
| args.append('-D' + name + '=' + value) |
| return args |
| |
| def download_distribution(version, nolib, temp): |
| if version == 'main': |
| return utils.R8_JAR if nolib else utils.R8LIB_JAR |
| if version == 'source': |
| return '%s:%s' % (utils.BUILD_JAVA_MAIN_DIR, utils.ALL_DEPS_JAR) |
| name = 'r8.jar' if nolib else 'r8lib.jar' |
| source = archive.GetUploadDestination(version, name, is_hash(version)) |
| dest = os.path.join(temp, 'r8.jar') |
| utils.download_file_from_cloud_storage(source, dest) |
| return dest |
| |
| def clean_configs(files, args): |
| for file in files: |
| clean_config(file, args) |
| |
| def clean_config(file, args): |
| with open(file) as f: |
| lines = f.readlines() |
| minify = args.minify |
| optimize = args.optimize |
| shrink = args.shrink |
| with open(file, 'w') as f: |
| if minify == 'force-disable': |
| print('Adding config line: -dontobfuscate') |
| f.write('-dontobfuscate\n') |
| if optimize == 'force-disable': |
| print('Adding config line: -dontoptimize') |
| f.write('-dontoptimize\n') |
| if shrink == 'force-disable': |
| print('Adding config line: -dontshrink') |
| f.write('-dontshrink\n') |
| for line in lines: |
| if clean_config_line(line, minify, optimize, shrink): |
| print('Removing from config line: \n%s' % line) |
| else: |
| f.write(line) |
| |
| def clean_config_line(line, minify, optimize, shrink): |
| if line.lstrip().startswith('#'): |
| return False |
| if ('-injars' in line or '-libraryjars' in line or |
| '-print' in line or '-applymapping' in line): |
| return True |
| if minify == 'force-enable' and '-dontobfuscate' in line: |
| return True |
| if optimize == 'force-enable' and '-dontoptimize' in line: |
| return True |
| if shrink == 'force-enable' and '-dontshrink' in line: |
| return True |
| return False |
| |
| def prepare_r8_wrapper(dist, temp, jdkhome): |
| compile_wrapper_with_javac( |
| dist, |
| temp, |
| jdkhome, |
| os.path.join( |
| utils.REPO_ROOT, |
| 'src/main/java/com/android/tools/r8/utils/CompileDumpCompatR8.java')) |
| |
| def prepare_d8_wrapper(dist, temp, jdkhome): |
| compile_wrapper_with_javac( |
| dist, |
| temp, |
| jdkhome, |
| os.path.join( |
| utils.REPO_ROOT, |
| 'src/main/java/com/android/tools/r8/utils/CompileDumpD8.java')) |
| |
| def compile_wrapper_with_javac(dist, temp, jdkhome, path): |
| base_path = os.path.join( |
| utils.REPO_ROOT, |
| 'src/main/java/com/android/tools/r8/utils/CompileDumpBase.java') |
| cmd = [ |
| jdk.GetJavacExecutable(jdkhome), |
| path, |
| base_path, |
| '-d', temp, |
| '-cp', dist, |
| ] |
| utils.PrintCmd(cmd) |
| subprocess.check_output(cmd) |
| |
| def is_hash(version): |
| return len(version) == 40 |
| |
| def run1(out, args, otherargs, jdkhome=None): |
| jvmargs = [] |
| compilerargs = [] |
| for arg in otherargs: |
| if arg.startswith('-D'): |
| jvmargs.append(arg) |
| else: |
| compilerargs.append(arg) |
| with utils.TempDir() as temp: |
| if out: |
| temp = out |
| if not os.path.exists(temp): |
| os.makedirs(temp) |
| dump = read_dump_from_args(args, temp) |
| if not dump.program_jar(): |
| error("Cannot compile dump with no program classes") |
| if not dump.library_jar(): |
| print("WARNING: Unexpected lack of library classes in dump") |
| build_properties = determine_build_properties(args, dump) |
| version = determine_version(args, dump) |
| compiler = determine_compiler(args, build_properties) |
| config_files = determine_config_files(args, dump, temp) |
| out = determine_output(args, temp) |
| min_api = determine_min_api(args, build_properties) |
| classfile = determine_class_file(args, build_properties) |
| android_platform_build = determine_android_platform_build(args, build_properties) |
| enable_missing_library_api_modeling = determine_enable_missing_library_api_modeling(args, build_properties) |
| mode = determine_compilation_mode(args, build_properties) |
| jar = args.r8_jar if args.r8_jar else download_distribution(version, args.nolib, temp) |
| if ':' not in jar and not os.path.exists(jar): |
| error("Distribution does not exist: " + jar) |
| cmd = [jdk.GetJavaExecutable(jdkhome)] |
| cmd.extend(jvmargs) |
| if args.debug_agent: |
| if not args.nolib: |
| print("WARNING: Running debugging agent on r8lib is questionable...") |
| cmd.append( |
| '-agentlib:jdwp=transport=dt_socket,server=y,suspend=y,address=*:5005') |
| if args.xmx: |
| cmd.append('-Xmx' + args.xmx) |
| if args.ea: |
| cmd.append('-ea') |
| cmd.append('-Dcom.android.tools.r8.enableTestAssertions=1') |
| if args.print_times: |
| cmd.append('-Dcom.android.tools.r8.printtimes=1') |
| if args.r8_flags: |
| cmd.extend(args.r8_flags.split(' ')) |
| if hasattr(args, 'properties'): |
| cmd.extend(args.properties) |
| cmd.extend(determine_properties(build_properties)) |
| cmd.extend(['-cp', '%s:%s' % (temp, jar)]) |
| if compiler == 'd8': |
| prepare_d8_wrapper(jar, temp, jdkhome) |
| cmd.append('com.android.tools.r8.utils.CompileDumpD8') |
| if is_l8_compiler(compiler): |
| cmd.append('com.android.tools.r8.L8') |
| if compiler == 'tracereferences': |
| cmd.append('com.android.tools.r8.tracereferences.TraceReferences') |
| cmd.extend(determine_trace_references_commands(build_properties, out)) |
| if compiler.startswith('r8'): |
| prepare_r8_wrapper(jar, temp, jdkhome) |
| cmd.append('com.android.tools.r8.utils.CompileDumpCompatR8') |
| if compiler == 'r8': |
| cmd.append('--compat') |
| if compiler != 'tracereferences': |
| assert mode == 'debug' or mode == 'release' |
| cmd.append('--' + mode) |
| # For recompilation of dumps run_on_app_dumps pass in a program jar. |
| program_jar = determine_program_jar(args, dump) |
| if compiler != 'tracereferences': |
| cmd.append(program_jar) |
| cmd.extend(['--output', out]) |
| else: |
| cmd.extend(['--source', program_jar]) |
| for feature_jar in dump.feature_jars(): |
| cmd.extend(['--feature-jar', feature_jar, |
| determine_feature_output(feature_jar, temp)]) |
| if dump.library_jar(): |
| cmd.extend(['--lib', dump.library_jar()]) |
| if dump.classpath_jar() and not is_l8_compiler(compiler): |
| cmd.extend( |
| ['--target' if compiler == 'tracereferences' else '--classpath', |
| dump.classpath_jar()]) |
| if dump.desugared_library_json() and not args.disable_desugared_lib: |
| cmd.extend(['--desugared-lib', dump.desugared_library_json()]) |
| if not is_l8_compiler(compiler): |
| cmd.extend([ |
| '--desugared-lib-pg-conf-output', |
| determine_desugared_lib_pg_conf_output(temp)]) |
| if (is_r8_compiler(compiler) or compiler == 'l8') and config_files: |
| if hasattr(args, 'config_files_consumer') and args.config_files_consumer: |
| args.config_files_consumer(config_files) |
| else: |
| # If we get a dump from the wild we can't use -injars, -libraryjars or |
| # -print{mapping,usage} |
| clean_configs(config_files, args) |
| for config_file in config_files: |
| cmd.extend(['--pg-conf', config_file]) |
| cmd.extend(['--pg-map-output', '%s.map' % out]) |
| if dump.main_dex_list_resource(): |
| cmd.extend(['--main-dex-list', dump.main_dex_list_resource()]) |
| if dump.main_dex_rules_resource(): |
| cmd.extend(['--main-dex-rules', dump.main_dex_rules_resource()]) |
| for art_profile_resource in dump.art_profile_resources(): |
| residual_art_profile_output = \ |
| determine_residual_art_profile_output(art_profile_resource, temp) |
| cmd.extend([ |
| '--art-profile', art_profile_resource, residual_art_profile_output]) |
| for startup_profile_resource in dump.startup_profile_resources(): |
| cmd.extend(['--startup-profile', startup_profile_resource]) |
| if min_api: |
| cmd.extend(['--min-api', min_api]) |
| if classfile: |
| cmd.extend(['--classfile']) |
| if android_platform_build: |
| cmd.extend(['--android-platform-build']) |
| if enable_missing_library_api_modeling: |
| cmd.extend(['--enable-missing-library-api-modeling']) |
| if args.threads: |
| cmd.extend(['--threads', args.threads]) |
| cmd.extend(compilerargs) |
| utils.PrintCmd(cmd) |
| try: |
| print(subprocess.check_output(cmd, stderr=subprocess.STDOUT).decode('utf-8')) |
| return 0 |
| except subprocess.CalledProcessError as e: |
| if args.nolib \ |
| or version == 'source' \ |
| or not try_retrace_output(e, version, temp): |
| print(e.output.decode('UTF-8')) |
| return 1 |
| |
| def try_retrace_output(e, version, temp): |
| try: |
| stacktrace = os.path.join(temp, 'stacktrace') |
| open(stacktrace, 'w+').write(e.output.decode('UTF-8')) |
| print("=" * 80) |
| print(" RETRACED OUTPUT") |
| print("=" * 80) |
| retrace.run(get_map_file(version, temp), stacktrace, no_r8lib=False) |
| return True |
| except Exception as e2: |
| print("Failed to retrace for version: %s" % version) |
| print(e2) |
| return False |
| |
| def get_map_file(version, temp): |
| if version == 'main': |
| return utils.R8LIB_MAP |
| download_path = archive.GetUploadDestination( |
| version, |
| 'r8lib.jar.map', |
| is_hash(version)) |
| if utils.file_exists_on_cloud_storage(download_path): |
| map_path = os.path.join(temp, 'mapping.map') |
| utils.download_file_from_cloud_storage(download_path, map_path) |
| return map_path |
| else: |
| print('Could not find map file from argument: %s.' % version) |
| return None |
| |
| def summarize_dump_files(dumpfiles): |
| if len(dumpfiles) == 0: |
| error('Summary command expects a list of dumps to summarize') |
| for f in dumpfiles: |
| print(f + ':') |
| try: |
| with utils.TempDir() as temp: |
| dump = read_dump(f, temp) |
| summarize_dump(dump) |
| except IOError as e: |
| print("Error: " + str(e)) |
| except zipfile.BadZipfile as e: |
| print("Error: " + str(e)) |
| |
| def summarize_dump(dump): |
| version = dump.version() |
| if not version: |
| print('No dump version info') |
| return |
| print('version=' + version) |
| props = dump.build_properties_file() |
| if props: |
| with open(props) as props_file: |
| print(props_file.read()) |
| if dump.library_jar(): |
| print('library.jar present') |
| if dump.classpath_jar(): |
| print('classpath.jar present') |
| prog = dump.program_jar() |
| if prog: |
| print('program.jar content:') |
| summarize_jar(prog) |
| |
| def summarize_jar(jar): |
| with zipfile.ZipFile(jar) as zip: |
| pkgs = {} |
| for info in zip.infolist(): |
| if info.filename.endswith('.class'): |
| pkg, clazz = os.path.split(info.filename) |
| count = pkgs.get(pkg, 0) |
| pkgs[pkg] = count + 1 |
| sorted = list(pkgs.keys()) |
| sorted.sort() |
| for p in sorted: |
| print(' ' + p + ': ' + str(pkgs[p])) |
| |
| def run(args, otherargs): |
| if args.summary: |
| summarize_dump_files(otherargs) |
| elif args.loop: |
| count = 1 |
| while True: |
| print('Iteration {:03d}'.format(count)) |
| out = args.temp |
| if out: |
| out = os.path.join(out, '{:03d}'.format(count)) |
| run1(out, args, otherargs) |
| count += 1 |
| else: |
| run1(args.temp, args, otherargs) |
| |
| if __name__ == '__main__': |
| (args, otherargs) = make_parser().parse_known_args(sys.argv[1:]) |
| sys.exit(run(args, otherargs)) |