| #!/usr/bin/env python3 | 
 | # Copyright (c) 2018, 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 optparse | 
 | import os | 
 | import subprocess | 
 | import sys | 
 |  | 
 | import utils | 
 |  | 
 | LINUX_DIR = os.path.join(utils.TOOLS_DIR, 'linux') | 
 |  | 
 | LATEST = '12.0.0' | 
 |  | 
 | VERSIONS = [ | 
 |     '12.0.0', | 
 |     # TODO(b/258170524): Fix the broken dex2oat versions. | 
 |     # 'default', | 
 |     # '9.0.0', | 
 |     # '8.1.0', | 
 |     # '7.0.0', | 
 |     '6.0.1', | 
 |     # '5.1.1', | 
 | ] | 
 |  | 
 | DIRS = { | 
 |     '12.0.0': 'host/art-12.0.0-beta4', | 
 |     'default': 'art', | 
 |     '9.0.0': 'art-9.0.0', | 
 |     '8.1.0': 'art-8.1.0', | 
 |     '7.0.0': 'art-7.0.0', | 
 |     '6.0.1': 'art-6.0.1', | 
 |     '5.1.1': 'art-5.1.1', | 
 | } | 
 |  | 
 | PRODUCTS = { | 
 |     '12.0.0': 'redfin', | 
 |     'default': 'angler', | 
 |     '9.0.0': 'marlin', | 
 |     '8.1.0': 'marlin', | 
 |     '7.0.0': 'angler', | 
 |     '6.0.1': 'angler', | 
 |     '5.1.1': 'mako', | 
 | } | 
 |  | 
 | ARCHS = { | 
 |     '12.0.0': 'x86_64', | 
 |     'default': 'arm64', | 
 |     '9.0.0': 'arm64', | 
 |     '8.1.0': 'arm64', | 
 |     '7.0.0': 'arm64', | 
 |     '6.0.1': 'arm64', | 
 |     '5.1.1': 'arm', | 
 | } | 
 |  | 
 | VERBOSE_OPTIONS = [ | 
 |     'verifier', | 
 |     'compiler', | 
 |     'gc', | 
 |     'jit', | 
 |     'jni', | 
 |     'class', | 
 |     'all', | 
 | ] | 
 |  | 
 | BOOT_IMAGE = {'12.0.0': 'apex/art_boot_images/javalib/boot.art'} | 
 |  | 
 |  | 
 | def ParseOptions(): | 
 |     parser = optparse.OptionParser() | 
 |     parser.add_option('--version', | 
 |                       help='Version of dex2oat. (defaults to latest: ' + | 
 |                       LATEST + ').', | 
 |                       choices=VERSIONS, | 
 |                       default=LATEST) | 
 |     parser.add_option( | 
 |         '--device', | 
 |         help='Run dex2oat on this device (this is passed as the -s SERIAL.') | 
 |     parser.add_option('--all', | 
 |                       help='Run dex2oat on all possible versions', | 
 |                       default=False, | 
 |                       action='store_true') | 
 |     parser.add_option( | 
 |         '--output', | 
 |         help= | 
 |         'Where to place the output oat (defaults to no output / temp file).', | 
 |         default=None) | 
 |     parser.add_option('--verbose', | 
 |                       help='Enable verbose dex2oat logging.', | 
 |                       choices=VERBOSE_OPTIONS, | 
 |                       default=None) | 
 |     return parser.parse_args() | 
 |  | 
 |  | 
 | def Main(): | 
 |     (options, args) = ParseOptions() | 
 |     if len(args) != 1: | 
 |         print("Can only take a single dex/zip/jar/apk file as input.") | 
 |         return 1 | 
 |     if (options.device): | 
 |         return run_device_dex2oat(options, args) | 
 |     else: | 
 |         return run_host_dex2oat(options, args) | 
 |  | 
 |  | 
 | def run_host_dex2oat(options, args): | 
 |     if options.all and options.output: | 
 |         print("Can't write output when running all versions.") | 
 |         return 1 | 
 |     dexfile = args[0] | 
 |     oatfile = options.output | 
 |     versions = VERSIONS if options.all else [options.version] | 
 |     for version in versions: | 
 |         run(options, dexfile, oatfile, version) | 
 |         print("") | 
 |     return 0 | 
 |  | 
 |  | 
 | def adb_cmd(serial, *args): | 
 |     cmd = ['adb', '-s', serial] | 
 |     cmd.extend(args) | 
 |     return cmd | 
 |  | 
 |  | 
 | def append_dex2oat_verbose_flags(options, cmd): | 
 |     verbose = [options.verbose] if options.verbose else [] | 
 |     if 'all' in verbose: | 
 |         verbose = [x for x in VERBOSE_OPTIONS if x != 'all'] | 
 |     for flag in verbose: | 
 |         cmd += ['--runtime-arg', '-verbose:' + flag] | 
 |     return cmd | 
 |  | 
 |  | 
 | def run_device_dex2oat(options, args): | 
 |     serial = options.device | 
 |     dexfile = args[0] | 
 |     device_dexfile = '/data/local/tmp/' + os.path.basename(dexfile) | 
 |     device_oatfile = '/data/local/tmp/unused.oat' | 
 |     cmd = adb_cmd(serial, 'shell', 'rm', '-f', device_dexfile, device_oatfile) | 
 |     utils.PrintCmd(cmd) | 
 |     subprocess.check_call(cmd) | 
 |     cmd = adb_cmd(serial, 'push', dexfile, device_dexfile) | 
 |     utils.PrintCmd(cmd) | 
 |     subprocess.check_call(cmd) | 
 |     cmd = adb_cmd(serial, 'logcat', '-c') | 
 |     utils.PrintCmd(cmd) | 
 |     subprocess.check_call(cmd) | 
 |     cmd = adb_cmd(serial, 'shell', 'dex2oat', '--dex-file=' + device_dexfile, | 
 |                   '--oat-file=/data/local/tmp/unused.oat') | 
 |     append_dex2oat_verbose_flags(options, cmd) | 
 |     utils.PrintCmd(cmd) | 
 |     subprocess.check_call(cmd) | 
 |     cmd = adb_cmd(serial, 'logcat', '-d', '-s', 'dex2oat', 'dex2oat_original') | 
 |     utils.PrintCmd(cmd) | 
 |     subprocess.check_call(cmd) | 
 |  | 
 |     return 0 | 
 |  | 
 |  | 
 | def run(options, dexfile, oatfile=None, version=None): | 
 |     if not version: | 
 |         version = LATEST | 
 |     # dex2oat accepts non-existent dex files, check here instead | 
 |     if not os.path.exists(dexfile): | 
 |         raise Exception('DEX file not found: "{}"'.format(dexfile)) | 
 |     with utils.TempDir() as temp: | 
 |         if not oatfile: | 
 |             oatfile = os.path.join(temp, "out.oat") | 
 |         base = os.path.join(LINUX_DIR, DIRS[version]) | 
 |         product = PRODUCTS[version] | 
 |         arch = ARCHS[version] | 
 |         cmd = [ | 
 |             os.path.join(base, 'bin', 'dex2oat'), | 
 |             '--android-root=' + | 
 |             os.path.join(base, 'product', product, 'system'), | 
 |             '--runtime-arg', | 
 |             '-Xnorelocate', | 
 |             '--dex-file=' + dexfile, | 
 |             '--oat-file=' + oatfile, | 
 |             '--instruction-set=' + arch, | 
 |         ] | 
 |         append_dex2oat_verbose_flags(options, cmd) | 
 |         if version in BOOT_IMAGE: | 
 |             cmd += ['--boot-image=' + BOOT_IMAGE[version]] | 
 |         env = {"LD_LIBRARY_PATH": os.path.join(base, 'lib')} | 
 |         utils.PrintCmd(cmd) | 
 |         with utils.ChangedWorkingDirectory(base): | 
 |             subprocess.check_call(cmd, env=env) | 
 |  | 
 |  | 
 | if __name__ == '__main__': | 
 |     sys.exit(Main()) |