1*6777b538SAndroid Build Coastguard Worker# Copyright 2018 The Chromium Authors 2*6777b538SAndroid Build Coastguard Worker# Use of this source code is governed by a BSD-style license that can be 3*6777b538SAndroid Build Coastguard Worker# found in the LICENSE file. 4*6777b538SAndroid Build Coastguard Worker 5*6777b538SAndroid Build Coastguard Workerimport contextlib 6*6777b538SAndroid Build Coastguard Workerimport logging 7*6777b538SAndroid Build Coastguard Workerimport os 8*6777b538SAndroid Build Coastguard Workerimport shutil 9*6777b538SAndroid Build Coastguard Workerimport subprocess 10*6777b538SAndroid Build Coastguard Workerimport sys 11*6777b538SAndroid Build Coastguard Workerimport tempfile 12*6777b538SAndroid Build Coastguard Worker 13*6777b538SAndroid Build Coastguard Workerfrom devil import devil_env 14*6777b538SAndroid Build Coastguard Workerfrom devil.android import device_signal, device_errors 15*6777b538SAndroid Build Coastguard Workerfrom devil.android.sdk import version_codes 16*6777b538SAndroid Build Coastguard Workerfrom pylib import constants 17*6777b538SAndroid Build Coastguard Worker 18*6777b538SAndroid Build Coastguard Worker 19*6777b538SAndroid Build Coastguard Workerdef _ProcessType(proc): 20*6777b538SAndroid Build Coastguard Worker _, _, suffix = proc.name.partition(':') 21*6777b538SAndroid Build Coastguard Worker if not suffix: 22*6777b538SAndroid Build Coastguard Worker return 'browser' 23*6777b538SAndroid Build Coastguard Worker if suffix.startswith('sandboxed_process'): 24*6777b538SAndroid Build Coastguard Worker return 'renderer' 25*6777b538SAndroid Build Coastguard Worker if suffix.startswith('privileged_process'): 26*6777b538SAndroid Build Coastguard Worker return 'gpu' 27*6777b538SAndroid Build Coastguard Worker return None 28*6777b538SAndroid Build Coastguard Worker 29*6777b538SAndroid Build Coastguard Worker 30*6777b538SAndroid Build Coastguard Workerdef _GetSpecifiedPID(device, package_name, process_specifier): 31*6777b538SAndroid Build Coastguard Worker if process_specifier is None: 32*6777b538SAndroid Build Coastguard Worker return None 33*6777b538SAndroid Build Coastguard Worker 34*6777b538SAndroid Build Coastguard Worker # Check for numeric PID 35*6777b538SAndroid Build Coastguard Worker try: 36*6777b538SAndroid Build Coastguard Worker pid = int(process_specifier) 37*6777b538SAndroid Build Coastguard Worker return pid 38*6777b538SAndroid Build Coastguard Worker except ValueError: 39*6777b538SAndroid Build Coastguard Worker pass 40*6777b538SAndroid Build Coastguard Worker 41*6777b538SAndroid Build Coastguard Worker # Check for exact process name; can be any of these formats: 42*6777b538SAndroid Build Coastguard Worker # <package>:<process name>, i.e. 'org.chromium.chrome:sandboxed_process0' 43*6777b538SAndroid Build Coastguard Worker # :<process name>, i.e. ':sandboxed_process0' 44*6777b538SAndroid Build Coastguard Worker # <process name>, i.e. 'sandboxed_process0' 45*6777b538SAndroid Build Coastguard Worker full_process_name = process_specifier 46*6777b538SAndroid Build Coastguard Worker if process_specifier.startswith(':'): 47*6777b538SAndroid Build Coastguard Worker full_process_name = package_name + process_specifier 48*6777b538SAndroid Build Coastguard Worker elif ':' not in process_specifier: 49*6777b538SAndroid Build Coastguard Worker full_process_name = '%s:%s' % (package_name, process_specifier) 50*6777b538SAndroid Build Coastguard Worker matching_processes = device.ListProcesses(full_process_name) 51*6777b538SAndroid Build Coastguard Worker if len(matching_processes) == 1: 52*6777b538SAndroid Build Coastguard Worker return matching_processes[0].pid 53*6777b538SAndroid Build Coastguard Worker if len(matching_processes) > 1: 54*6777b538SAndroid Build Coastguard Worker raise RuntimeError('Found %d processes with name "%s".' % ( 55*6777b538SAndroid Build Coastguard Worker len(matching_processes), process_specifier)) 56*6777b538SAndroid Build Coastguard Worker 57*6777b538SAndroid Build Coastguard Worker # Check for process type (i.e. 'renderer') 58*6777b538SAndroid Build Coastguard Worker package_processes = device.ListProcesses(package_name) 59*6777b538SAndroid Build Coastguard Worker matching_processes = [p for p in package_processes if ( 60*6777b538SAndroid Build Coastguard Worker _ProcessType(p) == process_specifier)] 61*6777b538SAndroid Build Coastguard Worker if process_specifier == 'renderer' and len(matching_processes) > 1: 62*6777b538SAndroid Build Coastguard Worker raise RuntimeError('Found %d renderer processes; please re-run with only ' 63*6777b538SAndroid Build Coastguard Worker 'one open tab.' % len(matching_processes)) 64*6777b538SAndroid Build Coastguard Worker if len(matching_processes) != 1: 65*6777b538SAndroid Build Coastguard Worker raise RuntimeError('Found %d processes of type "%s".' % ( 66*6777b538SAndroid Build Coastguard Worker len(matching_processes), process_specifier)) 67*6777b538SAndroid Build Coastguard Worker return matching_processes[0].pid 68*6777b538SAndroid Build Coastguard Worker 69*6777b538SAndroid Build Coastguard Worker 70*6777b538SAndroid Build Coastguard Workerdef _ThreadsForProcess(device, pid): 71*6777b538SAndroid Build Coastguard Worker # The thread list output format for 'ps' is the same regardless of version. 72*6777b538SAndroid Build Coastguard Worker # Here's the column headers, and a sample line for a thread belonging to 73*6777b538SAndroid Build Coastguard Worker # pid 12345 (note that the last few columns are not aligned with headers): 74*6777b538SAndroid Build Coastguard Worker # 75*6777b538SAndroid Build Coastguard Worker # USER PID TID PPID VSZ RSS WCHAN ADDR S CMD 76*6777b538SAndroid Build Coastguard Worker # u0_i101 12345 24680 567 1357902 97531 futex_wait_queue_me e85acd9c S \ 77*6777b538SAndroid Build Coastguard Worker # CrRendererMain 78*6777b538SAndroid Build Coastguard Worker if device.build_version_sdk >= version_codes.OREO: 79*6777b538SAndroid Build Coastguard Worker pid_regex = ( 80*6777b538SAndroid Build Coastguard Worker r'^[[:graph:]]\{1,\}[[:blank:]]\{1,\}%d[[:blank:]]\{1,\}' % pid) 81*6777b538SAndroid Build Coastguard Worker ps_cmd = "ps -T -e | grep '%s'" % pid_regex 82*6777b538SAndroid Build Coastguard Worker ps_output_lines = device.RunShellCommand( 83*6777b538SAndroid Build Coastguard Worker ps_cmd, shell=True, check_return=True) 84*6777b538SAndroid Build Coastguard Worker else: 85*6777b538SAndroid Build Coastguard Worker ps_cmd = ['ps', '-p', str(pid), '-t'] 86*6777b538SAndroid Build Coastguard Worker ps_output_lines = device.RunShellCommand(ps_cmd, check_return=True) 87*6777b538SAndroid Build Coastguard Worker result = [] 88*6777b538SAndroid Build Coastguard Worker for l in ps_output_lines: 89*6777b538SAndroid Build Coastguard Worker fields = l.split() 90*6777b538SAndroid Build Coastguard Worker # fields[2] is tid, fields[-1] is thread name. Output may include an entry 91*6777b538SAndroid Build Coastguard Worker # for the process itself with tid=pid; omit that one. 92*6777b538SAndroid Build Coastguard Worker if fields[2] == str(pid): 93*6777b538SAndroid Build Coastguard Worker continue 94*6777b538SAndroid Build Coastguard Worker result.append((int(fields[2]), fields[-1])) 95*6777b538SAndroid Build Coastguard Worker return result 96*6777b538SAndroid Build Coastguard Worker 97*6777b538SAndroid Build Coastguard Worker 98*6777b538SAndroid Build Coastguard Workerdef _ThreadType(thread_name): 99*6777b538SAndroid Build Coastguard Worker if not thread_name: 100*6777b538SAndroid Build Coastguard Worker return 'unknown' 101*6777b538SAndroid Build Coastguard Worker if (thread_name.startswith('Chrome_ChildIO') or 102*6777b538SAndroid Build Coastguard Worker thread_name.startswith('Chrome_IO')): 103*6777b538SAndroid Build Coastguard Worker return 'io' 104*6777b538SAndroid Build Coastguard Worker if thread_name.startswith('Compositor'): 105*6777b538SAndroid Build Coastguard Worker return 'compositor' 106*6777b538SAndroid Build Coastguard Worker if (thread_name.startswith('ChildProcessMai') or 107*6777b538SAndroid Build Coastguard Worker thread_name.startswith('CrGpuMain') or 108*6777b538SAndroid Build Coastguard Worker thread_name.startswith('CrRendererMain')): 109*6777b538SAndroid Build Coastguard Worker return 'main' 110*6777b538SAndroid Build Coastguard Worker if thread_name.startswith('RenderThread'): 111*6777b538SAndroid Build Coastguard Worker return 'render' 112*6777b538SAndroid Build Coastguard Worker raise ValueError('got no matching thread_name') 113*6777b538SAndroid Build Coastguard Worker 114*6777b538SAndroid Build Coastguard Worker 115*6777b538SAndroid Build Coastguard Workerdef _GetSpecifiedTID(device, pid, thread_specifier): 116*6777b538SAndroid Build Coastguard Worker if thread_specifier is None: 117*6777b538SAndroid Build Coastguard Worker return None 118*6777b538SAndroid Build Coastguard Worker 119*6777b538SAndroid Build Coastguard Worker # Check for numeric TID 120*6777b538SAndroid Build Coastguard Worker try: 121*6777b538SAndroid Build Coastguard Worker tid = int(thread_specifier) 122*6777b538SAndroid Build Coastguard Worker return tid 123*6777b538SAndroid Build Coastguard Worker except ValueError: 124*6777b538SAndroid Build Coastguard Worker pass 125*6777b538SAndroid Build Coastguard Worker 126*6777b538SAndroid Build Coastguard Worker # Check for thread type 127*6777b538SAndroid Build Coastguard Worker if pid is not None: 128*6777b538SAndroid Build Coastguard Worker matching_threads = [t for t in _ThreadsForProcess(device, pid) if ( 129*6777b538SAndroid Build Coastguard Worker _ThreadType(t[1]) == thread_specifier)] 130*6777b538SAndroid Build Coastguard Worker if len(matching_threads) != 1: 131*6777b538SAndroid Build Coastguard Worker raise RuntimeError('Found %d threads of type "%s".' % ( 132*6777b538SAndroid Build Coastguard Worker len(matching_threads), thread_specifier)) 133*6777b538SAndroid Build Coastguard Worker return matching_threads[0][0] 134*6777b538SAndroid Build Coastguard Worker 135*6777b538SAndroid Build Coastguard Worker return None 136*6777b538SAndroid Build Coastguard Worker 137*6777b538SAndroid Build Coastguard Worker 138*6777b538SAndroid Build Coastguard Workerdef PrepareDevice(device): 139*6777b538SAndroid Build Coastguard Worker if device.build_version_sdk < version_codes.NOUGAT: 140*6777b538SAndroid Build Coastguard Worker raise RuntimeError('Simpleperf profiling is only supported on Android N ' 141*6777b538SAndroid Build Coastguard Worker 'and later.') 142*6777b538SAndroid Build Coastguard Worker 143*6777b538SAndroid Build Coastguard Worker # Necessary for profiling 144*6777b538SAndroid Build Coastguard Worker # https://android-review.googlesource.com/c/platform/system/sepolicy/+/234400 145*6777b538SAndroid Build Coastguard Worker device.SetProp('security.perf_harden', '0') 146*6777b538SAndroid Build Coastguard Worker 147*6777b538SAndroid Build Coastguard Worker 148*6777b538SAndroid Build Coastguard Workerdef InstallSimpleperf(device, package_name): 149*6777b538SAndroid Build Coastguard Worker package_arch = device.GetPackageArchitecture(package_name) or 'armeabi-v7a' 150*6777b538SAndroid Build Coastguard Worker host_simpleperf_path = devil_env.config.LocalPath('simpleperf', package_arch) 151*6777b538SAndroid Build Coastguard Worker if not host_simpleperf_path: 152*6777b538SAndroid Build Coastguard Worker raise Exception('Could not get path to simpleperf executable on host.') 153*6777b538SAndroid Build Coastguard Worker device_simpleperf_path = '/'.join( 154*6777b538SAndroid Build Coastguard Worker ('/data/local/tmp/profilers', package_arch, 'simpleperf')) 155*6777b538SAndroid Build Coastguard Worker device.PushChangedFiles([(host_simpleperf_path, device_simpleperf_path)]) 156*6777b538SAndroid Build Coastguard Worker return device_simpleperf_path 157*6777b538SAndroid Build Coastguard Worker 158*6777b538SAndroid Build Coastguard Worker 159*6777b538SAndroid Build Coastguard Worker@contextlib.contextmanager 160*6777b538SAndroid Build Coastguard Workerdef RunSimpleperf(device, device_simpleperf_path, package_name, 161*6777b538SAndroid Build Coastguard Worker process_specifier, thread_specifier, events, 162*6777b538SAndroid Build Coastguard Worker profiler_args, host_out_path): 163*6777b538SAndroid Build Coastguard Worker pid = _GetSpecifiedPID(device, package_name, process_specifier) 164*6777b538SAndroid Build Coastguard Worker tid = _GetSpecifiedTID(device, pid, thread_specifier) 165*6777b538SAndroid Build Coastguard Worker if pid is None and tid is None: 166*6777b538SAndroid Build Coastguard Worker raise RuntimeError('Could not find specified process/thread running on ' 167*6777b538SAndroid Build Coastguard Worker 'device. Make sure the apk is already running before ' 168*6777b538SAndroid Build Coastguard Worker 'attempting to profile.') 169*6777b538SAndroid Build Coastguard Worker profiler_args = list(profiler_args) 170*6777b538SAndroid Build Coastguard Worker if profiler_args and profiler_args[0] == 'record': 171*6777b538SAndroid Build Coastguard Worker profiler_args.pop(0) 172*6777b538SAndroid Build Coastguard Worker profiler_args.extend(('-e', events)) 173*6777b538SAndroid Build Coastguard Worker if '--call-graph' not in profiler_args and '-g' not in profiler_args: 174*6777b538SAndroid Build Coastguard Worker profiler_args.append('-g') 175*6777b538SAndroid Build Coastguard Worker if '-f' not in profiler_args: 176*6777b538SAndroid Build Coastguard Worker profiler_args.extend(('-f', '1000')) 177*6777b538SAndroid Build Coastguard Worker 178*6777b538SAndroid Build Coastguard Worker device_out_path = '/data/local/tmp/perf.data' 179*6777b538SAndroid Build Coastguard Worker should_remove_device_out_path = True 180*6777b538SAndroid Build Coastguard Worker if '-o' in profiler_args: 181*6777b538SAndroid Build Coastguard Worker device_out_path = profiler_args[profiler_args.index('-o') + 1] 182*6777b538SAndroid Build Coastguard Worker should_remove_device_out_path = False 183*6777b538SAndroid Build Coastguard Worker else: 184*6777b538SAndroid Build Coastguard Worker profiler_args.extend(('-o', device_out_path)) 185*6777b538SAndroid Build Coastguard Worker 186*6777b538SAndroid Build Coastguard Worker # Remove the default output to avoid confusion if simpleperf opts not 187*6777b538SAndroid Build Coastguard Worker # to update the file. 188*6777b538SAndroid Build Coastguard Worker file_exists = True 189*6777b538SAndroid Build Coastguard Worker try: 190*6777b538SAndroid Build Coastguard Worker device.adb.Shell('readlink -e ' + device_out_path) 191*6777b538SAndroid Build Coastguard Worker except device_errors.AdbCommandFailedError: 192*6777b538SAndroid Build Coastguard Worker file_exists = False 193*6777b538SAndroid Build Coastguard Worker if file_exists: 194*6777b538SAndroid Build Coastguard Worker logging.warning('%s output file already exists on device', device_out_path) 195*6777b538SAndroid Build Coastguard Worker if not should_remove_device_out_path: 196*6777b538SAndroid Build Coastguard Worker raise RuntimeError('Specified output file \'{}\' already exists, not ' 197*6777b538SAndroid Build Coastguard Worker 'continuing'.format(device_out_path)) 198*6777b538SAndroid Build Coastguard Worker device.adb.Shell('rm -f ' + device_out_path) 199*6777b538SAndroid Build Coastguard Worker 200*6777b538SAndroid Build Coastguard Worker if tid: 201*6777b538SAndroid Build Coastguard Worker profiler_args.extend(('-t', str(tid))) 202*6777b538SAndroid Build Coastguard Worker else: 203*6777b538SAndroid Build Coastguard Worker profiler_args.extend(('-p', str(pid))) 204*6777b538SAndroid Build Coastguard Worker 205*6777b538SAndroid Build Coastguard Worker adb_shell_simpleperf_process = device.adb.StartShell( 206*6777b538SAndroid Build Coastguard Worker [device_simpleperf_path, 'record'] + profiler_args) 207*6777b538SAndroid Build Coastguard Worker 208*6777b538SAndroid Build Coastguard Worker completed = False 209*6777b538SAndroid Build Coastguard Worker try: 210*6777b538SAndroid Build Coastguard Worker yield 211*6777b538SAndroid Build Coastguard Worker completed = True 212*6777b538SAndroid Build Coastguard Worker 213*6777b538SAndroid Build Coastguard Worker finally: 214*6777b538SAndroid Build Coastguard Worker device.KillAll('simpleperf', signum=device_signal.SIGINT, blocking=True, 215*6777b538SAndroid Build Coastguard Worker quiet=True) 216*6777b538SAndroid Build Coastguard Worker if completed: 217*6777b538SAndroid Build Coastguard Worker adb_shell_simpleperf_process.wait() 218*6777b538SAndroid Build Coastguard Worker ret = adb_shell_simpleperf_process.returncode 219*6777b538SAndroid Build Coastguard Worker if ret == 0: 220*6777b538SAndroid Build Coastguard Worker # Successfully gathered a profile 221*6777b538SAndroid Build Coastguard Worker device.PullFile(device_out_path, host_out_path) 222*6777b538SAndroid Build Coastguard Worker else: 223*6777b538SAndroid Build Coastguard Worker logging.warning( 224*6777b538SAndroid Build Coastguard Worker 'simpleperf exited unusually, expected exit 0, got %d', ret 225*6777b538SAndroid Build Coastguard Worker ) 226*6777b538SAndroid Build Coastguard Worker stdout, stderr = adb_shell_simpleperf_process.communicate() 227*6777b538SAndroid Build Coastguard Worker logging.info('stdout: \'%s\', stderr: \'%s\'', stdout, stderr) 228*6777b538SAndroid Build Coastguard Worker raise RuntimeError('simpleperf exited with unexpected code {} ' 229*6777b538SAndroid Build Coastguard Worker '(run with -vv for full stdout/stderr)'.format(ret)) 230*6777b538SAndroid Build Coastguard Worker 231*6777b538SAndroid Build Coastguard Worker 232*6777b538SAndroid Build Coastguard Workerdef ConvertSimpleperfToPprof(simpleperf_out_path, build_directory, 233*6777b538SAndroid Build Coastguard Worker pprof_out_path): 234*6777b538SAndroid Build Coastguard Worker # The simpleperf scripts require the unstripped libs to be installed in the 235*6777b538SAndroid Build Coastguard Worker # same directory structure as the libs on the device. Much of the logic here 236*6777b538SAndroid Build Coastguard Worker # is just figuring out and creating the necessary directory structure, and 237*6777b538SAndroid Build Coastguard Worker # symlinking the unstripped shared libs. 238*6777b538SAndroid Build Coastguard Worker 239*6777b538SAndroid Build Coastguard Worker # Get the set of libs that we can symbolize 240*6777b538SAndroid Build Coastguard Worker unstripped_lib_dir = os.path.join(build_directory, 'lib.unstripped') 241*6777b538SAndroid Build Coastguard Worker unstripped_libs = set( 242*6777b538SAndroid Build Coastguard Worker f for f in os.listdir(unstripped_lib_dir) if f.endswith('.so')) 243*6777b538SAndroid Build Coastguard Worker 244*6777b538SAndroid Build Coastguard Worker # report.py will show the directory structure above the shared libs; 245*6777b538SAndroid Build Coastguard Worker # that is the directory structure we need to recreate on the host. 246*6777b538SAndroid Build Coastguard Worker script_dir = devil_env.config.LocalPath('simpleperf_scripts') 247*6777b538SAndroid Build Coastguard Worker report_path = os.path.join(script_dir, 'report.py') 248*6777b538SAndroid Build Coastguard Worker report_cmd = [sys.executable, report_path, '-i', simpleperf_out_path] 249*6777b538SAndroid Build Coastguard Worker device_lib_path = None 250*6777b538SAndroid Build Coastguard Worker output = subprocess.check_output(report_cmd, stderr=subprocess.STDOUT) 251*6777b538SAndroid Build Coastguard Worker if isinstance(output, bytes): 252*6777b538SAndroid Build Coastguard Worker output = output.decode() 253*6777b538SAndroid Build Coastguard Worker for line in output.splitlines(): 254*6777b538SAndroid Build Coastguard Worker fields = line.split() 255*6777b538SAndroid Build Coastguard Worker if len(fields) < 5: 256*6777b538SAndroid Build Coastguard Worker continue 257*6777b538SAndroid Build Coastguard Worker shlib_path = fields[4] 258*6777b538SAndroid Build Coastguard Worker shlib_dirname, shlib_basename = shlib_path.rpartition('/')[::2] 259*6777b538SAndroid Build Coastguard Worker if shlib_basename in unstripped_libs: 260*6777b538SAndroid Build Coastguard Worker device_lib_path = shlib_dirname 261*6777b538SAndroid Build Coastguard Worker break 262*6777b538SAndroid Build Coastguard Worker if not device_lib_path: 263*6777b538SAndroid Build Coastguard Worker raise RuntimeError('No chrome-related symbols in profiling data in %s. ' 264*6777b538SAndroid Build Coastguard Worker 'Either the process was idle for the entire profiling ' 265*6777b538SAndroid Build Coastguard Worker 'period, or something went very wrong (and you should ' 266*6777b538SAndroid Build Coastguard Worker 'file a bug at crbug.com/new with component ' 267*6777b538SAndroid Build Coastguard Worker 'Speed>Tracing, and assign it to [email protected]).' 268*6777b538SAndroid Build Coastguard Worker % simpleperf_out_path) 269*6777b538SAndroid Build Coastguard Worker 270*6777b538SAndroid Build Coastguard Worker # Recreate the directory structure locally, and symlink unstripped libs. 271*6777b538SAndroid Build Coastguard Worker processing_dir = tempfile.mkdtemp() 272*6777b538SAndroid Build Coastguard Worker try: 273*6777b538SAndroid Build Coastguard Worker processing_lib_dir = os.path.join( 274*6777b538SAndroid Build Coastguard Worker processing_dir, 'binary_cache', device_lib_path.lstrip('/')) 275*6777b538SAndroid Build Coastguard Worker os.makedirs(processing_lib_dir) 276*6777b538SAndroid Build Coastguard Worker for lib in unstripped_libs: 277*6777b538SAndroid Build Coastguard Worker unstripped_lib_path = os.path.join(unstripped_lib_dir, lib) 278*6777b538SAndroid Build Coastguard Worker processing_lib_path = os.path.join(processing_lib_dir, lib) 279*6777b538SAndroid Build Coastguard Worker os.symlink(unstripped_lib_path, processing_lib_path) 280*6777b538SAndroid Build Coastguard Worker 281*6777b538SAndroid Build Coastguard Worker # Run the script to annotate symbols and convert from simpleperf format to 282*6777b538SAndroid Build Coastguard Worker # pprof format. 283*6777b538SAndroid Build Coastguard Worker pprof_converter_script = os.path.join( 284*6777b538SAndroid Build Coastguard Worker script_dir, 'pprof_proto_generator.py') 285*6777b538SAndroid Build Coastguard Worker pprof_converter_cmd = [ 286*6777b538SAndroid Build Coastguard Worker sys.executable, pprof_converter_script, '-i', simpleperf_out_path, '-o', 287*6777b538SAndroid Build Coastguard Worker os.path.abspath(pprof_out_path), '--ndk_path', 288*6777b538SAndroid Build Coastguard Worker constants.ANDROID_NDK_ROOT 289*6777b538SAndroid Build Coastguard Worker ] 290*6777b538SAndroid Build Coastguard Worker subprocess.check_output(pprof_converter_cmd, stderr=subprocess.STDOUT, 291*6777b538SAndroid Build Coastguard Worker cwd=processing_dir) 292*6777b538SAndroid Build Coastguard Worker finally: 293*6777b538SAndroid Build Coastguard Worker shutil.rmtree(processing_dir, ignore_errors=True) 294