| Index: build/android/gradle/gn_to_cmake.py
 | 
| diff --git a/build/android/gradle/gn_to_cmake.py b/build/android/gradle/gn_to_cmake.py
 | 
| new file mode 100755
 | 
| index 0000000000000000000000000000000000000000..a790d63493b4df8f02e395d932c207109aa051e0
 | 
| --- /dev/null
 | 
| +++ b/build/android/gradle/gn_to_cmake.py
 | 
| @@ -0,0 +1,680 @@
 | 
| +#!/usr/bin/env python
 | 
| +# Copyright 2016 The Chromium Authors. All rights reserved.
 | 
| +# Use of this source code is governed by a BSD-style license that can be
 | 
| +# found in the LICENSE file.
 | 
| +
 | 
| +"""
 | 
| +Usage: gn_to_cmake.py <json_file_name>
 | 
| +
 | 
| +gn gen out/config --ide=json --json-ide-script=../../gn/gn_to_cmake.py
 | 
| +
 | 
| +or
 | 
| +
 | 
| +gn gen out/config --ide=json
 | 
| +python gn/gn_to_cmake.py out/config/project.json
 | 
| +
 | 
| +The first is recommended, as it will auto-update.
 | 
| +"""
 | 
| +
 | 
| +import functools
 | 
| +import json
 | 
| +import posixpath
 | 
| +import string
 | 
| +import sys
 | 
| +
 | 
| +
 | 
| +def CMakeStringEscape(a):
 | 
| +  """Escapes the string 'a' for use inside a CMake string.
 | 
| +
 | 
| +  This means escaping
 | 
| +  '\' otherwise it may be seen as modifying the next character
 | 
| +  '"' otherwise it will end the string
 | 
| +  ';' otherwise the string becomes a list
 | 
| +
 | 
| +  The following do not need to be escaped
 | 
| +  '#' when the lexer is in string state, this does not start a comment
 | 
| +  """
 | 
| +  return a.replace('\\', '\\\\').replace(';', '\\;').replace('"', '\\"')
 | 
| +
 | 
| +
 | 
| +def CMakeTargetEscape(a):
 | 
| +  """Escapes the string 'a' for use as a CMake target name.
 | 
| +
 | 
| +  CMP0037 in CMake 3.0 restricts target names to "^[A-Za-z0-9_.:+-]+$"
 | 
| +  The ':' is only allowed for imported targets.
 | 
| +  """
 | 
| +  def Escape(c):
 | 
| +    if c in string.ascii_letters or c in string.digits or c in '_.+-':
 | 
| +      return c
 | 
| +    else:
 | 
| +      return '__'
 | 
| +  return ''.join([Escape(c) for c in a])
 | 
| +
 | 
| +
 | 
| +def SetVariable(out, variable_name, value):
 | 
| +  """Sets a CMake variable."""
 | 
| +  out.write('set("')
 | 
| +  out.write(CMakeStringEscape(variable_name))
 | 
| +  out.write('" "')
 | 
| +  out.write(CMakeStringEscape(value))
 | 
| +  out.write('")\n')
 | 
| +
 | 
| +
 | 
| +def SetVariableList(out, variable_name, values):
 | 
| +  """Sets a CMake variable to a list."""
 | 
| +  if not values:
 | 
| +    return SetVariable(out, variable_name, "")
 | 
| +  if len(values) == 1:
 | 
| +    return SetVariable(out, variable_name, values[0])
 | 
| +  out.write('list(APPEND "')
 | 
| +  out.write(CMakeStringEscape(variable_name))
 | 
| +  out.write('"\n  "')
 | 
| +  out.write('"\n  "'.join([CMakeStringEscape(value) for value in values]))
 | 
| +  out.write('")\n')
 | 
| +
 | 
| +
 | 
| +def SetFilesProperty(output, variable, property_name, values, sep):
 | 
| +  """Given a set of source files, sets the given property on them."""
 | 
| +  output.write('set_source_files_properties(')
 | 
| +  WriteVariable(output, variable)
 | 
| +  output.write(' PROPERTIES ')
 | 
| +  output.write(property_name)
 | 
| +  output.write(' "')
 | 
| +  for value in values:
 | 
| +    output.write(CMakeStringEscape(value))
 | 
| +    output.write(sep)
 | 
| +  output.write('")\n')
 | 
| +
 | 
| +
 | 
| +def SetCurrentTargetProperty(out, property_name, values, sep=''):
 | 
| +  """Given a target, sets the given property."""
 | 
| +  out.write('set_target_properties("${target}" PROPERTIES ')
 | 
| +  out.write(property_name)
 | 
| +  out.write(' "')
 | 
| +  for value in values:
 | 
| +    out.write(CMakeStringEscape(value))
 | 
| +    out.write(sep)
 | 
| +  out.write('")\n')
 | 
| +
 | 
| +
 | 
| +def WriteVariable(output, variable_name, prepend=None):
 | 
| +  if prepend:
 | 
| +    output.write(prepend)
 | 
| +  output.write('${')
 | 
| +  output.write(variable_name)
 | 
| +  output.write('}')
 | 
| +
 | 
| +
 | 
| +# See GetSourceFileType in gn
 | 
| +source_file_types = {
 | 
| +  '.cc': 'cxx',
 | 
| +  '.cpp': 'cxx',
 | 
| +  '.cxx': 'cxx',
 | 
| +  '.c': 'c',
 | 
| +  '.s': 'asm',
 | 
| +  '.S': 'asm',
 | 
| +  '.asm': 'asm',
 | 
| +  '.o': 'obj',
 | 
| +  '.obj': 'obj',
 | 
| +}
 | 
| +
 | 
| +
 | 
| +class CMakeTargetType(object):
 | 
| +  def __init__(self, command, modifier, property_modifier, is_linkable):
 | 
| +    self.command = command
 | 
| +    self.modifier = modifier
 | 
| +    self.property_modifier = property_modifier
 | 
| +    self.is_linkable = is_linkable
 | 
| +CMakeTargetType.custom = CMakeTargetType('add_custom_target', 'SOURCES',
 | 
| +                                         None, False)
 | 
| +
 | 
| +# See GetStringForOutputType in gn
 | 
| +cmake_target_types = {
 | 
| +  'unknown': CMakeTargetType.custom,
 | 
| +  'group': CMakeTargetType.custom,
 | 
| +  'executable': CMakeTargetType('add_executable', None, 'RUNTIME', True),
 | 
| +  'loadable_module': CMakeTargetType('add_library', 'MODULE', 'LIBRARY', True),
 | 
| +  'shared_library': CMakeTargetType('add_library', 'SHARED', 'LIBRARY', True),
 | 
| +  'static_library': CMakeTargetType('add_library', 'STATIC', 'ARCHIVE', False),
 | 
| +  'source_set': CMakeTargetType('add_library', 'OBJECT', None, False),
 | 
| +  'copy': CMakeTargetType.custom,
 | 
| +  'action': CMakeTargetType.custom,
 | 
| +  'action_foreach': CMakeTargetType.custom,
 | 
| +  'bundle_data': CMakeTargetType.custom,
 | 
| +  'create_bundle': CMakeTargetType.custom,
 | 
| +}
 | 
| +
 | 
| +
 | 
| +def FindFirstOf(s, a):
 | 
| +  return min(s.find(i) for i in a if i in s)
 | 
| +
 | 
| +
 | 
| +def GetCMakeTargetName(gn_target_name):
 | 
| +  # See <chromium>/src/tools/gn/label.cc#Resolve
 | 
| +  # //base/test:test_support(//build/toolchain/win:msvc)
 | 
| +  path_separator = FindFirstOf(gn_target_name, (':', '('))
 | 
| +  location = None
 | 
| +  name = None
 | 
| +  toolchain = None
 | 
| +  if not path_separator:
 | 
| +    location = gn_target_name[2:]
 | 
| +  else:
 | 
| +    location = gn_target_name[2:path_separator]
 | 
| +    toolchain_separator = gn_target_name.find('(', path_separator)
 | 
| +    if toolchain_separator == -1:
 | 
| +      name = gn_target_name[path_separator + 1:]
 | 
| +    else:
 | 
| +      if toolchain_separator > path_separator:
 | 
| +        name = gn_target_name[path_separator + 1:toolchain_separator]
 | 
| +      assert gn_target_name.endswith(')')
 | 
| +      toolchain = gn_target_name[toolchain_separator + 1:-1]
 | 
| +  assert location or name
 | 
| +
 | 
| +  cmake_target_name = None
 | 
| +  if location.endswith('/' + name):
 | 
| +    cmake_target_name = location
 | 
| +  elif location:
 | 
| +    cmake_target_name = location + '_' + name
 | 
| +  else:
 | 
| +    cmake_target_name = name
 | 
| +  if toolchain:
 | 
| +    cmake_target_name += '--' + toolchain
 | 
| +  return CMakeTargetEscape(cmake_target_name)
 | 
| +
 | 
| +
 | 
| +class Project(object):
 | 
| +  def __init__(self, project_json):
 | 
| +    self.targets = project_json['targets']
 | 
| +    build_settings = project_json['build_settings']
 | 
| +    self.root_path = build_settings['root_path']
 | 
| +    self.build_path = posixpath.join(self.root_path,
 | 
| +                                     build_settings['build_dir'][2:])
 | 
| +
 | 
| +  def GetAbsolutePath(self, path):
 | 
| +    if path.startswith("//"):
 | 
| +      return self.root_path + "/" + path[2:]
 | 
| +    else:
 | 
| +      return path
 | 
| +
 | 
| +  def GetObjectSourceDependencies(self, gn_target_name, object_dependencies):
 | 
| +    """All OBJECT libraries whose sources have not been absorbed."""
 | 
| +    dependencies = self.targets[gn_target_name].get('deps', [])
 | 
| +    for dependency in dependencies:
 | 
| +      dependency_type = self.targets[dependency].get('type', None)
 | 
| +      if dependency_type == 'source_set':
 | 
| +        object_dependencies.add(dependency)
 | 
| +      if dependency_type not in gn_target_types_that_absorb_objects:
 | 
| +        self.GetObjectSourceDependencies(dependency, object_dependencies)
 | 
| +
 | 
| +  def GetObjectLibraryDependencies(self, gn_target_name, object_dependencies):
 | 
| +    """All OBJECT libraries whose libraries have not been absorbed."""
 | 
| +    dependencies = self.targets[gn_target_name].get('deps', [])
 | 
| +    for dependency in dependencies:
 | 
| +      dependency_type = self.targets[dependency].get('type', None)
 | 
| +      if dependency_type == 'source_set':
 | 
| +        object_dependencies.add(dependency)
 | 
| +        self.GetObjectLibraryDependencies(dependency, object_dependencies)
 | 
| +
 | 
| +
 | 
| +class Target(object):
 | 
| +  def __init__(self, gn_target_name, project):
 | 
| +    self.gn_name = gn_target_name
 | 
| +    self.properties = project.targets[self.gn_name]
 | 
| +    self.cmake_name = GetCMakeTargetName(self.gn_name)
 | 
| +    self.gn_type = self.properties.get('type', None)
 | 
| +    self.cmake_type = cmake_target_types.get(self.gn_type, None)
 | 
| +
 | 
| +
 | 
| +def WriteAction(out, target, project, sources, synthetic_dependencies):
 | 
| +  outputs = []
 | 
| +  output_directories = set()
 | 
| +  for output in target.properties.get('outputs', []):
 | 
| +    output_abs_path = project.GetAbsolutePath(output)
 | 
| +    outputs.append(output_abs_path)
 | 
| +    output_directory = posixpath.dirname(output_abs_path)
 | 
| +    if output_directory:
 | 
| +      output_directories.add(output_directory)
 | 
| +  outputs_name = '${target}__output'
 | 
| +  SetVariableList(out, outputs_name, outputs)
 | 
| +
 | 
| +  out.write('add_custom_command(OUTPUT ')
 | 
| +  WriteVariable(out, outputs_name)
 | 
| +  out.write('\n')
 | 
| +
 | 
| +  if output_directories:
 | 
| +    out.write('  COMMAND ${CMAKE_COMMAND} -E make_directory "')
 | 
| +    out.write('" "'.join([CMakeStringEscape(d) for d in output_directories]))
 | 
| +    out.write('"\n')
 | 
| +
 | 
| +  script = target.properties['script']
 | 
| +  arguments = target.properties['args']
 | 
| +  out.write('  COMMAND python "')
 | 
| +  out.write(CMakeStringEscape(project.GetAbsolutePath(script)))
 | 
| +  out.write('"')
 | 
| +  if arguments:
 | 
| +    out.write('\n    "')
 | 
| +    out.write('"\n    "'.join([CMakeStringEscape(a) for a in arguments]))
 | 
| +    out.write('"')
 | 
| +  out.write('\n')
 | 
| +
 | 
| +  out.write('  DEPENDS ')
 | 
| +  for sources_type_name in sources.values():
 | 
| +    WriteVariable(out, sources_type_name, ' ')
 | 
| +  out.write('\n')
 | 
| +
 | 
| +  #TODO: CMake 3.7 is introducing DEPFILE
 | 
| +
 | 
| +  out.write('  WORKING_DIRECTORY "')
 | 
| +  out.write(CMakeStringEscape(project.build_path))
 | 
| +  out.write('"\n')
 | 
| +
 | 
| +  out.write('  COMMENT "Action: ${target}"\n')
 | 
| +
 | 
| +  out.write('  VERBATIM)\n')
 | 
| +
 | 
| +  synthetic_dependencies.add(outputs_name)
 | 
| +
 | 
| +
 | 
| +def ExpandPlaceholders(source, a):
 | 
| +  source_dir, source_file_part = posixpath.split(source)
 | 
| +  source_name_part, _ = posixpath.splitext(source_file_part)
 | 
| +  #TODO: {{source_gen_dir}}, {{source_out_dir}}, {{response_file_name}}
 | 
| +  return a.replace('{{source}}', source) \
 | 
| +          .replace('{{source_file_part}}', source_file_part) \
 | 
| +          .replace('{{source_name_part}}', source_name_part) \
 | 
| +          .replace('{{source_dir}}', source_dir) \
 | 
| +          .replace('{{source_root_relative_dir}}', source_dir)
 | 
| +
 | 
| +
 | 
| +def WriteActionForEach(out, target, project, sources, synthetic_dependencies):
 | 
| +  all_outputs = target.properties.get('outputs', [])
 | 
| +  inputs = target.properties.get('sources', [])
 | 
| +  # TODO: consider expanding 'output_patterns' instead.
 | 
| +  outputs_per_input = len(all_outputs) / len(inputs)
 | 
| +  for count, source in enumerate(inputs):
 | 
| +    source_abs_path = project.GetAbsolutePath(source)
 | 
| +
 | 
| +    outputs = []
 | 
| +    output_directories = set()
 | 
| +    for output in all_outputs[outputs_per_input *  count:
 | 
| +                              outputs_per_input * (count+1)]:
 | 
| +      output_abs_path = project.GetAbsolutePath(output)
 | 
| +      outputs.append(output_abs_path)
 | 
| +      output_directory = posixpath.dirname(output_abs_path)
 | 
| +      if output_directory:
 | 
| +        output_directories.add(output_directory)
 | 
| +    outputs_name = '${target}__output_' + str(count)
 | 
| +    SetVariableList(out, outputs_name, outputs)
 | 
| +
 | 
| +    out.write('add_custom_command(OUTPUT ')
 | 
| +    WriteVariable(out, outputs_name)
 | 
| +    out.write('\n')
 | 
| +
 | 
| +    if output_directories:
 | 
| +      out.write('  COMMAND ${CMAKE_COMMAND} -E make_directory "')
 | 
| +      out.write('" "'.join([CMakeStringEscape(d) for d in output_directories]))
 | 
| +      out.write('"\n')
 | 
| +
 | 
| +    script = target.properties['script']
 | 
| +    # TODO: need to expand {{xxx}} in arguments
 | 
| +    arguments = target.properties['args']
 | 
| +    out.write('  COMMAND python "')
 | 
| +    out.write(CMakeStringEscape(project.GetAbsolutePath(script)))
 | 
| +    out.write('"')
 | 
| +    if arguments:
 | 
| +      out.write('\n    "')
 | 
| +      expand = functools.partial(ExpandPlaceholders, source_abs_path)
 | 
| +      out.write('"\n    "'.join(
 | 
| +          [CMakeStringEscape(expand(a)) for a in arguments]))
 | 
| +      out.write('"')
 | 
| +    out.write('\n')
 | 
| +
 | 
| +    out.write('  DEPENDS')
 | 
| +    if 'input' in sources:
 | 
| +      WriteVariable(out, sources['input'], ' ')
 | 
| +    out.write(' "')
 | 
| +    out.write(CMakeStringEscape(source_abs_path))
 | 
| +    out.write('"\n')
 | 
| +
 | 
| +    #TODO: CMake 3.7 is introducing DEPFILE
 | 
| +
 | 
| +    out.write('  WORKING_DIRECTORY "')
 | 
| +    out.write(CMakeStringEscape(project.build_path))
 | 
| +    out.write('"\n')
 | 
| +
 | 
| +    out.write('  COMMENT "Action ${target} on ')
 | 
| +    out.write(CMakeStringEscape(source_abs_path))
 | 
| +    out.write('"\n')
 | 
| +
 | 
| +    out.write('  VERBATIM)\n')
 | 
| +
 | 
| +    synthetic_dependencies.add(outputs_name)
 | 
| +
 | 
| +
 | 
| +def WriteCopy(out, target, project, sources, synthetic_dependencies):
 | 
| +  inputs = target.properties.get('sources', [])
 | 
| +  raw_outputs = target.properties.get('outputs', [])
 | 
| +
 | 
| +  # TODO: consider expanding 'output_patterns' instead.
 | 
| +  outputs = []
 | 
| +  for output in raw_outputs:
 | 
| +    output_abs_path = project.GetAbsolutePath(output)
 | 
| +    outputs.append(output_abs_path)
 | 
| +  outputs_name = '${target}__output'
 | 
| +  SetVariableList(out, outputs_name, outputs)
 | 
| +
 | 
| +  out.write('add_custom_command(OUTPUT ')
 | 
| +  WriteVariable(out, outputs_name)
 | 
| +  out.write('\n')
 | 
| +
 | 
| +  for src, dst in zip(inputs, outputs):
 | 
| +    out.write('  COMMAND ${CMAKE_COMMAND} -E copy "')
 | 
| +    out.write(CMakeStringEscape(project.GetAbsolutePath(src)))
 | 
| +    out.write('" "')
 | 
| +    out.write(CMakeStringEscape(dst))
 | 
| +    out.write('"\n')
 | 
| +
 | 
| +  out.write('  DEPENDS ')
 | 
| +  for sources_type_name in sources.values():
 | 
| +    WriteVariable(out, sources_type_name, ' ')
 | 
| +  out.write('\n')
 | 
| +
 | 
| +  out.write('  WORKING_DIRECTORY "')
 | 
| +  out.write(CMakeStringEscape(project.build_path))
 | 
| +  out.write('"\n')
 | 
| +
 | 
| +  out.write('  COMMENT "Copy ${target}"\n')
 | 
| +
 | 
| +  out.write('  VERBATIM)\n')
 | 
| +
 | 
| +  synthetic_dependencies.add(outputs_name)
 | 
| +
 | 
| +
 | 
| +def WriteCompilerFlags(out, target, project, sources):
 | 
| +  # Hack, set linker language to c if no c or cxx files present.
 | 
| +  if not 'c' in sources and not 'cxx' in sources:
 | 
| +    SetCurrentTargetProperty(out, 'LINKER_LANGUAGE', ['C'])
 | 
| +
 | 
| +  # Mark uncompiled sources as uncompiled.
 | 
| +  if 'input' in sources:
 | 
| +    SetFilesProperty(out, sources['input'], 'HEADER_FILE_ONLY', ('True',), '')
 | 
| +  if 'other' in sources:
 | 
| +    SetFilesProperty(out, sources['other'], 'HEADER_FILE_ONLY', ('True',), '')
 | 
| +
 | 
| +  # Mark object sources as linkable.
 | 
| +  if 'obj' in sources:
 | 
| +    SetFilesProperty(out, sources['obj'], 'EXTERNAL_OBJECT', ('True',), '')
 | 
| +
 | 
| +  # TODO: 'output_name', 'output_dir', 'output_extension'
 | 
| +  # This includes using 'source_outputs' to direct compiler output.
 | 
| +
 | 
| +  # Includes
 | 
| +  includes = target.properties.get('include_dirs', [])
 | 
| +  if includes:
 | 
| +    out.write('set_property(TARGET "${target}" ')
 | 
| +    out.write('APPEND PROPERTY INCLUDE_DIRECTORIES')
 | 
| +    for include_dir in includes:
 | 
| +      out.write('\n  "')
 | 
| +      out.write(project.GetAbsolutePath(include_dir))
 | 
| +      out.write('"')
 | 
| +    out.write(')\n')
 | 
| +
 | 
| +  # Defines
 | 
| +  defines = target.properties.get('defines', [])
 | 
| +  if defines:
 | 
| +    SetCurrentTargetProperty(out, 'COMPILE_DEFINITIONS', defines, ';')
 | 
| +
 | 
| +  # Compile flags
 | 
| +  # "arflags", "asmflags", "cflags",
 | 
| +  # "cflags_c", "clfags_cc", "cflags_objc", "clfags_objcc"
 | 
| +  # CMake does not have per target lang compile flags.
 | 
| +  # TODO: $<$<COMPILE_LANGUAGE:CXX>:cflags_cc style generator expression.
 | 
| +  #       http://public.kitware.com/Bug/view.php?id=14857
 | 
| +  flags = []
 | 
| +  flags.extend(target.properties.get('cflags', []))
 | 
| +  cflags_asm = target.properties.get('asmflags', [])
 | 
| +  cflags_c = target.properties.get('cflags_c', [])
 | 
| +  cflags_cxx = target.properties.get('cflags_cc', [])
 | 
| +  if 'c' in sources and not any(k in sources for k in ('asm', 'cxx')):
 | 
| +    flags.extend(cflags_c)
 | 
| +  elif 'cxx' in sources and not any(k in sources for k in ('asm', 'c')):
 | 
| +    flags.extend(cflags_cxx)
 | 
| +  else:
 | 
| +    # TODO: This is broken, one cannot generally set properties on files,
 | 
| +    # as other targets may require different properties on the same files.
 | 
| +    if 'asm' in sources and cflags_asm:
 | 
| +      SetFilesProperty(out, sources['asm'], 'COMPILE_FLAGS', cflags_asm, ' ')
 | 
| +    if 'c' in sources and cflags_c:
 | 
| +      SetFilesProperty(out, sources['c'], 'COMPILE_FLAGS', cflags_c, ' ')
 | 
| +    if 'cxx' in sources and cflags_cxx:
 | 
| +      SetFilesProperty(out, sources['cxx'], 'COMPILE_FLAGS', cflags_cxx, ' ')
 | 
| +  if flags:
 | 
| +    SetCurrentTargetProperty(out, 'COMPILE_FLAGS', flags, ' ')
 | 
| +
 | 
| +  # Linker flags
 | 
| +  ldflags = target.properties.get('ldflags', [])
 | 
| +  if ldflags:
 | 
| +    SetCurrentTargetProperty(out, 'LINK_FLAGS', ldflags, ' ')
 | 
| +
 | 
| +
 | 
| +gn_target_types_that_absorb_objects = (
 | 
| +  'executable',
 | 
| +  'loadable_module',
 | 
| +  'shared_library',
 | 
| +  'static_library'
 | 
| +)
 | 
| +
 | 
| +
 | 
| +def WriteSourceVariables(out, target, project):
 | 
| +  # gn separates the sheep from the goats based on file extensions.
 | 
| +  # A full separation is done here because of flag handing (see Compile flags).
 | 
| +  source_types = {'cxx':[], 'c':[], 'asm':[],
 | 
| +                  'obj':[], 'obj_target':[], 'input':[], 'other':[]}
 | 
| +
 | 
| +  # TODO .def files on Windows
 | 
| +  for source in target.properties.get('sources', []):
 | 
| +    _, ext = posixpath.splitext(source)
 | 
| +    source_abs_path = project.GetAbsolutePath(source)
 | 
| +    source_types[source_file_types.get(ext, 'other')].append(source_abs_path)
 | 
| +
 | 
| +  for input_path in target.properties.get('inputs', []):
 | 
| +    input_abs_path = project.GetAbsolutePath(input_path)
 | 
| +    source_types['input'].append(input_abs_path)
 | 
| +
 | 
| +  # OBJECT library dependencies need to be listed as sources.
 | 
| +  # Only executables and non-OBJECT libraries may reference an OBJECT library.
 | 
| +  # https://gitlab.kitware.com/cmake/cmake/issues/14778
 | 
| +  if target.gn_type in gn_target_types_that_absorb_objects:
 | 
| +    object_dependencies = set()
 | 
| +    project.GetObjectSourceDependencies(target.gn_name, object_dependencies)
 | 
| +    for dependency in object_dependencies:
 | 
| +      cmake_dependency_name = GetCMakeTargetName(dependency)
 | 
| +      obj_target_sources = '$<TARGET_OBJECTS:' + cmake_dependency_name + '>'
 | 
| +      source_types['obj_target'].append(obj_target_sources)
 | 
| +
 | 
| +  sources = {}
 | 
| +  for source_type, sources_of_type in source_types.items():
 | 
| +    if sources_of_type:
 | 
| +      sources[source_type] = '${target}__' + source_type + '_srcs'
 | 
| +      SetVariableList(out, sources[source_type], sources_of_type)
 | 
| +  return sources
 | 
| +
 | 
| +
 | 
| +def WriteTarget(out, target, project):
 | 
| +  out.write('\n#')
 | 
| +  out.write(target.gn_name)
 | 
| +  out.write('\n')
 | 
| +
 | 
| +  if target.cmake_type is None:
 | 
| +    print 'Target {} has unknown target type {}, skipping.'.format(
 | 
| +        target.gn_name, target.gn_type)
 | 
| +    return
 | 
| +
 | 
| +  SetVariable(out, 'target', target.cmake_name)
 | 
| +
 | 
| +  sources = WriteSourceVariables(out, target, project)
 | 
| +
 | 
| +  synthetic_dependencies = set()
 | 
| +  if target.gn_type == 'action':
 | 
| +    WriteAction(out, target, project, sources, synthetic_dependencies)
 | 
| +  if target.gn_type == 'action_foreach':
 | 
| +    WriteActionForEach(out, target, project, sources, synthetic_dependencies)
 | 
| +  if target.gn_type == 'copy':
 | 
| +    WriteCopy(out, target, project, sources, synthetic_dependencies)
 | 
| +
 | 
| +  out.write(target.cmake_type.command)
 | 
| +  out.write('("${target}"')
 | 
| +  if target.cmake_type.modifier is not None:
 | 
| +    out.write(' ')
 | 
| +    out.write(target.cmake_type.modifier)
 | 
| +  for sources_type_name in sources.values():
 | 
| +    WriteVariable(out, sources_type_name, ' ')
 | 
| +  if synthetic_dependencies:
 | 
| +    out.write(' DEPENDS')
 | 
| +    for synthetic_dependencie in synthetic_dependencies:
 | 
| +      WriteVariable(out, synthetic_dependencie, ' ')
 | 
| +  out.write(')\n')
 | 
| +
 | 
| +  if target.cmake_type.command != 'add_custom_target':
 | 
| +    WriteCompilerFlags(out, target, project, sources)
 | 
| +
 | 
| +  libraries = set()
 | 
| +  nonlibraries = set()
 | 
| +
 | 
| +  dependencies = set(target.properties.get('deps', []))
 | 
| +  # Transitive OBJECT libraries are in sources.
 | 
| +  # Those sources are dependent on the OBJECT library dependencies.
 | 
| +  # Those sources cannot bring in library dependencies.
 | 
| +  object_dependencies = set()
 | 
| +  if target.gn_type != 'source_set':
 | 
| +    project.GetObjectLibraryDependencies(target.gn_name, object_dependencies)
 | 
| +  for object_dependency in object_dependencies:
 | 
| +    dependencies.update(project.targets.get(object_dependency).get('deps', []))
 | 
| +
 | 
| +  for dependency in dependencies:
 | 
| +    gn_dependency_type = project.targets.get(dependency, {}).get('type', None)
 | 
| +    cmake_dependency_type = cmake_target_types.get(gn_dependency_type, None)
 | 
| +    cmake_dependency_name = GetCMakeTargetName(dependency)
 | 
| +    if cmake_dependency_type.command != 'add_library':
 | 
| +      nonlibraries.add(cmake_dependency_name)
 | 
| +    elif cmake_dependency_type.modifier != 'OBJECT':
 | 
| +      if target.cmake_type.is_linkable:
 | 
| +        libraries.add(cmake_dependency_name)
 | 
| +      else:
 | 
| +        nonlibraries.add(cmake_dependency_name)
 | 
| +
 | 
| +  # Non-library dependencies.
 | 
| +  if nonlibraries:
 | 
| +    out.write('add_dependencies("${target}"')
 | 
| +    for nonlibrary in nonlibraries:
 | 
| +      out.write('\n  "')
 | 
| +      out.write(nonlibrary)
 | 
| +      out.write('"')
 | 
| +    out.write(')\n')
 | 
| +
 | 
| +  # Non-OBJECT library dependencies.
 | 
| +  external_libraries = target.properties.get('libs', [])
 | 
| +  if target.cmake_type.is_linkable and (external_libraries or libraries):
 | 
| +    library_dirs = target.properties.get('lib_dirs', [])
 | 
| +    if library_dirs:
 | 
| +      SetVariableList(out, '${target}__library_directories', library_dirs)
 | 
| +
 | 
| +    system_libraries = []
 | 
| +    for external_library in external_libraries:
 | 
| +      if '/' in external_library:
 | 
| +        libraries.add(project.GetAbsolutePath(external_library))
 | 
| +      else:
 | 
| +        if external_library.endswith('.framework'):
 | 
| +          external_library = external_library[:-len('.framework')]
 | 
| +        system_library = 'library__' + external_library
 | 
| +        if library_dirs:
 | 
| +          system_library = system_library + '__for_${target}'
 | 
| +        out.write('find_library("')
 | 
| +        out.write(CMakeStringEscape(system_library))
 | 
| +        out.write('" "')
 | 
| +        out.write(CMakeStringEscape(external_library))
 | 
| +        out.write('"')
 | 
| +        if library_dirs:
 | 
| +          out.write(' PATHS "')
 | 
| +          WriteVariable(out, '${target}__library_directories')
 | 
| +          out.write('"')
 | 
| +        out.write(')\n')
 | 
| +        system_libraries.append(system_library)
 | 
| +    out.write('target_link_libraries("${target}"')
 | 
| +    for library in libraries:
 | 
| +      out.write('\n  "')
 | 
| +      out.write(CMakeStringEscape(library))
 | 
| +      out.write('"')
 | 
| +    for system_library in system_libraries:
 | 
| +      WriteVariable(out, system_library, '\n  "')
 | 
| +      out.write('"')
 | 
| +    out.write(')\n')
 | 
| +
 | 
| +
 | 
| +def WriteProject(project):
 | 
| +  out = open(posixpath.join(project.build_path, 'CMakeLists.txt'), 'w+')
 | 
| +  out.write('# Generated by gn_to_cmake.py.\n')
 | 
| +  out.write('cmake_minimum_required(VERSION 2.8.8 FATAL_ERROR)\n')
 | 
| +  out.write('cmake_policy(VERSION 2.8.8)\n\n')
 | 
| +
 | 
| +  # Update the gn generated ninja build.
 | 
| +  # If a build file has changed, this will update CMakeLists.ext if
 | 
| +  # gn gen out/config --ide=json --json-ide-script=../../gn/gn_to_cmake.py
 | 
| +  # style was used to create this config.
 | 
| +  out.write('execute_process(COMMAND ninja -C "')
 | 
| +  out.write(CMakeStringEscape(project.build_path))
 | 
| +  out.write('" build.ninja)\n')
 | 
| +
 | 
| +  out.write('include(CMakeLists.ext)\n')
 | 
| +  out.close()
 | 
| +
 | 
| +  out = open(posixpath.join(project.build_path, 'CMakeLists.ext'), 'w+')
 | 
| +  out.write('# Generated by gn_to_cmake.py.\n')
 | 
| +  out.write('cmake_minimum_required(VERSION 2.8.8 FATAL_ERROR)\n')
 | 
| +  out.write('cmake_policy(VERSION 2.8.8)\n')
 | 
| +
 | 
| +  # The following appears to be as-yet undocumented.
 | 
| +  # http://public.kitware.com/Bug/view.php?id=8392
 | 
| +  out.write('enable_language(ASM)\n\n')
 | 
| +  # ASM-ATT does not support .S files.
 | 
| +  # output.write('enable_language(ASM-ATT)\n')
 | 
| +
 | 
| +  # Current issues with automatic re-generation:
 | 
| +  # The gn generated build.ninja target uses build.ninja.d
 | 
| +  #   but build.ninja.d does not contain the ide or gn.
 | 
| +  # Currently the ide is not run if the project.json file is not changed
 | 
| +  #   but the ide needs to be run anyway if it has itself changed.
 | 
| +  #   This can be worked around by deleting the project.json file.
 | 
| +  out.write('file(READ "')
 | 
| +  gn_deps_file = posixpath.join(project.build_path, 'build.ninja.d')
 | 
| +  out.write(CMakeStringEscape(gn_deps_file))
 | 
| +  out.write('" "gn_deps_string" OFFSET ')
 | 
| +  out.write(str(len('build.ninja: ')))
 | 
| +  out.write(')\n')
 | 
| +  # One would think this would need to worry about escaped spaces
 | 
| +  # but gn doesn't escape spaces here (it generates invalid .d files).
 | 
| +  out.write('string(REPLACE " " ";" "gn_deps" ${gn_deps_string})\n')
 | 
| +  out.write('foreach("gn_dep" ${gn_deps})\n')
 | 
| +  out.write('  configure_file(${gn_dep} "CMakeLists.devnull" COPYONLY)\n')
 | 
| +  out.write('endforeach("gn_dep")\n')
 | 
| +
 | 
| +  for target_name in project.targets.keys():
 | 
| +    out.write('\n')
 | 
| +    WriteTarget(out, Target(target_name, project), project)
 | 
| +
 | 
| +
 | 
| +def main():
 | 
| +  if len(sys.argv) != 2:
 | 
| +    print 'Usage: ' + sys.argv[0] + ' <json_file_name>'
 | 
| +    exit(1)
 | 
| +
 | 
| +  json_path = sys.argv[1]
 | 
| +  project = None
 | 
| +  with open(json_path, 'r') as json_file:
 | 
| +    project = json.loads(json_file.read())
 | 
| +
 | 
| +  WriteProject(Project(project))
 | 
| +
 | 
| +
 | 
| +if __name__ == "__main__":
 | 
| +  main()
 | 
| 
 |