1*6dbdd20aSAndroid Build Coastguard Worker#!/usr/bin/env python3 2*6dbdd20aSAndroid Build Coastguard Worker# Copyright (C) 2017 The Android Open Source Project 3*6dbdd20aSAndroid Build Coastguard Worker# 4*6dbdd20aSAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 5*6dbdd20aSAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 6*6dbdd20aSAndroid Build Coastguard Worker# You may obtain a copy of the License at 7*6dbdd20aSAndroid Build Coastguard Worker# 8*6dbdd20aSAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 9*6dbdd20aSAndroid Build Coastguard Worker# 10*6dbdd20aSAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 11*6dbdd20aSAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 12*6dbdd20aSAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*6dbdd20aSAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 14*6dbdd20aSAndroid Build Coastguard Worker# limitations under the License. 15*6dbdd20aSAndroid Build Coastguard Worker 16*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import absolute_import 17*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import division 18*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import print_function 19*6dbdd20aSAndroid Build Coastguard Worker 20*6dbdd20aSAndroid Build Coastguard Workerimport argparse 21*6dbdd20aSAndroid Build Coastguard Workerimport atexit 22*6dbdd20aSAndroid Build Coastguard Workerimport os 23*6dbdd20aSAndroid Build Coastguard Workerimport shutil 24*6dbdd20aSAndroid Build Coastguard Workerimport signal 25*6dbdd20aSAndroid Build Coastguard Workerimport subprocess 26*6dbdd20aSAndroid Build Coastguard Workerimport sys 27*6dbdd20aSAndroid Build Coastguard Workerimport tempfile 28*6dbdd20aSAndroid Build Coastguard Workerimport time 29*6dbdd20aSAndroid Build Coastguard Workerimport uuid 30*6dbdd20aSAndroid Build Coastguard Worker 31*6dbdd20aSAndroid Build Coastguard Workerfrom perfetto.prebuilts.manifests.traceconv import * 32*6dbdd20aSAndroid Build Coastguard Workerfrom perfetto.prebuilts.perfetto_prebuilts import * 33*6dbdd20aSAndroid Build Coastguard Worker 34*6dbdd20aSAndroid Build Coastguard WorkerNULL = open(os.devnull) 35*6dbdd20aSAndroid Build Coastguard WorkerNOOUT = { 36*6dbdd20aSAndroid Build Coastguard Worker 'stdout': NULL, 37*6dbdd20aSAndroid Build Coastguard Worker 'stderr': NULL, 38*6dbdd20aSAndroid Build Coastguard Worker} 39*6dbdd20aSAndroid Build Coastguard Worker 40*6dbdd20aSAndroid Build Coastguard WorkerUUID = str(uuid.uuid4())[-6:] 41*6dbdd20aSAndroid Build Coastguard Worker 42*6dbdd20aSAndroid Build Coastguard WorkerPACKAGES_LIST_CFG = '''data_sources { 43*6dbdd20aSAndroid Build Coastguard Worker config { 44*6dbdd20aSAndroid Build Coastguard Worker name: "android.packages_list" 45*6dbdd20aSAndroid Build Coastguard Worker } 46*6dbdd20aSAndroid Build Coastguard Worker} 47*6dbdd20aSAndroid Build Coastguard Worker''' 48*6dbdd20aSAndroid Build Coastguard Worker 49*6dbdd20aSAndroid Build Coastguard WorkerCFG_INDENT = ' ' 50*6dbdd20aSAndroid Build Coastguard WorkerCFG = '''buffers {{ 51*6dbdd20aSAndroid Build Coastguard Worker size_kb: 63488 52*6dbdd20aSAndroid Build Coastguard Worker}} 53*6dbdd20aSAndroid Build Coastguard Worker 54*6dbdd20aSAndroid Build Coastguard Workerdata_sources {{ 55*6dbdd20aSAndroid Build Coastguard Worker config {{ 56*6dbdd20aSAndroid Build Coastguard Worker name: "android.heapprofd" 57*6dbdd20aSAndroid Build Coastguard Worker heapprofd_config {{ 58*6dbdd20aSAndroid Build Coastguard Worker shmem_size_bytes: {shmem_size} 59*6dbdd20aSAndroid Build Coastguard Worker sampling_interval_bytes: {interval} 60*6dbdd20aSAndroid Build Coastguard Worker{target_cfg} 61*6dbdd20aSAndroid Build Coastguard Worker }} 62*6dbdd20aSAndroid Build Coastguard Worker }} 63*6dbdd20aSAndroid Build Coastguard Worker}} 64*6dbdd20aSAndroid Build Coastguard Worker 65*6dbdd20aSAndroid Build Coastguard Workerduration_ms: {duration} 66*6dbdd20aSAndroid Build Coastguard Workerwrite_into_file: true 67*6dbdd20aSAndroid Build Coastguard Workerflush_timeout_ms: 30000 68*6dbdd20aSAndroid Build Coastguard Workerflush_period_ms: 604800000 69*6dbdd20aSAndroid Build Coastguard Worker''' 70*6dbdd20aSAndroid Build Coastguard Worker 71*6dbdd20aSAndroid Build Coastguard Worker# flush_period_ms of 1 week to suppress trace_processor_shell warning. 72*6dbdd20aSAndroid Build Coastguard Worker 73*6dbdd20aSAndroid Build Coastguard WorkerCONTINUOUS_DUMP = """ 74*6dbdd20aSAndroid Build Coastguard Worker continuous_dump_config {{ 75*6dbdd20aSAndroid Build Coastguard Worker dump_phase_ms: 0 76*6dbdd20aSAndroid Build Coastguard Worker dump_interval_ms: {dump_interval} 77*6dbdd20aSAndroid Build Coastguard Worker }} 78*6dbdd20aSAndroid Build Coastguard Worker""" 79*6dbdd20aSAndroid Build Coastguard Worker 80*6dbdd20aSAndroid Build Coastguard WorkerPROFILE_LOCAL_PATH = os.path.join(tempfile.gettempdir(), UUID) 81*6dbdd20aSAndroid Build Coastguard Worker 82*6dbdd20aSAndroid Build Coastguard WorkerIS_INTERRUPTED = False 83*6dbdd20aSAndroid Build Coastguard Worker 84*6dbdd20aSAndroid Build Coastguard Worker 85*6dbdd20aSAndroid Build Coastguard Workerdef sigint_handler(sig, frame): 86*6dbdd20aSAndroid Build Coastguard Worker global IS_INTERRUPTED 87*6dbdd20aSAndroid Build Coastguard Worker IS_INTERRUPTED = True 88*6dbdd20aSAndroid Build Coastguard Worker 89*6dbdd20aSAndroid Build Coastguard Worker 90*6dbdd20aSAndroid Build Coastguard Workerdef print_no_profile_error(): 91*6dbdd20aSAndroid Build Coastguard Worker print("No profiles generated", file=sys.stderr) 92*6dbdd20aSAndroid Build Coastguard Worker print( 93*6dbdd20aSAndroid Build Coastguard Worker "If this is unexpected, check " 94*6dbdd20aSAndroid Build Coastguard Worker "https://perfetto.dev/docs/data-sources/native-heap-profiler#troubleshooting.", 95*6dbdd20aSAndroid Build Coastguard Worker file=sys.stderr) 96*6dbdd20aSAndroid Build Coastguard Worker 97*6dbdd20aSAndroid Build Coastguard Worker 98*6dbdd20aSAndroid Build Coastguard Workerdef known_issues_url(number): 99*6dbdd20aSAndroid Build Coastguard Worker return ('https://perfetto.dev/docs/data-sources/native-heap-profiler' 100*6dbdd20aSAndroid Build Coastguard Worker '#known-issues-android{}'.format(number)) 101*6dbdd20aSAndroid Build Coastguard Worker 102*6dbdd20aSAndroid Build Coastguard Worker 103*6dbdd20aSAndroid Build Coastguard WorkerKNOWN_ISSUES = { 104*6dbdd20aSAndroid Build Coastguard Worker '10': known_issues_url(10), 105*6dbdd20aSAndroid Build Coastguard Worker 'Q': known_issues_url(10), 106*6dbdd20aSAndroid Build Coastguard Worker '11': known_issues_url(11), 107*6dbdd20aSAndroid Build Coastguard Worker 'R': known_issues_url(11), 108*6dbdd20aSAndroid Build Coastguard Worker} 109*6dbdd20aSAndroid Build Coastguard Worker 110*6dbdd20aSAndroid Build Coastguard Worker 111*6dbdd20aSAndroid Build Coastguard Workerdef maybe_known_issues(): 112*6dbdd20aSAndroid Build Coastguard Worker release_or_codename = subprocess.check_output( 113*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', 'getprop', 114*6dbdd20aSAndroid Build Coastguard Worker 'ro.build.version.release_or_codename']).decode('utf-8').strip() 115*6dbdd20aSAndroid Build Coastguard Worker return KNOWN_ISSUES.get(release_or_codename, None) 116*6dbdd20aSAndroid Build Coastguard Worker 117*6dbdd20aSAndroid Build Coastguard Worker 118*6dbdd20aSAndroid Build Coastguard WorkerSDK = { 119*6dbdd20aSAndroid Build Coastguard Worker 'R': 30, 120*6dbdd20aSAndroid Build Coastguard Worker} 121*6dbdd20aSAndroid Build Coastguard Worker 122*6dbdd20aSAndroid Build Coastguard Worker 123*6dbdd20aSAndroid Build Coastguard Workerdef release_or_newer(release): 124*6dbdd20aSAndroid Build Coastguard Worker sdk = int( 125*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_output( 126*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', 'getprop', 127*6dbdd20aSAndroid Build Coastguard Worker 'ro.system.build.version.sdk']).decode('utf-8').strip()) 128*6dbdd20aSAndroid Build Coastguard Worker if sdk >= SDK[release]: 129*6dbdd20aSAndroid Build Coastguard Worker return True 130*6dbdd20aSAndroid Build Coastguard Worker codename = subprocess.check_output( 131*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', 'getprop', 132*6dbdd20aSAndroid Build Coastguard Worker 'ro.build.version.codename']).decode('utf-8').strip() 133*6dbdd20aSAndroid Build Coastguard Worker return codename == release 134*6dbdd20aSAndroid Build Coastguard Worker 135*6dbdd20aSAndroid Build Coastguard Worker 136*6dbdd20aSAndroid Build Coastguard WorkerORDER = ['-n', '-p', '-i', '-o'] 137*6dbdd20aSAndroid Build Coastguard Worker 138*6dbdd20aSAndroid Build Coastguard Worker 139*6dbdd20aSAndroid Build Coastguard Workerdef arg_order(action): 140*6dbdd20aSAndroid Build Coastguard Worker result = len(ORDER) 141*6dbdd20aSAndroid Build Coastguard Worker for opt in action.option_strings: 142*6dbdd20aSAndroid Build Coastguard Worker if opt in ORDER: 143*6dbdd20aSAndroid Build Coastguard Worker result = min(ORDER.index(opt), result) 144*6dbdd20aSAndroid Build Coastguard Worker return result, action.option_strings[0].strip('-') 145*6dbdd20aSAndroid Build Coastguard Worker 146*6dbdd20aSAndroid Build Coastguard Worker 147*6dbdd20aSAndroid Build Coastguard Workerdef print_options(parser): 148*6dbdd20aSAndroid Build Coastguard Worker for action in sorted(parser._actions, key=arg_order): 149*6dbdd20aSAndroid Build Coastguard Worker if action.help is argparse.SUPPRESS: 150*6dbdd20aSAndroid Build Coastguard Worker continue 151*6dbdd20aSAndroid Build Coastguard Worker opts = ', '.join('`' + x + '`' for x in action.option_strings) 152*6dbdd20aSAndroid Build Coastguard Worker metavar = '' if action.metavar is None else ' _' + action.metavar + '_' 153*6dbdd20aSAndroid Build Coastguard Worker print('{}{}'.format(opts, metavar)) 154*6dbdd20aSAndroid Build Coastguard Worker print(': {}'.format(action.help)) 155*6dbdd20aSAndroid Build Coastguard Worker print() 156*6dbdd20aSAndroid Build Coastguard Worker 157*6dbdd20aSAndroid Build Coastguard Worker 158*6dbdd20aSAndroid Build Coastguard Workerdef main(argv): 159*6dbdd20aSAndroid Build Coastguard Worker parser = argparse.ArgumentParser(description="""Collect a heap profile 160*6dbdd20aSAndroid Build Coastguard Worker 161*6dbdd20aSAndroid Build Coastguard Worker The PERFETTO_PROGUARD_MAP=packagename=map_filename.txt[:packagename=map_filename.txt...] environment variable can be used to pass proguard deobfuscation maps for different packages""", formatter_class=argparse.RawDescriptionHelpFormatter) 162*6dbdd20aSAndroid Build Coastguard Worker 163*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 164*6dbdd20aSAndroid Build Coastguard Worker "-i", 165*6dbdd20aSAndroid Build Coastguard Worker "--interval", 166*6dbdd20aSAndroid Build Coastguard Worker help="Sampling interval. " 167*6dbdd20aSAndroid Build Coastguard Worker "Default 4096 (4KiB)", 168*6dbdd20aSAndroid Build Coastguard Worker type=int, 169*6dbdd20aSAndroid Build Coastguard Worker default=4096) 170*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 171*6dbdd20aSAndroid Build Coastguard Worker "-d", 172*6dbdd20aSAndroid Build Coastguard Worker "--duration", 173*6dbdd20aSAndroid Build Coastguard Worker help="Duration of profile (ms). 0 to run until interrupted. " 174*6dbdd20aSAndroid Build Coastguard Worker "Default: until interrupted by user.", 175*6dbdd20aSAndroid Build Coastguard Worker type=int, 176*6dbdd20aSAndroid Build Coastguard Worker default=0) 177*6dbdd20aSAndroid Build Coastguard Worker # This flag is a no-op now. We never start heapprofd explicitly using system 178*6dbdd20aSAndroid Build Coastguard Worker # properties. 179*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 180*6dbdd20aSAndroid Build Coastguard Worker "--no-start", help="Do not start heapprofd.", action='store_true') 181*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 182*6dbdd20aSAndroid Build Coastguard Worker "-p", 183*6dbdd20aSAndroid Build Coastguard Worker "--pid", 184*6dbdd20aSAndroid Build Coastguard Worker help="Comma-separated list of PIDs to " 185*6dbdd20aSAndroid Build Coastguard Worker "profile.", 186*6dbdd20aSAndroid Build Coastguard Worker metavar="PIDS") 187*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 188*6dbdd20aSAndroid Build Coastguard Worker "-n", 189*6dbdd20aSAndroid Build Coastguard Worker "--name", 190*6dbdd20aSAndroid Build Coastguard Worker help="Comma-separated list of process " 191*6dbdd20aSAndroid Build Coastguard Worker "names to profile.", 192*6dbdd20aSAndroid Build Coastguard Worker metavar="NAMES") 193*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 194*6dbdd20aSAndroid Build Coastguard Worker "-c", 195*6dbdd20aSAndroid Build Coastguard Worker "--continuous-dump", 196*6dbdd20aSAndroid Build Coastguard Worker help="Dump interval in ms. 0 to disable continuous dump.", 197*6dbdd20aSAndroid Build Coastguard Worker type=int, 198*6dbdd20aSAndroid Build Coastguard Worker default=0) 199*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 200*6dbdd20aSAndroid Build Coastguard Worker "--heaps", 201*6dbdd20aSAndroid Build Coastguard Worker help="Comma-separated list of heaps to collect, e.g: malloc,art. " 202*6dbdd20aSAndroid Build Coastguard Worker "Requires Android 12.", 203*6dbdd20aSAndroid Build Coastguard Worker metavar="HEAPS") 204*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 205*6dbdd20aSAndroid Build Coastguard Worker "--all-heaps", 206*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 207*6dbdd20aSAndroid Build Coastguard Worker help="Collect allocations from all heaps registered by target.") 208*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 209*6dbdd20aSAndroid Build Coastguard Worker "--no-android-tree-symbolization", 210*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 211*6dbdd20aSAndroid Build Coastguard Worker help="Do not symbolize using currently lunched target in the " 212*6dbdd20aSAndroid Build Coastguard Worker "Android tree.") 213*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 214*6dbdd20aSAndroid Build Coastguard Worker "--disable-selinux", 215*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 216*6dbdd20aSAndroid Build Coastguard Worker help="Disable SELinux enforcement for duration of " 217*6dbdd20aSAndroid Build Coastguard Worker "profile.") 218*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 219*6dbdd20aSAndroid Build Coastguard Worker "--no-versions", 220*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 221*6dbdd20aSAndroid Build Coastguard Worker help="Do not get version information about APKs.") 222*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 223*6dbdd20aSAndroid Build Coastguard Worker "--no-running", 224*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 225*6dbdd20aSAndroid Build Coastguard Worker help="Do not target already running processes. Requires Android 11.") 226*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 227*6dbdd20aSAndroid Build Coastguard Worker "--no-startup", 228*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 229*6dbdd20aSAndroid Build Coastguard Worker help="Do not target processes that start during " 230*6dbdd20aSAndroid Build Coastguard Worker "the profile. Requires Android 11.") 231*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 232*6dbdd20aSAndroid Build Coastguard Worker "--shmem-size", 233*6dbdd20aSAndroid Build Coastguard Worker help="Size of buffer between client and " 234*6dbdd20aSAndroid Build Coastguard Worker "heapprofd. Default 8MiB. Needs to be a power of two " 235*6dbdd20aSAndroid Build Coastguard Worker "multiple of 4096, at least 8192.", 236*6dbdd20aSAndroid Build Coastguard Worker type=int, 237*6dbdd20aSAndroid Build Coastguard Worker default=8 * 1048576) 238*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 239*6dbdd20aSAndroid Build Coastguard Worker "--block-client", 240*6dbdd20aSAndroid Build Coastguard Worker help="When buffer is full, block the " 241*6dbdd20aSAndroid Build Coastguard Worker "client to wait for buffer space. Use with caution as " 242*6dbdd20aSAndroid Build Coastguard Worker "this can significantly slow down the client. " 243*6dbdd20aSAndroid Build Coastguard Worker "This is the default", 244*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 245*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 246*6dbdd20aSAndroid Build Coastguard Worker "--block-client-timeout", 247*6dbdd20aSAndroid Build Coastguard Worker help="If --block-client is given, do not block any allocation for " 248*6dbdd20aSAndroid Build Coastguard Worker "longer than this timeout (us).", 249*6dbdd20aSAndroid Build Coastguard Worker type=int) 250*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 251*6dbdd20aSAndroid Build Coastguard Worker "--no-block-client", 252*6dbdd20aSAndroid Build Coastguard Worker help="When buffer is full, stop the " 253*6dbdd20aSAndroid Build Coastguard Worker "profile early.", 254*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 255*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 256*6dbdd20aSAndroid Build Coastguard Worker "--idle-allocations", 257*6dbdd20aSAndroid Build Coastguard Worker help="Keep track of how many " 258*6dbdd20aSAndroid Build Coastguard Worker "bytes were unused since the last dump, per " 259*6dbdd20aSAndroid Build Coastguard Worker "callstack", 260*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 261*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 262*6dbdd20aSAndroid Build Coastguard Worker "--dump-at-max", 263*6dbdd20aSAndroid Build Coastguard Worker help="Dump the maximum memory usage " 264*6dbdd20aSAndroid Build Coastguard Worker "rather than at the time of the dump.", 265*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 266*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 267*6dbdd20aSAndroid Build Coastguard Worker "--disable-fork-teardown", 268*6dbdd20aSAndroid Build Coastguard Worker help="Do not tear down client in forks. This can be useful for programs " 269*6dbdd20aSAndroid Build Coastguard Worker "that use vfork. Android 11+ only.", 270*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 271*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 272*6dbdd20aSAndroid Build Coastguard Worker "--simpleperf", 273*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 274*6dbdd20aSAndroid Build Coastguard Worker help="Get simpleperf profile of heapprofd. This is " 275*6dbdd20aSAndroid Build Coastguard Worker "only for heapprofd development.") 276*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 277*6dbdd20aSAndroid Build Coastguard Worker "--traceconv-binary", help="Path to local trace to text. For debugging.") 278*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 279*6dbdd20aSAndroid Build Coastguard Worker "--no-annotations", 280*6dbdd20aSAndroid Build Coastguard Worker help="Do not suffix the pprof function names with Android ART mode " 281*6dbdd20aSAndroid Build Coastguard Worker "annotations such as [jit].", 282*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 283*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 284*6dbdd20aSAndroid Build Coastguard Worker "--print-config", 285*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 286*6dbdd20aSAndroid Build Coastguard Worker help="Print config instead of running. For debugging.") 287*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 288*6dbdd20aSAndroid Build Coastguard Worker "-o", 289*6dbdd20aSAndroid Build Coastguard Worker "--output", 290*6dbdd20aSAndroid Build Coastguard Worker help="Output directory.", 291*6dbdd20aSAndroid Build Coastguard Worker metavar="DIRECTORY", 292*6dbdd20aSAndroid Build Coastguard Worker default=None) 293*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 294*6dbdd20aSAndroid Build Coastguard Worker "--print-options", action="store_true", help=argparse.SUPPRESS) 295*6dbdd20aSAndroid Build Coastguard Worker 296*6dbdd20aSAndroid Build Coastguard Worker args = parser.parse_args() 297*6dbdd20aSAndroid Build Coastguard Worker if args.print_options: 298*6dbdd20aSAndroid Build Coastguard Worker print_options(parser) 299*6dbdd20aSAndroid Build Coastguard Worker return 0 300*6dbdd20aSAndroid Build Coastguard Worker fail = False 301*6dbdd20aSAndroid Build Coastguard Worker if args.block_client and args.no_block_client: 302*6dbdd20aSAndroid Build Coastguard Worker print( 303*6dbdd20aSAndroid Build Coastguard Worker "FATAL: Both block-client and no-block-client given.", file=sys.stderr) 304*6dbdd20aSAndroid Build Coastguard Worker fail = True 305*6dbdd20aSAndroid Build Coastguard Worker if args.pid is None and args.name is None: 306*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: Neither PID nor NAME given.", file=sys.stderr) 307*6dbdd20aSAndroid Build Coastguard Worker fail = True 308*6dbdd20aSAndroid Build Coastguard Worker if args.duration is None: 309*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: No duration given.", file=sys.stderr) 310*6dbdd20aSAndroid Build Coastguard Worker fail = True 311*6dbdd20aSAndroid Build Coastguard Worker if args.interval is None: 312*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: No interval given.", file=sys.stderr) 313*6dbdd20aSAndroid Build Coastguard Worker fail = True 314*6dbdd20aSAndroid Build Coastguard Worker if args.shmem_size % 4096: 315*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: shmem-size is not a multiple of 4096.", file=sys.stderr) 316*6dbdd20aSAndroid Build Coastguard Worker fail = True 317*6dbdd20aSAndroid Build Coastguard Worker if args.shmem_size < 8192: 318*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: shmem-size is less than 8192.", file=sys.stderr) 319*6dbdd20aSAndroid Build Coastguard Worker fail = True 320*6dbdd20aSAndroid Build Coastguard Worker if args.shmem_size & (args.shmem_size - 1): 321*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: shmem-size is not a power of two.", file=sys.stderr) 322*6dbdd20aSAndroid Build Coastguard Worker fail = True 323*6dbdd20aSAndroid Build Coastguard Worker 324*6dbdd20aSAndroid Build Coastguard Worker target_cfg = "" 325*6dbdd20aSAndroid Build Coastguard Worker if not args.no_block_client: 326*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "block_client: true\n" 327*6dbdd20aSAndroid Build Coastguard Worker if args.block_client_timeout: 328*6dbdd20aSAndroid Build Coastguard Worker target_cfg += ( 329*6dbdd20aSAndroid Build Coastguard Worker CFG_INDENT + 330*6dbdd20aSAndroid Build Coastguard Worker "block_client_timeout_us: %s\n" % args.block_client_timeout) 331*6dbdd20aSAndroid Build Coastguard Worker if args.no_startup: 332*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "no_startup: true\n" 333*6dbdd20aSAndroid Build Coastguard Worker if args.no_running: 334*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "no_running: true\n" 335*6dbdd20aSAndroid Build Coastguard Worker if args.dump_at_max: 336*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "dump_at_max: true\n" 337*6dbdd20aSAndroid Build Coastguard Worker if args.disable_fork_teardown: 338*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "disable_fork_teardown: true\n" 339*6dbdd20aSAndroid Build Coastguard Worker if args.all_heaps: 340*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "all_heaps: true\n" 341*6dbdd20aSAndroid Build Coastguard Worker if args.pid: 342*6dbdd20aSAndroid Build Coastguard Worker for pid in args.pid.split(','): 343*6dbdd20aSAndroid Build Coastguard Worker try: 344*6dbdd20aSAndroid Build Coastguard Worker pid = int(pid) 345*6dbdd20aSAndroid Build Coastguard Worker except ValueError: 346*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: invalid PID %s" % pid, file=sys.stderr) 347*6dbdd20aSAndroid Build Coastguard Worker fail = True 348*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + 'pid: {}\n'.format(pid) 349*6dbdd20aSAndroid Build Coastguard Worker if args.name: 350*6dbdd20aSAndroid Build Coastguard Worker for name in args.name.split(','): 351*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + 'process_cmdline: "{}"\n'.format(name) 352*6dbdd20aSAndroid Build Coastguard Worker if args.heaps: 353*6dbdd20aSAndroid Build Coastguard Worker for heap in args.heaps.split(','): 354*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + 'heaps: "{}"\n'.format(heap) 355*6dbdd20aSAndroid Build Coastguard Worker 356*6dbdd20aSAndroid Build Coastguard Worker if fail: 357*6dbdd20aSAndroid Build Coastguard Worker parser.print_help() 358*6dbdd20aSAndroid Build Coastguard Worker return 1 359*6dbdd20aSAndroid Build Coastguard Worker 360*6dbdd20aSAndroid Build Coastguard Worker traceconv_binary = args.traceconv_binary 361*6dbdd20aSAndroid Build Coastguard Worker 362*6dbdd20aSAndroid Build Coastguard Worker if args.continuous_dump: 363*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CONTINUOUS_DUMP.format(dump_interval=args.continuous_dump) 364*6dbdd20aSAndroid Build Coastguard Worker cfg = CFG.format( 365*6dbdd20aSAndroid Build Coastguard Worker interval=args.interval, 366*6dbdd20aSAndroid Build Coastguard Worker duration=args.duration, 367*6dbdd20aSAndroid Build Coastguard Worker target_cfg=target_cfg, 368*6dbdd20aSAndroid Build Coastguard Worker shmem_size=args.shmem_size) 369*6dbdd20aSAndroid Build Coastguard Worker if not args.no_versions: 370*6dbdd20aSAndroid Build Coastguard Worker cfg += PACKAGES_LIST_CFG 371*6dbdd20aSAndroid Build Coastguard Worker 372*6dbdd20aSAndroid Build Coastguard Worker if args.print_config: 373*6dbdd20aSAndroid Build Coastguard Worker print(cfg) 374*6dbdd20aSAndroid Build Coastguard Worker return 0 375*6dbdd20aSAndroid Build Coastguard Worker 376*6dbdd20aSAndroid Build Coastguard Worker # Do this AFTER print_config so we do not download traceconv only to 377*6dbdd20aSAndroid Build Coastguard Worker # print out the config. 378*6dbdd20aSAndroid Build Coastguard Worker if traceconv_binary is None: 379*6dbdd20aSAndroid Build Coastguard Worker traceconv_binary = get_perfetto_prebuilt(TRACECONV_MANIFEST, soft_fail=True) 380*6dbdd20aSAndroid Build Coastguard Worker 381*6dbdd20aSAndroid Build Coastguard Worker known_issues = maybe_known_issues() 382*6dbdd20aSAndroid Build Coastguard Worker if known_issues: 383*6dbdd20aSAndroid Build Coastguard Worker print('If you are experiencing problems, please see the known issues for ' 384*6dbdd20aSAndroid Build Coastguard Worker 'your release: {}.'.format(known_issues)) 385*6dbdd20aSAndroid Build Coastguard Worker 386*6dbdd20aSAndroid Build Coastguard Worker # TODO(fmayer): Maybe feature detect whether we can remove traces instead of 387*6dbdd20aSAndroid Build Coastguard Worker # this. 388*6dbdd20aSAndroid Build Coastguard Worker uuid_trace = release_or_newer('R') 389*6dbdd20aSAndroid Build Coastguard Worker if uuid_trace: 390*6dbdd20aSAndroid Build Coastguard Worker profile_device_path = '/data/misc/perfetto-traces/profile-' + UUID 391*6dbdd20aSAndroid Build Coastguard Worker else: 392*6dbdd20aSAndroid Build Coastguard Worker user = subprocess.check_output(['adb', 'shell', 393*6dbdd20aSAndroid Build Coastguard Worker 'whoami']).decode('utf-8').strip() 394*6dbdd20aSAndroid Build Coastguard Worker profile_device_path = '/data/misc/perfetto-traces/profile-' + user 395*6dbdd20aSAndroid Build Coastguard Worker 396*6dbdd20aSAndroid Build Coastguard Worker perfetto_cmd = ('CFG=\'{cfg}\'; echo ${{CFG}} | ' 397*6dbdd20aSAndroid Build Coastguard Worker 'perfetto --txt -c - -o ' + profile_device_path + ' -d') 398*6dbdd20aSAndroid Build Coastguard Worker 399*6dbdd20aSAndroid Build Coastguard Worker if args.disable_selinux: 400*6dbdd20aSAndroid Build Coastguard Worker enforcing = subprocess.check_output(['adb', 'shell', 401*6dbdd20aSAndroid Build Coastguard Worker 'getenforce']).decode('utf-8').strip() 402*6dbdd20aSAndroid Build Coastguard Worker atexit.register( 403*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call, 404*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', 'su root setenforce %s' % enforcing]) 405*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['adb', 'shell', 'su root setenforce 0']) 406*6dbdd20aSAndroid Build Coastguard Worker 407*6dbdd20aSAndroid Build Coastguard Worker if args.simpleperf: 408*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call([ 409*6dbdd20aSAndroid Build Coastguard Worker 'adb', 'shell', 'mkdir -p /data/local/tmp/heapprofd_profile && ' 410*6dbdd20aSAndroid Build Coastguard Worker 'cd /data/local/tmp/heapprofd_profile &&' 411*6dbdd20aSAndroid Build Coastguard Worker '(nohup simpleperf record -g -p $(pidof heapprofd) 2>&1 &) ' 412*6dbdd20aSAndroid Build Coastguard Worker '> /dev/null' 413*6dbdd20aSAndroid Build Coastguard Worker ]) 414*6dbdd20aSAndroid Build Coastguard Worker 415*6dbdd20aSAndroid Build Coastguard Worker profile_target = PROFILE_LOCAL_PATH 416*6dbdd20aSAndroid Build Coastguard Worker if args.output is not None: 417*6dbdd20aSAndroid Build Coastguard Worker profile_target = args.output 418*6dbdd20aSAndroid Build Coastguard Worker else: 419*6dbdd20aSAndroid Build Coastguard Worker os.mkdir(profile_target) 420*6dbdd20aSAndroid Build Coastguard Worker 421*6dbdd20aSAndroid Build Coastguard Worker if not os.path.isdir(profile_target): 422*6dbdd20aSAndroid Build Coastguard Worker print( 423*6dbdd20aSAndroid Build Coastguard Worker "Output directory {} not found".format(profile_target), file=sys.stderr) 424*6dbdd20aSAndroid Build Coastguard Worker return 1 425*6dbdd20aSAndroid Build Coastguard Worker 426*6dbdd20aSAndroid Build Coastguard Worker if os.listdir(profile_target): 427*6dbdd20aSAndroid Build Coastguard Worker print( 428*6dbdd20aSAndroid Build Coastguard Worker "Output directory {} not empty".format(profile_target), file=sys.stderr) 429*6dbdd20aSAndroid Build Coastguard Worker return 1 430*6dbdd20aSAndroid Build Coastguard Worker 431*6dbdd20aSAndroid Build Coastguard Worker perfetto_pid = subprocess.check_output( 432*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'exec-out', perfetto_cmd.format(cfg=cfg)]).strip() 433*6dbdd20aSAndroid Build Coastguard Worker try: 434*6dbdd20aSAndroid Build Coastguard Worker perfetto_pid = int(perfetto_pid.strip()) 435*6dbdd20aSAndroid Build Coastguard Worker except ValueError: 436*6dbdd20aSAndroid Build Coastguard Worker print("Failed to invoke perfetto: {}".format(perfetto_pid), file=sys.stderr) 437*6dbdd20aSAndroid Build Coastguard Worker return 1 438*6dbdd20aSAndroid Build Coastguard Worker 439*6dbdd20aSAndroid Build Coastguard Worker old_handler = signal.signal(signal.SIGINT, sigint_handler) 440*6dbdd20aSAndroid Build Coastguard Worker print("Profiling active. Press Ctrl+C to terminate.") 441*6dbdd20aSAndroid Build Coastguard Worker print("You may disconnect your device.") 442*6dbdd20aSAndroid Build Coastguard Worker print() 443*6dbdd20aSAndroid Build Coastguard Worker exists = True 444*6dbdd20aSAndroid Build Coastguard Worker device_connected = True 445*6dbdd20aSAndroid Build Coastguard Worker while not device_connected or (exists and not IS_INTERRUPTED): 446*6dbdd20aSAndroid Build Coastguard Worker exists = subprocess.call( 447*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', '[ -d /proc/{} ]'.format(perfetto_pid)], **NOOUT) == 0 448*6dbdd20aSAndroid Build Coastguard Worker device_connected = subprocess.call(['adb', 'shell', 'true'], **NOOUT) == 0 449*6dbdd20aSAndroid Build Coastguard Worker time.sleep(1) 450*6dbdd20aSAndroid Build Coastguard Worker print("Waiting for profiler shutdown...") 451*6dbdd20aSAndroid Build Coastguard Worker signal.signal(signal.SIGINT, old_handler) 452*6dbdd20aSAndroid Build Coastguard Worker if IS_INTERRUPTED: 453*6dbdd20aSAndroid Build Coastguard Worker # Not check_call because it could have existed in the meantime. 454*6dbdd20aSAndroid Build Coastguard Worker subprocess.call(['adb', 'shell', 'kill', '-INT', str(perfetto_pid)]) 455*6dbdd20aSAndroid Build Coastguard Worker if args.simpleperf: 456*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['adb', 'shell', 'killall', '-INT', 'simpleperf']) 457*6dbdd20aSAndroid Build Coastguard Worker print("Waiting for simpleperf to exit.") 458*6dbdd20aSAndroid Build Coastguard Worker while subprocess.call( 459*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', '[ -f /proc/$(pidof simpleperf)/exe ]'], **NOOUT) == 0: 460*6dbdd20aSAndroid Build Coastguard Worker time.sleep(1) 461*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call( 462*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'pull', '/data/local/tmp/heapprofd_profile', profile_target]) 463*6dbdd20aSAndroid Build Coastguard Worker print("Pulled simpleperf profile to " + profile_target + 464*6dbdd20aSAndroid Build Coastguard Worker "/heapprofd_profile") 465*6dbdd20aSAndroid Build Coastguard Worker 466*6dbdd20aSAndroid Build Coastguard Worker # Wait for perfetto cmd to return. 467*6dbdd20aSAndroid Build Coastguard Worker while exists: 468*6dbdd20aSAndroid Build Coastguard Worker exists = subprocess.call( 469*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', '[ -d /proc/{} ]'.format(perfetto_pid)]) == 0 470*6dbdd20aSAndroid Build Coastguard Worker time.sleep(1) 471*6dbdd20aSAndroid Build Coastguard Worker 472*6dbdd20aSAndroid Build Coastguard Worker profile_host_path = os.path.join(profile_target, 'raw-trace') 473*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['adb', 'pull', profile_device_path, profile_host_path], 474*6dbdd20aSAndroid Build Coastguard Worker stdout=NULL) 475*6dbdd20aSAndroid Build Coastguard Worker if uuid_trace: 476*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['adb', 'shell', 'rm', profile_device_path], 477*6dbdd20aSAndroid Build Coastguard Worker stdout=NULL) 478*6dbdd20aSAndroid Build Coastguard Worker 479*6dbdd20aSAndroid Build Coastguard Worker if traceconv_binary is None: 480*6dbdd20aSAndroid Build Coastguard Worker print('Wrote profile to {}'.format(profile_host_path)) 481*6dbdd20aSAndroid Build Coastguard Worker print( 482*6dbdd20aSAndroid Build Coastguard Worker 'This file can be opened using the Perfetto UI, https://ui.perfetto.dev' 483*6dbdd20aSAndroid Build Coastguard Worker ) 484*6dbdd20aSAndroid Build Coastguard Worker return 0 485*6dbdd20aSAndroid Build Coastguard Worker 486*6dbdd20aSAndroid Build Coastguard Worker binary_path = os.getenv('PERFETTO_BINARY_PATH') 487*6dbdd20aSAndroid Build Coastguard Worker if not args.no_android_tree_symbolization: 488*6dbdd20aSAndroid Build Coastguard Worker product_out = os.getenv('ANDROID_PRODUCT_OUT') 489*6dbdd20aSAndroid Build Coastguard Worker if product_out: 490*6dbdd20aSAndroid Build Coastguard Worker product_out_symbols = product_out + '/symbols' 491*6dbdd20aSAndroid Build Coastguard Worker else: 492*6dbdd20aSAndroid Build Coastguard Worker product_out_symbols = None 493*6dbdd20aSAndroid Build Coastguard Worker 494*6dbdd20aSAndroid Build Coastguard Worker if binary_path is None: 495*6dbdd20aSAndroid Build Coastguard Worker binary_path = product_out_symbols 496*6dbdd20aSAndroid Build Coastguard Worker elif product_out_symbols is not None: 497*6dbdd20aSAndroid Build Coastguard Worker binary_path += os.pathsep + product_out_symbols 498*6dbdd20aSAndroid Build Coastguard Worker 499*6dbdd20aSAndroid Build Coastguard Worker trace_file = os.path.join(profile_target, 'raw-trace') 500*6dbdd20aSAndroid Build Coastguard Worker concat_files = [trace_file] 501*6dbdd20aSAndroid Build Coastguard Worker 502*6dbdd20aSAndroid Build Coastguard Worker if binary_path is not None: 503*6dbdd20aSAndroid Build Coastguard Worker with open(os.path.join(profile_target, 'symbols'), 'w') as fd: 504*6dbdd20aSAndroid Build Coastguard Worker ret = subprocess.call([ 505*6dbdd20aSAndroid Build Coastguard Worker traceconv_binary, 'symbolize', 506*6dbdd20aSAndroid Build Coastguard Worker os.path.join(profile_target, 'raw-trace') 507*6dbdd20aSAndroid Build Coastguard Worker ], 508*6dbdd20aSAndroid Build Coastguard Worker env=dict( 509*6dbdd20aSAndroid Build Coastguard Worker os.environ, PERFETTO_BINARY_PATH=binary_path), 510*6dbdd20aSAndroid Build Coastguard Worker stdout=fd) 511*6dbdd20aSAndroid Build Coastguard Worker if ret == 0: 512*6dbdd20aSAndroid Build Coastguard Worker concat_files.append(os.path.join(profile_target, 'symbols')) 513*6dbdd20aSAndroid Build Coastguard Worker else: 514*6dbdd20aSAndroid Build Coastguard Worker print("Failed to symbolize. Continuing without symbols.", file=sys.stderr) 515*6dbdd20aSAndroid Build Coastguard Worker 516*6dbdd20aSAndroid Build Coastguard Worker proguard_map = os.getenv('PERFETTO_PROGUARD_MAP') 517*6dbdd20aSAndroid Build Coastguard Worker if proguard_map is not None: 518*6dbdd20aSAndroid Build Coastguard Worker with open(os.path.join(profile_target, 'deobfuscation-packets'), 'w') as fd: 519*6dbdd20aSAndroid Build Coastguard Worker ret = subprocess.call([ 520*6dbdd20aSAndroid Build Coastguard Worker traceconv_binary, 'deobfuscate', 521*6dbdd20aSAndroid Build Coastguard Worker os.path.join(profile_target, 'raw-trace') 522*6dbdd20aSAndroid Build Coastguard Worker ], 523*6dbdd20aSAndroid Build Coastguard Worker env=dict( 524*6dbdd20aSAndroid Build Coastguard Worker os.environ, PERFETTO_PROGUARD_MAP=proguard_map), 525*6dbdd20aSAndroid Build Coastguard Worker stdout=fd) 526*6dbdd20aSAndroid Build Coastguard Worker if ret == 0: 527*6dbdd20aSAndroid Build Coastguard Worker concat_files.append(os.path.join(profile_target, 'deobfuscation-packets')) 528*6dbdd20aSAndroid Build Coastguard Worker else: 529*6dbdd20aSAndroid Build Coastguard Worker print( 530*6dbdd20aSAndroid Build Coastguard Worker "Failed to deobfuscate. Continuing without deobfuscated.", 531*6dbdd20aSAndroid Build Coastguard Worker file=sys.stderr) 532*6dbdd20aSAndroid Build Coastguard Worker 533*6dbdd20aSAndroid Build Coastguard Worker if len(concat_files) > 1: 534*6dbdd20aSAndroid Build Coastguard Worker with open(os.path.join(profile_target, 'symbolized-trace'), 'wb') as out: 535*6dbdd20aSAndroid Build Coastguard Worker for fn in concat_files: 536*6dbdd20aSAndroid Build Coastguard Worker with open(fn, 'rb') as inp: 537*6dbdd20aSAndroid Build Coastguard Worker while True: 538*6dbdd20aSAndroid Build Coastguard Worker buf = inp.read(4096) 539*6dbdd20aSAndroid Build Coastguard Worker if not buf: 540*6dbdd20aSAndroid Build Coastguard Worker break 541*6dbdd20aSAndroid Build Coastguard Worker out.write(buf) 542*6dbdd20aSAndroid Build Coastguard Worker trace_file = os.path.join(profile_target, 'symbolized-trace') 543*6dbdd20aSAndroid Build Coastguard Worker 544*6dbdd20aSAndroid Build Coastguard Worker conversion_args = [traceconv_binary, 'profile'] + ( 545*6dbdd20aSAndroid Build Coastguard Worker ['--no-annotations'] if args.no_annotations else []) + [trace_file] 546*6dbdd20aSAndroid Build Coastguard Worker traceconv_output = subprocess.check_output(conversion_args) 547*6dbdd20aSAndroid Build Coastguard Worker profile_path = None 548*6dbdd20aSAndroid Build Coastguard Worker for word in traceconv_output.decode('utf-8').split(): 549*6dbdd20aSAndroid Build Coastguard Worker if 'heap_profile-' in word: 550*6dbdd20aSAndroid Build Coastguard Worker profile_path = word 551*6dbdd20aSAndroid Build Coastguard Worker if profile_path is None: 552*6dbdd20aSAndroid Build Coastguard Worker print_no_profile_error() 553*6dbdd20aSAndroid Build Coastguard Worker return 1 554*6dbdd20aSAndroid Build Coastguard Worker 555*6dbdd20aSAndroid Build Coastguard Worker profile_files = os.listdir(profile_path) 556*6dbdd20aSAndroid Build Coastguard Worker if not profile_files: 557*6dbdd20aSAndroid Build Coastguard Worker print_no_profile_error() 558*6dbdd20aSAndroid Build Coastguard Worker return 1 559*6dbdd20aSAndroid Build Coastguard Worker 560*6dbdd20aSAndroid Build Coastguard Worker for profile_file in profile_files: 561*6dbdd20aSAndroid Build Coastguard Worker shutil.copy(os.path.join(profile_path, profile_file), profile_target) 562*6dbdd20aSAndroid Build Coastguard Worker 563*6dbdd20aSAndroid Build Coastguard Worker symlink_path = None 564*6dbdd20aSAndroid Build Coastguard Worker if not sys.platform.startswith('win'): 565*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call( 566*6dbdd20aSAndroid Build Coastguard Worker ['gzip'] + [os.path.join(profile_target, x) for x in profile_files]) 567*6dbdd20aSAndroid Build Coastguard Worker if args.output is None: 568*6dbdd20aSAndroid Build Coastguard Worker symlink_path = os.path.join( 569*6dbdd20aSAndroid Build Coastguard Worker os.path.dirname(profile_target), "heap_profile-latest") 570*6dbdd20aSAndroid Build Coastguard Worker if os.path.lexists(symlink_path): 571*6dbdd20aSAndroid Build Coastguard Worker os.unlink(symlink_path) 572*6dbdd20aSAndroid Build Coastguard Worker os.symlink(profile_target, symlink_path) 573*6dbdd20aSAndroid Build Coastguard Worker 574*6dbdd20aSAndroid Build Coastguard Worker if symlink_path is not None: 575*6dbdd20aSAndroid Build Coastguard Worker print("Wrote profiles to {} (symlink {})".format(profile_target, 576*6dbdd20aSAndroid Build Coastguard Worker symlink_path)) 577*6dbdd20aSAndroid Build Coastguard Worker else: 578*6dbdd20aSAndroid Build Coastguard Worker print("Wrote profiles to {}".format(profile_target)) 579*6dbdd20aSAndroid Build Coastguard Worker 580*6dbdd20aSAndroid Build Coastguard Worker print("The raw-trace file can be viewed using https://ui.perfetto.dev.") 581*6dbdd20aSAndroid Build Coastguard Worker print("The heap_dump.* files can be viewed using pprof/ (Googlers only) " + 582*6dbdd20aSAndroid Build Coastguard Worker "or https://www.speedscope.app/.") 583*6dbdd20aSAndroid Build Coastguard Worker print("The two above are equivalent. The raw-trace contains the union of " + 584*6dbdd20aSAndroid Build Coastguard Worker "all the heap dumps.") 585*6dbdd20aSAndroid Build Coastguard Worker 586*6dbdd20aSAndroid Build Coastguard Worker 587*6dbdd20aSAndroid Build Coastguard Workerif __name__ == '__main__': 588*6dbdd20aSAndroid Build Coastguard Worker sys.exit(main(sys.argv)) 589