blob: a2bed5652c1c2f571c8a61be3a30f5f69feadfe0 [file] [log] [blame]
ager@chromium.org9258b6b2008-09-11 09:11:10 +00001# Copyright 2008 the V8 project authors. All rights reserved.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +00002# Redistribution and use in source and binary forms, with or without
3# modification, are permitted provided that the following conditions are
4# met:
5#
6# * Redistributions of source code must retain the above copyright
7# notice, this list of conditions and the following disclaimer.
8# * Redistributions in binary form must reproduce the above
9# copyright notice, this list of conditions and the following
10# disclaimer in the documentation and/or other materials provided
11# with the distribution.
12# * Neither the name of Google Inc. nor the names of its
13# contributors may be used to endorse or promote products derived
14# from this software without specific prior written permission.
15#
16# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28import platform
kasper.lund212ac232008-07-16 07:07:30 +000029import re
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000030import sys
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +000031import os
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000032from os.path import join, dirname, abspath
ager@chromium.org9258b6b2008-09-11 09:11:10 +000033from types import DictType
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000034root_dir = dirname(File('SConstruct').rfile().abspath)
35sys.path.append(join(root_dir, 'tools'))
ager@chromium.orgc27e4e72008-09-04 13:52:27 +000036import js2c, utils
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000037
38
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +000039LIBRARY_FLAGS = {
40 'all': {
41 'CPPDEFINES': ['ENABLE_LOGGING_AND_PROFILING']
42 },
43 'gcc': {
44 'all': {
45 'DIALECTFLAGS': ['-ansi'],
46 'CCFLAGS': ['$DIALECTFLAGS', '$WARNINGFLAGS',
47 '-fno-strict-aliasing'],
48 'CXXFLAGS': ['$CCFLAGS', '-fno-rtti', '-fno-exceptions'],
49 'LIBS': ['pthread']
50 },
51 'mode:debug': {
52 'CCFLAGS': ['-g', '-O0'],
53 'CPPDEFINES': ['ENABLE_DISASSEMBLER', 'DEBUG']
54 },
55 'mode:release': {
56 'CCFLAGS': ['-O2']
57 },
58 'wordsize:64': {
ager@chromium.org9258b6b2008-09-11 09:11:10 +000059 'CCFLAGS': ['-m32'],
60 'LINKFLAGS': ['-m32']
61 }
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +000062 },
63 'msvc': {
64 'all': {
65 'DIALECTFLAGS': ['/nologo'],
66 'WARNINGFLAGS': ['/W3', '/WX', '/wd4355', '/wd4800'],
67 'CCFLAGS': ['$DIALECTFLAGS', '$WARNINGFLAGS'],
v8.team.kasperl727e9952008-09-02 14:56:44 +000068 'CXXFLAGS': ['$CCFLAGS', '/GR-', '/Gy'],
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +000069 'CPPDEFINES': ['WIN32', '_CRT_SECURE_NO_DEPRECATE',
70 '_CRT_NONSTDC_NO_DEPRECATE', '_USE_32BIT_TIME_T',
71 'PCRE_STATIC'],
72 'LINKFLAGS': ['/NOLOGO', '/MACHINE:X86', '/INCREMENTAL:NO',
73 '/NXCOMPAT', '/IGNORE:4221'],
74 'ARFLAGS': ['/NOLOGO'],
75 'CCPDBFLAGS': ['/Zi']
76 },
77 'mode:debug': {
78 'CCFLAGS': ['/Od', '/Gm', '/MTd'],
79 'CPPDEFINES': ['_DEBUG', 'ENABLE_DISASSEMBLER', 'DEBUG'],
80 'LINKFLAGS': ['/DEBUG']
81 },
82 'mode:release': {
83 'CCFLAGS': ['/Ox', '/MT', '/Ob2', '/Oi', '/Oy'],
v8.team.kasperl727e9952008-09-02 14:56:44 +000084 'LINKFLAGS': ['/OPT:REF', '/OPT:ICF']
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +000085 }
86 }
87}
88
89
90V8_EXTRA_FLAGS = {
91 'gcc': {
92 'all': {
93 'CXXFLAGS': [], #['-fvisibility=hidden'],
94 'WARNINGFLAGS': ['-pedantic', '-Wall', '-Werror', '-W',
95 '-Wno-unused-parameter']
96 },
97 'arch:arm': {
98 'CPPDEFINES': ['ARM']
99 },
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000100 'disassembler:on': {
101 'CPPDEFINES': ['ENABLE_DISASSEMBLER']
102 }
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000103 },
104 'msvc': {
105 'all': {
106 'WARNINGFLAGS': ['/W3', '/WX', '/wd4355', '/wd4800']
107 },
108 'library:shared': {
109 'CPPDEFINES': ['BUILDING_V8_SHARED']
110 },
111 'arch:arm': {
112 'CPPDEFINES': ['ARM']
113 },
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000114 'disassembler:on': {
115 'CPPDEFINES': ['ENABLE_DISASSEMBLER']
116 }
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000117 }
118}
119
120
121JSCRE_EXTRA_FLAGS = {
122 'gcc': {
123 'all': {
124 'CPPDEFINES': ['SUPPORT_UTF8', 'NO_RECURSE', 'SUPPORT_UCP'],
125 'WARNINGFLAGS': ['-w']
126 },
127 },
128 'msvc': {
129 'all': {
130 'CPPDEFINES': ['SUPPORT_UTF8', 'NO_RECURSE', 'SUPPORT_UCP'],
131 'WARNINGFLAGS': ['/W3', '/WX', '/wd4355', '/wd4800']
132 },
133 'library:shared': {
134 'CPPDEFINES': ['BUILDING_V8_SHARED']
135 }
136 }
137}
138
139
140DTOA_EXTRA_FLAGS = {
141 'gcc': {
142 'all': {
143 'WARNINGFLAGS': ['-Werror']
144 }
145 },
146 'msvc': {
147 'all': {
148 'WARNINGFLAGS': ['/WX', '/wd4018', '/wd4244']
149 }
150 }
151}
152
153
154CCTEST_EXTRA_FLAGS = {
155 'all': {
156 'CPPPATH': [join(root_dir, 'src')],
157 'LIBS': ['$LIBRARY']
158 },
159 'gcc': {
160 'all': {
161 'LIBPATH': [abspath('.')]
162 },
163 'wordsize:64': {
164 'CCFLAGS': ['-m32'],
165 'LINKFLAGS': ['-m32']
166 },
167 },
168 'msvc': {
169 'all': {
170 'CPPDEFINES': ['_HAS_EXCEPTIONS=0']
171 },
172 'library:shared': {
173 'CPPDEFINES': ['USING_V8_SHARED']
174 }
175 }
176}
177
178
179SAMPLE_FLAGS = {
180 'all': {
v8.team.kasperl727e9952008-09-02 14:56:44 +0000181 'CPPPATH': [join(abspath('.'), 'include')],
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000182 'LIBS': ['$LIBRARY'],
183 },
184 'gcc': {
185 'all': {
186 'LIBS': ['pthread'],
187 'LIBPATH': ['.']
188 },
189 'wordsize:64': {
190 'CCFLAGS': ['-m32'],
191 'LINKFLAGS': ['-m32']
192 },
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000193 'mode:debug': {
194 'CCFLAGS': ['-g', '-O0']
195 }
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000196 },
197 'msvc': {
198 'all': {
199 'CCFLAGS': ['/nologo'],
200 },
201 'library:shared': {
202 'CPPDEFINES': ['USING_V8_SHARED']
203 },
204 'mode:release': {
205 'CCFLAGS': ['/MT'],
206 },
207 'mode:debug': {
208 'CCFLAGS': ['/MTd']
209 }
210 }
211}
212
213
214SUFFIXES = {
215 'release': '',
216 'debug': '_g'
217}
218
219
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000220def Abort(message):
221 print message
222 sys.exit(1)
223
224
225def GuessOS():
226 id = platform.system()
227 if id == 'Linux':
228 return 'linux'
229 elif id == 'Darwin':
230 return 'macos'
231 elif id == 'Windows':
232 return 'win32'
233 else:
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000234 return None
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000235
236
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000237def GuessWordsize():
238 if '64' in platform.machine():
239 return '64'
240 else:
241 return '32'
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000242
243
244def GuessToolchain(os):
245 tools = Environment()['TOOLS']
246 if 'gcc' in tools:
kasper.lund7276f142008-07-30 08:49:36 +0000247 return 'gcc'
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000248 elif 'msvc' in tools:
249 return 'msvc'
250 else:
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000251 return None
252
253
254OS_GUESS = GuessOS()
255TOOLCHAIN_GUESS = GuessToolchain(OS_GUESS)
ager@chromium.orgc27e4e72008-09-04 13:52:27 +0000256ARCH_GUESS = utils.GuessArchitecture()
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000257WORDSIZE_GUESS = GuessWordsize()
258
259
260SIMPLE_OPTIONS = {
261 'toolchain': {
262 'values': ['gcc', 'msvc'],
263 'default': TOOLCHAIN_GUESS,
264 'help': 'the toolchain to use'
265 },
266 'os': {
267 'values': ['linux', 'macos', 'win32'],
268 'default': OS_GUESS,
269 'help': 'the os to build for'
270 },
271 'arch': {
272 'values':['arm', 'ia32'],
273 'default': ARCH_GUESS,
274 'help': 'the architecture to build for'
275 },
276 'snapshot': {
277 'values': ['on', 'off'],
278 'default': 'off',
279 'help': 'build using snapshots for faster start-up'
280 },
281 'library': {
282 'values': ['static', 'shared'],
283 'default': 'static',
284 'help': 'the type of library to produce'
285 },
286 'wordsize': {
287 'values': ['64', '32'],
288 'default': WORDSIZE_GUESS,
289 'help': 'the word size'
290 },
291 'simulator': {
292 'values': ['arm', 'none'],
293 'default': 'none',
294 'help': 'build with simulator'
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000295 },
296 'disassembler': {
297 'values': ['on', 'off'],
298 'default': 'off',
299 'help': 'enable the disassembler to inspect generated code'
300 },
301 'sourcesignatures': {
302 'values': ['MD5', 'timestamp'],
303 'default': 'MD5',
304 'help': 'set how the build system detects file changes'
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000305 }
306}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000307
308
309def GetOptions():
310 result = Options()
kasper.lund7276f142008-07-30 08:49:36 +0000311 result.Add('mode', 'compilation mode (debug, release)', 'release')
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000312 result.Add('sample', 'build sample (shell, process)', '')
v8.team.kasperl727e9952008-09-02 14:56:44 +0000313 result.Add('env', 'override environment settings (NAME1:value1,NAME2:value2)', '')
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000314 for (name, option) in SIMPLE_OPTIONS.items():
315 help = '%s (%s)' % (name, ", ".join(option['values']))
316 result.Add(name, help, option.get('default'))
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000317 return result
318
319
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000320def SplitList(str):
321 return [ s for s in str.split(",") if len(s) > 0 ]
322
323
324def IsLegal(env, option, values):
325 str = env[option]
326 for s in SplitList(str):
327 if not s in values:
328 Abort("Illegal value for option %s '%s'." % (option, s))
329 return False
330 return True
331
332
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000333def VerifyOptions(env):
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000334 if not IsLegal(env, 'mode', ['debug', 'release']):
335 return False
336 if not IsLegal(env, 'sample', ["shell", "process"]):
337 return False
338 for (name, option) in SIMPLE_OPTIONS.items():
339 if (not option.get('default')) and (name not in ARGUMENTS):
340 message = ("A value for option %s must be specified (%s)." %
341 (name, ", ".join(option['values'])))
342 Abort(message)
343 if not env[name] in option['values']:
344 message = ("Unknown %s value '%s'. Possible values are (%s)." %
345 (name, env[name], ", ".join(option['values'])))
346 Abort(message)
347
348
349class BuildContext(object):
350
v8.team.kasperl727e9952008-09-02 14:56:44 +0000351 def __init__(self, options, env_overrides, samples):
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000352 self.library_targets = []
353 self.cctest_targets = []
354 self.sample_targets = []
355 self.options = options
v8.team.kasperl727e9952008-09-02 14:56:44 +0000356 self.env_overrides = env_overrides
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000357 self.samples = samples
358 self.use_snapshot = (options['snapshot'] == 'on')
359 self.flags = None
360
361 def AddRelevantFlags(self, initial, flags):
362 result = initial.copy()
363 self.AppendFlags(result, flags.get('all'))
364 toolchain = self.options['toolchain']
365 self.AppendFlags(result, flags[toolchain].get('all'))
366 for option in sorted(self.options.keys()):
367 value = self.options[option]
368 self.AppendFlags(result, flags[toolchain].get(option + ':' + value))
369 return result
370
371 def GetRelevantSources(self, source):
372 result = []
373 result += source.get('all', [])
374 for (name, value) in self.options.items():
375 result += source.get(name + ':' + value, [])
376 return sorted(result)
377
378 def AppendFlags(self, options, added):
379 if not added:
380 return
381 for (key, value) in added.items():
382 if not key in options:
383 options[key] = value
384 else:
385 options[key] = options[key] + value
386
387 def ConfigureObject(self, env, input, **kw):
388 if self.options['library'] == 'static':
389 return env.StaticObject(input, **kw)
390 else:
391 return env.SharedObject(input, **kw)
392
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000393 def ApplyEnvOverrides(self, env):
394 if not self.env_overrides:
395 return
396 if type(env['ENV']) == DictType:
397 env['ENV'].update(**self.env_overrides)
398 else:
399 env['ENV'] = self.env_overrides
400
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000401
402def PostprocessOptions(options):
403 # Adjust architecture if the simulator option has been set
404 if (options['simulator'] != 'none') and (options['arch'] != options['simulator']):
405 if 'arch' in ARGUMENTS:
406 # Print a warning if arch has explicitly been set
407 print "Warning: forcing architecture to match simulator (%s)" % options['simulator']
408 options['arch'] = options['simulator']
409
410
v8.team.kasperl727e9952008-09-02 14:56:44 +0000411def ParseEnvOverrides(arg):
412 # The environment overrides are in the format NAME1:value1,NAME2:value2
413 overrides = {}
414 for override in arg.split(','):
415 pos = override.find(':')
416 if pos == -1:
417 continue
418 overrides[override[:pos].strip()] = override[pos+1:].strip()
419 return overrides
420
421
422def BuildSpecific(env, mode, env_overrides):
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000423 options = {'mode': mode}
424 for option in SIMPLE_OPTIONS:
425 options[option] = env[option]
426 PostprocessOptions(options)
427
v8.team.kasperl727e9952008-09-02 14:56:44 +0000428 context = BuildContext(options, env_overrides, samples=SplitList(env['sample']))
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000429
430 library_flags = context.AddRelevantFlags(os.environ, LIBRARY_FLAGS)
431 v8_flags = context.AddRelevantFlags(library_flags, V8_EXTRA_FLAGS)
432 jscre_flags = context.AddRelevantFlags(library_flags, JSCRE_EXTRA_FLAGS)
433 dtoa_flags = context.AddRelevantFlags(library_flags, DTOA_EXTRA_FLAGS)
434 cctest_flags = context.AddRelevantFlags(v8_flags, CCTEST_EXTRA_FLAGS)
435 sample_flags = context.AddRelevantFlags(os.environ, SAMPLE_FLAGS)
436
437 context.flags = {
438 'v8': v8_flags,
439 'jscre': jscre_flags,
440 'dtoa': dtoa_flags,
441 'cctest': cctest_flags,
442 'sample': sample_flags
443 }
444
445 target_id = mode
446 suffix = SUFFIXES[target_id]
447 library_name = 'v8' + suffix
448 env['LIBRARY'] = library_name
449
450 # Build the object files by invoking SCons recursively.
451 object_files = env.SConscript(
452 join('src', 'SConscript'),
453 build_dir=join('obj', target_id),
454 exports='context',
455 duplicate=False
456 )
457
458 # Link the object files into a library.
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000459 context.ApplyEnvOverrides(env)
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000460 if context.options['library'] == 'static':
461 library = env.StaticLibrary(library_name, object_files)
462 else:
463 # There seems to be a glitch in the way scons decides where to put
464 # PDB files when compiling using MSVC so we specify it manually.
465 # This should not affect any other platforms.
466 pdb_name = library_name + '.dll.pdb'
467 library = env.SharedLibrary(library_name, object_files, PDB=pdb_name)
468 context.library_targets.append(library)
469
470 for sample in context.samples:
471 sample_env = Environment(LIBRARY=library_name)
472 sample_env.Replace(**context.flags['sample'])
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000473 context.ApplyEnvOverrides(sample_env)
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000474 sample_object = sample_env.SConscript(
475 join('samples', 'SConscript'),
476 build_dir=join('obj', 'sample', sample, target_id),
477 exports='sample context',
478 duplicate=False
479 )
480 sample_name = sample + suffix
481 sample_program = sample_env.Program(sample_name, sample_object)
482 sample_env.Depends(sample_program, library)
483 context.sample_targets.append(sample_program)
484
485 cctest_program = env.SConscript(
486 join('test', 'cctest', 'SConscript'),
487 build_dir=join('obj', 'test', target_id),
488 exports='context object_files',
489 duplicate=False
490 )
491 context.cctest_targets.append(cctest_program)
492
493 return context
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000494
495
kasper.lund7276f142008-07-30 08:49:36 +0000496def Build():
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000497 opts = GetOptions()
498 env = Environment(options=opts)
499 Help(opts.GenerateHelpText(env))
500 VerifyOptions(env)
v8.team.kasperl727e9952008-09-02 14:56:44 +0000501 env_overrides = ParseEnvOverrides(env['env'])
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000502
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000503 SourceSignatures(env['sourcesignatures'])
504
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000505 libraries = []
506 cctests = []
507 samples = []
508 modes = SplitList(env['mode'])
509 for mode in modes:
v8.team.kasperl727e9952008-09-02 14:56:44 +0000510 context = BuildSpecific(env.Copy(), mode, env_overrides)
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000511 libraries += context.library_targets
512 cctests += context.cctest_targets
513 samples += context.sample_targets
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000514
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000515 env.Alias('library', libraries)
516 env.Alias('cctests', cctests)
517 env.Alias('sample', samples)
518
519 if env['sample']:
520 env.Default('sample')
kasper.lund7276f142008-07-30 08:49:36 +0000521 else:
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000522 env.Default('library')
kasper.lund7276f142008-07-30 08:49:36 +0000523
524
v8.team.kasperl727e9952008-09-02 14:56:44 +0000525# We disable deprecation warnings because we need to be able to use
526# env.Copy without getting warnings for compatibility with older
527# version of scons. Also, there's a bug in some revisions that
528# doesn't allow this flag to be set, so we swallow any exceptions.
529# Lovely.
530try:
531 SetOption('warn', 'no-deprecated')
532except:
533 pass
534
535
kasper.lund7276f142008-07-30 08:49:36 +0000536Build()