#!/usr/bin/env python # Copyright (C) 2018 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. from __future__ import absolute_import from __future__ import division from __future__ import print_function import os import re import sys import argparse import tempfile import subprocess import hashlib import textwrap SOURCE_TARGET = { 'protos/perfetto/config/perfetto_config.proto': 'src/perfetto_cmd/perfetto_config.descriptor.h', } ROOT_DIR = os.path.dirname(os.path.dirname(os.path.realpath(__file__))) SCRIPT_PATH = 'tools/gen_binary_descriptors' def hash_path(path): hash = hashlib.sha1() with open(os.path.join(ROOT_DIR, path)) as f: hash.update(f.read()) return hash.hexdigest() def find_protoc(): for root, dirs, files in os.walk(ROOT_DIR): if 'protoc' in files: return os.path.join(root, 'protoc') for name in ('src', 'buildtools'): if name in dirs: dirs.remove(name) return None def check(source, target): assert os.path.exists(os.path.join(ROOT_DIR, target)), \ 'Output file {} does not exist and so cannot be checked'.format(target) with open(target, 'rb') as f: s = f.read() hashes = re.findall(r'// SHA1\((.*)\)\n// (.*)\n', s) assert sorted([SCRIPT_PATH, source]) == sorted([key for key, _ in hashes]) for path, expected_sha1 in hashes: actual_sha1 = hash_path(os.path.join(ROOT_DIR, path)) assert actual_sha1 == expected_sha1, \ 'In {} hash given for {} did not match'.format(target, path) def generate(source, target, protoc_path): _, source_name = os.path.split(source) _, target_name = os.path.split(target) assert source_name.replace('.proto', '.descriptor.h') == target_name with tempfile.NamedTemporaryFile() as fdescriptor: subprocess.check_call([ protoc_path, '--proto_path=protos', '-o{}'.format(fdescriptor.name), source, ], cwd=ROOT_DIR) s = fdescriptor.read() proto_name = source_name[:-len('.proto')].title().replace("_", "") constant_name = 'k' + proto_name + 'Descriptor' binary = '{' + ', '.join('{0:#04x}'.format(ord(c)) for c in s) + '}' binary = textwrap.fill(binary, width=80, initial_indent=' ', subsequent_indent=' ') include_guard = target.replace('/', '_').replace('.', '_').upper() + '_' with open(os.path.join(ROOT_DIR, target), 'wb') as f: f.write(""" #ifndef {include_guard} #define {include_guard} #include <stddef.h> #include <stdint.h> #include <array> // This file was autogenerated by tools/gen_binary_descriptors. Do not edit. // SHA1({script_path}) // {script_hash} // SHA1({source_path}) // {source_hash} // This is the proto {proto_name} encoded as a ProtoFileDescriptor to allow // for reflection without libprotobuf full/non-lite protos. namespace perfetto {{ constexpr std::array<uint8_t, {size}> {constant_name}{{ {binary}}}; }} // namespace perfetto #endif // {include_guard} """.format(**{ 'proto_name': proto_name, 'size': len(s), 'constant_name': constant_name, 'binary': binary, 'include_guard': include_guard, 'script_path': SCRIPT_PATH, 'script_hash': hash_path(__file__), 'source_path': source, 'source_hash': hash_path(os.path.join(source)), })) def main(): parser = argparse.ArgumentParser() parser.add_argument('--check-only', action='store_true') parser.add_argument('--protoc') args = parser.parse_args() try: for source, target in SOURCE_TARGET.iteritems(): if args.check_only: check(source, target) else: protoc = args.protoc or find_protoc() assert protoc, 'protoc not found specific (--protoc PROTOC_PATH)' assert os.path.exists(protoc), '{} does not exist'.format(protoc) if protoc is not args.protoc: print('Using protoc: {}'.format(protoc)) generate(source, target, protoc) except AssertionError as e: if not str(e): raise print('Error: {}'.format(e)) return 1 if __name__ == '__main__': exit(main())