156 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
			
		
		
	
	
			156 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
| #!/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).replace('\\', '/')
 | |
| 
 | |
| 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;
 | |
| 
 | |
| option go_package = "github.com/google/perfetto/perfetto_proto";
 | |
| '''
 | |
| 
 | |
| 
 | |
| 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())
 |