2 # Copyright 2011 Google Inc. All Rights Reserved.
4 # Licensed under the Apache License, Version 2.0 (the "License");
5 # you may not use this file except in compliance with the License.
6 # You may obtain a copy of the License at
8 # http://www.apache.org/licenses/LICENSE-2.0
10 # Unless required by applicable law or agreed to in writing, software
11 # distributed under the License is distributed on an "AS IS" BASIS,
12 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 # See the License for the specific language governing permissions and
14 # limitations under the License.
16 from __future__ import print_function
18 from optparse import OptionParser
26 import platform_helper
28 os.chdir(os.path.dirname(os.path.abspath(__file__)))
30 parser = OptionParser()
32 parser.add_option('--verbose', action='store_true',
33 help='enable verbose build',)
34 parser.add_option('--x64', action='store_true',
35 help='force 64-bit build (Windows)',)
36 parser.add_option('--platform',
37 help='target platform (' + '/'.join(platform_helper.platforms()) + ')',
38 choices=platform_helper.platforms())
39 (options, conf_args) = parser.parse_args()
42 platform = platform_helper.Platform(options.platform)
43 conf_args.append("--platform=" + platform.platform())
45 def run(*args, **kwargs):
46 returncode = subprocess.call(*args, **kwargs)
50 # Compute system-specific CFLAGS/LDFLAGS as used in both in the below
51 # g++ call as well as in the later configure.py.
52 cflags = os.environ.get('CFLAGS', '').split()
53 ldflags = os.environ.get('LDFLAGS', '').split()
54 if platform.is_freebsd() or platform.is_openbsd():
55 cflags.append('-I/usr/local/include')
56 ldflags.append('-L/usr/local/lib')
58 print('Building ninja manually...')
64 if e.errno != errno.EEXIST:
68 for src in glob.glob('src/*.cc'):
69 if src.endswith('test.cc') or src.endswith('.in.cc'):
71 if src.endswith('bench.cc'):
74 filename = os.path.basename(src)
75 if filename == 'browse.cc': # Depends on generated header.
78 if platform.is_windows():
79 if src.endswith('-posix.cc'):
82 if src.endswith('-win32.cc'):
87 if platform.is_windows():
88 sources.append('src/getopt.c')
90 if platform.is_msvc():
92 vcdir = os.environ.get('VCINSTALLDIR')
95 cl = os.path.join(vcdir, 'bin', 'x86_amd64', 'cl.exe')
96 if not os.path.exists(cl):
97 cl = os.path.join(vcdir, 'bin', 'amd64', 'cl.exe')
99 cl = os.path.join(vcdir, 'bin', 'cl.exe')
100 args = [cl, '/nologo', '/EHsc', '/DNOMINMAX']
102 args = shlex.split(os.environ.get('CXX', 'g++'))
103 cflags.extend(['-Wno-deprecated',
104 '-DNINJA_PYTHON="' + sys.executable + '"',
105 '-DNINJA_BOOTSTRAP'])
106 if platform.is_windows():
107 cflags.append('-D_WIN32_WINNT=0x0501')
109 cflags.append('-m64')
112 binary = 'ninja.bootstrap'
113 if platform.is_windows():
114 binary = 'ninja.bootstrap.exe'
116 if platform.is_msvc():
117 args.extend(['/link', '/out:' + binary])
119 args.extend(['-o', binary])
122 print(' '.join(args))
127 print('Failure running:', args)
134 if platform.is_windows():
135 print('Building ninja using itself...')
136 run([sys.executable, 'configure.py'] + conf_args)
137 run(['./' + binary] + verbose)
139 # Copy the new executable over the bootstrap one.
140 shutil.copyfile('ninja.exe', binary)
143 for obj in glob.glob('*.obj'):
149 Note: to work around Windows file locking, where you can't rebuild an
150 in-use binary, to run ninja after making any changes to build ninja itself
151 you should run ninja.bootstrap instead.""")
153 print('Building ninja using itself...')
154 run([sys.executable, 'configure.py'] + conf_args)
155 run(['./' + binary] + verbose)