xref: /aosp_15_r20/external/perfetto/tools/gen_merged_protos (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
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