gn 13.0 KB
Newer Older
1
#!/usr/bin/env python
M
Michael Goderbauer 已提交
2
# Copyright 2013 The Flutter Authors. All rights reserved.
3 4 5 6 7 8 9 10
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.

import argparse
import subprocess
import sys
import os

11
SRC_ROOT = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
12

13
def get_out_dir(args):
14 15 16 17 18
    if args.target_os is not None:
        target_dir = [args.target_os]
    else:
        target_dir = ['host']

19 20
    runtime_mode = args.runtime_mode
    if args.dynamic and runtime_mode in ['profile', 'release']:
21 22
        target_dir.append('dynamic')

23
    target_dir.append(args.runtime_mode)
24 25

    if args.simulator:
26
        target_dir.append('sim')
27

28 29
    if args.unoptimized:
        target_dir.append('unopt')
30

31 32 33
    if args.target_os != 'ios' and args.interpreter:
        target_dir.append('interpreter')

34
    if args.android_cpu != 'arm':
35
        target_dir.append(args.android_cpu)
36

37 38 39
    if args.ios_cpu != 'arm64':
        target_dir.append(args.ios_cpu)

40 41 42
    if args.linux_cpu is not None:
      target_dir.append(args.linux_cpu)

43 44 45
    if args.enable_vulkan:
        target_dir.append('vulkan')

46
    return os.path.join(args.out_dir, 'out', '_'.join(target_dir))
47 48 49 50

def to_command_line(gn_args):
    def merge(key, value):
        if type(value) is bool:
51 52
            return '%s=%s' % (key, 'true' if value else 'false')
        return '%s="%s"' % (key, value)
53 54
    return [merge(x, y) for x, y in gn_args.iteritems()]

55 56 57 58 59 60 61 62
def cpu_for_target_arch(arch):
  if arch in ['ia32', 'arm', 'armv6', 'armv5te', 'mips',
              'simarm', 'simarmv6', 'simarmv5te', 'simmips', 'simdbc',
              'armsimdbc']:
    return 'x86'
  if arch in ['x64', 'arm64', 'simarm64', 'simdbc64', 'armsimdbc64']:
    return 'x64'

63
def to_gn_args(args):
64 65 66 67
    if args.simulator:
        if args.target_os == 'android':
            raise Exception('--simulator is not supported on Android')
        elif args.target_os == 'ios':
68 69
            if args.runtime_mode != 'debug':
                raise Exception('iOS simulator only supports the debug runtime mode')
70

71 72 73
    if args.target_os != 'android' and args.enable_vulkan:
      raise Exception('--enable-vulkan is only supported on Android')

74 75 76 77
    runtime_mode = args.runtime_mode
    if args.dynamic and runtime_mode in ['profile', 'release']:
      runtime_mode = 'dynamic_' + runtime_mode

78 79
    gn_args = {}

80
    # Skia GN args.
81
    gn_args['skia_enable_flutter_defines'] = True # Enable Flutter API guards in Skia.
82
    gn_args['skia_use_dng_sdk'] = False    # RAW image handling.
83 84
    gn_args['skia_use_sfntly'] = False     # PDF handling depedency.
    gn_args['skia_enable_pdf'] = False     # PDF handling.
85
    gn_args['skia_use_x11'] = False        # Never add the X11 dependency (only takes effect on Linux).
86
    gn_args['skia_use_expat'] = args.target_os == 'android'
87
    gn_args['skia_use_fontconfig'] = False # Use the custom font manager instead.
88
    gn_args['is_official_build'] = True    # Disable Skia test utilities.
89

90
    gn_args['is_debug'] = args.unoptimized
91
    gn_args['android_full_debug'] = args.target_os == 'android' and args.unoptimized
92
    gn_args['is_clang'] = not sys.platform.startswith(('cygwin', 'win'))
93

94 95 96
    if not sys.platform.startswith(('cygwin', 'win')):
      gn_args['use_clang_static_analyzer'] = args.clang_static_analyzer

97
    gn_args['embedder_for_target'] = args.embedder_for_target
98

99 100
    gn_args['enable_coverage'] = args.coverage

101 102 103
    if args.operator_new_alignment is not None:
      gn_args['operator_new_alignment'] = args.operator_new_alignment

104 105 106 107 108
    enable_lto = args.lto
    if args.unoptimized:
      # There is no point in enabling LTO in unoptimized builds.
      enable_lto = False

109
    if not sys.platform.startswith('win'):
110 111
      # The GN arg is not available in the windows toolchain.
      gn_args['enable_lto'] = enable_lto
112

113
    aot = not runtime_mode in ['debug', 'dynamic_profile', 'dynamic_release']
114
    if args.target_os == 'android':
115
        gn_args['target_os'] = 'android'
116
    elif args.target_os == 'ios':
117
        gn_args['target_os'] = 'ios'
118
        gn_args['use_ios_simulator'] = args.simulator
119
        if not args.simulator:
120
          aot = True
121 122
    elif args.target_os is not None:
        gn_args['target_os'] = args.target_os
123
    else:
124
      aot = False
125

126 127
    gn_args['dart_lib_export_symbols'] = False

128
    if runtime_mode == 'debug':
129
        gn_args['dart_runtime_mode'] = 'develop'
130 131 132 133
    elif runtime_mode == 'dynamic_profile':
        gn_args['dart_runtime_mode'] = 'profile'
    elif runtime_mode == 'dynamic_release':
        gn_args['dart_runtime_mode'] = 'release'
134
    else:
135
        gn_args['dart_runtime_mode'] = runtime_mode
136

137 138 139
    if args.dart_debug:
        gn_args['dart_debug'] = True

140 141 142 143
    if args.full_dart_debug:
      gn_args['dart_debug'] = True
      gn_args['dart_debug_optimization_level'] = '0'

144 145 146 147 148 149
    if args.target_os == 'android':
        gn_args['target_cpu'] = args.android_cpu
    elif args.target_os == 'ios':
        if args.simulator:
            gn_args['target_cpu'] = 'x64'
        else:
150
            gn_args['target_cpu'] = args.ios_cpu
151 152
    elif args.target_os == 'linux':
      gn_args['target_cpu'] = args.linux_cpu
153 154 155
    else:
        # Building host artifacts
        gn_args['target_cpu'] = 'x64'
156

157 158 159
    # On iOS Devices, use the Dart bytecode interpreter so we don't incur
    # snapshotting and linking costs of the precompiler during development.
    # We can still use the JIT on the simulator though.
160 161 162 163 164 165 166
    can_use_dbc = runtime_mode in ['debug', 'dynamic_profile', 'dynamic_release']
    use_dbc = args.target_os == 'ios' and not args.simulator and can_use_dbc
    # Use dbc if it is requested on the command line and supported by the
    # requested runtime mode.
    if args.interpreter and not can_use_dbc:
      raise Exception('--interpreter not supported with --runtime-mode=' + runtime_mode)
    use_dbc = use_dbc or (args.interpreter and can_use_dbc)
167 168
    if use_dbc:
      gn_args['dart_target_arch'] = 'dbc'
169 170 171 172 173 174 175 176 177
    else:
      gn_args['dart_target_arch'] = gn_args['target_cpu']

    # No cross-compilation on Windows (for now).
    if sys.platform.startswith(('cygwin', 'win')):
      if 'target_os' in gn_args:
        gn_args['target_os'] = 'win'
      if 'target_cpu' in gn_args:
        gn_args['target_cpu'] = cpu_for_target_arch(gn_args['target_cpu'])
178

179 180 181 182 183
    if args.target_os is None:
      if sys.platform.startswith(('cygwin', 'win')):
        gn_args['dart_use_fallback_root_certificates'] = True


184 185
    gn_args['dart_custom_version_for_pub'] = 'flutter'

186
    # Make sure host_cpu matches the bit width of target_cpu.
187 188
    target_is_32_bit = gn_args['target_cpu'] == 'arm' or gn_args['target_cpu'] == 'x86'
    if target_is_32_bit:
189
      gn_args["host_cpu"] = "x86"
190

191
    gn_args['flutter_runtime_mode'] = runtime_mode
192
    gn_args['flutter_aot'] = aot
193

G
George Kulakowski 已提交
194 195
    if args.target_sysroot:
      gn_args['target_sysroot'] = args.target_sysroot
196 197 198 199 200 201 202
      gn_args['custom_sysroot'] = args.target_sysroot

    if args.target_toolchain:
      gn_args['custom_toolchain'] = args.target_toolchain

    if args.target_triple:
      gn_args['custom_target_triple'] = args.target_triple
G
George Kulakowski 已提交
203 204 205 206

    if args.toolchain_prefix:
      gn_args['toolchain_prefix'] = args.toolchain_prefix

E
Eric Seidel 已提交
207 208
    goma_dir = os.environ.get('GOMA_DIR')
    goma_home_dir = os.path.join(os.getenv('HOME', ''), 'goma')
209 210 211 212 213

    # GOMA has a different default (home) path on gWindows.
    if not os.path.exists(goma_home_dir) and sys.platform.startswith(('cygwin', 'win')):
      goma_home_dir = os.path.join('c:\\', 'src', 'goma', 'goma-win64')

E
Eric Seidel 已提交
214 215 216 217 218 219 220 221 222 223
    if args.goma and goma_dir:
      gn_args['use_goma'] = True
      gn_args['goma_dir'] = goma_dir
    elif args.goma and os.path.exists(goma_home_dir):
      gn_args['use_goma'] = True
      gn_args['goma_dir'] = goma_home_dir
    else:
      gn_args['use_goma'] = False
      gn_args['goma_dir'] = None

224 225 226 227 228 229
    if args.enable_vulkan:
      # Enable vulkan in the Flutter shell.
      gn_args['shell_enable_vulkan'] = True
      # Configure Skia for Vulkan support.
      gn_args['skia_use_vulkan'] = True

230 231
    # We should not need a special case for x86, but this seems to introduce text relocations
    # even with -fPIC everywhere.
232
    # gn_args['enable_profiling'] = args.runtime_mode != 'release' and args.android_cpu != 'x86'
233

234 235 236
    if args.arm_float_abi:
      gn_args['arm_float_abi'] = args.arm_float_abi

237 238
    # Whether to build trained Dart SDK snapshots of dart2js and dartdevc,
    # including the web sdk kernel and source files.
239
    gn_args['dart_platform_sdk'] = not args.full_dart_sdk
240
    gn_args['full_dart_sdk'] = args.full_dart_sdk
241

242 243
    return gn_args

244
def parse_args(args):
E
Eric Seidel 已提交
245
  args = args[1:]
246
  parser = argparse.ArgumentParser(description='A script run` gn gen`.')
E
Eric Seidel 已提交
247

248
  parser.add_argument('--unoptimized', default=False, action='store_true')
249

250
  parser.add_argument('--runtime-mode', type=str, choices=['debug', 'profile', 'release'], default='debug')
251 252
  parser.add_argument('--dynamic', default=False, action='store_true')
  parser.add_argument('--interpreter', default=False, action='store_true')
253 254 255 256
  parser.add_argument('--dart-debug', default=False, action='store_true', help='Enables assertsion in the Dart VM. ' +
      'Does not affect affect optimization levels. If you need to disable optimizations in Dart, use --full-dart-debug')
  parser.add_argument('--full-dart-debug', default=False, action='store_true', help='Implies --dart-debug ' +
      'and also disables optimizations in the Dart VM making it easier to step through VM code in the debugger.')
E
Eric Seidel 已提交
257

258
  parser.add_argument('--target-os', type=str, choices=['android', 'ios', 'linux'])
259
  parser.add_argument('--android', dest='target_os', action='store_const', const='android')
260
  parser.add_argument('--android-cpu', type=str, choices=['arm', 'x64', 'x86', 'arm64'], default='arm')
261
  parser.add_argument('--ios', dest='target_os', action='store_const', const='ios')
262
  parser.add_argument('--ios-cpu', type=str, choices=['arm', 'arm64'], default='arm64')
E
Eric Seidel 已提交
263
  parser.add_argument('--simulator', action='store_true', default=False)
264 265
  parser.add_argument('--linux-cpu', type=str, choices=['x64', 'x86', 'arm64', 'arm'])
  parser.add_argument('--arm-float-abi', type=str, choices=['hard', 'soft', 'softfp'])
E
Eric Seidel 已提交
266 267 268 269

  parser.add_argument('--goma', default=True, action='store_true')
  parser.add_argument('--no-goma', dest='goma', action='store_false')

270 271 272
  parser.add_argument('--lto', default=True, action='store_true')
  parser.add_argument('--no-lto', dest='lto', action='store_false')

A
Adam Barth 已提交
273 274 275
  parser.add_argument('--clang', default=True, action='store_true')
  parser.add_argument('--no-clang', dest='clang', action='store_false')

276 277 278
  parser.add_argument('--clang-static-analyzer', default=False, action='store_true')
  parser.add_argument('--no-clang-static-analyzer', dest='clang_static_analyzer', action='store_false')

G
George Kulakowski 已提交
279
  parser.add_argument('--target-sysroot', type=str)
280 281
  parser.add_argument('--target-toolchain', type=str)
  parser.add_argument('--target-triple', type=str)
G
George Kulakowski 已提交
282
  parser.add_argument('--toolchain-prefix', type=str)
283
  parser.add_argument('--operator-new-alignment', dest='operator_new_alignment', type=str, default=None)
G
George Kulakowski 已提交
284

285
  parser.add_argument('--enable-vulkan', action='store_true', default=False)
286

287 288
  parser.add_argument('--embedder-for-target', dest='embedder_for_target', action='store_true', default=False)

289 290
  parser.add_argument('--coverage', default=False, action='store_true')

291 292
  parser.add_argument('--out-dir', default='', type=str)

293 294 295
  parser.add_argument('--full-dart-sdk', default=False, action='store_true',
                      help='include trained dart2js and dartdevc snapshots. Enable only on steps that create an SDK')
  parser.add_argument('--no-full-dart-sdk', dest='full_dart_sdk', action='store_false')
296

297
  return parser.parse_args(args)
298

299 300
def main(argv):
  args = parse_args(argv)
301 302 303 304

  if sys.platform.startswith(('cygwin', 'win')):
    subdir = 'win'
  elif sys.platform == 'darwin':
305
    subdir = 'mac-x64'
306
  elif sys.platform.startswith('linux'):
307
     subdir = 'linux-x64'
308 309 310 311 312 313
  else:
    raise Error('Unknown platform: ' + sys.platform)

  command = [
    '%s/buildtools/%s/gn' % (SRC_ROOT, subdir),
    'gen',
314
    '--check',
315
  ]
316 317 318 319 320

  if sys.platform == 'darwin':
    # On the Mac, also generate Xcode projects for ease of editing.
    command.append('--ide=xcode')

321 322 323 324
  if sys.platform.startswith('win'):
    # On Windows, also generate Visual Studio project for ease of editing.
    command.append('--ide=vs')

325 326
  gn_args = to_command_line(to_gn_args(args))
  out_dir = get_out_dir(args)
E
Eric Seidel 已提交
327
  print "gn gen --check in %s" % out_dir
328 329
  command.append(out_dir)
  command.append('--args=%s' % ' '.join(gn_args))
330 331 332 333 334
  gn_call_result = subprocess.call(command, cwd=SRC_ROOT)

  if gn_call_result == 0:
    # Generate/Replace the compile commands database in out.
    compile_cmd_gen_cmd = [
335
      'ninja',
336 337 338 339 340
      '-C',
      out_dir,
      '-t',
      'compdb',
      'cc',
341 342 343 344
      'cxx',
      'objc',
      'objcxx',
      'asm',
345 346 347 348 349 350 351 352
    ]

    contents = subprocess.check_output(compile_cmd_gen_cmd, cwd=SRC_ROOT)
    compile_commands = open('%s/out/compile_commands.json' % SRC_ROOT, 'w+')
    compile_commands.write(contents)
    compile_commands.close()

  return gn_call_result
353 354

if __name__ == '__main__':
355
    sys.exit(main(sys.argv))