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 Worker# !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! 17*6dbdd20aSAndroid Build Coastguard Worker# DO NOT EDIT. Auto-generated by tools/gen_amalgamated_python_tools 18*6dbdd20aSAndroid Build Coastguard Worker# !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! 19*6dbdd20aSAndroid Build Coastguard Worker 20*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import absolute_import 21*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import division 22*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import print_function 23*6dbdd20aSAndroid Build Coastguard Worker 24*6dbdd20aSAndroid Build Coastguard Workerimport argparse 25*6dbdd20aSAndroid Build Coastguard Workerimport atexit 26*6dbdd20aSAndroid Build Coastguard Workerimport os 27*6dbdd20aSAndroid Build Coastguard Workerimport shutil 28*6dbdd20aSAndroid Build Coastguard Workerimport signal 29*6dbdd20aSAndroid Build Coastguard Workerimport subprocess 30*6dbdd20aSAndroid Build Coastguard Workerimport sys 31*6dbdd20aSAndroid Build Coastguard Workerimport tempfile 32*6dbdd20aSAndroid Build Coastguard Workerimport time 33*6dbdd20aSAndroid Build Coastguard Workerimport uuid 34*6dbdd20aSAndroid Build Coastguard Worker 35*6dbdd20aSAndroid Build Coastguard Worker 36*6dbdd20aSAndroid Build Coastguard Worker# ----- Amalgamator: begin of python/perfetto/prebuilts/manifests/traceconv.py 37*6dbdd20aSAndroid Build Coastguard Worker# This file has been generated by: tools/roll-prebuilts v48.1 38*6dbdd20aSAndroid Build Coastguard WorkerTRACECONV_MANIFEST = [{ 39*6dbdd20aSAndroid Build Coastguard Worker 'arch': 40*6dbdd20aSAndroid Build Coastguard Worker 'mac-amd64', 41*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 42*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 43*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 44*6dbdd20aSAndroid Build Coastguard Worker 9041560, 45*6dbdd20aSAndroid Build Coastguard Worker 'url': 46*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/mac-amd64/traceconv', 47*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 48*6dbdd20aSAndroid Build Coastguard Worker 'cec2da5cb771a4812d0b2d15604d5023954d28e0af12e87313da2ab70d26b970', 49*6dbdd20aSAndroid Build Coastguard Worker 'platform': 50*6dbdd20aSAndroid Build Coastguard Worker 'darwin', 51*6dbdd20aSAndroid Build Coastguard Worker 'machine': ['x86_64'] 52*6dbdd20aSAndroid Build Coastguard Worker}, { 53*6dbdd20aSAndroid Build Coastguard Worker 'arch': 54*6dbdd20aSAndroid Build Coastguard Worker 'mac-arm64', 55*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 56*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 57*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 58*6dbdd20aSAndroid Build Coastguard Worker 8375512, 59*6dbdd20aSAndroid Build Coastguard Worker 'url': 60*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/mac-arm64/traceconv', 61*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 62*6dbdd20aSAndroid Build Coastguard Worker '64e200a58ea9c9f366e1071dd274d0023d1fd14043f75dbba3fe0cc138ff5fc7', 63*6dbdd20aSAndroid Build Coastguard Worker 'platform': 64*6dbdd20aSAndroid Build Coastguard Worker 'darwin', 65*6dbdd20aSAndroid Build Coastguard Worker 'machine': ['arm64'] 66*6dbdd20aSAndroid Build Coastguard Worker}, { 67*6dbdd20aSAndroid Build Coastguard Worker 'arch': 68*6dbdd20aSAndroid Build Coastguard Worker 'linux-amd64', 69*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 70*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 71*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 72*6dbdd20aSAndroid Build Coastguard Worker 9134136, 73*6dbdd20aSAndroid Build Coastguard Worker 'url': 74*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/linux-amd64/traceconv', 75*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 76*6dbdd20aSAndroid Build Coastguard Worker '87b87e1778367c1e3b99fc77439a28b4911125d2751f9909fd1b51f6bd60b6f4', 77*6dbdd20aSAndroid Build Coastguard Worker 'platform': 78*6dbdd20aSAndroid Build Coastguard Worker 'linux', 79*6dbdd20aSAndroid Build Coastguard Worker 'machine': ['x86_64'] 80*6dbdd20aSAndroid Build Coastguard Worker}, { 81*6dbdd20aSAndroid Build Coastguard Worker 'arch': 82*6dbdd20aSAndroid Build Coastguard Worker 'linux-arm', 83*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 84*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 85*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 86*6dbdd20aSAndroid Build Coastguard Worker 6753020, 87*6dbdd20aSAndroid Build Coastguard Worker 'url': 88*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/linux-arm/traceconv', 89*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 90*6dbdd20aSAndroid Build Coastguard Worker '804c4e13aca5798731056952d9cb0c6ee58795c03477c69514ccd39703060812', 91*6dbdd20aSAndroid Build Coastguard Worker 'platform': 92*6dbdd20aSAndroid Build Coastguard Worker 'linux', 93*6dbdd20aSAndroid Build Coastguard Worker 'machine': ['armv6l', 'armv7l', 'armv8l'] 94*6dbdd20aSAndroid Build Coastguard Worker}, { 95*6dbdd20aSAndroid Build Coastguard Worker 'arch': 96*6dbdd20aSAndroid Build Coastguard Worker 'linux-arm64', 97*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 98*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 99*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 100*6dbdd20aSAndroid Build Coastguard Worker 8740064, 101*6dbdd20aSAndroid Build Coastguard Worker 'url': 102*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/linux-arm64/traceconv', 103*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 104*6dbdd20aSAndroid Build Coastguard Worker '0d781886531d11e1d573a1ec5e06376ef139bb479eec38c16c8735821c35b895', 105*6dbdd20aSAndroid Build Coastguard Worker 'platform': 106*6dbdd20aSAndroid Build Coastguard Worker 'linux', 107*6dbdd20aSAndroid Build Coastguard Worker 'machine': ['aarch64'] 108*6dbdd20aSAndroid Build Coastguard Worker}, { 109*6dbdd20aSAndroid Build Coastguard Worker 'arch': 110*6dbdd20aSAndroid Build Coastguard Worker 'android-arm', 111*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 112*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 113*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 114*6dbdd20aSAndroid Build Coastguard Worker 6792280, 115*6dbdd20aSAndroid Build Coastguard Worker 'url': 116*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/android-arm/traceconv', 117*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 118*6dbdd20aSAndroid Build Coastguard Worker '7d91e4133184a3722a25488edd3692c5a195148eba56621014311d3f85d3fc15' 119*6dbdd20aSAndroid Build Coastguard Worker}, { 120*6dbdd20aSAndroid Build Coastguard Worker 'arch': 121*6dbdd20aSAndroid Build Coastguard Worker 'android-arm64', 122*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 123*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 124*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 125*6dbdd20aSAndroid Build Coastguard Worker 8677992, 126*6dbdd20aSAndroid Build Coastguard Worker 'url': 127*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/android-arm64/traceconv', 128*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 129*6dbdd20aSAndroid Build Coastguard Worker 'c03c4a901ed23f1e20a12c98ce4556353a62bddcd260fb4d797cd29ff6c49a05' 130*6dbdd20aSAndroid Build Coastguard Worker}, { 131*6dbdd20aSAndroid Build Coastguard Worker 'arch': 132*6dbdd20aSAndroid Build Coastguard Worker 'android-x86', 133*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 134*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 135*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 136*6dbdd20aSAndroid Build Coastguard Worker 9503704, 137*6dbdd20aSAndroid Build Coastguard Worker 'url': 138*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/android-x86/traceconv', 139*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 140*6dbdd20aSAndroid Build Coastguard Worker '704e58a7249de56aadec64d4c0d83bab0821d2c4fd77114a9b71705ff4224539' 141*6dbdd20aSAndroid Build Coastguard Worker}, { 142*6dbdd20aSAndroid Build Coastguard Worker 'arch': 143*6dbdd20aSAndroid Build Coastguard Worker 'android-x64', 144*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 145*6dbdd20aSAndroid Build Coastguard Worker 'traceconv', 146*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 147*6dbdd20aSAndroid Build Coastguard Worker 8964488, 148*6dbdd20aSAndroid Build Coastguard Worker 'url': 149*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/android-x64/traceconv', 150*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 151*6dbdd20aSAndroid Build Coastguard Worker 'e4f07836fc2a5fb7cd997a9acc4183af7a06997d1e73aac71021af5114b921bc' 152*6dbdd20aSAndroid Build Coastguard Worker}, { 153*6dbdd20aSAndroid Build Coastguard Worker 'arch': 154*6dbdd20aSAndroid Build Coastguard Worker 'windows-amd64', 155*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 156*6dbdd20aSAndroid Build Coastguard Worker 'traceconv.exe', 157*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 158*6dbdd20aSAndroid Build Coastguard Worker 8763904, 159*6dbdd20aSAndroid Build Coastguard Worker 'url': 160*6dbdd20aSAndroid Build Coastguard Worker 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v48.1/windows-amd64/traceconv.exe', 161*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 162*6dbdd20aSAndroid Build Coastguard Worker '084670ac28ed59a9642782a30e051735c1b7474b8cd569b9bc94c305af68290e', 163*6dbdd20aSAndroid Build Coastguard Worker 'platform': 164*6dbdd20aSAndroid Build Coastguard Worker 'win32', 165*6dbdd20aSAndroid Build Coastguard Worker 'machine': ['amd64'] 166*6dbdd20aSAndroid Build Coastguard Worker}] 167*6dbdd20aSAndroid Build Coastguard Worker 168*6dbdd20aSAndroid Build Coastguard Worker# ----- Amalgamator: end of python/perfetto/prebuilts/manifests/traceconv.py 169*6dbdd20aSAndroid Build Coastguard Worker 170*6dbdd20aSAndroid Build Coastguard Worker# ----- Amalgamator: begin of python/perfetto/prebuilts/perfetto_prebuilts.py 171*6dbdd20aSAndroid Build Coastguard Worker# Copyright (C) 2021 The Android Open Source Project 172*6dbdd20aSAndroid Build Coastguard Worker# 173*6dbdd20aSAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 174*6dbdd20aSAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 175*6dbdd20aSAndroid Build Coastguard Worker# You may obtain a copy of the License at 176*6dbdd20aSAndroid Build Coastguard Worker# 177*6dbdd20aSAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 178*6dbdd20aSAndroid Build Coastguard Worker# 179*6dbdd20aSAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 180*6dbdd20aSAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 181*6dbdd20aSAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 182*6dbdd20aSAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 183*6dbdd20aSAndroid Build Coastguard Worker# limitations under the License. 184*6dbdd20aSAndroid Build Coastguard Worker""" 185*6dbdd20aSAndroid Build Coastguard WorkerFunctions to fetch pre-pinned Perfetto prebuilts. 186*6dbdd20aSAndroid Build Coastguard Worker 187*6dbdd20aSAndroid Build Coastguard WorkerThis function is used in different places: 188*6dbdd20aSAndroid Build Coastguard Worker- Into the //tools/{trace_processor, traceconv} scripts, which are just plain 189*6dbdd20aSAndroid Build Coastguard Worker wrappers around executables. 190*6dbdd20aSAndroid Build Coastguard Worker- Into the //tools/{heap_profiler, record_android_trace} scripts, which contain 191*6dbdd20aSAndroid Build Coastguard Worker some other hand-written python code. 192*6dbdd20aSAndroid Build Coastguard Worker 193*6dbdd20aSAndroid Build Coastguard WorkerThe manifest argument looks as follows: 194*6dbdd20aSAndroid Build Coastguard WorkerTRACECONV_MANIFEST = [ 195*6dbdd20aSAndroid Build Coastguard Worker { 196*6dbdd20aSAndroid Build Coastguard Worker 'arch': 'mac-amd64', 197*6dbdd20aSAndroid Build Coastguard Worker 'file_name': 'traceconv', 198*6dbdd20aSAndroid Build Coastguard Worker 'file_size': 7087080, 199*6dbdd20aSAndroid Build Coastguard Worker 'url': https://commondatastorage.googleapis.com/.../trace_to_text', 200*6dbdd20aSAndroid Build Coastguard Worker 'sha256': 7d957c005b0dc130f5bd855d6cec27e060d38841b320d04840afc569f9087490', 201*6dbdd20aSAndroid Build Coastguard Worker 'platform': 'darwin', 202*6dbdd20aSAndroid Build Coastguard Worker 'machine': 'x86_64' 203*6dbdd20aSAndroid Build Coastguard Worker }, 204*6dbdd20aSAndroid Build Coastguard Worker ... 205*6dbdd20aSAndroid Build Coastguard Worker] 206*6dbdd20aSAndroid Build Coastguard Worker 207*6dbdd20aSAndroid Build Coastguard WorkerThe intended usage is: 208*6dbdd20aSAndroid Build Coastguard Worker 209*6dbdd20aSAndroid Build Coastguard Worker from perfetto.prebuilts.manifests.traceconv import TRACECONV_MANIFEST 210*6dbdd20aSAndroid Build Coastguard Worker bin_path = get_perfetto_prebuilt(TRACECONV_MANIFEST) 211*6dbdd20aSAndroid Build Coastguard Worker subprocess.call(bin_path, ...) 212*6dbdd20aSAndroid Build Coastguard Worker""" 213*6dbdd20aSAndroid Build Coastguard Worker 214*6dbdd20aSAndroid Build Coastguard Workerimport hashlib 215*6dbdd20aSAndroid Build Coastguard Workerimport os 216*6dbdd20aSAndroid Build Coastguard Workerimport platform 217*6dbdd20aSAndroid Build Coastguard Workerimport random 218*6dbdd20aSAndroid Build Coastguard Workerimport subprocess 219*6dbdd20aSAndroid Build Coastguard Workerimport sys 220*6dbdd20aSAndroid Build Coastguard Worker 221*6dbdd20aSAndroid Build Coastguard Worker 222*6dbdd20aSAndroid Build Coastguard Workerdef download_or_get_cached(file_name, url, sha256): 223*6dbdd20aSAndroid Build Coastguard Worker """ Downloads a prebuilt or returns a cached version 224*6dbdd20aSAndroid Build Coastguard Worker 225*6dbdd20aSAndroid Build Coastguard Worker The first time this is invoked, it downloads the |url| and caches it into 226*6dbdd20aSAndroid Build Coastguard Worker ~/.local/share/perfetto/prebuilts/$tool_name. On subsequent invocations it 227*6dbdd20aSAndroid Build Coastguard Worker just runs the cached version. 228*6dbdd20aSAndroid Build Coastguard Worker """ 229*6dbdd20aSAndroid Build Coastguard Worker dir = os.path.join( 230*6dbdd20aSAndroid Build Coastguard Worker os.path.expanduser('~'), '.local', 'share', 'perfetto', 'prebuilts') 231*6dbdd20aSAndroid Build Coastguard Worker os.makedirs(dir, exist_ok=True) 232*6dbdd20aSAndroid Build Coastguard Worker bin_path = os.path.join(dir, file_name) 233*6dbdd20aSAndroid Build Coastguard Worker sha256_path = os.path.join(dir, file_name + '.sha256') 234*6dbdd20aSAndroid Build Coastguard Worker needs_download = True 235*6dbdd20aSAndroid Build Coastguard Worker 236*6dbdd20aSAndroid Build Coastguard Worker # Avoid recomputing the SHA-256 on each invocation. The SHA-256 of the last 237*6dbdd20aSAndroid Build Coastguard Worker # download is cached into file_name.sha256, just check if that matches. 238*6dbdd20aSAndroid Build Coastguard Worker if os.path.exists(bin_path) and os.path.exists(sha256_path): 239*6dbdd20aSAndroid Build Coastguard Worker with open(sha256_path, 'rb') as f: 240*6dbdd20aSAndroid Build Coastguard Worker digest = f.read().decode() 241*6dbdd20aSAndroid Build Coastguard Worker if digest == sha256: 242*6dbdd20aSAndroid Build Coastguard Worker needs_download = False 243*6dbdd20aSAndroid Build Coastguard Worker 244*6dbdd20aSAndroid Build Coastguard Worker if needs_download: # The file doesn't exist or the SHA256 doesn't match. 245*6dbdd20aSAndroid Build Coastguard Worker # Use a unique random file to guard against concurrent executions. 246*6dbdd20aSAndroid Build Coastguard Worker # See https://github.com/google/perfetto/issues/786 . 247*6dbdd20aSAndroid Build Coastguard Worker tmp_path = '%s.%d.tmp' % (bin_path, random.randint(0, 100000)) 248*6dbdd20aSAndroid Build Coastguard Worker print('Downloading ' + url) 249*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['curl', '-f', '-L', '-#', '-o', tmp_path, url]) 250*6dbdd20aSAndroid Build Coastguard Worker with open(tmp_path, 'rb') as fd: 251*6dbdd20aSAndroid Build Coastguard Worker actual_sha256 = hashlib.sha256(fd.read()).hexdigest() 252*6dbdd20aSAndroid Build Coastguard Worker if actual_sha256 != sha256: 253*6dbdd20aSAndroid Build Coastguard Worker raise Exception('Checksum mismatch for %s (actual: %s, expected: %s)' % 254*6dbdd20aSAndroid Build Coastguard Worker (url, actual_sha256, sha256)) 255*6dbdd20aSAndroid Build Coastguard Worker os.chmod(tmp_path, 0o755) 256*6dbdd20aSAndroid Build Coastguard Worker os.replace(tmp_path, bin_path) 257*6dbdd20aSAndroid Build Coastguard Worker with open(tmp_path, 'w') as f: 258*6dbdd20aSAndroid Build Coastguard Worker f.write(sha256) 259*6dbdd20aSAndroid Build Coastguard Worker os.replace(tmp_path, sha256_path) 260*6dbdd20aSAndroid Build Coastguard Worker return bin_path 261*6dbdd20aSAndroid Build Coastguard Worker 262*6dbdd20aSAndroid Build Coastguard Worker 263*6dbdd20aSAndroid Build Coastguard Workerdef get_perfetto_prebuilt(manifest, soft_fail=False, arch=None): 264*6dbdd20aSAndroid Build Coastguard Worker """ Downloads the prebuilt, if necessary, and returns its path on disk. """ 265*6dbdd20aSAndroid Build Coastguard Worker plat = sys.platform.lower() 266*6dbdd20aSAndroid Build Coastguard Worker machine = platform.machine().lower() 267*6dbdd20aSAndroid Build Coastguard Worker manifest_entry = None 268*6dbdd20aSAndroid Build Coastguard Worker for entry in manifest: 269*6dbdd20aSAndroid Build Coastguard Worker # If the caller overrides the arch, just match that (for Android prebuilts). 270*6dbdd20aSAndroid Build Coastguard Worker if arch: 271*6dbdd20aSAndroid Build Coastguard Worker if entry.get('arch') == arch: 272*6dbdd20aSAndroid Build Coastguard Worker manifest_entry = entry 273*6dbdd20aSAndroid Build Coastguard Worker break 274*6dbdd20aSAndroid Build Coastguard Worker continue 275*6dbdd20aSAndroid Build Coastguard Worker # Otherwise guess the local machine arch. 276*6dbdd20aSAndroid Build Coastguard Worker if entry.get('platform') == plat and machine in entry.get('machine', []): 277*6dbdd20aSAndroid Build Coastguard Worker manifest_entry = entry 278*6dbdd20aSAndroid Build Coastguard Worker break 279*6dbdd20aSAndroid Build Coastguard Worker if manifest_entry is None: 280*6dbdd20aSAndroid Build Coastguard Worker if soft_fail: 281*6dbdd20aSAndroid Build Coastguard Worker return None 282*6dbdd20aSAndroid Build Coastguard Worker raise Exception( 283*6dbdd20aSAndroid Build Coastguard Worker ('No prebuilts available for %s-%s\n' % (plat, machine)) + 284*6dbdd20aSAndroid Build Coastguard Worker 'See https://perfetto.dev/docs/contributing/build-instructions') 285*6dbdd20aSAndroid Build Coastguard Worker 286*6dbdd20aSAndroid Build Coastguard Worker return download_or_get_cached( 287*6dbdd20aSAndroid Build Coastguard Worker file_name=manifest_entry['file_name'], 288*6dbdd20aSAndroid Build Coastguard Worker url=manifest_entry['url'], 289*6dbdd20aSAndroid Build Coastguard Worker sha256=manifest_entry['sha256']) 290*6dbdd20aSAndroid Build Coastguard Worker 291*6dbdd20aSAndroid Build Coastguard Worker 292*6dbdd20aSAndroid Build Coastguard Workerdef run_perfetto_prebuilt(manifest): 293*6dbdd20aSAndroid Build Coastguard Worker bin_path = get_perfetto_prebuilt(manifest) 294*6dbdd20aSAndroid Build Coastguard Worker if sys.platform.lower() == 'win32': 295*6dbdd20aSAndroid Build Coastguard Worker sys.exit(subprocess.check_call([bin_path, *sys.argv[1:]])) 296*6dbdd20aSAndroid Build Coastguard Worker os.execv(bin_path, [bin_path] + sys.argv[1:]) 297*6dbdd20aSAndroid Build Coastguard Worker 298*6dbdd20aSAndroid Build Coastguard Worker# ----- Amalgamator: end of python/perfetto/prebuilts/perfetto_prebuilts.py 299*6dbdd20aSAndroid Build Coastguard Worker 300*6dbdd20aSAndroid Build Coastguard WorkerNULL = open(os.devnull) 301*6dbdd20aSAndroid Build Coastguard WorkerNOOUT = { 302*6dbdd20aSAndroid Build Coastguard Worker 'stdout': NULL, 303*6dbdd20aSAndroid Build Coastguard Worker 'stderr': NULL, 304*6dbdd20aSAndroid Build Coastguard Worker} 305*6dbdd20aSAndroid Build Coastguard Worker 306*6dbdd20aSAndroid Build Coastguard WorkerUUID = str(uuid.uuid4())[-6:] 307*6dbdd20aSAndroid Build Coastguard Worker 308*6dbdd20aSAndroid Build Coastguard WorkerPACKAGES_LIST_CFG = '''data_sources { 309*6dbdd20aSAndroid Build Coastguard Worker config { 310*6dbdd20aSAndroid Build Coastguard Worker name: "android.packages_list" 311*6dbdd20aSAndroid Build Coastguard Worker } 312*6dbdd20aSAndroid Build Coastguard Worker} 313*6dbdd20aSAndroid Build Coastguard Worker''' 314*6dbdd20aSAndroid Build Coastguard Worker 315*6dbdd20aSAndroid Build Coastguard WorkerCFG_INDENT = ' ' 316*6dbdd20aSAndroid Build Coastguard WorkerCFG = '''buffers {{ 317*6dbdd20aSAndroid Build Coastguard Worker size_kb: 63488 318*6dbdd20aSAndroid Build Coastguard Worker}} 319*6dbdd20aSAndroid Build Coastguard Worker 320*6dbdd20aSAndroid Build Coastguard Workerdata_sources {{ 321*6dbdd20aSAndroid Build Coastguard Worker config {{ 322*6dbdd20aSAndroid Build Coastguard Worker name: "android.heapprofd" 323*6dbdd20aSAndroid Build Coastguard Worker heapprofd_config {{ 324*6dbdd20aSAndroid Build Coastguard Worker shmem_size_bytes: {shmem_size} 325*6dbdd20aSAndroid Build Coastguard Worker sampling_interval_bytes: {interval} 326*6dbdd20aSAndroid Build Coastguard Worker{target_cfg} 327*6dbdd20aSAndroid Build Coastguard Worker }} 328*6dbdd20aSAndroid Build Coastguard Worker }} 329*6dbdd20aSAndroid Build Coastguard Worker}} 330*6dbdd20aSAndroid Build Coastguard Worker 331*6dbdd20aSAndroid Build Coastguard Workerduration_ms: {duration} 332*6dbdd20aSAndroid Build Coastguard Workerwrite_into_file: true 333*6dbdd20aSAndroid Build Coastguard Workerflush_timeout_ms: 30000 334*6dbdd20aSAndroid Build Coastguard Workerflush_period_ms: 604800000 335*6dbdd20aSAndroid Build Coastguard Worker''' 336*6dbdd20aSAndroid Build Coastguard Worker 337*6dbdd20aSAndroid Build Coastguard Worker# flush_period_ms of 1 week to suppress trace_processor_shell warning. 338*6dbdd20aSAndroid Build Coastguard Worker 339*6dbdd20aSAndroid Build Coastguard WorkerCONTINUOUS_DUMP = """ 340*6dbdd20aSAndroid Build Coastguard Worker continuous_dump_config {{ 341*6dbdd20aSAndroid Build Coastguard Worker dump_phase_ms: 0 342*6dbdd20aSAndroid Build Coastguard Worker dump_interval_ms: {dump_interval} 343*6dbdd20aSAndroid Build Coastguard Worker }} 344*6dbdd20aSAndroid Build Coastguard Worker""" 345*6dbdd20aSAndroid Build Coastguard Worker 346*6dbdd20aSAndroid Build Coastguard WorkerPROFILE_LOCAL_PATH = os.path.join(tempfile.gettempdir(), UUID) 347*6dbdd20aSAndroid Build Coastguard Worker 348*6dbdd20aSAndroid Build Coastguard WorkerIS_INTERRUPTED = False 349*6dbdd20aSAndroid Build Coastguard Worker 350*6dbdd20aSAndroid Build Coastguard Worker 351*6dbdd20aSAndroid Build Coastguard Workerdef sigint_handler(sig, frame): 352*6dbdd20aSAndroid Build Coastguard Worker global IS_INTERRUPTED 353*6dbdd20aSAndroid Build Coastguard Worker IS_INTERRUPTED = True 354*6dbdd20aSAndroid Build Coastguard Worker 355*6dbdd20aSAndroid Build Coastguard Worker 356*6dbdd20aSAndroid Build Coastguard Workerdef print_no_profile_error(): 357*6dbdd20aSAndroid Build Coastguard Worker print("No profiles generated", file=sys.stderr) 358*6dbdd20aSAndroid Build Coastguard Worker print( 359*6dbdd20aSAndroid Build Coastguard Worker "If this is unexpected, check " 360*6dbdd20aSAndroid Build Coastguard Worker "https://perfetto.dev/docs/data-sources/native-heap-profiler#troubleshooting.", 361*6dbdd20aSAndroid Build Coastguard Worker file=sys.stderr) 362*6dbdd20aSAndroid Build Coastguard Worker 363*6dbdd20aSAndroid Build Coastguard Worker 364*6dbdd20aSAndroid Build Coastguard Workerdef known_issues_url(number): 365*6dbdd20aSAndroid Build Coastguard Worker return ('https://perfetto.dev/docs/data-sources/native-heap-profiler' 366*6dbdd20aSAndroid Build Coastguard Worker '#known-issues-android{}'.format(number)) 367*6dbdd20aSAndroid Build Coastguard Worker 368*6dbdd20aSAndroid Build Coastguard Worker 369*6dbdd20aSAndroid Build Coastguard WorkerKNOWN_ISSUES = { 370*6dbdd20aSAndroid Build Coastguard Worker '10': known_issues_url(10), 371*6dbdd20aSAndroid Build Coastguard Worker 'Q': known_issues_url(10), 372*6dbdd20aSAndroid Build Coastguard Worker '11': known_issues_url(11), 373*6dbdd20aSAndroid Build Coastguard Worker 'R': known_issues_url(11), 374*6dbdd20aSAndroid Build Coastguard Worker} 375*6dbdd20aSAndroid Build Coastguard Worker 376*6dbdd20aSAndroid Build Coastguard Worker 377*6dbdd20aSAndroid Build Coastguard Workerdef maybe_known_issues(): 378*6dbdd20aSAndroid Build Coastguard Worker release_or_codename = subprocess.check_output( 379*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', 'getprop', 380*6dbdd20aSAndroid Build Coastguard Worker 'ro.build.version.release_or_codename']).decode('utf-8').strip() 381*6dbdd20aSAndroid Build Coastguard Worker return KNOWN_ISSUES.get(release_or_codename, None) 382*6dbdd20aSAndroid Build Coastguard Worker 383*6dbdd20aSAndroid Build Coastguard Worker 384*6dbdd20aSAndroid Build Coastguard WorkerSDK = { 385*6dbdd20aSAndroid Build Coastguard Worker 'R': 30, 386*6dbdd20aSAndroid Build Coastguard Worker} 387*6dbdd20aSAndroid Build Coastguard Worker 388*6dbdd20aSAndroid Build Coastguard Worker 389*6dbdd20aSAndroid Build Coastguard Workerdef release_or_newer(release): 390*6dbdd20aSAndroid Build Coastguard Worker sdk = int( 391*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_output( 392*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', 'getprop', 393*6dbdd20aSAndroid Build Coastguard Worker 'ro.system.build.version.sdk']).decode('utf-8').strip()) 394*6dbdd20aSAndroid Build Coastguard Worker if sdk >= SDK[release]: 395*6dbdd20aSAndroid Build Coastguard Worker return True 396*6dbdd20aSAndroid Build Coastguard Worker codename = subprocess.check_output( 397*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', 'getprop', 398*6dbdd20aSAndroid Build Coastguard Worker 'ro.build.version.codename']).decode('utf-8').strip() 399*6dbdd20aSAndroid Build Coastguard Worker return codename == release 400*6dbdd20aSAndroid Build Coastguard Worker 401*6dbdd20aSAndroid Build Coastguard Worker 402*6dbdd20aSAndroid Build Coastguard WorkerORDER = ['-n', '-p', '-i', '-o'] 403*6dbdd20aSAndroid Build Coastguard Worker 404*6dbdd20aSAndroid Build Coastguard Worker 405*6dbdd20aSAndroid Build Coastguard Workerdef arg_order(action): 406*6dbdd20aSAndroid Build Coastguard Worker result = len(ORDER) 407*6dbdd20aSAndroid Build Coastguard Worker for opt in action.option_strings: 408*6dbdd20aSAndroid Build Coastguard Worker if opt in ORDER: 409*6dbdd20aSAndroid Build Coastguard Worker result = min(ORDER.index(opt), result) 410*6dbdd20aSAndroid Build Coastguard Worker return result, action.option_strings[0].strip('-') 411*6dbdd20aSAndroid Build Coastguard Worker 412*6dbdd20aSAndroid Build Coastguard Worker 413*6dbdd20aSAndroid Build Coastguard Workerdef print_options(parser): 414*6dbdd20aSAndroid Build Coastguard Worker for action in sorted(parser._actions, key=arg_order): 415*6dbdd20aSAndroid Build Coastguard Worker if action.help is argparse.SUPPRESS: 416*6dbdd20aSAndroid Build Coastguard Worker continue 417*6dbdd20aSAndroid Build Coastguard Worker opts = ', '.join('`' + x + '`' for x in action.option_strings) 418*6dbdd20aSAndroid Build Coastguard Worker metavar = '' if action.metavar is None else ' _' + action.metavar + '_' 419*6dbdd20aSAndroid Build Coastguard Worker print('{}{}'.format(opts, metavar)) 420*6dbdd20aSAndroid Build Coastguard Worker print(': {}'.format(action.help)) 421*6dbdd20aSAndroid Build Coastguard Worker print() 422*6dbdd20aSAndroid Build Coastguard Worker 423*6dbdd20aSAndroid Build Coastguard Worker 424*6dbdd20aSAndroid Build Coastguard Workerdef main(argv): 425*6dbdd20aSAndroid Build Coastguard Worker parser = argparse.ArgumentParser(description="""Collect a heap profile 426*6dbdd20aSAndroid Build Coastguard Worker 427*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) 428*6dbdd20aSAndroid Build Coastguard Worker 429*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 430*6dbdd20aSAndroid Build Coastguard Worker "-i", 431*6dbdd20aSAndroid Build Coastguard Worker "--interval", 432*6dbdd20aSAndroid Build Coastguard Worker help="Sampling interval. " 433*6dbdd20aSAndroid Build Coastguard Worker "Default 4096 (4KiB)", 434*6dbdd20aSAndroid Build Coastguard Worker type=int, 435*6dbdd20aSAndroid Build Coastguard Worker default=4096) 436*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 437*6dbdd20aSAndroid Build Coastguard Worker "-d", 438*6dbdd20aSAndroid Build Coastguard Worker "--duration", 439*6dbdd20aSAndroid Build Coastguard Worker help="Duration of profile (ms). 0 to run until interrupted. " 440*6dbdd20aSAndroid Build Coastguard Worker "Default: until interrupted by user.", 441*6dbdd20aSAndroid Build Coastguard Worker type=int, 442*6dbdd20aSAndroid Build Coastguard Worker default=0) 443*6dbdd20aSAndroid Build Coastguard Worker # This flag is a no-op now. We never start heapprofd explicitly using system 444*6dbdd20aSAndroid Build Coastguard Worker # properties. 445*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 446*6dbdd20aSAndroid Build Coastguard Worker "--no-start", help="Do not start heapprofd.", action='store_true') 447*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 448*6dbdd20aSAndroid Build Coastguard Worker "-p", 449*6dbdd20aSAndroid Build Coastguard Worker "--pid", 450*6dbdd20aSAndroid Build Coastguard Worker help="Comma-separated list of PIDs to " 451*6dbdd20aSAndroid Build Coastguard Worker "profile.", 452*6dbdd20aSAndroid Build Coastguard Worker metavar="PIDS") 453*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 454*6dbdd20aSAndroid Build Coastguard Worker "-n", 455*6dbdd20aSAndroid Build Coastguard Worker "--name", 456*6dbdd20aSAndroid Build Coastguard Worker help="Comma-separated list of process " 457*6dbdd20aSAndroid Build Coastguard Worker "names to profile.", 458*6dbdd20aSAndroid Build Coastguard Worker metavar="NAMES") 459*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 460*6dbdd20aSAndroid Build Coastguard Worker "-c", 461*6dbdd20aSAndroid Build Coastguard Worker "--continuous-dump", 462*6dbdd20aSAndroid Build Coastguard Worker help="Dump interval in ms. 0 to disable continuous dump.", 463*6dbdd20aSAndroid Build Coastguard Worker type=int, 464*6dbdd20aSAndroid Build Coastguard Worker default=0) 465*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 466*6dbdd20aSAndroid Build Coastguard Worker "--heaps", 467*6dbdd20aSAndroid Build Coastguard Worker help="Comma-separated list of heaps to collect, e.g: malloc,art. " 468*6dbdd20aSAndroid Build Coastguard Worker "Requires Android 12.", 469*6dbdd20aSAndroid Build Coastguard Worker metavar="HEAPS") 470*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 471*6dbdd20aSAndroid Build Coastguard Worker "--all-heaps", 472*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 473*6dbdd20aSAndroid Build Coastguard Worker help="Collect allocations from all heaps registered by target.") 474*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 475*6dbdd20aSAndroid Build Coastguard Worker "--no-android-tree-symbolization", 476*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 477*6dbdd20aSAndroid Build Coastguard Worker help="Do not symbolize using currently lunched target in the " 478*6dbdd20aSAndroid Build Coastguard Worker "Android tree.") 479*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 480*6dbdd20aSAndroid Build Coastguard Worker "--disable-selinux", 481*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 482*6dbdd20aSAndroid Build Coastguard Worker help="Disable SELinux enforcement for duration of " 483*6dbdd20aSAndroid Build Coastguard Worker "profile.") 484*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 485*6dbdd20aSAndroid Build Coastguard Worker "--no-versions", 486*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 487*6dbdd20aSAndroid Build Coastguard Worker help="Do not get version information about APKs.") 488*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 489*6dbdd20aSAndroid Build Coastguard Worker "--no-running", 490*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 491*6dbdd20aSAndroid Build Coastguard Worker help="Do not target already running processes. Requires Android 11.") 492*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 493*6dbdd20aSAndroid Build Coastguard Worker "--no-startup", 494*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 495*6dbdd20aSAndroid Build Coastguard Worker help="Do not target processes that start during " 496*6dbdd20aSAndroid Build Coastguard Worker "the profile. Requires Android 11.") 497*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 498*6dbdd20aSAndroid Build Coastguard Worker "--shmem-size", 499*6dbdd20aSAndroid Build Coastguard Worker help="Size of buffer between client and " 500*6dbdd20aSAndroid Build Coastguard Worker "heapprofd. Default 8MiB. Needs to be a power of two " 501*6dbdd20aSAndroid Build Coastguard Worker "multiple of 4096, at least 8192.", 502*6dbdd20aSAndroid Build Coastguard Worker type=int, 503*6dbdd20aSAndroid Build Coastguard Worker default=8 * 1048576) 504*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 505*6dbdd20aSAndroid Build Coastguard Worker "--block-client", 506*6dbdd20aSAndroid Build Coastguard Worker help="When buffer is full, block the " 507*6dbdd20aSAndroid Build Coastguard Worker "client to wait for buffer space. Use with caution as " 508*6dbdd20aSAndroid Build Coastguard Worker "this can significantly slow down the client. " 509*6dbdd20aSAndroid Build Coastguard Worker "This is the default", 510*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 511*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 512*6dbdd20aSAndroid Build Coastguard Worker "--block-client-timeout", 513*6dbdd20aSAndroid Build Coastguard Worker help="If --block-client is given, do not block any allocation for " 514*6dbdd20aSAndroid Build Coastguard Worker "longer than this timeout (us).", 515*6dbdd20aSAndroid Build Coastguard Worker type=int) 516*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 517*6dbdd20aSAndroid Build Coastguard Worker "--no-block-client", 518*6dbdd20aSAndroid Build Coastguard Worker help="When buffer is full, stop the " 519*6dbdd20aSAndroid Build Coastguard Worker "profile early.", 520*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 521*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 522*6dbdd20aSAndroid Build Coastguard Worker "--idle-allocations", 523*6dbdd20aSAndroid Build Coastguard Worker help="Keep track of how many " 524*6dbdd20aSAndroid Build Coastguard Worker "bytes were unused since the last dump, per " 525*6dbdd20aSAndroid Build Coastguard Worker "callstack", 526*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 527*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 528*6dbdd20aSAndroid Build Coastguard Worker "--dump-at-max", 529*6dbdd20aSAndroid Build Coastguard Worker help="Dump the maximum memory usage " 530*6dbdd20aSAndroid Build Coastguard Worker "rather than at the time of the dump.", 531*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 532*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 533*6dbdd20aSAndroid Build Coastguard Worker "--disable-fork-teardown", 534*6dbdd20aSAndroid Build Coastguard Worker help="Do not tear down client in forks. This can be useful for programs " 535*6dbdd20aSAndroid Build Coastguard Worker "that use vfork. Android 11+ only.", 536*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 537*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 538*6dbdd20aSAndroid Build Coastguard Worker "--simpleperf", 539*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 540*6dbdd20aSAndroid Build Coastguard Worker help="Get simpleperf profile of heapprofd. This is " 541*6dbdd20aSAndroid Build Coastguard Worker "only for heapprofd development.") 542*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 543*6dbdd20aSAndroid Build Coastguard Worker "--traceconv-binary", help="Path to local trace to text. For debugging.") 544*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 545*6dbdd20aSAndroid Build Coastguard Worker "--no-annotations", 546*6dbdd20aSAndroid Build Coastguard Worker help="Do not suffix the pprof function names with Android ART mode " 547*6dbdd20aSAndroid Build Coastguard Worker "annotations such as [jit].", 548*6dbdd20aSAndroid Build Coastguard Worker action="store_true") 549*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 550*6dbdd20aSAndroid Build Coastguard Worker "--print-config", 551*6dbdd20aSAndroid Build Coastguard Worker action="store_true", 552*6dbdd20aSAndroid Build Coastguard Worker help="Print config instead of running. For debugging.") 553*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 554*6dbdd20aSAndroid Build Coastguard Worker "-o", 555*6dbdd20aSAndroid Build Coastguard Worker "--output", 556*6dbdd20aSAndroid Build Coastguard Worker help="Output directory.", 557*6dbdd20aSAndroid Build Coastguard Worker metavar="DIRECTORY", 558*6dbdd20aSAndroid Build Coastguard Worker default=None) 559*6dbdd20aSAndroid Build Coastguard Worker parser.add_argument( 560*6dbdd20aSAndroid Build Coastguard Worker "--print-options", action="store_true", help=argparse.SUPPRESS) 561*6dbdd20aSAndroid Build Coastguard Worker 562*6dbdd20aSAndroid Build Coastguard Worker args = parser.parse_args() 563*6dbdd20aSAndroid Build Coastguard Worker if args.print_options: 564*6dbdd20aSAndroid Build Coastguard Worker print_options(parser) 565*6dbdd20aSAndroid Build Coastguard Worker return 0 566*6dbdd20aSAndroid Build Coastguard Worker fail = False 567*6dbdd20aSAndroid Build Coastguard Worker if args.block_client and args.no_block_client: 568*6dbdd20aSAndroid Build Coastguard Worker print( 569*6dbdd20aSAndroid Build Coastguard Worker "FATAL: Both block-client and no-block-client given.", file=sys.stderr) 570*6dbdd20aSAndroid Build Coastguard Worker fail = True 571*6dbdd20aSAndroid Build Coastguard Worker if args.pid is None and args.name is None: 572*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: Neither PID nor NAME given.", file=sys.stderr) 573*6dbdd20aSAndroid Build Coastguard Worker fail = True 574*6dbdd20aSAndroid Build Coastguard Worker if args.duration is None: 575*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: No duration given.", file=sys.stderr) 576*6dbdd20aSAndroid Build Coastguard Worker fail = True 577*6dbdd20aSAndroid Build Coastguard Worker if args.interval is None: 578*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: No interval given.", file=sys.stderr) 579*6dbdd20aSAndroid Build Coastguard Worker fail = True 580*6dbdd20aSAndroid Build Coastguard Worker if args.shmem_size % 4096: 581*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: shmem-size is not a multiple of 4096.", file=sys.stderr) 582*6dbdd20aSAndroid Build Coastguard Worker fail = True 583*6dbdd20aSAndroid Build Coastguard Worker if args.shmem_size < 8192: 584*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: shmem-size is less than 8192.", file=sys.stderr) 585*6dbdd20aSAndroid Build Coastguard Worker fail = True 586*6dbdd20aSAndroid Build Coastguard Worker if args.shmem_size & (args.shmem_size - 1): 587*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: shmem-size is not a power of two.", file=sys.stderr) 588*6dbdd20aSAndroid Build Coastguard Worker fail = True 589*6dbdd20aSAndroid Build Coastguard Worker 590*6dbdd20aSAndroid Build Coastguard Worker target_cfg = "" 591*6dbdd20aSAndroid Build Coastguard Worker if not args.no_block_client: 592*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "block_client: true\n" 593*6dbdd20aSAndroid Build Coastguard Worker if args.block_client_timeout: 594*6dbdd20aSAndroid Build Coastguard Worker target_cfg += ( 595*6dbdd20aSAndroid Build Coastguard Worker CFG_INDENT + 596*6dbdd20aSAndroid Build Coastguard Worker "block_client_timeout_us: %s\n" % args.block_client_timeout) 597*6dbdd20aSAndroid Build Coastguard Worker if args.no_startup: 598*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "no_startup: true\n" 599*6dbdd20aSAndroid Build Coastguard Worker if args.no_running: 600*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "no_running: true\n" 601*6dbdd20aSAndroid Build Coastguard Worker if args.dump_at_max: 602*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "dump_at_max: true\n" 603*6dbdd20aSAndroid Build Coastguard Worker if args.disable_fork_teardown: 604*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "disable_fork_teardown: true\n" 605*6dbdd20aSAndroid Build Coastguard Worker if args.all_heaps: 606*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + "all_heaps: true\n" 607*6dbdd20aSAndroid Build Coastguard Worker if args.pid: 608*6dbdd20aSAndroid Build Coastguard Worker for pid in args.pid.split(','): 609*6dbdd20aSAndroid Build Coastguard Worker try: 610*6dbdd20aSAndroid Build Coastguard Worker pid = int(pid) 611*6dbdd20aSAndroid Build Coastguard Worker except ValueError: 612*6dbdd20aSAndroid Build Coastguard Worker print("FATAL: invalid PID %s" % pid, file=sys.stderr) 613*6dbdd20aSAndroid Build Coastguard Worker fail = True 614*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + 'pid: {}\n'.format(pid) 615*6dbdd20aSAndroid Build Coastguard Worker if args.name: 616*6dbdd20aSAndroid Build Coastguard Worker for name in args.name.split(','): 617*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + 'process_cmdline: "{}"\n'.format(name) 618*6dbdd20aSAndroid Build Coastguard Worker if args.heaps: 619*6dbdd20aSAndroid Build Coastguard Worker for heap in args.heaps.split(','): 620*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CFG_INDENT + 'heaps: "{}"\n'.format(heap) 621*6dbdd20aSAndroid Build Coastguard Worker 622*6dbdd20aSAndroid Build Coastguard Worker if fail: 623*6dbdd20aSAndroid Build Coastguard Worker parser.print_help() 624*6dbdd20aSAndroid Build Coastguard Worker return 1 625*6dbdd20aSAndroid Build Coastguard Worker 626*6dbdd20aSAndroid Build Coastguard Worker traceconv_binary = args.traceconv_binary 627*6dbdd20aSAndroid Build Coastguard Worker 628*6dbdd20aSAndroid Build Coastguard Worker if args.continuous_dump: 629*6dbdd20aSAndroid Build Coastguard Worker target_cfg += CONTINUOUS_DUMP.format(dump_interval=args.continuous_dump) 630*6dbdd20aSAndroid Build Coastguard Worker cfg = CFG.format( 631*6dbdd20aSAndroid Build Coastguard Worker interval=args.interval, 632*6dbdd20aSAndroid Build Coastguard Worker duration=args.duration, 633*6dbdd20aSAndroid Build Coastguard Worker target_cfg=target_cfg, 634*6dbdd20aSAndroid Build Coastguard Worker shmem_size=args.shmem_size) 635*6dbdd20aSAndroid Build Coastguard Worker if not args.no_versions: 636*6dbdd20aSAndroid Build Coastguard Worker cfg += PACKAGES_LIST_CFG 637*6dbdd20aSAndroid Build Coastguard Worker 638*6dbdd20aSAndroid Build Coastguard Worker if args.print_config: 639*6dbdd20aSAndroid Build Coastguard Worker print(cfg) 640*6dbdd20aSAndroid Build Coastguard Worker return 0 641*6dbdd20aSAndroid Build Coastguard Worker 642*6dbdd20aSAndroid Build Coastguard Worker # Do this AFTER print_config so we do not download traceconv only to 643*6dbdd20aSAndroid Build Coastguard Worker # print out the config. 644*6dbdd20aSAndroid Build Coastguard Worker if traceconv_binary is None: 645*6dbdd20aSAndroid Build Coastguard Worker traceconv_binary = get_perfetto_prebuilt(TRACECONV_MANIFEST, soft_fail=True) 646*6dbdd20aSAndroid Build Coastguard Worker 647*6dbdd20aSAndroid Build Coastguard Worker known_issues = maybe_known_issues() 648*6dbdd20aSAndroid Build Coastguard Worker if known_issues: 649*6dbdd20aSAndroid Build Coastguard Worker print('If you are experiencing problems, please see the known issues for ' 650*6dbdd20aSAndroid Build Coastguard Worker 'your release: {}.'.format(known_issues)) 651*6dbdd20aSAndroid Build Coastguard Worker 652*6dbdd20aSAndroid Build Coastguard Worker # TODO(fmayer): Maybe feature detect whether we can remove traces instead of 653*6dbdd20aSAndroid Build Coastguard Worker # this. 654*6dbdd20aSAndroid Build Coastguard Worker uuid_trace = release_or_newer('R') 655*6dbdd20aSAndroid Build Coastguard Worker if uuid_trace: 656*6dbdd20aSAndroid Build Coastguard Worker profile_device_path = '/data/misc/perfetto-traces/profile-' + UUID 657*6dbdd20aSAndroid Build Coastguard Worker else: 658*6dbdd20aSAndroid Build Coastguard Worker user = subprocess.check_output(['adb', 'shell', 659*6dbdd20aSAndroid Build Coastguard Worker 'whoami']).decode('utf-8').strip() 660*6dbdd20aSAndroid Build Coastguard Worker profile_device_path = '/data/misc/perfetto-traces/profile-' + user 661*6dbdd20aSAndroid Build Coastguard Worker 662*6dbdd20aSAndroid Build Coastguard Worker perfetto_cmd = ('CFG=\'{cfg}\'; echo ${{CFG}} | ' 663*6dbdd20aSAndroid Build Coastguard Worker 'perfetto --txt -c - -o ' + profile_device_path + ' -d') 664*6dbdd20aSAndroid Build Coastguard Worker 665*6dbdd20aSAndroid Build Coastguard Worker if args.disable_selinux: 666*6dbdd20aSAndroid Build Coastguard Worker enforcing = subprocess.check_output(['adb', 'shell', 667*6dbdd20aSAndroid Build Coastguard Worker 'getenforce']).decode('utf-8').strip() 668*6dbdd20aSAndroid Build Coastguard Worker atexit.register( 669*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call, 670*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', 'su root setenforce %s' % enforcing]) 671*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['adb', 'shell', 'su root setenforce 0']) 672*6dbdd20aSAndroid Build Coastguard Worker 673*6dbdd20aSAndroid Build Coastguard Worker if args.simpleperf: 674*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call([ 675*6dbdd20aSAndroid Build Coastguard Worker 'adb', 'shell', 'mkdir -p /data/local/tmp/heapprofd_profile && ' 676*6dbdd20aSAndroid Build Coastguard Worker 'cd /data/local/tmp/heapprofd_profile &&' 677*6dbdd20aSAndroid Build Coastguard Worker '(nohup simpleperf record -g -p $(pidof heapprofd) 2>&1 &) ' 678*6dbdd20aSAndroid Build Coastguard Worker '> /dev/null' 679*6dbdd20aSAndroid Build Coastguard Worker ]) 680*6dbdd20aSAndroid Build Coastguard Worker 681*6dbdd20aSAndroid Build Coastguard Worker profile_target = PROFILE_LOCAL_PATH 682*6dbdd20aSAndroid Build Coastguard Worker if args.output is not None: 683*6dbdd20aSAndroid Build Coastguard Worker profile_target = args.output 684*6dbdd20aSAndroid Build Coastguard Worker else: 685*6dbdd20aSAndroid Build Coastguard Worker os.mkdir(profile_target) 686*6dbdd20aSAndroid Build Coastguard Worker 687*6dbdd20aSAndroid Build Coastguard Worker if not os.path.isdir(profile_target): 688*6dbdd20aSAndroid Build Coastguard Worker print( 689*6dbdd20aSAndroid Build Coastguard Worker "Output directory {} not found".format(profile_target), file=sys.stderr) 690*6dbdd20aSAndroid Build Coastguard Worker return 1 691*6dbdd20aSAndroid Build Coastguard Worker 692*6dbdd20aSAndroid Build Coastguard Worker if os.listdir(profile_target): 693*6dbdd20aSAndroid Build Coastguard Worker print( 694*6dbdd20aSAndroid Build Coastguard Worker "Output directory {} not empty".format(profile_target), file=sys.stderr) 695*6dbdd20aSAndroid Build Coastguard Worker return 1 696*6dbdd20aSAndroid Build Coastguard Worker 697*6dbdd20aSAndroid Build Coastguard Worker perfetto_pid = subprocess.check_output( 698*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'exec-out', perfetto_cmd.format(cfg=cfg)]).strip() 699*6dbdd20aSAndroid Build Coastguard Worker try: 700*6dbdd20aSAndroid Build Coastguard Worker perfetto_pid = int(perfetto_pid.strip()) 701*6dbdd20aSAndroid Build Coastguard Worker except ValueError: 702*6dbdd20aSAndroid Build Coastguard Worker print("Failed to invoke perfetto: {}".format(perfetto_pid), file=sys.stderr) 703*6dbdd20aSAndroid Build Coastguard Worker return 1 704*6dbdd20aSAndroid Build Coastguard Worker 705*6dbdd20aSAndroid Build Coastguard Worker old_handler = signal.signal(signal.SIGINT, sigint_handler) 706*6dbdd20aSAndroid Build Coastguard Worker print("Profiling active. Press Ctrl+C to terminate.") 707*6dbdd20aSAndroid Build Coastguard Worker print("You may disconnect your device.") 708*6dbdd20aSAndroid Build Coastguard Worker print() 709*6dbdd20aSAndroid Build Coastguard Worker exists = True 710*6dbdd20aSAndroid Build Coastguard Worker device_connected = True 711*6dbdd20aSAndroid Build Coastguard Worker while not device_connected or (exists and not IS_INTERRUPTED): 712*6dbdd20aSAndroid Build Coastguard Worker exists = subprocess.call( 713*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', '[ -d /proc/{} ]'.format(perfetto_pid)], **NOOUT) == 0 714*6dbdd20aSAndroid Build Coastguard Worker device_connected = subprocess.call(['adb', 'shell', 'true'], **NOOUT) == 0 715*6dbdd20aSAndroid Build Coastguard Worker time.sleep(1) 716*6dbdd20aSAndroid Build Coastguard Worker print("Waiting for profiler shutdown...") 717*6dbdd20aSAndroid Build Coastguard Worker signal.signal(signal.SIGINT, old_handler) 718*6dbdd20aSAndroid Build Coastguard Worker if IS_INTERRUPTED: 719*6dbdd20aSAndroid Build Coastguard Worker # Not check_call because it could have existed in the meantime. 720*6dbdd20aSAndroid Build Coastguard Worker subprocess.call(['adb', 'shell', 'kill', '-INT', str(perfetto_pid)]) 721*6dbdd20aSAndroid Build Coastguard Worker if args.simpleperf: 722*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['adb', 'shell', 'killall', '-INT', 'simpleperf']) 723*6dbdd20aSAndroid Build Coastguard Worker print("Waiting for simpleperf to exit.") 724*6dbdd20aSAndroid Build Coastguard Worker while subprocess.call( 725*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', '[ -f /proc/$(pidof simpleperf)/exe ]'], **NOOUT) == 0: 726*6dbdd20aSAndroid Build Coastguard Worker time.sleep(1) 727*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call( 728*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'pull', '/data/local/tmp/heapprofd_profile', profile_target]) 729*6dbdd20aSAndroid Build Coastguard Worker print("Pulled simpleperf profile to " + profile_target + 730*6dbdd20aSAndroid Build Coastguard Worker "/heapprofd_profile") 731*6dbdd20aSAndroid Build Coastguard Worker 732*6dbdd20aSAndroid Build Coastguard Worker # Wait for perfetto cmd to return. 733*6dbdd20aSAndroid Build Coastguard Worker while exists: 734*6dbdd20aSAndroid Build Coastguard Worker exists = subprocess.call( 735*6dbdd20aSAndroid Build Coastguard Worker ['adb', 'shell', '[ -d /proc/{} ]'.format(perfetto_pid)]) == 0 736*6dbdd20aSAndroid Build Coastguard Worker time.sleep(1) 737*6dbdd20aSAndroid Build Coastguard Worker 738*6dbdd20aSAndroid Build Coastguard Worker profile_host_path = os.path.join(profile_target, 'raw-trace') 739*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['adb', 'pull', profile_device_path, profile_host_path], 740*6dbdd20aSAndroid Build Coastguard Worker stdout=NULL) 741*6dbdd20aSAndroid Build Coastguard Worker if uuid_trace: 742*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call(['adb', 'shell', 'rm', profile_device_path], 743*6dbdd20aSAndroid Build Coastguard Worker stdout=NULL) 744*6dbdd20aSAndroid Build Coastguard Worker 745*6dbdd20aSAndroid Build Coastguard Worker if traceconv_binary is None: 746*6dbdd20aSAndroid Build Coastguard Worker print('Wrote profile to {}'.format(profile_host_path)) 747*6dbdd20aSAndroid Build Coastguard Worker print( 748*6dbdd20aSAndroid Build Coastguard Worker 'This file can be opened using the Perfetto UI, https://ui.perfetto.dev' 749*6dbdd20aSAndroid Build Coastguard Worker ) 750*6dbdd20aSAndroid Build Coastguard Worker return 0 751*6dbdd20aSAndroid Build Coastguard Worker 752*6dbdd20aSAndroid Build Coastguard Worker binary_path = os.getenv('PERFETTO_BINARY_PATH') 753*6dbdd20aSAndroid Build Coastguard Worker if not args.no_android_tree_symbolization: 754*6dbdd20aSAndroid Build Coastguard Worker product_out = os.getenv('ANDROID_PRODUCT_OUT') 755*6dbdd20aSAndroid Build Coastguard Worker if product_out: 756*6dbdd20aSAndroid Build Coastguard Worker product_out_symbols = product_out + '/symbols' 757*6dbdd20aSAndroid Build Coastguard Worker else: 758*6dbdd20aSAndroid Build Coastguard Worker product_out_symbols = None 759*6dbdd20aSAndroid Build Coastguard Worker 760*6dbdd20aSAndroid Build Coastguard Worker if binary_path is None: 761*6dbdd20aSAndroid Build Coastguard Worker binary_path = product_out_symbols 762*6dbdd20aSAndroid Build Coastguard Worker elif product_out_symbols is not None: 763*6dbdd20aSAndroid Build Coastguard Worker binary_path += os.pathsep + product_out_symbols 764*6dbdd20aSAndroid Build Coastguard Worker 765*6dbdd20aSAndroid Build Coastguard Worker trace_file = os.path.join(profile_target, 'raw-trace') 766*6dbdd20aSAndroid Build Coastguard Worker concat_files = [trace_file] 767*6dbdd20aSAndroid Build Coastguard Worker 768*6dbdd20aSAndroid Build Coastguard Worker if binary_path is not None: 769*6dbdd20aSAndroid Build Coastguard Worker with open(os.path.join(profile_target, 'symbols'), 'w') as fd: 770*6dbdd20aSAndroid Build Coastguard Worker ret = subprocess.call([ 771*6dbdd20aSAndroid Build Coastguard Worker traceconv_binary, 'symbolize', 772*6dbdd20aSAndroid Build Coastguard Worker os.path.join(profile_target, 'raw-trace') 773*6dbdd20aSAndroid Build Coastguard Worker ], 774*6dbdd20aSAndroid Build Coastguard Worker env=dict( 775*6dbdd20aSAndroid Build Coastguard Worker os.environ, PERFETTO_BINARY_PATH=binary_path), 776*6dbdd20aSAndroid Build Coastguard Worker stdout=fd) 777*6dbdd20aSAndroid Build Coastguard Worker if ret == 0: 778*6dbdd20aSAndroid Build Coastguard Worker concat_files.append(os.path.join(profile_target, 'symbols')) 779*6dbdd20aSAndroid Build Coastguard Worker else: 780*6dbdd20aSAndroid Build Coastguard Worker print("Failed to symbolize. Continuing without symbols.", file=sys.stderr) 781*6dbdd20aSAndroid Build Coastguard Worker 782*6dbdd20aSAndroid Build Coastguard Worker proguard_map = os.getenv('PERFETTO_PROGUARD_MAP') 783*6dbdd20aSAndroid Build Coastguard Worker if proguard_map is not None: 784*6dbdd20aSAndroid Build Coastguard Worker with open(os.path.join(profile_target, 'deobfuscation-packets'), 'w') as fd: 785*6dbdd20aSAndroid Build Coastguard Worker ret = subprocess.call([ 786*6dbdd20aSAndroid Build Coastguard Worker traceconv_binary, 'deobfuscate', 787*6dbdd20aSAndroid Build Coastguard Worker os.path.join(profile_target, 'raw-trace') 788*6dbdd20aSAndroid Build Coastguard Worker ], 789*6dbdd20aSAndroid Build Coastguard Worker env=dict( 790*6dbdd20aSAndroid Build Coastguard Worker os.environ, PERFETTO_PROGUARD_MAP=proguard_map), 791*6dbdd20aSAndroid Build Coastguard Worker stdout=fd) 792*6dbdd20aSAndroid Build Coastguard Worker if ret == 0: 793*6dbdd20aSAndroid Build Coastguard Worker concat_files.append(os.path.join(profile_target, 'deobfuscation-packets')) 794*6dbdd20aSAndroid Build Coastguard Worker else: 795*6dbdd20aSAndroid Build Coastguard Worker print( 796*6dbdd20aSAndroid Build Coastguard Worker "Failed to deobfuscate. Continuing without deobfuscated.", 797*6dbdd20aSAndroid Build Coastguard Worker file=sys.stderr) 798*6dbdd20aSAndroid Build Coastguard Worker 799*6dbdd20aSAndroid Build Coastguard Worker if len(concat_files) > 1: 800*6dbdd20aSAndroid Build Coastguard Worker with open(os.path.join(profile_target, 'symbolized-trace'), 'wb') as out: 801*6dbdd20aSAndroid Build Coastguard Worker for fn in concat_files: 802*6dbdd20aSAndroid Build Coastguard Worker with open(fn, 'rb') as inp: 803*6dbdd20aSAndroid Build Coastguard Worker while True: 804*6dbdd20aSAndroid Build Coastguard Worker buf = inp.read(4096) 805*6dbdd20aSAndroid Build Coastguard Worker if not buf: 806*6dbdd20aSAndroid Build Coastguard Worker break 807*6dbdd20aSAndroid Build Coastguard Worker out.write(buf) 808*6dbdd20aSAndroid Build Coastguard Worker trace_file = os.path.join(profile_target, 'symbolized-trace') 809*6dbdd20aSAndroid Build Coastguard Worker 810*6dbdd20aSAndroid Build Coastguard Worker conversion_args = [traceconv_binary, 'profile'] + ( 811*6dbdd20aSAndroid Build Coastguard Worker ['--no-annotations'] if args.no_annotations else []) + [trace_file] 812*6dbdd20aSAndroid Build Coastguard Worker traceconv_output = subprocess.check_output(conversion_args) 813*6dbdd20aSAndroid Build Coastguard Worker profile_path = None 814*6dbdd20aSAndroid Build Coastguard Worker for word in traceconv_output.decode('utf-8').split(): 815*6dbdd20aSAndroid Build Coastguard Worker if 'heap_profile-' in word: 816*6dbdd20aSAndroid Build Coastguard Worker profile_path = word 817*6dbdd20aSAndroid Build Coastguard Worker if profile_path is None: 818*6dbdd20aSAndroid Build Coastguard Worker print_no_profile_error() 819*6dbdd20aSAndroid Build Coastguard Worker return 1 820*6dbdd20aSAndroid Build Coastguard Worker 821*6dbdd20aSAndroid Build Coastguard Worker profile_files = os.listdir(profile_path) 822*6dbdd20aSAndroid Build Coastguard Worker if not profile_files: 823*6dbdd20aSAndroid Build Coastguard Worker print_no_profile_error() 824*6dbdd20aSAndroid Build Coastguard Worker return 1 825*6dbdd20aSAndroid Build Coastguard Worker 826*6dbdd20aSAndroid Build Coastguard Worker for profile_file in profile_files: 827*6dbdd20aSAndroid Build Coastguard Worker shutil.copy(os.path.join(profile_path, profile_file), profile_target) 828*6dbdd20aSAndroid Build Coastguard Worker 829*6dbdd20aSAndroid Build Coastguard Worker symlink_path = None 830*6dbdd20aSAndroid Build Coastguard Worker if not sys.platform.startswith('win'): 831*6dbdd20aSAndroid Build Coastguard Worker subprocess.check_call( 832*6dbdd20aSAndroid Build Coastguard Worker ['gzip'] + [os.path.join(profile_target, x) for x in profile_files]) 833*6dbdd20aSAndroid Build Coastguard Worker if args.output is None: 834*6dbdd20aSAndroid Build Coastguard Worker symlink_path = os.path.join( 835*6dbdd20aSAndroid Build Coastguard Worker os.path.dirname(profile_target), "heap_profile-latest") 836*6dbdd20aSAndroid Build Coastguard Worker if os.path.lexists(symlink_path): 837*6dbdd20aSAndroid Build Coastguard Worker os.unlink(symlink_path) 838*6dbdd20aSAndroid Build Coastguard Worker os.symlink(profile_target, symlink_path) 839*6dbdd20aSAndroid Build Coastguard Worker 840*6dbdd20aSAndroid Build Coastguard Worker if symlink_path is not None: 841*6dbdd20aSAndroid Build Coastguard Worker print("Wrote profiles to {} (symlink {})".format(profile_target, 842*6dbdd20aSAndroid Build Coastguard Worker symlink_path)) 843*6dbdd20aSAndroid Build Coastguard Worker else: 844*6dbdd20aSAndroid Build Coastguard Worker print("Wrote profiles to {}".format(profile_target)) 845*6dbdd20aSAndroid Build Coastguard Worker 846*6dbdd20aSAndroid Build Coastguard Worker print("The raw-trace file can be viewed using https://ui.perfetto.dev.") 847*6dbdd20aSAndroid Build Coastguard Worker print("The heap_dump.* files can be viewed using pprof/ (Googlers only) " + 848*6dbdd20aSAndroid Build Coastguard Worker "or https://www.speedscope.app/.") 849*6dbdd20aSAndroid Build Coastguard Worker print("The two above are equivalent. The raw-trace contains the union of " + 850*6dbdd20aSAndroid Build Coastguard Worker "all the heap dumps.") 851*6dbdd20aSAndroid Build Coastguard Worker 852*6dbdd20aSAndroid Build Coastguard Worker 853*6dbdd20aSAndroid Build Coastguard Workerif __name__ == '__main__': 854*6dbdd20aSAndroid Build Coastguard Worker sys.exit(main(sys.argv)) 855