#!/usr/bin/env python3 # 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 from codecs import open PROJECT_ROOT = os.path.dirname(os.path.dirname(os.path.realpath(__file__))) SELF_PATH = os.path.relpath(__file__, PROJECT_ROOT) CONFIG_PROTO_ROOTS = [ 'protos/perfetto/common/data_source_descriptor.proto', 'protos/perfetto/common/tracing_service_state.proto', 'protos/perfetto/config/trace_config.proto' ] MERGED_CONFIG_PROTO = 'protos/perfetto/config/perfetto_config.proto' TRACE_PROTO_ROOTS = CONFIG_PROTO_ROOTS + [ 'protos/perfetto/trace/trace.proto', ] MERGED_TRACE_PROTO = 'protos/perfetto/trace/perfetto_trace.proto' METRICS_PROTOS_ROOTS = ['protos/perfetto/metrics/metrics.proto'] MERGED_METRICS_PROTO = 'protos/perfetto/metrics/perfetto_merged_metrics.proto' REPLACEMENT_HEADER = ''' // AUTOGENERATED - DO NOT EDIT // --------------------------- // This file has been generated by // AOSP://external/perfetto/%s // merging the perfetto config protos. // This fused proto is intended to be copied in: // - Android tree, for statsd. // - Google internal repos. syntax = "proto2"; package perfetto.protos; ''' def get_transitive_imports(rel_path, visited): if rel_path in visited: return [] visited.add(rel_path) with open(os.path.join(PROJECT_ROOT, rel_path), 'r', encoding='utf-8') as f: content = f.read() imports = re.findall(r'^import "(.*)";\n', content, flags=re.MULTILINE) res = [] for child in sorted(imports): res += get_transitive_imports(child, visited) res += [rel_path] return res def merge_protos_content(proto_paths): merged_content = REPLACEMENT_HEADER.lstrip() % SELF_PATH added_files = set() for proto in proto_paths: if proto in added_files: continue added_files.add(proto) path = os.path.join(PROJECT_ROOT, proto) with open(path, 'r', encoding='utf-8') as f: content = f.read() # Remove header header = re.match(r'\/(\*|\/)(?:.|\s)*?package .*;\n', content) header = header.group(0) content = content[len(header):] content = re.sub(r'^import.*?\n\n?', '', content, flags=re.MULTILINE) merged_content += '\n// Begin of %s\n' % proto merged_content += content merged_content += '\n// End of %s\n' % proto definitions_re = r'^ *(?:message|enum) ([A-Z][A-Za-z0-9].*) {' definitions = re.finditer(definitions_re, merged_content, re.MULTILINE) types = set((match.group(1) for match in definitions)) # Limitation: |types| doesn't track the nesting of messages, so a reference to # a nested message (optional One.Two f = 1;) is simplified to its leafmost # name (Two in this example). uses_re = r'^( +)(?:repeated)?(?:optional)?\s?'\ r'(?:[A-Z]\w+\.)*([A-Z]\w+)\s+[a-z]\w*\s*=\s*(\d+);' uses = re.finditer(uses_re, merged_content, re.MULTILINE) substitutions = [] for use in uses: everything = use.group(0) indentation = use.group(1) used_type = use.group(2) field_number = use.group(3) if used_type not in types: replacement = '{}// removed field with id {}'.format( indentation, field_number) substitutions.append((everything, replacement)) for before, after in substitutions: merged_content = merged_content.replace(before, after) return merged_content def merge_protos(root_paths, output_path): all_protos = [] for root_path in root_paths: all_protos += get_transitive_imports(root_path, visited=set()) merged_content = merge_protos_content(all_protos) out_path = os.path.join(PROJECT_ROOT, output_path) prev_content = None if os.path.exists(out_path): with open(out_path, 'r', encoding='utf-8') as fprev: prev_content = fprev.read() if prev_content == merged_content: return True if '--check-only' in sys.argv: return False print('Updating {}'.format(output_path)) with open(out_path, 'w', encoding='utf-8') as fout: fout.write(merged_content) return True def main(): result = merge_protos(CONFIG_PROTO_ROOTS, MERGED_CONFIG_PROTO) result &= merge_protos(TRACE_PROTO_ROOTS, MERGED_TRACE_PROTO) result &= merge_protos(METRICS_PROTOS_ROOTS, MERGED_METRICS_PROTO) return 0 if result else 1 if __name__ == '__main__': sys.exit(main())