2011-04-14 12:38:55 +04:00
|
|
|
#!/usr/bin/env python
|
2009-09-16 03:52:14 +04:00
|
|
|
|
2012-05-14 22:39:27 +04:00
|
|
|
# Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
2009-09-16 23:44:02 +04:00
|
|
|
# Use of this source code is governed by a BSD-style license that can be
|
|
|
|
# found in the LICENSE file.
|
|
|
|
|
2009-09-16 03:52:14 +04:00
|
|
|
# This script is wrapper for Chromium that adds some support for how GYP
|
2009-09-16 23:44:02 +04:00
|
|
|
# is invoked by Chromium beyond what can be done in the gclient hooks.
|
2009-09-16 03:52:14 +04:00
|
|
|
|
|
|
|
import glob
|
2012-11-14 08:59:48 +04:00
|
|
|
import gyp_helper
|
2009-09-16 03:52:14 +04:00
|
|
|
import os
|
2013-12-03 21:48:26 +04:00
|
|
|
import pipes
|
2009-09-16 03:52:14 +04:00
|
|
|
import shlex
|
2011-02-23 05:00:06 +03:00
|
|
|
import subprocess
|
2013-12-05 00:28:10 +04:00
|
|
|
import string
|
2009-09-16 03:52:14 +04:00
|
|
|
import sys
|
|
|
|
|
2012-07-11 23:34:17 +04:00
|
|
|
script_dir = os.path.dirname(os.path.realpath(__file__))
|
2011-04-19 13:33:24 +04:00
|
|
|
chrome_src = os.path.abspath(os.path.join(script_dir, os.pardir))
|
2009-09-16 23:44:02 +04:00
|
|
|
|
2010-07-13 01:56:56 +04:00
|
|
|
sys.path.insert(0, os.path.join(chrome_src, 'tools', 'gyp', 'pylib'))
|
2009-09-17 02:37:47 +04:00
|
|
|
import gyp
|
2009-09-16 03:52:14 +04:00
|
|
|
|
2013-12-04 03:14:46 +04:00
|
|
|
# Assume this file is in a one-level-deep subdirectory of the source root.
|
|
|
|
SRC_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
|
|
|
|
2011-08-17 00:43:27 +04:00
|
|
|
# Add paths so that pymod_do_main(...) can import files.
|
2012-12-18 05:51:37 +04:00
|
|
|
sys.path.insert(1, os.path.join(chrome_src, 'tools', 'generate_shim_headers'))
|
2011-05-28 06:11:24 +04:00
|
|
|
sys.path.insert(1, os.path.join(chrome_src, 'tools', 'grit'))
|
2011-08-17 00:43:27 +04:00
|
|
|
sys.path.insert(1, os.path.join(chrome_src, 'chrome', 'tools', 'build'))
|
2012-05-18 04:33:11 +04:00
|
|
|
sys.path.insert(1, os.path.join(chrome_src, 'native_client', 'build'))
|
2013-11-14 10:10:14 +04:00
|
|
|
sys.path.insert(1, os.path.join(chrome_src, 'native_client_sdk', 'src',
|
|
|
|
'build_tools'))
|
2013-07-12 02:01:30 +04:00
|
|
|
sys.path.insert(1, os.path.join(chrome_src, 'remoting', 'tools', 'build'))
|
2013-12-04 07:34:57 +04:00
|
|
|
sys.path.insert(1, os.path.join(chrome_src, 'third_party', 'liblouis'))
|
2013-01-12 01:15:19 +04:00
|
|
|
sys.path.insert(1, os.path.join(chrome_src, 'third_party', 'WebKit',
|
2013-10-10 18:50:47 +04:00
|
|
|
'Source', 'build', 'scripts'))
|
2011-05-28 06:11:24 +04:00
|
|
|
|
2011-03-30 22:52:23 +04:00
|
|
|
# On Windows, Psyco shortens warm runs of build/gyp_chromium by about
|
|
|
|
# 20 seconds on a z600 machine with 12 GB of RAM, from 90 down to 70
|
|
|
|
# seconds. Conversely, memory usage of build/gyp_chromium with Psyco
|
|
|
|
# maxes out at about 158 MB vs. 132 MB without it.
|
|
|
|
#
|
|
|
|
# Psyco uses native libraries, so we need to load a different
|
|
|
|
# installation depending on which OS we are running under. It has not
|
|
|
|
# been tested whether using Psyco on our Mac and Linux builds is worth
|
|
|
|
# it (the GYP running time is a lot shorter, so the JIT startup cost
|
|
|
|
# may not be worth it).
|
|
|
|
if sys.platform == 'win32':
|
|
|
|
try:
|
|
|
|
sys.path.insert(0, os.path.join(chrome_src, 'third_party', 'psyco_win32'))
|
|
|
|
import psyco
|
|
|
|
except:
|
|
|
|
psyco = None
|
|
|
|
else:
|
|
|
|
psyco = None
|
|
|
|
|
2013-12-04 03:14:46 +04:00
|
|
|
|
|
|
|
def GetSupplementalFiles():
|
|
|
|
"""Returns a list of the supplemental files that are included in all GYP
|
|
|
|
sources."""
|
|
|
|
return glob.glob(os.path.join(chrome_src, '*', 'supplement.gypi'))
|
|
|
|
|
|
|
|
|
2013-12-05 00:28:10 +04:00
|
|
|
def FormatKeyForGN(key):
|
|
|
|
"""Returns the given GYP key reformatted for GN.
|
|
|
|
|
|
|
|
GYP dictionary keys can be almost anything, but in GN they are identifiers
|
|
|
|
and must follow the same rules. This reformats such keys to be valid GN
|
|
|
|
identifiers."""
|
|
|
|
return ''.join([c if c in string.ascii_letters else '_' for c in key])
|
|
|
|
|
|
|
|
|
2013-12-05 07:51:46 +04:00
|
|
|
def EscapeStringForGN(s):
|
|
|
|
"""Converts a string to a GN string literal."""
|
|
|
|
# Escape $ characters which have special meaning to GN.
|
|
|
|
return '"' + s.replace('$', '\\$').replace('"', '\\"') + '"'
|
|
|
|
|
|
|
|
|
2013-12-14 00:28:08 +04:00
|
|
|
def GetGypVarsForGN(supplemental_files):
|
|
|
|
"""Returns a dictionary of all GYP vars that we will be passing to GN."""
|
2013-12-04 03:14:46 +04:00
|
|
|
vars_dict = {}
|
|
|
|
|
|
|
|
for supplement in supplemental_files:
|
|
|
|
with open(supplement, 'r') as f:
|
|
|
|
try:
|
|
|
|
file_data = eval(f.read(), {'__builtins__': None}, None)
|
|
|
|
except SyntaxError, e:
|
|
|
|
e.filename = os.path.abspath(supplement)
|
|
|
|
raise
|
2013-12-05 06:11:23 +04:00
|
|
|
variables = file_data.get('variables', [])
|
2013-12-04 03:14:46 +04:00
|
|
|
for v in variables:
|
2013-12-05 07:51:46 +04:00
|
|
|
vars_dict[FormatKeyForGN(v)] = EscapeStringForGN(str(variables[v]))
|
2013-12-04 03:14:46 +04:00
|
|
|
|
Fix crash during build/gyp_chromium time.
Without GYP_DEFINES set, build/gyp_chromium would crash with
Generating gyp files from GN...
Traceback (most recent call last):
File "./build/gyp_chromium", line 207, in <module>
if not RunGN(supplemental_includes):
File "./build/gyp_chromium", line 147, in RunGN
gyp_vars = GetVarsStringForGN(supplemental_includes)
File "./build/gyp_chromium", line 79, in GetVarsStringForGN
items = shlex.split(env_string)
File "/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/shlex.py", line 279, in split
return list(lex)
File "/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/shlex.py", line 269, in next
token = self.get_token()
File "/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/shlex.py", line 96, in get_token
raw = self.read_token()
File "/System/Library/Frameworks/Python.framework/Versions/2.7/lib/python2.7/shlex.py", line 124, in read_token
nextchar = self.instream.read(1)
AttributeError: 'list' object has no attribute 'read'
This was broken by r238475
BUG=321352
Review URL: https://codereview.chromium.org/103253002
git-svn-id: http://src.chromium.org/svn/trunk/src/build@238486 4ff67af0-8c30-449e-8e8b-ad334ec8d88c
2013-12-04 03:36:53 +04:00
|
|
|
env_string = os.environ.get('GYP_DEFINES', '')
|
2013-12-04 03:14:46 +04:00
|
|
|
items = shlex.split(env_string)
|
|
|
|
for item in items:
|
|
|
|
tokens = item.split('=', 1)
|
2013-12-04 04:27:35 +04:00
|
|
|
# Some GYP variables have hyphens, which we don't support.
|
2013-12-05 00:28:10 +04:00
|
|
|
key = FormatKeyForGN(tokens[0])
|
2013-12-04 03:14:46 +04:00
|
|
|
if len(tokens) == 2:
|
2013-12-14 00:28:08 +04:00
|
|
|
vars_dict[key] = tokens[1]
|
2013-12-04 03:14:46 +04:00
|
|
|
else:
|
|
|
|
# No value supplied, treat it as a boolean and set it.
|
2013-12-04 04:27:35 +04:00
|
|
|
vars_dict[key] = 'true'
|
2013-12-04 03:14:46 +04:00
|
|
|
|
2013-12-14 00:28:08 +04:00
|
|
|
return vars_dict
|
|
|
|
|
|
|
|
|
|
|
|
def GetArgsStringForGN(supplemental_files):
|
|
|
|
"""Returns the args to pass to GN.
|
|
|
|
Based on a subset of the GYP variables that have been rewritten a bit."""
|
|
|
|
|
|
|
|
vars_dict = GetGypVarsForGN(supplemental_files)
|
|
|
|
gn_args = ''
|
|
|
|
|
2013-12-16 21:58:16 +04:00
|
|
|
# These tuples of (key, value, gn_arg_string) use the gn_arg_string for
|
|
|
|
# gn when the key is set to the given value in the GYP arguments.
|
|
|
|
remap_cases = [
|
|
|
|
('branding', 'Chrome', 'is_chrome_branded=true'),
|
|
|
|
('buildtype', 'Official', 'is_official_build=true'),
|
|
|
|
('component', 'shared_library', 'is_component_build=true'),
|
|
|
|
]
|
|
|
|
for i in remap_cases:
|
|
|
|
if i[0] in vars_dict and vars_dict[i[0]] == i[1]:
|
|
|
|
gn_args += ' ' + i[2]
|
2013-12-14 00:28:08 +04:00
|
|
|
|
|
|
|
# These string arguments get passed directly.
|
2013-12-16 21:58:16 +04:00
|
|
|
for v in ['windows_sdk_path']:
|
2013-12-14 00:28:08 +04:00
|
|
|
if v in vars_dict:
|
2013-12-17 04:02:45 +04:00
|
|
|
gn_args += ' ' + v + '=' + EscapeStringForGN(vars_dict[v])
|
2013-12-14 00:28:08 +04:00
|
|
|
|
|
|
|
# Set the GYP flag so BUILD files know they're being invoked in GYP mode.
|
|
|
|
gn_args += ' is_gyp=true'
|
|
|
|
return gn_args.strip()
|
2013-12-04 03:14:46 +04:00
|
|
|
|
|
|
|
|
|
|
|
def additional_include_files(supplemental_files, args=[]):
|
2009-09-23 07:49:50 +04:00
|
|
|
"""
|
2013-12-04 03:14:46 +04:00
|
|
|
Returns a list of additional (.gypi) files to include, without duplicating
|
|
|
|
ones that are already specified on the command line. The list of supplemental
|
|
|
|
include files is passed in as an argument.
|
2009-09-23 07:49:50 +04:00
|
|
|
"""
|
|
|
|
# Determine the include files specified on the command line.
|
|
|
|
# This doesn't cover all the different option formats you can use,
|
|
|
|
# but it's mainly intended to avoid duplicating flags on the automatic
|
|
|
|
# makefile regeneration which only uses this format.
|
|
|
|
specified_includes = set()
|
|
|
|
for arg in args:
|
|
|
|
if arg.startswith('-I') and len(arg) > 2:
|
|
|
|
specified_includes.add(os.path.realpath(arg[2:]))
|
|
|
|
|
|
|
|
result = []
|
|
|
|
def AddInclude(path):
|
|
|
|
if os.path.realpath(path) not in specified_includes:
|
|
|
|
result.append(path)
|
|
|
|
|
2011-06-11 00:44:47 +04:00
|
|
|
# Always include common.gypi.
|
2009-09-23 07:49:50 +04:00
|
|
|
AddInclude(os.path.join(script_dir, 'common.gypi'))
|
|
|
|
|
|
|
|
# Optionally add supplemental .gypi files if present.
|
2013-12-04 03:14:46 +04:00
|
|
|
for supplement in supplemental_files:
|
2009-09-23 07:49:50 +04:00
|
|
|
AddInclude(supplement)
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
2013-12-04 03:14:46 +04:00
|
|
|
|
|
|
|
def RunGN(supplemental_includes):
|
|
|
|
"""Runs GN, returning True if it succeeded, printing an error and returning
|
|
|
|
false if not."""
|
|
|
|
|
|
|
|
# The binaries in platform-specific subdirectories in src/tools/gn/bin.
|
|
|
|
gnpath = SRC_DIR + '/tools/gn/bin/'
|
2013-12-06 00:35:31 +04:00
|
|
|
if sys.platform in ('cygwin', 'win32'):
|
2013-12-04 03:14:46 +04:00
|
|
|
gnpath += 'win/gn.exe'
|
|
|
|
elif sys.platform.startswith('linux'):
|
2013-12-04 04:05:44 +04:00
|
|
|
# On Linux we have 32-bit and 64-bit versions.
|
|
|
|
if subprocess.check_output(["getconf", "LONG_BIT"]).find("64") >= 0:
|
2013-12-04 03:14:46 +04:00
|
|
|
gnpath += 'linux/gn'
|
|
|
|
else:
|
|
|
|
gnpath += 'linux/gn32'
|
|
|
|
elif sys.platform == 'darwin':
|
|
|
|
gnpath += 'mac/gn'
|
|
|
|
else:
|
|
|
|
print 'Unknown platform for GN: ', sys.platform
|
|
|
|
return False
|
|
|
|
|
|
|
|
print 'Generating gyp files from GN...'
|
|
|
|
|
|
|
|
# Need to pass both the source root (the bots don't run this command from
|
|
|
|
# within the source tree) as well as set the is_gyp value so the BUILD files
|
|
|
|
# to know they're being run under GYP.
|
|
|
|
args = [gnpath, 'gyp', '-q',
|
|
|
|
'--root=' + chrome_src,
|
2013-12-14 00:28:08 +04:00
|
|
|
'--args=' + GetArgsStringForGN(supplemental_includes)]
|
2013-12-04 03:14:46 +04:00
|
|
|
return subprocess.call(args) == 0
|
|
|
|
|
|
|
|
|
2009-09-16 03:52:14 +04:00
|
|
|
if __name__ == '__main__':
|
|
|
|
args = sys.argv[1:]
|
|
|
|
|
2013-11-21 01:30:30 +04:00
|
|
|
if int(os.environ.get('GYP_CHROMIUM_NO_ACTION', 0)):
|
|
|
|
print 'Skipping gyp_chromium due to GYP_CHROMIUM_NO_ACTION env var.'
|
|
|
|
sys.exit(0)
|
|
|
|
|
2011-03-30 22:52:23 +04:00
|
|
|
# Use the Psyco JIT if available.
|
|
|
|
if psyco:
|
|
|
|
psyco.profile()
|
|
|
|
print "Enabled Psyco JIT."
|
|
|
|
|
2011-02-23 05:00:06 +03:00
|
|
|
# Fall back on hermetic python if we happen to get run under cygwin.
|
|
|
|
# TODO(bradnelson): take this out once this issue is fixed:
|
|
|
|
# http://code.google.com/p/gyp/issues/detail?id=177
|
|
|
|
if sys.platform == 'cygwin':
|
|
|
|
python_dir = os.path.join(chrome_src, 'third_party', 'python_26')
|
|
|
|
env = os.environ.copy()
|
|
|
|
env['PATH'] = python_dir + os.pathsep + env.get('PATH', '')
|
|
|
|
p = subprocess.Popen(
|
|
|
|
[os.path.join(python_dir, 'python.exe')] + sys.argv,
|
|
|
|
env=env, shell=False)
|
|
|
|
p.communicate()
|
|
|
|
sys.exit(p.returncode)
|
|
|
|
|
2012-11-14 08:59:48 +04:00
|
|
|
gyp_helper.apply_chromium_gyp_env()
|
2010-11-08 23:09:23 +03:00
|
|
|
|
2009-09-17 03:16:05 +04:00
|
|
|
# This could give false positives since it doesn't actually do real option
|
|
|
|
# parsing. Oh well.
|
|
|
|
gyp_file_specified = False
|
|
|
|
for arg in args:
|
|
|
|
if arg.endswith('.gyp'):
|
|
|
|
gyp_file_specified = True
|
|
|
|
break
|
|
|
|
|
2009-09-16 03:52:14 +04:00
|
|
|
# If we didn't get a file, check an env var, and then fall back to
|
2009-09-17 02:37:47 +04:00
|
|
|
# assuming 'all.gyp' from the same directory as the script.
|
2009-09-17 03:16:05 +04:00
|
|
|
if not gyp_file_specified:
|
|
|
|
gyp_file = os.environ.get('CHROMIUM_GYP_FILE')
|
|
|
|
if gyp_file:
|
|
|
|
# Note that CHROMIUM_GYP_FILE values can't have backslashes as
|
|
|
|
# path separators even on Windows due to the use of shlex.split().
|
|
|
|
args.extend(shlex.split(gyp_file))
|
|
|
|
else:
|
|
|
|
args.append(os.path.join(script_dir, 'all.gyp'))
|
|
|
|
|
2013-12-04 03:14:46 +04:00
|
|
|
supplemental_includes = GetSupplementalFiles()
|
|
|
|
|
2013-12-07 00:35:07 +04:00
|
|
|
if not RunGN(supplemental_includes):
|
|
|
|
sys.exit(1)
|
2013-12-04 03:14:46 +04:00
|
|
|
|
|
|
|
args.extend(
|
|
|
|
['-I' + i for i in additional_include_files(supplemental_includes, args)])
|
2009-09-16 03:52:14 +04:00
|
|
|
|
2010-02-16 23:14:26 +03:00
|
|
|
# There shouldn't be a circular dependency relationship between .gyp files,
|
|
|
|
# but in Chromium's .gyp files, on non-Mac platforms, circular relationships
|
|
|
|
# currently exist. The check for circular dependencies is currently
|
|
|
|
# bypassed on other platforms, but is left enabled on the Mac, where a
|
|
|
|
# violation of the rule causes Xcode to misbehave badly.
|
|
|
|
# TODO(mark): Find and kill remaining circular dependencies, and remove this
|
|
|
|
# option. http://crbug.com/35878.
|
2010-05-24 04:48:16 +04:00
|
|
|
# TODO(tc): Fix circular dependencies in ChromiumOS then add linux2 to the
|
|
|
|
# list.
|
|
|
|
if sys.platform not in ('darwin',):
|
2010-02-16 23:14:26 +03:00
|
|
|
args.append('--no-circular-check')
|
|
|
|
|
2013-05-11 07:18:37 +04:00
|
|
|
# Default to ninja on linux, but only if no generator has explicitly been set.
|
2013-10-30 04:04:06 +04:00
|
|
|
# Also default to ninja on mac, but only when not building chrome/ios.
|
2013-05-11 07:18:37 +04:00
|
|
|
# . -f / --format has precedence over the env var, no need to check for it
|
|
|
|
# . set the env var only if it hasn't been set yet
|
|
|
|
# . chromium.gyp_env has been applied to os.environ at this point already
|
|
|
|
if sys.platform.startswith('linux') and not os.environ.get('GYP_GENERATORS'):
|
|
|
|
os.environ['GYP_GENERATORS'] = 'ninja'
|
2013-10-30 04:04:06 +04:00
|
|
|
elif sys.platform == 'darwin' and not os.environ.get('GYP_GENERATORS') and \
|
2013-10-31 10:18:11 +04:00
|
|
|
not 'OS=ios' in os.environ.get('GYP_DEFINES', []):
|
2013-10-30 04:04:06 +04:00
|
|
|
os.environ['GYP_GENERATORS'] = 'ninja'
|
2013-05-11 07:18:37 +04:00
|
|
|
|
2013-12-03 21:48:26 +04:00
|
|
|
# If using ninja on windows, and not opting out of the the automatic
|
|
|
|
# toolchain, then set up variables for the automatic toolchain. Opt-out is
|
|
|
|
# on by default, for now.
|
|
|
|
if (sys.platform in ('win32', 'cygwin') and
|
|
|
|
os.environ.get('GYP_GENERATORS') == 'ninja' and
|
|
|
|
os.environ.get('GYP_MSVS_USE_SYSTEM_TOOLCHAIN', '1') != '1'):
|
|
|
|
# For now, call the acquisition script here so that there's only one
|
|
|
|
# opt-in step required. This will be moved to a separate DEPS step once
|
|
|
|
# it's on by default.
|
|
|
|
subprocess.check_call([
|
|
|
|
sys.executable,
|
|
|
|
os.path.normpath(os.path.join(script_dir, '..', 'tools', 'win',
|
|
|
|
'toolchain',
|
|
|
|
'get_toolchain_if_necessary.py'))])
|
|
|
|
toolchain = os.path.normpath(os.path.join(
|
|
|
|
script_dir, '..', 'third_party', 'win_toolchain', 'files'))
|
|
|
|
os.environ['GYP_MSVS_OVERRIDE_PATH'] = toolchain
|
|
|
|
os.environ['GYP_MSVS_VERSION'] = '2013'
|
|
|
|
# We need to make sure windows_sdk_path is set to the automated toolchain
|
|
|
|
# values in GYP_DEFINES, but don't want to override any other values there.
|
|
|
|
gyp_defines_dict = gyp.NameValueListToDict(gyp.ShlexEnv('GYP_DEFINES'))
|
|
|
|
win8sdk = os.path.join(toolchain, 'win8sdk')
|
|
|
|
gyp_defines_dict['windows_sdk_path'] = win8sdk
|
|
|
|
os.environ['WINDOWSSDKDIR'] = win8sdk
|
|
|
|
os.environ['GYP_DEFINES'] = ' '.join('%s=%s' % (k, pipes.quote(str(v)))
|
|
|
|
for k, v in gyp_defines_dict.iteritems())
|
2013-12-10 09:32:16 +04:00
|
|
|
# Include the VS runtime in the PATH in case it's not machine-installed.
|
|
|
|
runtime_path = ';'.join(
|
|
|
|
os.path.normpath(os.path.join(
|
|
|
|
script_dir, '..', 'third_party', 'win_toolchain', 'files', s))
|
|
|
|
for s in ('sys64', 'sys32'))
|
|
|
|
os.environ['PATH'] = runtime_path + os.environ['PATH']
|
2013-12-03 21:48:26 +04:00
|
|
|
print('Using automatic toolchain in %s.' % toolchain)
|
|
|
|
|
2010-03-26 21:40:49 +03:00
|
|
|
# If CHROMIUM_GYP_SYNTAX_CHECK is set to 1, it will invoke gyp with --check
|
|
|
|
# to enfore syntax checking.
|
|
|
|
syntax_check = os.environ.get('CHROMIUM_GYP_SYNTAX_CHECK')
|
|
|
|
if syntax_check and int(syntax_check):
|
|
|
|
args.append('--check')
|
|
|
|
|
2009-09-17 02:37:47 +04:00
|
|
|
print 'Updating projects from gyp files...'
|
|
|
|
sys.stdout.flush()
|
|
|
|
|
2009-09-16 03:52:14 +04:00
|
|
|
# Off we go...
|
|
|
|
sys.exit(gyp.main(args))
|