1*6dbdd20aSAndroid Build Coastguard Worker#!/usr/bin/env python3 2*6dbdd20aSAndroid Build Coastguard Worker# Copyright (C) 2018 The Android Open Source Project 3*6dbdd20aSAndroid Build Coastguard Worker# 4*6dbdd20aSAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 5*6dbdd20aSAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 6*6dbdd20aSAndroid Build Coastguard Worker# You may obtain a copy of the License at 7*6dbdd20aSAndroid Build Coastguard Worker# 8*6dbdd20aSAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 9*6dbdd20aSAndroid Build Coastguard Worker# 10*6dbdd20aSAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 11*6dbdd20aSAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 12*6dbdd20aSAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*6dbdd20aSAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 14*6dbdd20aSAndroid Build Coastguard Worker# limitations under the License. 15*6dbdd20aSAndroid Build Coastguard Worker 16*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import absolute_import 17*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import division 18*6dbdd20aSAndroid Build Coastguard Workerfrom __future__ import print_function 19*6dbdd20aSAndroid Build Coastguard Workerimport os 20*6dbdd20aSAndroid Build Coastguard Workerimport re 21*6dbdd20aSAndroid Build Coastguard Workerimport sys 22*6dbdd20aSAndroid Build Coastguard Workerfrom codecs import open 23*6dbdd20aSAndroid Build Coastguard Worker 24*6dbdd20aSAndroid Build Coastguard WorkerPROJECT_ROOT = os.path.dirname(os.path.dirname(os.path.realpath(__file__))) 25*6dbdd20aSAndroid Build Coastguard WorkerSELF_PATH = os.path.relpath(__file__, PROJECT_ROOT).replace('\\', '/') 26*6dbdd20aSAndroid Build Coastguard Worker 27*6dbdd20aSAndroid Build Coastguard WorkerCONFIG_PROTO_ROOTS = [ 28*6dbdd20aSAndroid Build Coastguard Worker 'protos/perfetto/common/data_source_descriptor.proto', 29*6dbdd20aSAndroid Build Coastguard Worker 'protos/perfetto/common/tracing_service_state.proto', 30*6dbdd20aSAndroid Build Coastguard Worker 'protos/perfetto/config/trace_config.proto' 31*6dbdd20aSAndroid Build Coastguard Worker] 32*6dbdd20aSAndroid Build Coastguard WorkerMERGED_CONFIG_PROTO = 'protos/perfetto/config/perfetto_config.proto' 33*6dbdd20aSAndroid Build Coastguard Worker 34*6dbdd20aSAndroid Build Coastguard WorkerTRACE_PROTO_ROOTS = CONFIG_PROTO_ROOTS + [ 35*6dbdd20aSAndroid Build Coastguard Worker 'protos/perfetto/trace/trace.proto', 36*6dbdd20aSAndroid Build Coastguard Worker] 37*6dbdd20aSAndroid Build Coastguard WorkerMERGED_TRACE_PROTO = 'protos/perfetto/trace/perfetto_trace.proto' 38*6dbdd20aSAndroid Build Coastguard Worker 39*6dbdd20aSAndroid Build Coastguard WorkerMETRICS_PROTOS_ROOTS = ['protos/perfetto/metrics/metrics.proto'] 40*6dbdd20aSAndroid Build Coastguard WorkerMERGED_METRICS_PROTO = 'protos/perfetto/metrics/perfetto_merged_metrics.proto' 41*6dbdd20aSAndroid Build Coastguard Worker 42*6dbdd20aSAndroid Build Coastguard WorkerREPLACEMENT_HEADER = ''' 43*6dbdd20aSAndroid Build Coastguard Worker// AUTOGENERATED - DO NOT EDIT 44*6dbdd20aSAndroid Build Coastguard Worker// --------------------------- 45*6dbdd20aSAndroid Build Coastguard Worker// This file has been generated by 46*6dbdd20aSAndroid Build Coastguard Worker// AOSP://external/perfetto/%s 47*6dbdd20aSAndroid Build Coastguard Worker// merging the perfetto config protos. 48*6dbdd20aSAndroid Build Coastguard Worker// This fused proto is intended to be copied in: 49*6dbdd20aSAndroid Build Coastguard Worker// - Android tree, for statsd. 50*6dbdd20aSAndroid Build Coastguard Worker// - Google internal repos. 51*6dbdd20aSAndroid Build Coastguard Worker 52*6dbdd20aSAndroid Build Coastguard Workersyntax = "proto2"; 53*6dbdd20aSAndroid Build Coastguard Worker 54*6dbdd20aSAndroid Build Coastguard Workerpackage perfetto.protos; 55*6dbdd20aSAndroid Build Coastguard Worker 56*6dbdd20aSAndroid Build Coastguard Workeroption go_package = "github.com/google/perfetto/perfetto_proto"; 57*6dbdd20aSAndroid Build Coastguard Worker''' 58*6dbdd20aSAndroid Build Coastguard Worker 59*6dbdd20aSAndroid Build Coastguard Worker 60*6dbdd20aSAndroid Build Coastguard Workerdef get_transitive_imports(rel_path, visited): 61*6dbdd20aSAndroid Build Coastguard Worker if rel_path in visited: 62*6dbdd20aSAndroid Build Coastguard Worker return [] 63*6dbdd20aSAndroid Build Coastguard Worker visited.add(rel_path) 64*6dbdd20aSAndroid Build Coastguard Worker with open(os.path.join(PROJECT_ROOT, rel_path), 'r', encoding='utf-8') as f: 65*6dbdd20aSAndroid Build Coastguard Worker content = f.read() 66*6dbdd20aSAndroid Build Coastguard Worker imports = re.findall(r'^import "(.*)";\n', content, flags=re.MULTILINE) 67*6dbdd20aSAndroid Build Coastguard Worker res = [] 68*6dbdd20aSAndroid Build Coastguard Worker for child in sorted(imports): 69*6dbdd20aSAndroid Build Coastguard Worker res += get_transitive_imports(child, visited) 70*6dbdd20aSAndroid Build Coastguard Worker res += [rel_path] 71*6dbdd20aSAndroid Build Coastguard Worker return res 72*6dbdd20aSAndroid Build Coastguard Worker 73*6dbdd20aSAndroid Build Coastguard Worker 74*6dbdd20aSAndroid Build Coastguard Workerdef merge_protos_content(proto_paths): 75*6dbdd20aSAndroid Build Coastguard Worker merged_content = REPLACEMENT_HEADER.lstrip() % SELF_PATH 76*6dbdd20aSAndroid Build Coastguard Worker added_files = set() 77*6dbdd20aSAndroid Build Coastguard Worker for proto in proto_paths: 78*6dbdd20aSAndroid Build Coastguard Worker if proto in added_files: 79*6dbdd20aSAndroid Build Coastguard Worker continue 80*6dbdd20aSAndroid Build Coastguard Worker added_files.add(proto) 81*6dbdd20aSAndroid Build Coastguard Worker 82*6dbdd20aSAndroid Build Coastguard Worker path = os.path.join(PROJECT_ROOT, proto) 83*6dbdd20aSAndroid Build Coastguard Worker with open(path, 'r', encoding='utf-8') as f: 84*6dbdd20aSAndroid Build Coastguard Worker content = f.read() 85*6dbdd20aSAndroid Build Coastguard Worker 86*6dbdd20aSAndroid Build Coastguard Worker # Remove header 87*6dbdd20aSAndroid Build Coastguard Worker header = re.match(r'\/(\*|\/)(?:.|\s)*?package .*;\n', content) 88*6dbdd20aSAndroid Build Coastguard Worker header = header.group(0) 89*6dbdd20aSAndroid Build Coastguard Worker content = content[len(header):] 90*6dbdd20aSAndroid Build Coastguard Worker 91*6dbdd20aSAndroid Build Coastguard Worker content = re.sub(r'^import.*?\n\n?', '', content, flags=re.MULTILINE) 92*6dbdd20aSAndroid Build Coastguard Worker merged_content += '\n// Begin of %s\n' % proto 93*6dbdd20aSAndroid Build Coastguard Worker merged_content += content 94*6dbdd20aSAndroid Build Coastguard Worker merged_content += '\n// End of %s\n' % proto 95*6dbdd20aSAndroid Build Coastguard Worker 96*6dbdd20aSAndroid Build Coastguard Worker definitions_re = r'^ *(?:message|enum) ([A-Z][A-Za-z0-9].*) {' 97*6dbdd20aSAndroid Build Coastguard Worker definitions = re.finditer(definitions_re, merged_content, re.MULTILINE) 98*6dbdd20aSAndroid Build Coastguard Worker types = set((match.group(1) for match in definitions)) 99*6dbdd20aSAndroid Build Coastguard Worker 100*6dbdd20aSAndroid Build Coastguard Worker # Limitation: |types| doesn't track the nesting of messages, so a reference to 101*6dbdd20aSAndroid Build Coastguard Worker # a nested message (optional One.Two f = 1;) is simplified to its leafmost 102*6dbdd20aSAndroid Build Coastguard Worker # name (Two in this example). 103*6dbdd20aSAndroid Build Coastguard Worker uses_re = r'^( +)(?:repeated)?(?:optional)?\s?'\ 104*6dbdd20aSAndroid Build Coastguard Worker r'(?:[A-Z]\w+\.)*([A-Z]\w+)\s+[a-z]\w*\s*=\s*(\d+);' 105*6dbdd20aSAndroid Build Coastguard Worker uses = re.finditer(uses_re, merged_content, re.MULTILINE) 106*6dbdd20aSAndroid Build Coastguard Worker substitutions = [] 107*6dbdd20aSAndroid Build Coastguard Worker for use in uses: 108*6dbdd20aSAndroid Build Coastguard Worker everything = use.group(0) 109*6dbdd20aSAndroid Build Coastguard Worker indentation = use.group(1) 110*6dbdd20aSAndroid Build Coastguard Worker used_type = use.group(2) 111*6dbdd20aSAndroid Build Coastguard Worker field_number = use.group(3) 112*6dbdd20aSAndroid Build Coastguard Worker if used_type not in types: 113*6dbdd20aSAndroid Build Coastguard Worker replacement = '{}// removed field with id {}'.format( 114*6dbdd20aSAndroid Build Coastguard Worker indentation, field_number) 115*6dbdd20aSAndroid Build Coastguard Worker substitutions.append((everything, replacement)) 116*6dbdd20aSAndroid Build Coastguard Worker 117*6dbdd20aSAndroid Build Coastguard Worker for before, after in substitutions: 118*6dbdd20aSAndroid Build Coastguard Worker merged_content = merged_content.replace(before, after) 119*6dbdd20aSAndroid Build Coastguard Worker 120*6dbdd20aSAndroid Build Coastguard Worker return merged_content 121*6dbdd20aSAndroid Build Coastguard Worker 122*6dbdd20aSAndroid Build Coastguard Worker 123*6dbdd20aSAndroid Build Coastguard Workerdef merge_protos(root_paths, output_path): 124*6dbdd20aSAndroid Build Coastguard Worker all_protos = [] 125*6dbdd20aSAndroid Build Coastguard Worker for root_path in root_paths: 126*6dbdd20aSAndroid Build Coastguard Worker all_protos += get_transitive_imports(root_path, visited=set()) 127*6dbdd20aSAndroid Build Coastguard Worker merged_content = merge_protos_content(all_protos) 128*6dbdd20aSAndroid Build Coastguard Worker 129*6dbdd20aSAndroid Build Coastguard Worker out_path = os.path.join(PROJECT_ROOT, output_path) 130*6dbdd20aSAndroid Build Coastguard Worker prev_content = None 131*6dbdd20aSAndroid Build Coastguard Worker if os.path.exists(out_path): 132*6dbdd20aSAndroid Build Coastguard Worker with open(out_path, 'r', encoding='utf-8') as fprev: 133*6dbdd20aSAndroid Build Coastguard Worker prev_content = fprev.read() 134*6dbdd20aSAndroid Build Coastguard Worker 135*6dbdd20aSAndroid Build Coastguard Worker if prev_content == merged_content: 136*6dbdd20aSAndroid Build Coastguard Worker return True 137*6dbdd20aSAndroid Build Coastguard Worker 138*6dbdd20aSAndroid Build Coastguard Worker if '--check-only' in sys.argv: 139*6dbdd20aSAndroid Build Coastguard Worker return False 140*6dbdd20aSAndroid Build Coastguard Worker 141*6dbdd20aSAndroid Build Coastguard Worker print('Updating {}'.format(output_path)) 142*6dbdd20aSAndroid Build Coastguard Worker with open(out_path, 'w', encoding='utf-8') as fout: 143*6dbdd20aSAndroid Build Coastguard Worker fout.write(merged_content) 144*6dbdd20aSAndroid Build Coastguard Worker return True 145*6dbdd20aSAndroid Build Coastguard Worker 146*6dbdd20aSAndroid Build Coastguard Worker 147*6dbdd20aSAndroid Build Coastguard Workerdef main(): 148*6dbdd20aSAndroid Build Coastguard Worker result = merge_protos(CONFIG_PROTO_ROOTS, MERGED_CONFIG_PROTO) 149*6dbdd20aSAndroid Build Coastguard Worker result &= merge_protos(TRACE_PROTO_ROOTS, MERGED_TRACE_PROTO) 150*6dbdd20aSAndroid Build Coastguard Worker result &= merge_protos(METRICS_PROTOS_ROOTS, MERGED_METRICS_PROTO) 151*6dbdd20aSAndroid Build Coastguard Worker return 0 if result else 1 152*6dbdd20aSAndroid Build Coastguard Worker 153*6dbdd20aSAndroid Build Coastguard Worker 154*6dbdd20aSAndroid Build Coastguard Workerif __name__ == '__main__': 155*6dbdd20aSAndroid Build Coastguard Worker sys.exit(main()) 156