|  | #!/usr/bin/env python | 
|  |  | 
|  | # Copyright 2017 the V8 project authors. All rights reserved. | 
|  | # Use of this source code is governed by a BSD-style license that can be | 
|  | # found in the LICENSE file. | 
|  |  | 
|  | # Runs an android build of d8 over adb, with any given arguments. Files | 
|  | # requested by d8 are transferred on-demand from the caller, by reverse port | 
|  | # forwarding a simple TCP file server from the computer to the android device. | 
|  | # | 
|  | # Usage: | 
|  | #    adb-d8.py <build_dir> [<d8_args>...] | 
|  | # | 
|  | # Options: | 
|  | #    <build_dir>    The directory containing the android build of d8. | 
|  | #    <d8_args>...   The arguments passed through to d8. | 
|  | # | 
|  | # Run adb-d8.py --help for complete usage information. | 
|  |  | 
|  | from __future__ import print_function | 
|  |  | 
|  | import os | 
|  | import sys | 
|  | import struct | 
|  | import threading | 
|  | import subprocess | 
|  | import SocketServer # TODO(leszeks): python 3 compatibility | 
|  |  | 
|  | def CreateFileHandlerClass(root_dirs, verbose): | 
|  | class FileHandler(SocketServer.BaseRequestHandler): | 
|  | def handle(self): | 
|  | data = self.request.recv(1024); | 
|  | while data[-1] != "\0": | 
|  | data += self.request.recv(1024); | 
|  |  | 
|  | filename = data[0:-1] | 
|  |  | 
|  | try: | 
|  | filename = os.path.abspath(filename) | 
|  |  | 
|  | if not any(filename.startswith(root) for root in root_dirs): | 
|  | raise Exception("{} not in roots {}".format(filename, root_dirs)) | 
|  | if not os.path.isfile(filename): | 
|  | raise Exception("{} is not a file".format(filename)) | 
|  |  | 
|  | if verbose: | 
|  | sys.stdout.write("Serving {}\r\n".format(os.path.relpath(filename))) | 
|  |  | 
|  | with open(filename) as f: | 
|  | contents = f.read(); | 
|  | self.request.sendall(struct.pack("!i", len(contents))) | 
|  | self.request.sendall(contents) | 
|  |  | 
|  | except Exception as e: | 
|  | if verbose: | 
|  | sys.stderr.write( | 
|  | "Request failed ({})\n".format(e).replace('\n','\r\n')) | 
|  | self.request.sendall(struct.pack("!i", -1)) | 
|  |  | 
|  | return FileHandler | 
|  |  | 
|  |  | 
|  | def TransferD8ToDevice(adb, build_dir, device_d8_dir, verbose): | 
|  | files_to_copy = ["d8", "snapshot_blob.bin"] | 
|  |  | 
|  | # Pipe the output of md5sum from the local computer to the device, checking | 
|  | # the md5 hashes on the device. | 
|  | local_md5_sum_proc = subprocess.Popen( | 
|  | ["md5sum"] + files_to_copy, | 
|  | cwd=build_dir, | 
|  | stdout=subprocess.PIPE | 
|  | ) | 
|  | device_md5_check_proc = subprocess.Popen( | 
|  | [ | 
|  | adb, "shell", | 
|  | "mkdir -p '{0}' ; cd '{0}' ; md5sum -c -".format(device_d8_dir) | 
|  | ], | 
|  | stdin=local_md5_sum_proc.stdout, | 
|  | stdout=subprocess.PIPE, | 
|  | stderr=subprocess.PIPE | 
|  | ) | 
|  |  | 
|  | # Push any files which failed the md5 check. | 
|  | (stdoutdata, stderrdata) = device_md5_check_proc.communicate() | 
|  | for line in stdoutdata.split('\n'): | 
|  | if line.endswith(": FAILED"): | 
|  | filename = line[:-len(": FAILED")] | 
|  | if verbose: | 
|  | print("Updating {}...".format(filename)) | 
|  | subprocess.check_call([ | 
|  | adb, "push", | 
|  | os.path.join(build_dir, filename), | 
|  | device_d8_dir | 
|  | ], stdout=sys.stdout if verbose else open(os.devnull, 'wb')) | 
|  |  | 
|  |  | 
|  | def AdbForwardDeviceToLocal(adb, device_port, server_port, verbose): | 
|  | if verbose: | 
|  | print("Forwarding device:{} to localhost:{}...".format( | 
|  | device_port, server_port)) | 
|  |  | 
|  | subprocess.check_call([ | 
|  | adb, "reverse", | 
|  | "tcp:{}".format(device_port), | 
|  | "tcp:{}".format(server_port) | 
|  | ]) | 
|  |  | 
|  |  | 
|  | def AdbRunD8(adb, device_d8_dir, device_port, d8_args, verbose): | 
|  | # Single-quote the arguments to d8, and concatenate them into a string. | 
|  | d8_arg_str = " ".join("'{}'".format(a) for a in d8_args) | 
|  | d8_arg_str = "--read-from-tcp-port='{}' ".format(device_port) + d8_arg_str | 
|  |  | 
|  | # Don't use os.path.join for d8 because we care about the device's os, not | 
|  | # the host os. | 
|  | d8_str = "{}/d8 {}".format(device_d8_dir, d8_arg_str) | 
|  |  | 
|  | if sys.stdout.isatty(): | 
|  | # Run adb shell with -t to have a tty if we run d8 without a script. | 
|  | cmd = [adb, "shell", "-t", d8_str] | 
|  | else: | 
|  | cmd = [adb, "shell", d8_str] | 
|  |  | 
|  | if verbose: | 
|  | print("Running {}".format(" ".join(cmd))) | 
|  | return subprocess.call(cmd) | 
|  |  | 
|  |  | 
|  | def PrintUsage(file=sys.stdout): | 
|  | print("Usage: adb-d8.py [-v|--verbose] [--] <build_dir> [<d8 args>...]", | 
|  | file=file) | 
|  |  | 
|  |  | 
|  | def PrintHelp(file=sys.stdout): | 
|  | print("""Usage: | 
|  | adb-d8.py [options] [--] <build_dir> [<d8_args>...] | 
|  | adb-d8.py -h|--help | 
|  |  | 
|  | Options: | 
|  | -h|--help             Show this help message and exit. | 
|  | -v|--verbose          Print verbose output. | 
|  | --device-dir=DIR      Specify which directory on the device should be used | 
|  | for the d8 binary. [default: /data/local/tmp/v8] | 
|  | --extra-root-dir=DIR  In addition to the current directory, allow d8 to | 
|  | access files inside DIR. Multiple additional roots | 
|  | can be specified. | 
|  | <build_dir>           The directory containing the android build of d8. | 
|  | <d8_args>...          The arguments passed through to d8.""", file=file) | 
|  |  | 
|  |  | 
|  | def Main(): | 
|  | if len(sys.argv) < 2: | 
|  | PrintUsage(sys.stderr) | 
|  | return 1 | 
|  |  | 
|  | script_dir = os.path.dirname(sys.argv[0]) | 
|  | # Use the platform-tools version of adb so that we know it has the reverse | 
|  | # command. | 
|  | adb = os.path.join( | 
|  | script_dir, | 
|  | "../third_party/android_sdk/public/platform-tools/adb" | 
|  | ) | 
|  |  | 
|  | # Read off any command line flags before build_dir (or --). Do this | 
|  | # manually, rather than using something like argparse, to be able to split | 
|  | # the adb-d8 options from the passthrough d8 options. | 
|  | verbose = False | 
|  | device_d8_dir = '/data/local/tmp/v8' | 
|  | root_dirs = [] | 
|  | arg_index = 1 | 
|  | while arg_index < len(sys.argv): | 
|  | arg = sys.argv[arg_index] | 
|  | if not arg.startswith("-"): | 
|  | break | 
|  | elif arg == "--": | 
|  | arg_index += 1 | 
|  | break | 
|  | elif arg == "-h" or arg == "--help": | 
|  | PrintHelp(sys.stdout) | 
|  | return 0 | 
|  | elif arg == "-v" or arg == "--verbose": | 
|  | verbose = True | 
|  |  | 
|  | elif arg == "--device-dir": | 
|  | arg_index += 1 | 
|  | device_d8_dir = sys.argv[arg_index] | 
|  | elif arg.startswith("--device-dir="): | 
|  | device_d8_dir = arg[len("--device-dir="):] | 
|  |  | 
|  | elif arg == "--extra-root-dir": | 
|  | arg_index += 1 | 
|  | root_dirs.append(sys.argv[arg_index]) | 
|  | elif arg.startswith("--extra-root-dir="): | 
|  | root_dirs.append(arg[len("--extra-root-dir="):]) | 
|  |  | 
|  | else: | 
|  | print("ERROR: Unrecognised option: {}".format(arg)) | 
|  | PrintUsage(sys.stderr) | 
|  | return 1 | 
|  |  | 
|  | arg_index += 1 | 
|  |  | 
|  | # Transfer d8 (and dependencies) to the device. | 
|  | build_dir = os.path.abspath(sys.argv[arg_index]) | 
|  |  | 
|  | TransferD8ToDevice(adb, build_dir, device_d8_dir, verbose) | 
|  |  | 
|  | # Start a file server for the files d8 might need. | 
|  | script_root_dir = os.path.abspath(os.curdir) | 
|  | root_dirs.append(script_root_dir) | 
|  | server = SocketServer.TCPServer( | 
|  | ("localhost", 0), # 0 means an arbitrary unused port. | 
|  | CreateFileHandlerClass(root_dirs, verbose) | 
|  | ) | 
|  |  | 
|  | try: | 
|  | # Start the file server in its own thread. | 
|  | server_thread = threading.Thread(target=server.serve_forever) | 
|  | server_thread.daemon = True | 
|  | server_thread.start() | 
|  |  | 
|  | # Port-forward the given device port to the file server. | 
|  | # TODO(leszeks): Pick an unused device port. | 
|  | # TODO(leszeks): Remove the port forwarding on exit. | 
|  | server_ip, server_port = server.server_address | 
|  | device_port = 4444 | 
|  | AdbForwardDeviceToLocal(adb, device_port, server_port, verbose) | 
|  |  | 
|  | # Run d8 over adb with the remaining arguments, using the given device | 
|  | # port to forward file reads. | 
|  | return AdbRunD8( | 
|  | adb, device_d8_dir, device_port, sys.argv[arg_index+1:], verbose) | 
|  |  | 
|  | finally: | 
|  | if verbose: | 
|  | print("Shutting down file server...") | 
|  | server.shutdown() | 
|  | server.server_close() | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | sys.exit(Main()) |