Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 1 | #!/usr/bin/env python |
| 2 | # Copyright (C) 2017 The Android Open Source Project |
| 3 | # |
| 4 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | # you may not use this file except in compliance with the License. |
| 6 | # You may obtain a copy of the License at |
| 7 | # |
| 8 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | # |
| 10 | # Unless required by applicable law or agreed to in writing, software |
| 11 | # distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | # See the License for the specific language governing permissions and |
| 14 | # limitations under the License. |
| 15 | |
| 16 | # This tool translates a collection of BUILD.gn files into a mostly equivalent |
| 17 | # Android.bp file for the Android Soong build system. The input to the tool is a |
| 18 | # JSON description of the GN build definition generated with the following |
| 19 | # command: |
| 20 | # |
| 21 | # gn desc out --format=json --all-toolchains "//*" > desc.json |
| 22 | # |
| 23 | # The tool is then given a list of GN labels for which to generate Android.bp |
| 24 | # build rules. The dependencies for the GN labels are squashed to the generated |
| 25 | # Android.bp target, except for actions which get their own genrule. Some |
| 26 | # libraries are also mapped to their Android equivalents -- see |builtin_deps|. |
| 27 | |
| 28 | import argparse |
Primiano Tucci | edf099c | 2018-01-08 18:27:56 +0000 | [diff] [blame] | 29 | import errno |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 30 | import json |
| 31 | import os |
| 32 | import re |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 33 | import shutil |
| 34 | import subprocess |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 35 | import sys |
| 36 | |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 37 | # Default targets to translate to the blueprint file. |
Primiano Tucci | 4e49c02 | 2017-12-21 18:22:44 +0100 | [diff] [blame] | 38 | default_targets = [ |
Primiano Tucci | edf099c | 2018-01-08 18:27:56 +0000 | [diff] [blame] | 39 | '//:libtraced_shared', |
Primiano Tucci | 4e49c02 | 2017-12-21 18:22:44 +0100 | [diff] [blame] | 40 | '//:perfetto_tests', |
Primiano Tucci | 3b72910 | 2018-01-08 18:16:36 +0000 | [diff] [blame] | 41 | '//:perfetto', |
Primiano Tucci | 4e49c02 | 2017-12-21 18:22:44 +0100 | [diff] [blame] | 42 | '//:traced', |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 43 | '//:traced_probes', |
Primiano Tucci | 4e49c02 | 2017-12-21 18:22:44 +0100 | [diff] [blame] | 44 | ] |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 45 | |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 46 | # Defines a custom init_rc argument to be applied to the corresponding output |
| 47 | # blueprint target. |
Primiano Tucci | 5a30453 | 2018-01-09 14:15:43 +0000 | [diff] [blame] | 48 | target_initrc = { |
| 49 | '//:traced': 'perfetto.rc', |
| 50 | } |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 51 | |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 52 | # Arguments for the GN output directory. |
Primiano Tucci | edf099c | 2018-01-08 18:27:56 +0000 | [diff] [blame] | 53 | gn_args = 'target_os="android" target_cpu="arm" is_debug=false build_with_android=true' |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 54 | |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 55 | # All module names are prefixed with this string to avoid collisions. |
| 56 | module_prefix = 'perfetto_' |
| 57 | |
| 58 | # Shared libraries which are directly translated to Android system equivalents. |
| 59 | library_whitelist = [ |
| 60 | 'android', |
Sami Kyostila | b5b7169 | 2018-01-12 12:16:44 +0000 | [diff] [blame] | 61 | 'binder', |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 62 | 'log', |
Sami Kyostila | b5b7169 | 2018-01-12 12:16:44 +0000 | [diff] [blame] | 63 | 'services', |
Primiano Tucci | edf099c | 2018-01-08 18:27:56 +0000 | [diff] [blame] | 64 | 'utils', |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 65 | ] |
| 66 | |
| 67 | # Name of the module which settings such as compiler flags for all other |
| 68 | # modules. |
| 69 | defaults_module = module_prefix + 'defaults' |
| 70 | |
| 71 | # Location of the project in the Android source tree. |
| 72 | tree_path = 'external/perfetto' |
| 73 | |
Primiano Tucci | edf099c | 2018-01-08 18:27:56 +0000 | [diff] [blame] | 74 | # Compiler flags which are passed through to the blueprint. |
| 75 | cflag_whitelist = r'^-DPERFETTO.*$' |
| 76 | |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 77 | |
| 78 | def enable_gmock(module): |
| 79 | module.static_libs.append('libgmock') |
| 80 | |
| 81 | |
Hector Dearman | 3e712a0 | 2017-12-19 16:39:59 +0000 | [diff] [blame] | 82 | def enable_gtest_prod(module): |
| 83 | module.static_libs.append('libgtest_prod') |
| 84 | |
| 85 | |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 86 | def enable_gtest(module): |
| 87 | assert module.type == 'cc_test' |
| 88 | |
| 89 | |
| 90 | def enable_protobuf_full(module): |
| 91 | module.shared_libs.append('libprotobuf-cpp-full') |
| 92 | |
| 93 | |
| 94 | def enable_protobuf_lite(module): |
| 95 | module.shared_libs.append('libprotobuf-cpp-lite') |
| 96 | |
| 97 | |
| 98 | def enable_protoc_lib(module): |
| 99 | module.shared_libs.append('libprotoc') |
| 100 | |
| 101 | |
| 102 | def enable_libunwind(module): |
Sami Kyostila | fc074d4 | 2017-12-15 10:33:42 +0000 | [diff] [blame] | 103 | # libunwind is disabled on Darwin so we cannot depend on it. |
| 104 | pass |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 105 | |
| 106 | |
| 107 | # Android equivalents for third-party libraries that the upstream project |
| 108 | # depends on. |
| 109 | builtin_deps = { |
| 110 | '//buildtools:gmock': enable_gmock, |
| 111 | '//buildtools:gtest': enable_gtest, |
Hector Dearman | 3e712a0 | 2017-12-19 16:39:59 +0000 | [diff] [blame] | 112 | '//gn:gtest_prod_config': enable_gtest_prod, |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 113 | '//buildtools:gtest_main': enable_gtest, |
| 114 | '//buildtools:libunwind': enable_libunwind, |
| 115 | '//buildtools:protobuf_full': enable_protobuf_full, |
| 116 | '//buildtools:protobuf_lite': enable_protobuf_lite, |
| 117 | '//buildtools:protoc_lib': enable_protoc_lib, |
| 118 | } |
| 119 | |
| 120 | # ---------------------------------------------------------------------------- |
| 121 | # End of configuration. |
| 122 | # ---------------------------------------------------------------------------- |
| 123 | |
| 124 | |
| 125 | class Error(Exception): |
| 126 | pass |
| 127 | |
| 128 | |
| 129 | class ThrowingArgumentParser(argparse.ArgumentParser): |
| 130 | def __init__(self, context): |
| 131 | super(ThrowingArgumentParser, self).__init__() |
| 132 | self.context = context |
| 133 | |
| 134 | def error(self, message): |
| 135 | raise Error('%s: %s' % (self.context, message)) |
| 136 | |
| 137 | |
| 138 | class Module(object): |
| 139 | """A single module (e.g., cc_binary, cc_test) in a blueprint.""" |
| 140 | |
| 141 | def __init__(self, mod_type, name): |
| 142 | self.type = mod_type |
| 143 | self.name = name |
| 144 | self.srcs = [] |
| 145 | self.comment = None |
| 146 | self.shared_libs = [] |
| 147 | self.static_libs = [] |
| 148 | self.tools = [] |
| 149 | self.cmd = None |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 150 | self.init_rc = [] |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 151 | self.out = [] |
| 152 | self.export_include_dirs = [] |
| 153 | self.generated_headers = [] |
Lalit Maganti | c5bcd79 | 2018-01-12 18:38:11 +0000 | [diff] [blame] | 154 | self.export_generated_headers = [] |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 155 | self.defaults = [] |
| 156 | self.cflags = [] |
| 157 | self.local_include_dirs = [] |
| 158 | |
| 159 | def to_string(self, output): |
| 160 | if self.comment: |
| 161 | output.append('// %s' % self.comment) |
| 162 | output.append('%s {' % self.type) |
| 163 | self._output_field(output, 'name') |
| 164 | self._output_field(output, 'srcs') |
| 165 | self._output_field(output, 'shared_libs') |
| 166 | self._output_field(output, 'static_libs') |
| 167 | self._output_field(output, 'tools') |
| 168 | self._output_field(output, 'cmd', sort=False) |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 169 | self._output_field(output, 'init_rc') |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 170 | self._output_field(output, 'out') |
| 171 | self._output_field(output, 'export_include_dirs') |
| 172 | self._output_field(output, 'generated_headers') |
Lalit Maganti | c5bcd79 | 2018-01-12 18:38:11 +0000 | [diff] [blame] | 173 | self._output_field(output, 'export_generated_headers') |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 174 | self._output_field(output, 'defaults') |
| 175 | self._output_field(output, 'cflags') |
| 176 | self._output_field(output, 'local_include_dirs') |
| 177 | output.append('}') |
| 178 | output.append('') |
| 179 | |
| 180 | def _output_field(self, output, name, sort=True): |
| 181 | value = getattr(self, name) |
| 182 | if not value: |
| 183 | return |
| 184 | if isinstance(value, list): |
| 185 | output.append(' %s: [' % name) |
| 186 | for item in sorted(value) if sort else value: |
| 187 | output.append(' "%s",' % item) |
| 188 | output.append(' ],') |
| 189 | else: |
| 190 | output.append(' %s: "%s",' % (name, value)) |
| 191 | |
| 192 | |
| 193 | class Blueprint(object): |
| 194 | """In-memory representation of an Android.bp file.""" |
| 195 | |
| 196 | def __init__(self): |
| 197 | self.modules = {} |
| 198 | |
| 199 | def add_module(self, module): |
| 200 | """Adds a new module to the blueprint, replacing any existing module |
| 201 | with the same name. |
| 202 | |
| 203 | Args: |
| 204 | module: Module instance. |
| 205 | """ |
| 206 | self.modules[module.name] = module |
| 207 | |
| 208 | def to_string(self, output): |
Sami Kyostila | ebba0fe | 2017-12-19 14:01:52 +0000 | [diff] [blame] | 209 | for m in sorted(self.modules.itervalues(), key=lambda m: m.name): |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 210 | m.to_string(output) |
| 211 | |
| 212 | |
| 213 | def label_to_path(label): |
| 214 | """Turn a GN output label (e.g., //some_dir/file.cc) into a path.""" |
| 215 | assert label.startswith('//') |
| 216 | return label[2:] |
| 217 | |
| 218 | |
| 219 | def label_to_module_name(label): |
| 220 | """Turn a GN label (e.g., //:perfetto_tests) into a module name.""" |
Primiano Tucci | 4e49c02 | 2017-12-21 18:22:44 +0100 | [diff] [blame] | 221 | module = re.sub(r'^//:?', '', label) |
| 222 | module = re.sub(r'[^a-zA-Z0-9_]', '_', module) |
| 223 | if not module.startswith(module_prefix) and label not in default_targets: |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 224 | return module_prefix + module |
| 225 | return module |
| 226 | |
| 227 | |
| 228 | def label_without_toolchain(label): |
| 229 | """Strips the toolchain from a GN label. |
| 230 | |
| 231 | Return a GN label (e.g //buildtools:protobuf(//gn/standalone/toolchain: |
| 232 | gcc_like_host) without the parenthesised toolchain part. |
| 233 | """ |
| 234 | return label.split('(')[0] |
| 235 | |
| 236 | |
| 237 | def is_supported_source_file(name): |
| 238 | """Returns True if |name| can appear in a 'srcs' list.""" |
| 239 | return os.path.splitext(name)[1] in ['.c', '.cc', '.proto'] |
| 240 | |
| 241 | |
| 242 | def is_generated_by_action(desc, label): |
| 243 | """Checks if a label is generated by an action. |
| 244 | |
| 245 | Returns True if a GN output label |label| is an output for any action, |
| 246 | i.e., the file is generated dynamically. |
| 247 | """ |
| 248 | for target in desc.itervalues(): |
| 249 | if target['type'] == 'action' and label in target['outputs']: |
| 250 | return True |
| 251 | return False |
| 252 | |
| 253 | |
| 254 | def apply_module_dependency(blueprint, desc, module, dep_name): |
| 255 | """Recursively collect dependencies for a given module. |
| 256 | |
| 257 | Walk the transitive dependencies for a GN target and apply them to a given |
| 258 | module. This effectively flattens the dependency tree so that |module| |
| 259 | directly contains all the sources, libraries, etc. in the corresponding GN |
| 260 | dependency tree. |
| 261 | |
| 262 | Args: |
| 263 | blueprint: Blueprint instance which is being generated. |
| 264 | desc: JSON GN description. |
| 265 | module: Module to which dependencies should be added. |
| 266 | dep_name: GN target of the dependency. |
| 267 | """ |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 268 | # If the dependency refers to a library which we can replace with an Android |
| 269 | # equivalent, stop recursing and patch the dependency in. |
| 270 | if label_without_toolchain(dep_name) in builtin_deps: |
| 271 | builtin_deps[label_without_toolchain(dep_name)](module) |
| 272 | return |
| 273 | |
| 274 | # Similarly some shared libraries are directly mapped to Android |
| 275 | # equivalents. |
| 276 | target = desc[dep_name] |
| 277 | for lib in target.get('libs', []): |
| 278 | android_lib = 'lib' + lib |
| 279 | if lib in library_whitelist and not android_lib in module.shared_libs: |
| 280 | module.shared_libs.append(android_lib) |
| 281 | |
| 282 | type = target['type'] |
| 283 | if type == 'action': |
| 284 | create_modules_from_target(blueprint, desc, dep_name) |
| 285 | # Depend both on the generated sources and headers -- see |
| 286 | # make_genrules_for_action. |
| 287 | module.srcs.append(':' + label_to_module_name(dep_name)) |
| 288 | module.generated_headers.append( |
| 289 | label_to_module_name(dep_name) + '_headers') |
Sami Kyostila | ebba0fe | 2017-12-19 14:01:52 +0000 | [diff] [blame] | 290 | elif type == 'static_library' and label_to_module_name( |
| 291 | dep_name) != module.name: |
| 292 | create_modules_from_target(blueprint, desc, dep_name) |
| 293 | module.static_libs.append(label_to_module_name(dep_name)) |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 294 | elif type == 'shared_library' and label_to_module_name( |
| 295 | dep_name) != module.name: |
| 296 | module.shared_libs.append(label_to_module_name(dep_name)) |
Sami Kyostila | ebba0fe | 2017-12-19 14:01:52 +0000 | [diff] [blame] | 297 | elif type in ['group', 'source_set', 'executable', 'static_library' |
| 298 | ] and 'sources' in target: |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 299 | # Ignore source files that are generated by actions since they will be |
| 300 | # implicitly added by the genrule dependencies. |
| 301 | module.srcs.extend( |
| 302 | label_to_path(src) for src in target['sources'] |
| 303 | if is_supported_source_file(src) |
| 304 | and not is_generated_by_action(desc, src)) |
| 305 | |
| 306 | |
| 307 | def make_genrules_for_action(blueprint, desc, target_name): |
| 308 | """Generate genrules for a GN action. |
| 309 | |
| 310 | GN actions are used to dynamically generate files during the build. The |
| 311 | Soong equivalent is a genrule. This function turns a specific kind of |
| 312 | genrule which turns .proto files into source and header files into a pair |
Sami Kyostila | 71625d7 | 2017-12-18 10:29:49 +0000 | [diff] [blame] | 313 | equivalent genrules. |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 314 | |
| 315 | Args: |
| 316 | blueprint: Blueprint instance which is being generated. |
| 317 | desc: JSON GN description. |
| 318 | target_name: GN target for genrule generation. |
| 319 | |
| 320 | Returns: |
| 321 | A (source_genrule, header_genrule) module tuple. |
| 322 | """ |
| 323 | target = desc[target_name] |
| 324 | |
| 325 | # We only support genrules which call protoc (with or without a plugin) to |
| 326 | # turn .proto files into header and source files. |
| 327 | args = target['args'] |
| 328 | if not args[0].endswith('/protoc'): |
| 329 | raise Error('Unsupported action in target %s: %s' % (target_name, |
| 330 | target['args'])) |
Primiano Tucci | 20b760c | 2018-01-19 12:36:12 +0000 | [diff] [blame] | 331 | parser = ThrowingArgumentParser('Action in target %s (%s)' % |
| 332 | (target_name, ' '.join(target['args']))) |
| 333 | parser.add_argument('--proto_path') |
| 334 | parser.add_argument('--cpp_out') |
| 335 | parser.add_argument('--plugin') |
| 336 | parser.add_argument('--plugin_out') |
| 337 | parser.add_argument('protos', nargs=argparse.REMAINDER) |
| 338 | args = parser.parse_args(args[1:]) |
| 339 | |
| 340 | # Depending on whether we are using the default protoc C++ generator or the |
| 341 | # protozero plugin, the output dir is passed as: |
| 342 | # --cpp_out=gen/xxx or |
| 343 | # --plugin_out=:gen/xxx or |
| 344 | # --plugin_out=wrapper_namespace=pbzero:gen/xxx |
| 345 | gen_dir = args.cpp_out if args.cpp_out else args.plugin_out.split(':')[1] |
| 346 | assert gen_dir.startswith('gen/') |
| 347 | gen_dir = gen_dir[4:] |
| 348 | cpp_out_dir = ('$(genDir)/%s/%s' % (tree_path, gen_dir)).rstrip('/') |
| 349 | |
| 350 | # TODO(skyostil): Is there a way to avoid hardcoding the tree path here? |
| 351 | # TODO(skyostil): Find a way to avoid creating the directory. |
| 352 | cmd = [ |
| 353 | 'mkdir -p %s &&' % cpp_out_dir, |
| 354 | '$(location aprotoc)', |
| 355 | '--cpp_out=%s' % cpp_out_dir |
| 356 | ] |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 357 | |
| 358 | # We create two genrules for each action: one for the protobuf headers and |
| 359 | # another for the sources. This is because the module that depends on the |
| 360 | # generated files needs to declare two different types of dependencies -- |
| 361 | # source files in 'srcs' and headers in 'generated_headers' -- and it's not |
| 362 | # valid to generate .h files from a source dependency and vice versa. |
Sami Kyostila | 71625d7 | 2017-12-18 10:29:49 +0000 | [diff] [blame] | 363 | source_module = Module('genrule', label_to_module_name(target_name)) |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 364 | source_module.srcs.extend(label_to_path(src) for src in target['sources']) |
| 365 | source_module.tools = ['aprotoc'] |
| 366 | |
Sami Kyostila | 71625d7 | 2017-12-18 10:29:49 +0000 | [diff] [blame] | 367 | header_module = Module('genrule', |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 368 | label_to_module_name(target_name) + '_headers') |
| 369 | header_module.srcs = source_module.srcs[:] |
| 370 | header_module.tools = source_module.tools[:] |
Primiano Tucci | 20b760c | 2018-01-19 12:36:12 +0000 | [diff] [blame] | 371 | header_module.export_include_dirs = [gen_dir or '.'] |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 372 | |
Primiano Tucci | 20b760c | 2018-01-19 12:36:12 +0000 | [diff] [blame] | 373 | # In GN builds the proto path is always relative to the output directory |
| 374 | # (out/tmp.xxx). |
| 375 | assert args.proto_path.startswith('../../') |
| 376 | cmd += [ '--proto_path=%s/%s' % (tree_path, args.proto_path[6:])] |
| 377 | |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 378 | namespaces = ['pb'] |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 379 | if args.plugin: |
| 380 | _, plugin = os.path.split(args.plugin) |
| 381 | # TODO(skyostil): Can we detect this some other way? |
| 382 | if plugin == 'ipc_plugin': |
| 383 | namespaces.append('ipc') |
| 384 | elif plugin == 'protoc_plugin': |
| 385 | namespaces = ['pbzero'] |
| 386 | for dep in target['deps']: |
| 387 | if desc[dep]['type'] != 'executable': |
| 388 | continue |
| 389 | _, executable = os.path.split(desc[dep]['outputs'][0]) |
| 390 | if executable == plugin: |
| 391 | cmd += [ |
| 392 | '--plugin=protoc-gen-plugin=$(location %s)' % |
| 393 | label_to_module_name(dep) |
| 394 | ] |
| 395 | source_module.tools.append(label_to_module_name(dep)) |
| 396 | # Also make sure the module for the tool is generated. |
| 397 | create_modules_from_target(blueprint, desc, dep) |
| 398 | break |
| 399 | else: |
| 400 | raise Error('Unrecognized protoc plugin in target %s: %s' % |
| 401 | (target_name, args[i])) |
| 402 | if args.plugin_out: |
| 403 | plugin_args = args.plugin_out.split(':')[0] |
Primiano Tucci | 20b760c | 2018-01-19 12:36:12 +0000 | [diff] [blame] | 404 | cmd += ['--plugin_out=%s:%s' % (plugin_args, cpp_out_dir)] |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 405 | |
| 406 | cmd += ['$(in)'] |
| 407 | source_module.cmd = ' '.join(cmd) |
| 408 | header_module.cmd = source_module.cmd |
| 409 | header_module.tools = source_module.tools[:] |
| 410 | |
| 411 | for ns in namespaces: |
| 412 | source_module.out += [ |
| 413 | '%s/%s' % (tree_path, src.replace('.proto', '.%s.cc' % ns)) |
| 414 | for src in source_module.srcs |
| 415 | ] |
| 416 | header_module.out += [ |
| 417 | '%s/%s' % (tree_path, src.replace('.proto', '.%s.h' % ns)) |
| 418 | for src in header_module.srcs |
| 419 | ] |
| 420 | return source_module, header_module |
| 421 | |
| 422 | |
| 423 | def create_modules_from_target(blueprint, desc, target_name): |
| 424 | """Generate module(s) for a given GN target. |
| 425 | |
| 426 | Given a GN target name, generate one or more corresponding modules into a |
| 427 | blueprint. |
| 428 | |
| 429 | Args: |
| 430 | blueprint: Blueprint instance which is being generated. |
| 431 | desc: JSON GN description. |
| 432 | target_name: GN target for module generation. |
| 433 | """ |
| 434 | target = desc[target_name] |
| 435 | if target['type'] == 'executable': |
| 436 | if 'host' in target['toolchain']: |
| 437 | module_type = 'cc_binary_host' |
| 438 | elif target.get('testonly'): |
| 439 | module_type = 'cc_test' |
| 440 | else: |
| 441 | module_type = 'cc_binary' |
| 442 | modules = [Module(module_type, label_to_module_name(target_name))] |
| 443 | elif target['type'] == 'action': |
| 444 | modules = make_genrules_for_action(blueprint, desc, target_name) |
Sami Kyostila | ebba0fe | 2017-12-19 14:01:52 +0000 | [diff] [blame] | 445 | elif target['type'] == 'static_library': |
Lalit Maganti | c5bcd79 | 2018-01-12 18:38:11 +0000 | [diff] [blame] | 446 | module = Module('cc_library_static', label_to_module_name(target_name)) |
| 447 | module.export_include_dirs = ['include'] |
| 448 | modules = [module] |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 449 | elif target['type'] == 'shared_library': |
| 450 | modules = [ |
| 451 | Module('cc_library_shared', label_to_module_name(target_name)) |
| 452 | ] |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 453 | else: |
| 454 | raise Error('Unknown target type: %s' % target['type']) |
| 455 | |
| 456 | for module in modules: |
| 457 | module.comment = 'GN target: %s' % target_name |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 458 | if target_name in target_initrc: |
| 459 | module.init_rc = [target_initrc[target_name]] |
| 460 | |
Sami Kyostila | ebba0fe | 2017-12-19 14:01:52 +0000 | [diff] [blame] | 461 | # Don't try to inject library/source dependencies into genrules because |
| 462 | # they are not compiled in the traditional sense. |
Sami Kyostila | 71625d7 | 2017-12-18 10:29:49 +0000 | [diff] [blame] | 463 | if module.type != 'genrule': |
Primiano Tucci | edf099c | 2018-01-08 18:27:56 +0000 | [diff] [blame] | 464 | for flag in target.get('cflags', []): |
| 465 | if re.match(cflag_whitelist, flag): |
| 466 | module.cflags.append(flag) |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 467 | module.defaults = [defaults_module] |
Sami Kyostila | ebba0fe | 2017-12-19 14:01:52 +0000 | [diff] [blame] | 468 | apply_module_dependency(blueprint, desc, module, target_name) |
| 469 | for dep in resolve_dependencies(desc, target_name): |
| 470 | apply_module_dependency(blueprint, desc, module, dep) |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 471 | |
Lalit Maganti | c5bcd79 | 2018-01-12 18:38:11 +0000 | [diff] [blame] | 472 | # If the module is a static library, export all the generated headers. |
| 473 | if module.type == 'cc_library_static': |
| 474 | module.export_generated_headers = module.generated_headers |
| 475 | |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 476 | blueprint.add_module(module) |
| 477 | |
| 478 | |
| 479 | def resolve_dependencies(desc, target_name): |
| 480 | """Return the transitive set of dependent-on targets for a GN target. |
| 481 | |
| 482 | Args: |
| 483 | blueprint: Blueprint instance which is being generated. |
| 484 | desc: JSON GN description. |
| 485 | |
| 486 | Returns: |
| 487 | A set of transitive dependencies in the form of GN targets. |
| 488 | """ |
| 489 | |
| 490 | if label_without_toolchain(target_name) in builtin_deps: |
| 491 | return set() |
| 492 | target = desc[target_name] |
| 493 | resolved_deps = set() |
| 494 | for dep in target.get('deps', []): |
| 495 | resolved_deps.add(dep) |
| 496 | # Ignore the transitive dependencies of actions because they are |
| 497 | # explicitly converted to genrules. |
| 498 | if desc[dep]['type'] == 'action': |
| 499 | continue |
Primiano Tucci | 6067e73 | 2018-01-08 16:19:40 +0000 | [diff] [blame] | 500 | # Dependencies on shared libraries shouldn't propagate any transitive |
| 501 | # dependencies but only depend on the shared library target |
| 502 | if desc[dep]['type'] == 'shared_library': |
| 503 | continue |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 504 | resolved_deps.update(resolve_dependencies(desc, dep)) |
| 505 | return resolved_deps |
| 506 | |
| 507 | |
| 508 | def create_blueprint_for_targets(desc, targets): |
| 509 | """Generate a blueprint for a list of GN targets.""" |
| 510 | blueprint = Blueprint() |
| 511 | |
| 512 | # Default settings used by all modules. |
| 513 | defaults = Module('cc_defaults', defaults_module) |
| 514 | defaults.local_include_dirs = ['include'] |
| 515 | defaults.cflags = [ |
| 516 | '-Wno-error=return-type', |
| 517 | '-Wno-sign-compare', |
| 518 | '-Wno-sign-promo', |
| 519 | '-Wno-unused-parameter', |
Florian Mayer | cc424fd | 2018-01-15 11:19:01 +0000 | [diff] [blame] | 520 | '-fvisibility=hidden', |
Florian Mayer | c2a38ea | 2018-01-19 11:48:43 +0000 | [diff] [blame] | 521 | '-Oz', |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 522 | ] |
| 523 | |
| 524 | blueprint.add_module(defaults) |
| 525 | for target in targets: |
| 526 | create_modules_from_target(blueprint, desc, target) |
| 527 | return blueprint |
| 528 | |
| 529 | |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 530 | def repo_root(): |
| 531 | """Returns an absolute path to the repository root.""" |
| 532 | |
| 533 | return os.path.join( |
| 534 | os.path.realpath(os.path.dirname(__file__)), os.path.pardir) |
| 535 | |
| 536 | |
| 537 | def create_build_description(): |
| 538 | """Creates the JSON build description by running GN.""" |
| 539 | |
| 540 | out = os.path.join(repo_root(), 'out', 'tmp.gen_android_bp') |
| 541 | try: |
| 542 | try: |
| 543 | os.makedirs(out) |
| 544 | except OSError as e: |
| 545 | if e.errno != errno.EEXIST: |
| 546 | raise |
| 547 | subprocess.check_output( |
| 548 | ['gn', 'gen', out, '--args=%s' % gn_args], cwd=repo_root()) |
| 549 | desc = subprocess.check_output( |
| 550 | ['gn', 'desc', out, '--format=json', '--all-toolchains', '//*'], |
| 551 | cwd=repo_root()) |
| 552 | return json.loads(desc) |
| 553 | finally: |
| 554 | shutil.rmtree(out) |
| 555 | |
| 556 | |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 557 | def main(): |
| 558 | parser = argparse.ArgumentParser( |
| 559 | description='Generate Android.bp from a GN description.') |
| 560 | parser.add_argument( |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 561 | '--desc', |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 562 | help= |
| 563 | 'GN description (e.g., gn desc out --format=json --all-toolchains "//*"' |
| 564 | ) |
| 565 | parser.add_argument( |
Lalit Maganti | c5bcd79 | 2018-01-12 18:38:11 +0000 | [diff] [blame] | 566 | '--extras', |
| 567 | help='Extra targets to include at the end of the Blueprint file', |
| 568 | default=os.path.join(repo_root(), 'Android.bp.extras'), |
| 569 | ) |
| 570 | parser.add_argument( |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 571 | '--output', |
| 572 | help='Blueprint file to create', |
| 573 | default=os.path.join(repo_root(), 'Android.bp'), |
| 574 | ) |
| 575 | parser.add_argument( |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 576 | 'targets', |
| 577 | nargs=argparse.REMAINDER, |
| 578 | help='Targets to include in the blueprint (e.g., "//:perfetto_tests")') |
| 579 | args = parser.parse_args() |
| 580 | |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 581 | if args.desc: |
| 582 | with open(args.desc) as f: |
| 583 | desc = json.load(f) |
| 584 | else: |
| 585 | desc = create_build_description() |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 586 | |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 587 | blueprint = create_blueprint_for_targets(desc, args.targets |
| 588 | or default_targets) |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 589 | output = [ |
| 590 | """// Copyright (C) 2017 The Android Open Source Project |
| 591 | // |
| 592 | // Licensed under the Apache License, Version 2.0 (the "License"); |
| 593 | // you may not use this file except in compliance with the License. |
| 594 | // You may obtain a copy of the License at |
| 595 | // |
| 596 | // http://www.apache.org/licenses/LICENSE-2.0 |
| 597 | // |
| 598 | // Unless required by applicable law or agreed to in writing, software |
| 599 | // distributed under the License is distributed on an "AS IS" BASIS, |
| 600 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 601 | // See the License for the specific language governing permissions and |
| 602 | // limitations under the License. |
| 603 | // |
| 604 | // This file is automatically generated by %s. Do not edit. |
| 605 | """ % (__file__) |
| 606 | ] |
| 607 | blueprint.to_string(output) |
Lalit Maganti | c5bcd79 | 2018-01-12 18:38:11 +0000 | [diff] [blame] | 608 | with open(args.extras, 'r') as r: |
| 609 | for line in r: |
| 610 | output.append(line.rstrip("\n\r")) |
Sami Kyostila | b27619f | 2017-12-13 19:22:16 +0000 | [diff] [blame] | 611 | with open(args.output, 'w') as f: |
| 612 | f.write('\n'.join(output)) |
Sami Kyostila | 865d1d3 | 2017-12-12 18:37:04 +0000 | [diff] [blame] | 613 | |
| 614 | |
| 615 | if __name__ == '__main__': |
| 616 | sys.exit(main()) |