|  | #!/usr/bin/env python | 
|  | # | 
|  | # Copyright (C) 2019 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. | 
|  | # | 
|  |  | 
|  | """ | 
|  | This script is part of controlling simpleperf recording in user code. It is used to prepare | 
|  | profiling environment (upload simpleperf to device and enable profiling) before recording | 
|  | and collect recording data on host after recording. | 
|  | Controlling simpleperf recording is done in below steps: | 
|  | 1. Add simpleperf Java API/C++ API to the app's source code. And call the API in user code. | 
|  | 2. Run `api_profiler.py prepare` to prepare profiling environment. | 
|  | 3. Run the app one or more times to generate recording data. | 
|  | 4. Run `api_profiler.py collect` to collect recording data on host. | 
|  | """ | 
|  |  | 
|  | from __future__ import print_function | 
|  | import argparse | 
|  | import os | 
|  | import os.path | 
|  | import shutil | 
|  | import zipfile | 
|  |  | 
|  | from utils import AdbHelper, get_target_binary_path, log_exit, log_info, remove | 
|  |  | 
|  | def prepare_recording(args): | 
|  | adb = AdbHelper() | 
|  | enable_profiling_on_device(adb, args) | 
|  | upload_simpleperf_to_device(adb) | 
|  | run_simpleperf_prepare_cmd(adb) | 
|  |  | 
|  | def enable_profiling_on_device(adb, args): | 
|  | android_version = adb.get_android_version() | 
|  | if android_version >= 10: | 
|  | adb.set_property('debug.perf_event_max_sample_rate', str(args.max_sample_rate[0])) | 
|  | adb.set_property('debug.perf_cpu_time_max_percent', str(args.max_cpu_percent[0])) | 
|  | adb.set_property('debug.perf_event_mlock_kb', str(args.max_memory_in_kb[0])) | 
|  | adb.set_property('security.perf_harden', '0') | 
|  |  | 
|  | def upload_simpleperf_to_device(adb): | 
|  | device_arch = adb.get_device_arch() | 
|  | simpleperf_binary = get_target_binary_path(device_arch, 'simpleperf') | 
|  | adb.check_run(['push', simpleperf_binary, '/data/local/tmp']) | 
|  | adb.check_run(['shell', 'chmod', 'a+x', '/data/local/tmp/simpleperf']) | 
|  |  | 
|  | def run_simpleperf_prepare_cmd(adb): | 
|  | adb.check_run(['shell', '/data/local/tmp/simpleperf', 'api-prepare']) | 
|  |  | 
|  |  | 
|  | def collect_data(args): | 
|  | adb = AdbHelper() | 
|  | if not os.path.isdir(args.out_dir): | 
|  | os.makedirs(args.out_dir) | 
|  | download_recording_data(adb, args) | 
|  | unzip_recording_data(args) | 
|  |  | 
|  | def download_recording_data(adb, args): | 
|  | """ download recording data to simpleperf_data.zip.""" | 
|  | upload_simpleperf_to_device(adb) | 
|  | adb.check_run(['shell', '/data/local/tmp/simpleperf', 'api-collect', '--app', args.app[0], | 
|  | '-o', '/data/local/tmp/simpleperf_data.zip']) | 
|  | adb.check_run(['pull', '/data/local/tmp/simpleperf_data.zip', args.out_dir]) | 
|  | adb.check_run(['shell', 'rm', '-rf', '/data/local/tmp/simpleperf_data']) | 
|  |  | 
|  | def unzip_recording_data(args): | 
|  | zip_file_path = os.path.join(args.out_dir, 'simpleperf_data.zip') | 
|  | with zipfile.ZipFile(zip_file_path, 'r') as zip_fh: | 
|  | names = zip_fh.namelist() | 
|  | log_info('There are %d recording data files.' % len(names)) | 
|  | for name in names: | 
|  | log_info('recording file: %s' % os.path.join(args.out_dir, name)) | 
|  | zip_fh.extract(name, args.out_dir) | 
|  | remove(zip_file_path) | 
|  |  | 
|  | class ArgumentHelpFormatter(argparse.ArgumentDefaultsHelpFormatter, | 
|  | argparse.RawDescriptionHelpFormatter): | 
|  | pass | 
|  |  | 
|  | def main(): | 
|  | parser = argparse.ArgumentParser(description=__doc__, | 
|  | formatter_class=ArgumentHelpFormatter) | 
|  | subparsers = parser.add_subparsers() | 
|  | prepare_parser = subparsers.add_parser('prepare', help='Prepare recording on device.', | 
|  | formatter_class=ArgumentHelpFormatter) | 
|  | prepare_parser.add_argument('--max-sample-rate', nargs=1, type=int, default=[100000], help=""" | 
|  | Set max sample rate (only on Android >= Q).""") | 
|  | prepare_parser.add_argument('--max-cpu-percent', nargs=1, type=int, default=[25], help=""" | 
|  | Set max cpu percent for recording (only on Android >= Q).""") | 
|  | prepare_parser.add_argument('--max-memory-in-kb', nargs=1, type=int, | 
|  | default=[(1024 + 1) * 4 * 8], help=""" | 
|  | Set max kernel buffer size for recording (only on Android >= Q). | 
|  | """) | 
|  | prepare_parser.set_defaults(func=prepare_recording) | 
|  | collect_parser = subparsers.add_parser('collect', help='Collect recording data.', | 
|  | formatter_class=ArgumentHelpFormatter) | 
|  | collect_parser.add_argument('-p', '--app', nargs=1, required=True, help=""" | 
|  | The app package name of the app profiled.""") | 
|  | collect_parser.add_argument('-o', '--out-dir', default='simpleperf_data', help=""" | 
|  | The directory to store recording data.""") | 
|  | collect_parser.set_defaults(func=collect_data) | 
|  | args = parser.parse_args() | 
|  | args.func(args) | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | main() |