| #!/usr/bin/env python3 | 
 | # Copyright (C) 2024 The Android Open Source Project | 
 | # | 
 | # Licensed under the Apache License, Version 2.0 (the "License"); | 
 | # you may not use this file except in compliance with the License. | 
 | # You may obtain a copy of the License at | 
 | # | 
 | #      http://www.apache.org/licenses/LICENSE-2.0 | 
 | # | 
 | # Unless required by applicable law or agreed to in writing, software | 
 | # distributed under the License is distributed on an "AS IS" BASIS, | 
 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | # See the License for the specific language governing permissions and | 
 | # limitations under the License. | 
 |  | 
 | # !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! | 
 | # DO NOT EDIT. Auto-generated by tools/gen_amalgamated_python_tools | 
 | # !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! | 
 |  | 
 | import argparse | 
 | import os | 
 | import re | 
 | import sys | 
 | import tempfile | 
 | import time | 
 |  | 
 |  | 
 | # ----- Amalgamator: begin of python/perfetto/prebuilts/perfetto_prebuilts.py | 
 | # Copyright (C) 2021 The Android Open Source Project | 
 | # | 
 | # Licensed under the Apache License, Version 2.0 (the "License"); | 
 | # you may not use this file except in compliance with the License. | 
 | # You may obtain a copy of the License at | 
 | # | 
 | #      http://www.apache.org/licenses/LICENSE-2.0 | 
 | # | 
 | # Unless required by applicable law or agreed to in writing, software | 
 | # distributed under the License is distributed on an "AS IS" BASIS, | 
 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | # See the License for the specific language governing permissions and | 
 | # limitations under the License. | 
 | """ | 
 | Functions to fetch pre-pinned Perfetto prebuilts. | 
 |  | 
 | This function is used in different places: | 
 | - Into the //tools/{trace_processor, traceconv} scripts, which are just plain | 
 |   wrappers around executables. | 
 | - Into the //tools/{heap_profiler, record_android_trace} scripts, which contain | 
 |   some other hand-written python code. | 
 |  | 
 | The manifest argument looks as follows: | 
 | TRACECONV_MANIFEST = [ | 
 |   { | 
 |     'arch': 'mac-amd64', | 
 |     'file_name': 'traceconv', | 
 |     'file_size': 7087080, | 
 |     'url': https://commondatastorage.googleapis.com/.../trace_to_text', | 
 |     'sha256': 7d957c005b0dc130f5bd855d6cec27e060d38841b320d04840afc569f9087490', | 
 |     'platform': 'darwin', | 
 |     'machine': 'x86_64' | 
 |   }, | 
 |   ... | 
 | ] | 
 |  | 
 | The intended usage is: | 
 |  | 
 |   from perfetto.prebuilts.manifests.traceconv import TRACECONV_MANIFEST | 
 |   bin_path = get_perfetto_prebuilt(TRACECONV_MANIFEST) | 
 |   subprocess.call(bin_path, ...) | 
 | """ | 
 |  | 
 | import hashlib | 
 | import os | 
 | import platform | 
 | import random | 
 | import subprocess | 
 | import sys | 
 |  | 
 |  | 
 | def download_or_get_cached(file_name, url, sha256): | 
 |   """ Downloads a prebuilt or returns a cached version | 
 |  | 
 |   The first time this is invoked, it downloads the |url| and caches it into | 
 |   ~/.local/share/perfetto/prebuilts/$tool_name. On subsequent invocations it | 
 |   just runs the cached version. | 
 |   """ | 
 |   dir = os.path.join( | 
 |       os.path.expanduser('~'), '.local', 'share', 'perfetto', 'prebuilts') | 
 |   os.makedirs(dir, exist_ok=True) | 
 |   bin_path = os.path.join(dir, file_name) | 
 |   sha256_path = os.path.join(dir, file_name + '.sha256') | 
 |   needs_download = True | 
 |  | 
 |   # Avoid recomputing the SHA-256 on each invocation. The SHA-256 of the last | 
 |   # download is cached into file_name.sha256, just check if that matches. | 
 |   if os.path.exists(bin_path) and os.path.exists(sha256_path): | 
 |     with open(sha256_path, 'rb') as f: | 
 |       digest = f.read().decode() | 
 |       if digest == sha256: | 
 |         needs_download = False | 
 |  | 
 |   if needs_download:  # The file doesn't exist or the SHA256 doesn't match. | 
 |     # Use a unique random file to guard against concurrent executions. | 
 |     # See https://github.com/google/perfetto/issues/786 . | 
 |     tmp_path = '%s.%d.tmp' % (bin_path, random.randint(0, 100000)) | 
 |     print('Downloading ' + url) | 
 |     subprocess.check_call(['curl', '-f', '-L', '-#', '-o', tmp_path, url]) | 
 |     with open(tmp_path, 'rb') as fd: | 
 |       actual_sha256 = hashlib.sha256(fd.read()).hexdigest() | 
 |     if actual_sha256 != sha256: | 
 |       raise Exception('Checksum mismatch for %s (actual: %s, expected: %s)' % | 
 |                       (url, actual_sha256, sha256)) | 
 |     os.chmod(tmp_path, 0o755) | 
 |     os.replace(tmp_path, bin_path) | 
 |     with open(tmp_path, 'w') as f: | 
 |       f.write(sha256) | 
 |     os.replace(tmp_path, sha256_path) | 
 |   return bin_path | 
 |  | 
 |  | 
 | def get_perfetto_prebuilt(manifest, soft_fail=False, arch=None): | 
 |   """ Downloads the prebuilt, if necessary, and returns its path on disk. """ | 
 |   plat = sys.platform.lower() | 
 |   machine = platform.machine().lower() | 
 |   manifest_entry = None | 
 |   for entry in manifest: | 
 |     # If the caller overrides the arch, just match that (for Android prebuilts). | 
 |     if arch: | 
 |       if entry.get('arch') == arch: | 
 |         manifest_entry = entry | 
 |         break | 
 |       continue | 
 |     # Otherwise guess the local machine arch. | 
 |     if entry.get('platform') == plat and machine in entry.get('machine', []): | 
 |       manifest_entry = entry | 
 |       break | 
 |   if manifest_entry is None: | 
 |     if soft_fail: | 
 |       return None | 
 |     raise Exception( | 
 |         ('No prebuilts available for %s-%s\n' % (plat, machine)) + | 
 |         'See https://perfetto.dev/docs/contributing/build-instructions') | 
 |  | 
 |   return download_or_get_cached( | 
 |       file_name=manifest_entry['file_name'], | 
 |       url=manifest_entry['url'], | 
 |       sha256=manifest_entry['sha256']) | 
 |  | 
 |  | 
 | def run_perfetto_prebuilt(manifest): | 
 |   bin_path = get_perfetto_prebuilt(manifest) | 
 |   if sys.platform.lower() == 'win32': | 
 |     sys.exit(subprocess.check_call([bin_path, *sys.argv[1:]])) | 
 |   os.execv(bin_path, [bin_path] + sys.argv[1:]) | 
 |  | 
 | # ----- Amalgamator: end of python/perfetto/prebuilts/perfetto_prebuilts.py | 
 |  | 
 | PERMSISION_REGEX = re.compile(r'''uses-permission: name='(.*)'.*''') | 
 | NAME_REGEX = re.compile(r'''package: name='(.*?)' .*''') | 
 |  | 
 |  | 
 | def cmd(args: list[str]): | 
 |   print('Running command ' + ' '.join(args)) | 
 |   return subprocess.check_output(args) | 
 |  | 
 |  | 
 | def main(): | 
 |   parser = argparse.ArgumentParser() | 
 |   parser.add_argument('--apk', help='Local APK to use instead of builtin') | 
 |  | 
 |   args = parser.parse_args() | 
 |  | 
 |   if args.apk: | 
 |     apk = args.apk | 
 |   else: | 
 |     apk = download_or_get_cached( | 
 |         'CtsPerfettoReporterApp.apk', | 
 |         'https://storage.googleapis.com/perfetto/CtsPerfettoReporterApp.apk', | 
 |         'f21dda36668c368793500b13724ab2a6231d12ded05746f7cfaaba4adedd7d46') | 
 |  | 
 |   # Figure out the package name and the permissions we need | 
 |   aapt = subprocess.check_output(['aapt', 'dump', 'badging', apk]).decode() | 
 |   permission_names = [] | 
 |   name = '' | 
 |   for l in aapt.splitlines(): | 
 |     name_match = NAME_REGEX.match(l) | 
 |     if name_match: | 
 |       name = name_match[1] | 
 |       continue | 
 |  | 
 |     permission_match = PERMSISION_REGEX.match(l) | 
 |     if permission_match: | 
 |       permission_names.append(permission_match[1]) | 
 |       continue | 
 |  | 
 |   # Root and remount the device. | 
 |   cmd(['adb', 'root']) | 
 |   cmd(['adb', 'wait-for-device']) | 
 |   cmd(['adb', 'remount', '-R']) | 
 |   input('The device might now reboot. If so, please unlock the device then ' | 
 |         'press enter to continue') | 
 |   cmd(['adb', 'wait-for-device']) | 
 |   cmd(['adb', 'root']) | 
 |   cmd(['adb', 'wait-for-device']) | 
 |   cmd(['adb', 'remount', '-R']) | 
 |  | 
 |   # Write out the permission file on device. | 
 |   permissions = '\n'.join( | 
 |       f'''<permission name='{p}' />''' for p in permission_names) | 
 |   permission_file_contents = f''' | 
 |     <permissions> | 
 |       <privapp-permissions package="{name}"> | 
 |         {permissions} | 
 |       </privapp-permissions> | 
 |     </permissions> | 
 |   ''' | 
 |   with tempfile.NamedTemporaryFile() as f: | 
 |     f.write(permission_file_contents.encode()) | 
 |     f.flush() | 
 |  | 
 |     cmd([ | 
 |         'adb', 'push', f.name, | 
 |         f'/system/etc/permissions/privapp-permissions-{name}.xml' | 
 |     ]) | 
 |  | 
 |   # Stop system_server, push the apk on device and restart system_server | 
 |   priv_app_path = f'/system/priv-app/{name}/{name}.apk' | 
 |   cmd(['adb', 'shell', 'stop']) | 
 |   cmd(['adb', 'push', apk, priv_app_path]) | 
 |   cmd(['adb', 'shell', 'start']) | 
 |   cmd(['adb', 'wait-for-device']) | 
 |   time.sleep(10) | 
 |  | 
 |   # Wait for system_server and package manager to come up. | 
 |   while 'system_server' not in cmd(['adb', 'shell', 'ps']).decode(): | 
 |     time.sleep(1) | 
 |   while True: | 
 |     ps = set([ | 
 |         l.strip() | 
 |         for l in cmd(['adb', 'shell', 'dumpsys', '-l']).decode().splitlines() | 
 |     ]) | 
 |     if 'storaged' in ps and 'settings' in ps and 'package' in ps: | 
 |       break | 
 |     time.sleep(1) | 
 |  | 
 |   # Install the actual APK. | 
 |   cmd(['adb', 'shell', 'pm', 'install', '-r', '-d', '-g', '-t', priv_app_path]) | 
 |  | 
 |   return 0 | 
 |  | 
 |  | 
 | sys.exit(main()) |