bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 1 | #!/usr/bin/env python |
| 2 | # |
| 3 | # Copyright 2016 Google Inc. |
| 4 | # |
| 5 | # Use of this source code is governed by a BSD-style license that can be |
| 6 | # found in the LICENSE file. |
| 7 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 8 | |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 9 | """ |
| 10 | Usage: gn_to_cmake.py <json_file_name> |
| 11 | |
| 12 | gn gen out/config --ide=json --json-ide-script=../../gn/gn_to_cmake.py |
| 13 | |
| 14 | or |
| 15 | |
| 16 | gn gen out/config --ide=json |
| 17 | python gn/gn_to_cmake.py out/config/project.json |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 18 | |
| 19 | The first is recommended, as it will auto-update. |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 20 | """ |
| 21 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 22 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 23 | import itertools |
| 24 | import functools |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 25 | import json |
| 26 | import posixpath |
| 27 | import os |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 28 | import string |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 29 | import sys |
| 30 | |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 31 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 32 | def CMakeStringEscape(a): |
| 33 | """Escapes the string 'a' for use inside a CMake string. |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 34 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 35 | This means escaping |
| 36 | '\' otherwise it may be seen as modifying the next character |
| 37 | '"' otherwise it will end the string |
| 38 | ';' otherwise the string becomes a list |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 39 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 40 | The following do not need to be escaped |
| 41 | '#' when the lexer is in string state, this does not start a comment |
| 42 | """ |
| 43 | return a.replace('\\', '\\\\').replace(';', '\\;').replace('"', '\\"') |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 44 | |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 45 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 46 | def CMakeTargetEscape(a): |
| 47 | """Escapes the string 'a' for use as a CMake target name. |
| 48 | |
| 49 | CMP0037 in CMake 3.0 restricts target names to "^[A-Za-z0-9_.:+-]+$" |
| 50 | The ':' is only allowed for imported targets. |
| 51 | """ |
| 52 | def Escape(c): |
| 53 | if c in string.ascii_letters or c in string.digits or c in '_.+-': |
| 54 | return c |
| 55 | else: |
| 56 | return '__' |
| 57 | return ''.join(map(Escape, a)) |
| 58 | |
| 59 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 60 | def SetVariable(out, variable_name, value): |
| 61 | """Sets a CMake variable.""" |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 62 | out.write('set("') |
| 63 | out.write(CMakeStringEscape(variable_name)) |
| 64 | out.write('" "') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 65 | out.write(CMakeStringEscape(value)) |
| 66 | out.write('")\n') |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 67 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 68 | |
| 69 | def SetVariableList(out, variable_name, values): |
| 70 | """Sets a CMake variable to a list.""" |
| 71 | if not values: |
| 72 | return SetVariable(out, variable_name, "") |
| 73 | if len(values) == 1: |
| 74 | return SetVariable(out, variable_name, values[0]) |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 75 | out.write('list(APPEND "') |
| 76 | out.write(CMakeStringEscape(variable_name)) |
| 77 | out.write('"\n "') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 78 | out.write('"\n "'.join([CMakeStringEscape(value) for value in values])) |
| 79 | out.write('")\n') |
| 80 | |
| 81 | |
| 82 | def SetFilesProperty(output, variable, property_name, values, sep): |
| 83 | """Given a set of source files, sets the given property on them.""" |
| 84 | output.write('set_source_files_properties(') |
| 85 | WriteVariable(output, variable) |
| 86 | output.write(' PROPERTIES ') |
| 87 | output.write(property_name) |
| 88 | output.write(' "') |
| 89 | for value in values: |
| 90 | output.write(CMakeStringEscape(value)) |
| 91 | output.write(sep) |
| 92 | output.write('")\n') |
| 93 | |
| 94 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 95 | def SetCurrentTargetProperty(out, property_name, values, sep=''): |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 96 | """Given a target, sets the given property.""" |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 97 | out.write('set_target_properties("${target}" PROPERTIES ') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 98 | out.write(property_name) |
| 99 | out.write(' "') |
| 100 | for value in values: |
| 101 | out.write(CMakeStringEscape(value)) |
| 102 | out.write(sep) |
| 103 | out.write('")\n') |
| 104 | |
| 105 | |
| 106 | def WriteVariable(output, variable_name, prepend=None): |
| 107 | if prepend: |
| 108 | output.write(prepend) |
| 109 | output.write('${') |
| 110 | output.write(variable_name) |
| 111 | output.write('}') |
| 112 | |
| 113 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 114 | # See GetSourceFileType in gn |
| 115 | source_file_types = { |
| 116 | '.cc': 'cxx', |
| 117 | '.cpp': 'cxx', |
| 118 | '.cxx': 'cxx', |
Brian Salomon | 4ce6989 | 2019-07-12 14:36:03 -0400 | [diff] [blame] | 119 | '.m': 'objc', |
| 120 | '.mm': 'objcc', |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 121 | '.c': 'c', |
| 122 | '.s': 'asm', |
| 123 | '.S': 'asm', |
| 124 | '.asm': 'asm', |
| 125 | '.o': 'obj', |
| 126 | '.obj': 'obj', |
| 127 | } |
| 128 | |
| 129 | |
| 130 | class CMakeTargetType(object): |
| 131 | def __init__(self, command, modifier, property_modifier, is_linkable): |
| 132 | self.command = command |
| 133 | self.modifier = modifier |
| 134 | self.property_modifier = property_modifier |
| 135 | self.is_linkable = is_linkable |
| 136 | CMakeTargetType.custom = CMakeTargetType('add_custom_target', 'SOURCES', |
| 137 | None, False) |
| 138 | |
| 139 | # See GetStringForOutputType in gn |
| 140 | cmake_target_types = { |
| 141 | 'unknown': CMakeTargetType.custom, |
| 142 | 'group': CMakeTargetType.custom, |
| 143 | 'executable': CMakeTargetType('add_executable', None, 'RUNTIME', True), |
| 144 | 'loadable_module': CMakeTargetType('add_library', 'MODULE', 'LIBRARY', True), |
| 145 | 'shared_library': CMakeTargetType('add_library', 'SHARED', 'LIBRARY', True), |
Kevin Lubick | d5c6e16 | 2019-02-01 15:34:04 -0500 | [diff] [blame] | 146 | 'static_library': CMakeTargetType('add_library', 'STATIC', 'ARCHIVE', True), |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 147 | 'source_set': CMakeTargetType('add_library', 'OBJECT', None, False), |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 148 | 'copy': CMakeTargetType.custom, |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 149 | 'action': CMakeTargetType.custom, |
| 150 | 'action_foreach': CMakeTargetType.custom, |
| 151 | 'bundle_data': CMakeTargetType.custom, |
| 152 | 'create_bundle': CMakeTargetType.custom, |
| 153 | } |
| 154 | |
| 155 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 156 | def FindFirstOf(s, a): |
| 157 | return min(s.find(i) for i in a if i in s) |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 158 | |
| 159 | |
| 160 | class Project(object): |
| 161 | def __init__(self, project_json): |
| 162 | self.targets = project_json['targets'] |
| 163 | build_settings = project_json['build_settings'] |
| 164 | self.root_path = build_settings['root_path'] |
| 165 | self.build_path = posixpath.join(self.root_path, |
| 166 | build_settings['build_dir'][2:]) |
| 167 | |
| 168 | def GetAbsolutePath(self, path): |
| 169 | if path.startswith("//"): |
| 170 | return self.root_path + "/" + path[2:] |
| 171 | else: |
| 172 | return path |
| 173 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 174 | def GetObjectSourceDependencies(self, gn_target_name, object_dependencies): |
| 175 | """All OBJECT libraries whose sources have not been absorbed.""" |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 176 | dependencies = self.targets[gn_target_name].get('deps', []) |
| 177 | for dependency in dependencies: |
| 178 | dependency_type = self.targets[dependency].get('type', None) |
| 179 | if dependency_type == 'source_set': |
| 180 | object_dependencies.add(dependency) |
| 181 | if dependency_type not in gn_target_types_that_absorb_objects: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 182 | self.GetObjectSourceDependencies(dependency, object_dependencies) |
| 183 | |
| 184 | def GetObjectLibraryDependencies(self, gn_target_name, object_dependencies): |
| 185 | """All OBJECT libraries whose libraries have not been absorbed.""" |
| 186 | dependencies = self.targets[gn_target_name].get('deps', []) |
| 187 | for dependency in dependencies: |
| 188 | dependency_type = self.targets[dependency].get('type', None) |
| 189 | if dependency_type == 'source_set': |
| 190 | object_dependencies.add(dependency) |
| 191 | self.GetObjectLibraryDependencies(dependency, object_dependencies) |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 192 | |
| 193 | def GetCMakeTargetName(self, gn_target_name): |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 194 | # See <chromium>/src/tools/gn/label.cc#Resolve |
| 195 | # //base/test:test_support(//build/toolchain/win:msvc) |
| 196 | path_separator = FindFirstOf(gn_target_name, (':', '(')) |
| 197 | location = None |
| 198 | name = None |
| 199 | toolchain = None |
| 200 | if not path_separator: |
| 201 | location = gn_target_name[2:] |
| 202 | else: |
| 203 | location = gn_target_name[2:path_separator] |
| 204 | toolchain_separator = gn_target_name.find('(', path_separator) |
| 205 | if toolchain_separator == -1: |
| 206 | name = gn_target_name[path_separator + 1:] |
| 207 | else: |
| 208 | if toolchain_separator > path_separator: |
| 209 | name = gn_target_name[path_separator + 1:toolchain_separator] |
| 210 | assert gn_target_name.endswith(')') |
| 211 | toolchain = gn_target_name[toolchain_separator + 1:-1] |
| 212 | assert location or name |
| 213 | |
| 214 | cmake_target_name = None |
| 215 | if location.endswith('/' + name): |
| 216 | cmake_target_name = location |
| 217 | elif location: |
| 218 | cmake_target_name = location + '_' + name |
| 219 | else: |
| 220 | cmake_target_name = name |
| 221 | if toolchain: |
| 222 | cmake_target_name += '--' + toolchain |
| 223 | return CMakeTargetEscape(cmake_target_name) |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 224 | |
| 225 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 226 | class Target(object): |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 227 | def __init__(self, gn_target_name, project): |
| 228 | self.gn_name = gn_target_name |
| 229 | self.properties = project.targets[self.gn_name] |
| 230 | self.cmake_name = project.GetCMakeTargetName(self.gn_name) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 231 | self.gn_type = self.properties.get('type', None) |
| 232 | self.cmake_type = cmake_target_types.get(self.gn_type, None) |
| 233 | |
| 234 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 235 | def WriteAction(out, target, project, sources, synthetic_dependencies): |
| 236 | outputs = [] |
| 237 | output_directories = set() |
| 238 | for output in target.properties.get('outputs', []): |
| 239 | output_abs_path = project.GetAbsolutePath(output) |
| 240 | outputs.append(output_abs_path) |
| 241 | output_directory = posixpath.dirname(output_abs_path) |
| 242 | if output_directory: |
| 243 | output_directories.add(output_directory) |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 244 | outputs_name = '${target}__output' |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 245 | SetVariableList(out, outputs_name, outputs) |
| 246 | |
| 247 | out.write('add_custom_command(OUTPUT ') |
| 248 | WriteVariable(out, outputs_name) |
| 249 | out.write('\n') |
| 250 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 251 | if output_directories: |
| 252 | out.write(' COMMAND ${CMAKE_COMMAND} -E make_directory "') |
| 253 | out.write('" "'.join(map(CMakeStringEscape, output_directories))) |
| 254 | out.write('"\n') |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 255 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 256 | script = target.properties['script'] |
| 257 | arguments = target.properties['args'] |
| 258 | out.write(' COMMAND python "') |
| 259 | out.write(CMakeStringEscape(project.GetAbsolutePath(script))) |
| 260 | out.write('"') |
| 261 | if arguments: |
| 262 | out.write('\n "') |
| 263 | out.write('"\n "'.join(map(CMakeStringEscape, arguments))) |
| 264 | out.write('"') |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 265 | out.write('\n') |
| 266 | |
| 267 | out.write(' DEPENDS ') |
| 268 | for sources_type_name in sources.values(): |
| 269 | WriteVariable(out, sources_type_name, ' ') |
| 270 | out.write('\n') |
| 271 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 272 | #TODO: CMake 3.7 is introducing DEPFILE |
| 273 | |
| 274 | out.write(' WORKING_DIRECTORY "') |
| 275 | out.write(CMakeStringEscape(project.build_path)) |
| 276 | out.write('"\n') |
| 277 | |
| 278 | out.write(' COMMENT "Action: ${target}"\n') |
| 279 | |
| 280 | out.write(' VERBATIM)\n') |
| 281 | |
| 282 | synthetic_dependencies.add(outputs_name) |
| 283 | |
| 284 | |
| 285 | def ExpandPlaceholders(source, a): |
| 286 | source_dir, source_file_part = posixpath.split(source) |
| 287 | source_name_part, _ = posixpath.splitext(source_file_part) |
| 288 | #TODO: {{source_gen_dir}}, {{source_out_dir}}, {{response_file_name}} |
| 289 | return a.replace('{{source}}', source) \ |
| 290 | .replace('{{source_file_part}}', source_file_part) \ |
| 291 | .replace('{{source_name_part}}', source_name_part) \ |
| 292 | .replace('{{source_dir}}', source_dir) \ |
| 293 | .replace('{{source_root_relative_dir}}', source_dir) |
| 294 | |
| 295 | |
| 296 | def WriteActionForEach(out, target, project, sources, synthetic_dependencies): |
| 297 | all_outputs = target.properties.get('outputs', []) |
| 298 | inputs = target.properties.get('sources', []) |
| 299 | # TODO: consider expanding 'output_patterns' instead. |
| 300 | outputs_per_input = len(all_outputs) / len(inputs) |
| 301 | for count, source in enumerate(inputs): |
| 302 | source_abs_path = project.GetAbsolutePath(source) |
| 303 | |
| 304 | outputs = [] |
| 305 | output_directories = set() |
| 306 | for output in all_outputs[outputs_per_input * count: |
| 307 | outputs_per_input * (count+1)]: |
| 308 | output_abs_path = project.GetAbsolutePath(output) |
| 309 | outputs.append(output_abs_path) |
| 310 | output_directory = posixpath.dirname(output_abs_path) |
| 311 | if output_directory: |
| 312 | output_directories.add(output_directory) |
| 313 | outputs_name = '${target}__output_' + str(count) |
| 314 | SetVariableList(out, outputs_name, outputs) |
| 315 | |
| 316 | out.write('add_custom_command(OUTPUT ') |
| 317 | WriteVariable(out, outputs_name) |
| 318 | out.write('\n') |
| 319 | |
| 320 | if output_directories: |
| 321 | out.write(' COMMAND ${CMAKE_COMMAND} -E make_directory "') |
| 322 | out.write('" "'.join(map(CMakeStringEscape, output_directories))) |
| 323 | out.write('"\n') |
| 324 | |
| 325 | script = target.properties['script'] |
| 326 | # TODO: need to expand {{xxx}} in arguments |
| 327 | arguments = target.properties['args'] |
| 328 | out.write(' COMMAND python "') |
| 329 | out.write(CMakeStringEscape(project.GetAbsolutePath(script))) |
| 330 | out.write('"') |
| 331 | if arguments: |
| 332 | out.write('\n "') |
| 333 | expand = functools.partial(ExpandPlaceholders, source_abs_path) |
| 334 | out.write('"\n "'.join(map(CMakeStringEscape, map(expand,arguments)))) |
| 335 | out.write('"') |
| 336 | out.write('\n') |
| 337 | |
| 338 | out.write(' DEPENDS') |
| 339 | if 'input' in sources: |
| 340 | WriteVariable(out, sources['input'], ' ') |
| 341 | out.write(' "') |
| 342 | out.write(CMakeStringEscape(source_abs_path)) |
| 343 | out.write('"\n') |
| 344 | |
| 345 | #TODO: CMake 3.7 is introducing DEPFILE |
| 346 | |
| 347 | out.write(' WORKING_DIRECTORY "') |
| 348 | out.write(CMakeStringEscape(project.build_path)) |
| 349 | out.write('"\n') |
| 350 | |
| 351 | out.write(' COMMENT "Action ${target} on ') |
| 352 | out.write(CMakeStringEscape(source_abs_path)) |
| 353 | out.write('"\n') |
| 354 | |
| 355 | out.write(' VERBATIM)\n') |
| 356 | |
| 357 | synthetic_dependencies.add(outputs_name) |
| 358 | |
| 359 | |
| 360 | def WriteCopy(out, target, project, sources, synthetic_dependencies): |
| 361 | inputs = target.properties.get('sources', []) |
| 362 | raw_outputs = target.properties.get('outputs', []) |
| 363 | |
| 364 | # TODO: consider expanding 'output_patterns' instead. |
| 365 | outputs = [] |
| 366 | for output in raw_outputs: |
| 367 | output_abs_path = project.GetAbsolutePath(output) |
| 368 | outputs.append(output_abs_path) |
| 369 | outputs_name = '${target}__output' |
| 370 | SetVariableList(out, outputs_name, outputs) |
| 371 | |
| 372 | out.write('add_custom_command(OUTPUT ') |
| 373 | WriteVariable(out, outputs_name) |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 374 | out.write('\n') |
| 375 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 376 | for src, dst in zip(inputs, outputs): |
Kevin Lubick | 9a2bb09 | 2019-02-01 13:32:44 -0500 | [diff] [blame] | 377 | abs_src_path = CMakeStringEscape(project.GetAbsolutePath(src)) |
| 378 | # CMake distinguishes between copying files and copying directories but |
| 379 | # gn does not. We assume if the src has a period in its name then it is |
| 380 | # a file and otherwise a directory. |
| 381 | if "." in os.path.basename(abs_src_path): |
| 382 | out.write(' COMMAND ${CMAKE_COMMAND} -E copy "') |
| 383 | else: |
| 384 | out.write(' COMMAND ${CMAKE_COMMAND} -E copy_directory "') |
| 385 | out.write(abs_src_path) |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 386 | out.write('" "') |
| 387 | out.write(CMakeStringEscape(dst)) |
| 388 | out.write('"\n') |
| 389 | |
| 390 | out.write(' DEPENDS ') |
| 391 | for sources_type_name in sources.values(): |
| 392 | WriteVariable(out, sources_type_name, ' ') |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 393 | out.write('\n') |
| 394 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 395 | out.write(' WORKING_DIRECTORY "') |
| 396 | out.write(CMakeStringEscape(project.build_path)) |
| 397 | out.write('"\n') |
| 398 | |
| 399 | out.write(' COMMENT "Copy ${target}"\n') |
| 400 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 401 | out.write(' VERBATIM)\n') |
| 402 | |
| 403 | synthetic_dependencies.add(outputs_name) |
| 404 | |
| 405 | |
| 406 | def WriteCompilerFlags(out, target, project, sources): |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 407 | # Hack, set linker language to c if no c or cxx files present. |
| 408 | if not 'c' in sources and not 'cxx' in sources: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 409 | SetCurrentTargetProperty(out, 'LINKER_LANGUAGE', ['C']) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 410 | |
| 411 | # Mark uncompiled sources as uncompiled. |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 412 | if 'input' in sources: |
| 413 | SetFilesProperty(out, sources['input'], 'HEADER_FILE_ONLY', ('True',), '') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 414 | if 'other' in sources: |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 415 | SetFilesProperty(out, sources['other'], 'HEADER_FILE_ONLY', ('True',), '') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 416 | |
| 417 | # Mark object sources as linkable. |
| 418 | if 'obj' in sources: |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 419 | SetFilesProperty(out, sources['obj'], 'EXTERNAL_OBJECT', ('True',), '') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 420 | |
| 421 | # TODO: 'output_name', 'output_dir', 'output_extension' |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 422 | # This includes using 'source_outputs' to direct compiler output. |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 423 | |
| 424 | # Includes |
| 425 | includes = target.properties.get('include_dirs', []) |
| 426 | if includes: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 427 | out.write('set_property(TARGET "${target}" ') |
| 428 | out.write('APPEND PROPERTY INCLUDE_DIRECTORIES') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 429 | for include_dir in includes: |
| 430 | out.write('\n "') |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 431 | out.write(project.GetAbsolutePath(include_dir)) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 432 | out.write('"') |
| 433 | out.write(')\n') |
| 434 | |
| 435 | # Defines |
| 436 | defines = target.properties.get('defines', []) |
| 437 | if defines: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 438 | SetCurrentTargetProperty(out, 'COMPILE_DEFINITIONS', defines, ';') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 439 | |
| 440 | # Compile flags |
| 441 | # "arflags", "asmflags", "cflags", |
| 442 | # "cflags_c", "clfags_cc", "cflags_objc", "clfags_objcc" |
| 443 | # CMake does not have per target lang compile flags. |
| 444 | # TODO: $<$<COMPILE_LANGUAGE:CXX>:cflags_cc style generator expression. |
| 445 | # http://public.kitware.com/Bug/view.php?id=14857 |
| 446 | flags = [] |
| 447 | flags.extend(target.properties.get('cflags', [])) |
| 448 | cflags_asm = target.properties.get('asmflags', []) |
| 449 | cflags_c = target.properties.get('cflags_c', []) |
| 450 | cflags_cxx = target.properties.get('cflags_cc', []) |
Brian Salomon | 4ce6989 | 2019-07-12 14:36:03 -0400 | [diff] [blame] | 451 | cflags_objc = cflags_c[:] |
| 452 | cflags_objc.extend(target.properties.get('cflags_objc', [])) |
| 453 | cflags_objcc = cflags_cxx[:] |
| 454 | cflags_objcc.extend(target.properties.get('cflags_objcc', [])) |
| 455 | |
| 456 | if 'c' in sources and not any(k in sources for k in ('asm', 'cxx', 'objc', 'objcc')): |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 457 | flags.extend(cflags_c) |
Brian Salomon | 4ce6989 | 2019-07-12 14:36:03 -0400 | [diff] [blame] | 458 | elif 'cxx' in sources and not any(k in sources for k in ('asm', 'c', 'objc', 'objcc')): |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 459 | flags.extend(cflags_cxx) |
Brian Salomon | 4ce6989 | 2019-07-12 14:36:03 -0400 | [diff] [blame] | 460 | elif 'objc' in sources and not any(k in sources for k in ('asm', 'c', 'cxx', 'objcc')): |
| 461 | flags.extend(cflags_objc) |
| 462 | elif 'objcc' in sources and not any(k in sources for k in ('asm', 'c', 'cxx', 'objc')): |
| 463 | flags.extend(cflags_objcc) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 464 | else: |
| 465 | # TODO: This is broken, one cannot generally set properties on files, |
| 466 | # as other targets may require different properties on the same files. |
| 467 | if 'asm' in sources and cflags_asm: |
| 468 | SetFilesProperty(out, sources['asm'], 'COMPILE_FLAGS', cflags_asm, ' ') |
| 469 | if 'c' in sources and cflags_c: |
| 470 | SetFilesProperty(out, sources['c'], 'COMPILE_FLAGS', cflags_c, ' ') |
| 471 | if 'cxx' in sources and cflags_cxx: |
| 472 | SetFilesProperty(out, sources['cxx'], 'COMPILE_FLAGS', cflags_cxx, ' ') |
Brian Salomon | 4ce6989 | 2019-07-12 14:36:03 -0400 | [diff] [blame] | 473 | if 'objc' in sources and cflags_objc: |
| 474 | SetFilesProperty(out, sources['objc'], 'COMPILE_FLAGS', cflags_objc, ' ') |
| 475 | if 'objcc' in sources and cflags_objcc: |
| 476 | SetFilesProperty(out, sources['objcc'], 'COMPILE_FLAGS', cflags_objcc, ' ') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 477 | if flags: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 478 | SetCurrentTargetProperty(out, 'COMPILE_FLAGS', flags, ' ') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 479 | |
| 480 | # Linker flags |
| 481 | ldflags = target.properties.get('ldflags', []) |
| 482 | if ldflags: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 483 | SetCurrentTargetProperty(out, 'LINK_FLAGS', ldflags, ' ') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 484 | |
| 485 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 486 | gn_target_types_that_absorb_objects = ( |
| 487 | 'executable', |
| 488 | 'loadable_module', |
| 489 | 'shared_library', |
| 490 | 'static_library' |
| 491 | ) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 492 | |
| 493 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 494 | def WriteSourceVariables(out, target, project): |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 495 | # gn separates the sheep from the goats based on file extensions. |
| 496 | # A full separation is done here because of flag handing (see Compile flags). |
Brian Salomon | 4ce6989 | 2019-07-12 14:36:03 -0400 | [diff] [blame] | 497 | source_types = {'cxx':[], 'c':[], 'asm':[], 'objc':[], 'objcc':[], |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 498 | 'obj':[], 'obj_target':[], 'input':[], 'other':[]} |
| 499 | |
Herb Derby | 5699278 | 2018-05-18 15:09:41 -0400 | [diff] [blame] | 500 | all_sources = target.properties.get('sources', []) |
| 501 | |
| 502 | # As of cmake 3.11 add_library must have sources. If there are |
| 503 | # no sources, add empty.cpp as the file to compile. |
| 504 | if len(all_sources) == 0: |
| 505 | all_sources.append(posixpath.join(project.build_path, 'empty.cpp')) |
| 506 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 507 | # TODO .def files on Windows |
Herb Derby | 5699278 | 2018-05-18 15:09:41 -0400 | [diff] [blame] | 508 | for source in all_sources: |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 509 | _, ext = posixpath.splitext(source) |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 510 | source_abs_path = project.GetAbsolutePath(source) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 511 | source_types[source_file_types.get(ext, 'other')].append(source_abs_path) |
| 512 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 513 | for input_path in target.properties.get('inputs', []): |
| 514 | input_abs_path = project.GetAbsolutePath(input_path) |
| 515 | source_types['input'].append(input_abs_path) |
| 516 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 517 | # OBJECT library dependencies need to be listed as sources. |
| 518 | # Only executables and non-OBJECT libraries may reference an OBJECT library. |
| 519 | # https://gitlab.kitware.com/cmake/cmake/issues/14778 |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 520 | if target.gn_type in gn_target_types_that_absorb_objects: |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 521 | object_dependencies = set() |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 522 | project.GetObjectSourceDependencies(target.gn_name, object_dependencies) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 523 | for dependency in object_dependencies: |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 524 | cmake_dependency_name = project.GetCMakeTargetName(dependency) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 525 | obj_target_sources = '$<TARGET_OBJECTS:' + cmake_dependency_name + '>' |
| 526 | source_types['obj_target'].append(obj_target_sources) |
| 527 | |
| 528 | sources = {} |
| 529 | for source_type, sources_of_type in source_types.items(): |
| 530 | if sources_of_type: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 531 | sources[source_type] = '${target}__' + source_type + '_srcs' |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 532 | SetVariableList(out, sources[source_type], sources_of_type) |
| 533 | return sources |
| 534 | |
| 535 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 536 | def WriteTarget(out, target, project): |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 537 | out.write('\n#') |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 538 | out.write(target.gn_name) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 539 | out.write('\n') |
| 540 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 541 | if target.cmake_type is None: |
| 542 | print ('Target %s has unknown target type %s, skipping.' % |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 543 | ( target.gn_name, target.gn_type ) ) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 544 | return |
| 545 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 546 | SetVariable(out, 'target', target.cmake_name) |
| 547 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 548 | sources = WriteSourceVariables(out, target, project) |
| 549 | |
| 550 | synthetic_dependencies = set() |
| 551 | if target.gn_type == 'action': |
| 552 | WriteAction(out, target, project, sources, synthetic_dependencies) |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 553 | if target.gn_type == 'action_foreach': |
| 554 | WriteActionForEach(out, target, project, sources, synthetic_dependencies) |
| 555 | if target.gn_type == 'copy': |
| 556 | WriteCopy(out, target, project, sources, synthetic_dependencies) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 557 | |
| 558 | out.write(target.cmake_type.command) |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 559 | out.write('("${target}"') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 560 | if target.cmake_type.modifier is not None: |
| 561 | out.write(' ') |
| 562 | out.write(target.cmake_type.modifier) |
| 563 | for sources_type_name in sources.values(): |
| 564 | WriteVariable(out, sources_type_name, ' ') |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 565 | if synthetic_dependencies: |
| 566 | out.write(' DEPENDS') |
| 567 | for synthetic_dependencie in synthetic_dependencies: |
| 568 | WriteVariable(out, synthetic_dependencie, ' ') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 569 | out.write(')\n') |
| 570 | |
| 571 | if target.cmake_type.command != 'add_custom_target': |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 572 | WriteCompilerFlags(out, target, project, sources) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 573 | |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 574 | libraries = set() |
| 575 | nonlibraries = set() |
| 576 | |
| 577 | dependencies = set(target.properties.get('deps', [])) |
| 578 | # Transitive OBJECT libraries are in sources. |
| 579 | # Those sources are dependent on the OBJECT library dependencies. |
| 580 | # Those sources cannot bring in library dependencies. |
| 581 | object_dependencies = set() |
| 582 | if target.gn_type != 'source_set': |
| 583 | project.GetObjectLibraryDependencies(target.gn_name, object_dependencies) |
| 584 | for object_dependency in object_dependencies: |
| 585 | dependencies.update(project.targets.get(object_dependency).get('deps', [])) |
| 586 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 587 | for dependency in dependencies: |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 588 | gn_dependency_type = project.targets.get(dependency, {}).get('type', None) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 589 | cmake_dependency_type = cmake_target_types.get(gn_dependency_type, None) |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 590 | cmake_dependency_name = project.GetCMakeTargetName(dependency) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 591 | if cmake_dependency_type.command != 'add_library': |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 592 | nonlibraries.add(cmake_dependency_name) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 593 | elif cmake_dependency_type.modifier != 'OBJECT': |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 594 | if target.cmake_type.is_linkable: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 595 | libraries.add(cmake_dependency_name) |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 596 | else: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 597 | nonlibraries.add(cmake_dependency_name) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 598 | |
| 599 | # Non-library dependencies. |
| 600 | if nonlibraries: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 601 | out.write('add_dependencies("${target}"') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 602 | for nonlibrary in nonlibraries: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 603 | out.write('\n "') |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 604 | out.write(nonlibrary) |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 605 | out.write('"') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 606 | out.write(')\n') |
| 607 | |
| 608 | # Non-OBJECT library dependencies. |
| 609 | external_libraries = target.properties.get('libs', []) |
| 610 | if target.cmake_type.is_linkable and (external_libraries or libraries): |
Ben Wagner | 187a771 | 2016-10-27 18:47:54 -0400 | [diff] [blame] | 611 | library_dirs = target.properties.get('lib_dirs', []) |
| 612 | if library_dirs: |
| 613 | SetVariableList(out, '${target}__library_directories', library_dirs) |
| 614 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 615 | system_libraries = [] |
| 616 | for external_library in external_libraries: |
| 617 | if '/' in external_library: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 618 | libraries.add(project.GetAbsolutePath(external_library)) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 619 | else: |
| 620 | if external_library.endswith('.framework'): |
| 621 | external_library = external_library[:-len('.framework')] |
Ben Wagner | 187a771 | 2016-10-27 18:47:54 -0400 | [diff] [blame] | 622 | system_library = 'library__' + external_library |
| 623 | if library_dirs: |
| 624 | system_library = system_library + '__for_${target}' |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 625 | out.write('find_library("') |
| 626 | out.write(CMakeStringEscape(system_library)) |
| 627 | out.write('" "') |
| 628 | out.write(CMakeStringEscape(external_library)) |
Ben Wagner | 187a771 | 2016-10-27 18:47:54 -0400 | [diff] [blame] | 629 | out.write('"') |
| 630 | if library_dirs: |
| 631 | out.write(' PATHS "') |
| 632 | WriteVariable(out, '${target}__library_directories') |
| 633 | out.write('"') |
| 634 | out.write(')\n') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 635 | system_libraries.append(system_library) |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 636 | out.write('target_link_libraries("${target}"') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 637 | for library in libraries: |
| 638 | out.write('\n "') |
| 639 | out.write(CMakeStringEscape(library)) |
| 640 | out.write('"') |
| 641 | for system_library in system_libraries: |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 642 | WriteVariable(out, system_library, '\n "') |
| 643 | out.write('"') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 644 | out.write(')\n') |
| 645 | |
| 646 | |
| 647 | def WriteProject(project): |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 648 | out = open(posixpath.join(project.build_path, 'CMakeLists.txt'), 'w+') |
Ben Wagner | 82cecb6 | 2018-05-31 15:04:43 -0400 | [diff] [blame] | 649 | extName = posixpath.join(project.build_path, 'CMakeLists.ext') |
Ben Wagner | dc69dc7 | 2016-10-04 16:44:44 -0400 | [diff] [blame] | 650 | out.write('# Generated by gn_to_cmake.py.\n') |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 651 | out.write('cmake_minimum_required(VERSION 2.8.8 FATAL_ERROR)\n') |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 652 | out.write('cmake_policy(VERSION 2.8.8)\n\n') |
| 653 | |
Herb Derby | 5699278 | 2018-05-18 15:09:41 -0400 | [diff] [blame] | 654 | out.write('file(WRITE "') |
Ben Wagner | 82cecb6 | 2018-05-31 15:04:43 -0400 | [diff] [blame] | 655 | out.write(CMakeStringEscape(posixpath.join(project.build_path, "empty.cpp"))) |
Herb Derby | 5699278 | 2018-05-18 15:09:41 -0400 | [diff] [blame] | 656 | out.write('")\n') |
| 657 | |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 658 | # Update the gn generated ninja build. |
| 659 | # If a build file has changed, this will update CMakeLists.ext if |
| 660 | # gn gen out/config --ide=json --json-ide-script=../../gn/gn_to_cmake.py |
| 661 | # style was used to create this config. |
Ben Wagner | 4f0a011 | 2018-06-25 15:35:28 -0400 | [diff] [blame] | 662 | out.write('execute_process(COMMAND\n') |
| 663 | out.write(' ninja -C "') |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 664 | out.write(CMakeStringEscape(project.build_path)) |
Ben Wagner | 4f0a011 | 2018-06-25 15:35:28 -0400 | [diff] [blame] | 665 | out.write('" build.ninja\n') |
| 666 | out.write(' RESULT_VARIABLE ninja_result)\n') |
| 667 | out.write('if (ninja_result)\n') |
| 668 | out.write(' message(WARNING ') |
| 669 | out.write('"Regeneration failed running ninja: ${ninja_result}")\n') |
| 670 | out.write('endif()\n') |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 671 | |
Ben Wagner | 82cecb6 | 2018-05-31 15:04:43 -0400 | [diff] [blame] | 672 | out.write('include("') |
| 673 | out.write(CMakeStringEscape(extName)) |
| 674 | out.write('")\n') |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 675 | out.close() |
| 676 | |
Ben Wagner | 82cecb6 | 2018-05-31 15:04:43 -0400 | [diff] [blame] | 677 | out = open(extName, 'w+') |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 678 | out.write('# Generated by gn_to_cmake.py.\n') |
| 679 | out.write('cmake_minimum_required(VERSION 2.8.8 FATAL_ERROR)\n') |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 680 | out.write('cmake_policy(VERSION 2.8.8)\n') |
| 681 | |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 682 | # The following appears to be as-yet undocumented. |
| 683 | # http://public.kitware.com/Bug/view.php?id=8392 |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 684 | out.write('enable_language(ASM)\n\n') |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 685 | # ASM-ATT does not support .S files. |
| 686 | # output.write('enable_language(ASM-ATT)\n') |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 687 | |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 688 | # Current issues with automatic re-generation: |
| 689 | # The gn generated build.ninja target uses build.ninja.d |
| 690 | # but build.ninja.d does not contain the ide or gn. |
| 691 | # Currently the ide is not run if the project.json file is not changed |
| 692 | # but the ide needs to be run anyway if it has itself changed. |
| 693 | # This can be worked around by deleting the project.json file. |
| 694 | out.write('file(READ "') |
| 695 | gn_deps_file = posixpath.join(project.build_path, 'build.ninja.d') |
| 696 | out.write(CMakeStringEscape(gn_deps_file)) |
| 697 | out.write('" "gn_deps_string" OFFSET ') |
| 698 | out.write(str(len('build.ninja: '))) |
| 699 | out.write(')\n') |
| 700 | # One would think this would need to worry about escaped spaces |
| 701 | # but gn doesn't escape spaces here (it generates invalid .d files). |
| 702 | out.write('string(REPLACE " " ";" "gn_deps" ${gn_deps_string})\n') |
| 703 | out.write('foreach("gn_dep" ${gn_deps})\n') |
Ben Wagner | 4f0a011 | 2018-06-25 15:35:28 -0400 | [diff] [blame] | 704 | out.write(' configure_file("') |
| 705 | out.write(CMakeStringEscape(project.build_path)) |
| 706 | out.write('${gn_dep}" "CMakeLists.devnull" COPYONLY)\n') |
Ben Wagner | 388faa0 | 2016-10-05 17:18:38 -0400 | [diff] [blame] | 707 | out.write('endforeach("gn_dep")\n') |
| 708 | |
Ben Wagner | 4f0a011 | 2018-06-25 15:35:28 -0400 | [diff] [blame] | 709 | out.write('list(APPEND other_deps "') |
| 710 | out.write(CMakeStringEscape(os.path.abspath(__file__))) |
| 711 | out.write('")\n') |
| 712 | out.write('foreach("other_dep" ${other_deps})\n') |
| 713 | out.write(' configure_file("${other_dep}" "CMakeLists.devnull" COPYONLY)\n') |
| 714 | out.write('endforeach("other_dep")\n') |
| 715 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 716 | for target_name in project.targets.keys(): |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 717 | out.write('\n') |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 718 | WriteTarget(out, Target(target_name, project), project) |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 719 | |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 720 | |
| 721 | def main(): |
| 722 | if len(sys.argv) != 2: |
| 723 | print('Usage: ' + sys.argv[0] + ' <json_file_name>') |
| 724 | exit(1) |
| 725 | |
| 726 | json_path = sys.argv[1] |
| 727 | project = None |
| 728 | with open(json_path, 'r') as json_file: |
| 729 | project = json.loads(json_file.read()) |
| 730 | |
Ben Wagner | bc34404 | 2016-09-29 15:41:53 -0400 | [diff] [blame] | 731 | WriteProject(Project(project)) |
bungeman | e95ea08 | 2016-09-28 16:45:35 -0400 | [diff] [blame] | 732 | |
bungeman | 623ef92 | 2016-09-23 08:16:04 -0700 | [diff] [blame] | 733 | |
| 734 | if __name__ == "__main__": |
| 735 | main() |