#!/usr/bin/env python3 # 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. # !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! # DO NOT EDIT. Auto-generated by tools/gen_amalgamated_python_tools # !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! # This file should do the same thing when being invoked in any of these ways: # ./traceconv # python traceconv # bash traceconv # cat ./traceconv | bash # cat ./traceconv | python - BASH_FALLBACK=""" " exec python3 - "$@" <<'#'EOF #""" # yapf: disable # ----- Amalgamator: begin of python/perfetto/prebuilts/manifests/traceconv.py # This file has been generated by: tools/release/roll-prebuilts v54.0 TRACECONV_MANIFEST = [{ 'arch': 'mac-amd64', 'file_name': 'traceconv', 'file_size': 11174856, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/mac-amd64/traceconv', 'sha256': '554ecfa7fff1ed35ed0d046e244d42d27b5e9f0067b08db68e5cb7f9eace879a', 'platform': 'darwin', 'machine': ['x86_64'] }, { 'arch': 'mac-arm64', 'file_name': 'traceconv', 'file_size': 10357864, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/mac-arm64/traceconv', 'sha256': '3976ef9871626ce28ceb32a36c657505a21f63c02837bb0ea51081412d79f4d7', 'platform': 'darwin', 'machine': ['arm64'] }, { 'arch': 'linux-amd64', 'file_name': 'traceconv', 'file_size': 11352048, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/linux-amd64/traceconv', 'sha256': '419bc40b76f23a5705def8624d5fdf31fd42ba1e3816fd056bf93f431cc156d1', 'platform': 'linux', 'machine': ['x86_64'] }, { 'arch': 'linux-arm', 'file_name': 'traceconv', 'file_size': 8655236, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/linux-arm/traceconv', 'sha256': '086721b87c8d3d865fb063f3f6e65127457884eb8e5da9f78eae2313a7383772', 'platform': 'linux', 'machine': ['armv6l', 'armv7l', 'armv8l'] }, { 'arch': 'linux-arm64', 'file_name': 'traceconv', 'file_size': 10772720, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/linux-arm64/traceconv', 'sha256': '1faf695042ccc907d66d663928a2c62d8f5ece16623f97a3a3260cda67e8e15a', 'platform': 'linux', 'machine': ['aarch64'] }, { 'arch': 'android-arm', 'file_name': 'traceconv', 'file_size': 8664936, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/android-arm/traceconv', 'sha256': 'ccd20eb089b956e2229062680b991ec6f526332353ada57ca034dae2667c481f' }, { 'arch': 'android-arm64', 'file_name': 'traceconv', 'file_size': 10696848, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/android-arm64/traceconv', 'sha256': 'ba2ecc2978fd28e1a6e629e13a54d365c7f9cb0828dbf9aea27e585f1b79a030' }, { 'arch': 'android-x86', 'file_name': 'traceconv', 'file_size': 11869456, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/android-x86/traceconv', 'sha256': '5c659c672126e844ec8465bcb224ebeddf702f411756f5bbc1168d61d3312ce7' }, { 'arch': 'android-x64', 'file_name': 'traceconv', 'file_size': 11188840, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/android-x64/traceconv', 'sha256': '8d253f5a24e10a1f1022317670bb82cc1b0c6a5545d5ca3df41762d93152d01e' }, { 'arch': 'windows-amd64', 'file_name': 'traceconv.exe', 'file_size': 11050496, 'url': 'https://commondatastorage.googleapis.com/perfetto-luci-artifacts/v54.0/windows-amd64/traceconv.exe', 'sha256': '795e60d3649c9c243be52e9e310327c3f88b441e1514b666fa690aaace770cdd', 'platform': 'win32', 'machine': ['amd64'] }] # ----- Amalgamator: end of python/perfetto/prebuilts/manifests/traceconv.py # ----- 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') # Placeholder entries (e.g. before a release has been rolled) have an empty # URL. Treat them the same as a missing entry when soft_fail is set. if not manifest_entry.get('url'): if soft_fail: return None raise Exception('No prebuilt URL available for %s on %s-%s. ' 'The prebuilt may not have been rolled yet.' % (manifest_entry.get('file_name', '?'), plat, machine)) 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 if __name__ == '__main__': run_perfetto_prebuilt(TRACECONV_MANIFEST) #EOF