blob: b6fc04f1f630e7f6b5454db159c9f346e4a367ea [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'],
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +000066 'CCFLAGS': ['$DIALECTFLAGS', '$WARNINGFLAGS'],
v8.team.kasperl727e9952008-09-02 14:56:44 +000067 'CXXFLAGS': ['$CCFLAGS', '/GR-', '/Gy'],
kasperl@chromium.orgb9123622008-09-17 14:05:56 +000068 'CPPDEFINES': ['WIN32', '_USE_32BIT_TIME_T', 'PCRE_STATIC'],
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +000069 'LINKFLAGS': ['/NOLOGO', '/MACHINE:X86', '/INCREMENTAL:NO',
70 '/NXCOMPAT', '/IGNORE:4221'],
71 'ARFLAGS': ['/NOLOGO'],
72 'CCPDBFLAGS': ['/Zi']
73 },
74 'mode:debug': {
75 'CCFLAGS': ['/Od', '/Gm', '/MTd'],
76 'CPPDEFINES': ['_DEBUG', 'ENABLE_DISASSEMBLER', 'DEBUG'],
77 'LINKFLAGS': ['/DEBUG']
78 },
79 'mode:release': {
80 'CCFLAGS': ['/Ox', '/MT', '/Ob2', '/Oi', '/Oy'],
v8.team.kasperl727e9952008-09-02 14:56:44 +000081 'LINKFLAGS': ['/OPT:REF', '/OPT:ICF']
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +000082 }
83 }
84}
85
86
87V8_EXTRA_FLAGS = {
88 'gcc': {
89 'all': {
90 'CXXFLAGS': [], #['-fvisibility=hidden'],
91 'WARNINGFLAGS': ['-pedantic', '-Wall', '-Werror', '-W',
92 '-Wno-unused-parameter']
93 },
94 'arch:arm': {
95 'CPPDEFINES': ['ARM']
96 },
ager@chromium.org9258b6b2008-09-11 09:11:10 +000097 'disassembler:on': {
98 'CPPDEFINES': ['ENABLE_DISASSEMBLER']
99 }
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000100 },
101 'msvc': {
102 'all': {
103 'WARNINGFLAGS': ['/W3', '/WX', '/wd4355', '/wd4800']
104 },
105 'library:shared': {
106 'CPPDEFINES': ['BUILDING_V8_SHARED']
107 },
108 'arch:arm': {
109 'CPPDEFINES': ['ARM']
110 },
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000111 'disassembler:on': {
112 'CPPDEFINES': ['ENABLE_DISASSEMBLER']
113 }
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000114 }
115}
116
117
118JSCRE_EXTRA_FLAGS = {
119 'gcc': {
120 'all': {
121 'CPPDEFINES': ['SUPPORT_UTF8', 'NO_RECURSE', 'SUPPORT_UCP'],
122 'WARNINGFLAGS': ['-w']
123 },
124 },
125 'msvc': {
126 'all': {
127 'CPPDEFINES': ['SUPPORT_UTF8', 'NO_RECURSE', 'SUPPORT_UCP'],
128 'WARNINGFLAGS': ['/W3', '/WX', '/wd4355', '/wd4800']
129 },
130 'library:shared': {
131 'CPPDEFINES': ['BUILDING_V8_SHARED']
132 }
133 }
134}
135
136
137DTOA_EXTRA_FLAGS = {
138 'gcc': {
139 'all': {
140 'WARNINGFLAGS': ['-Werror']
141 }
142 },
143 'msvc': {
144 'all': {
145 'WARNINGFLAGS': ['/WX', '/wd4018', '/wd4244']
146 }
147 }
148}
149
150
151CCTEST_EXTRA_FLAGS = {
152 'all': {
153 'CPPPATH': [join(root_dir, 'src')],
154 'LIBS': ['$LIBRARY']
155 },
156 'gcc': {
157 'all': {
158 'LIBPATH': [abspath('.')]
159 },
160 'wordsize:64': {
161 'CCFLAGS': ['-m32'],
162 'LINKFLAGS': ['-m32']
163 },
164 },
165 'msvc': {
166 'all': {
167 'CPPDEFINES': ['_HAS_EXCEPTIONS=0']
168 },
169 'library:shared': {
170 'CPPDEFINES': ['USING_V8_SHARED']
171 }
172 }
173}
174
175
176SAMPLE_FLAGS = {
177 'all': {
v8.team.kasperl727e9952008-09-02 14:56:44 +0000178 'CPPPATH': [join(abspath('.'), 'include')],
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000179 'LIBS': ['$LIBRARY'],
180 },
181 'gcc': {
182 'all': {
183 'LIBS': ['pthread'],
184 'LIBPATH': ['.']
185 },
186 'wordsize:64': {
187 'CCFLAGS': ['-m32'],
188 'LINKFLAGS': ['-m32']
189 },
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000190 'mode:debug': {
191 'CCFLAGS': ['-g', '-O0']
192 }
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000193 },
194 'msvc': {
195 'all': {
196 'CCFLAGS': ['/nologo'],
197 },
198 'library:shared': {
199 'CPPDEFINES': ['USING_V8_SHARED']
200 },
201 'mode:release': {
202 'CCFLAGS': ['/MT'],
203 },
204 'mode:debug': {
205 'CCFLAGS': ['/MTd']
206 }
207 }
208}
209
210
211SUFFIXES = {
212 'release': '',
213 'debug': '_g'
214}
215
216
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000217def Abort(message):
218 print message
219 sys.exit(1)
220
221
222def GuessOS():
223 id = platform.system()
224 if id == 'Linux':
225 return 'linux'
226 elif id == 'Darwin':
227 return 'macos'
228 elif id == 'Windows':
229 return 'win32'
230 else:
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000231 return None
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000232
233
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000234def GuessWordsize():
235 if '64' in platform.machine():
236 return '64'
237 else:
238 return '32'
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000239
240
241def GuessToolchain(os):
242 tools = Environment()['TOOLS']
243 if 'gcc' in tools:
kasper.lund7276f142008-07-30 08:49:36 +0000244 return 'gcc'
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000245 elif 'msvc' in tools:
246 return 'msvc'
247 else:
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000248 return None
249
250
251OS_GUESS = GuessOS()
252TOOLCHAIN_GUESS = GuessToolchain(OS_GUESS)
ager@chromium.orgc27e4e72008-09-04 13:52:27 +0000253ARCH_GUESS = utils.GuessArchitecture()
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000254WORDSIZE_GUESS = GuessWordsize()
255
256
257SIMPLE_OPTIONS = {
258 'toolchain': {
259 'values': ['gcc', 'msvc'],
260 'default': TOOLCHAIN_GUESS,
261 'help': 'the toolchain to use'
262 },
263 'os': {
264 'values': ['linux', 'macos', 'win32'],
265 'default': OS_GUESS,
266 'help': 'the os to build for'
267 },
268 'arch': {
269 'values':['arm', 'ia32'],
270 'default': ARCH_GUESS,
271 'help': 'the architecture to build for'
272 },
273 'snapshot': {
274 'values': ['on', 'off'],
275 'default': 'off',
276 'help': 'build using snapshots for faster start-up'
277 },
278 'library': {
279 'values': ['static', 'shared'],
280 'default': 'static',
281 'help': 'the type of library to produce'
282 },
283 'wordsize': {
284 'values': ['64', '32'],
285 'default': WORDSIZE_GUESS,
286 'help': 'the word size'
287 },
288 'simulator': {
289 'values': ['arm', 'none'],
290 'default': 'none',
291 'help': 'build with simulator'
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000292 },
293 'disassembler': {
294 'values': ['on', 'off'],
295 'default': 'off',
296 'help': 'enable the disassembler to inspect generated code'
297 },
298 'sourcesignatures': {
299 'values': ['MD5', 'timestamp'],
300 'default': 'MD5',
301 'help': 'set how the build system detects file changes'
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000302 }
303}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000304
305
306def GetOptions():
307 result = Options()
kasper.lund7276f142008-07-30 08:49:36 +0000308 result.Add('mode', 'compilation mode (debug, release)', 'release')
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000309 result.Add('sample', 'build sample (shell, process)', '')
v8.team.kasperl727e9952008-09-02 14:56:44 +0000310 result.Add('env', 'override environment settings (NAME1:value1,NAME2:value2)', '')
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000311 for (name, option) in SIMPLE_OPTIONS.items():
312 help = '%s (%s)' % (name, ", ".join(option['values']))
313 result.Add(name, help, option.get('default'))
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000314 return result
315
316
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000317def SplitList(str):
318 return [ s for s in str.split(",") if len(s) > 0 ]
319
320
321def IsLegal(env, option, values):
322 str = env[option]
323 for s in SplitList(str):
324 if not s in values:
325 Abort("Illegal value for option %s '%s'." % (option, s))
326 return False
327 return True
328
329
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000330def VerifyOptions(env):
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000331 if not IsLegal(env, 'mode', ['debug', 'release']):
332 return False
333 if not IsLegal(env, 'sample', ["shell", "process"]):
334 return False
335 for (name, option) in SIMPLE_OPTIONS.items():
336 if (not option.get('default')) and (name not in ARGUMENTS):
337 message = ("A value for option %s must be specified (%s)." %
338 (name, ", ".join(option['values'])))
339 Abort(message)
340 if not env[name] in option['values']:
341 message = ("Unknown %s value '%s'. Possible values are (%s)." %
342 (name, env[name], ", ".join(option['values'])))
343 Abort(message)
344
345
346class BuildContext(object):
347
v8.team.kasperl727e9952008-09-02 14:56:44 +0000348 def __init__(self, options, env_overrides, samples):
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000349 self.library_targets = []
350 self.cctest_targets = []
351 self.sample_targets = []
352 self.options = options
v8.team.kasperl727e9952008-09-02 14:56:44 +0000353 self.env_overrides = env_overrides
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000354 self.samples = samples
355 self.use_snapshot = (options['snapshot'] == 'on')
356 self.flags = None
357
358 def AddRelevantFlags(self, initial, flags):
359 result = initial.copy()
360 self.AppendFlags(result, flags.get('all'))
361 toolchain = self.options['toolchain']
362 self.AppendFlags(result, flags[toolchain].get('all'))
363 for option in sorted(self.options.keys()):
364 value = self.options[option]
365 self.AppendFlags(result, flags[toolchain].get(option + ':' + value))
366 return result
367
368 def GetRelevantSources(self, source):
369 result = []
370 result += source.get('all', [])
371 for (name, value) in self.options.items():
372 result += source.get(name + ':' + value, [])
373 return sorted(result)
374
375 def AppendFlags(self, options, added):
376 if not added:
377 return
378 for (key, value) in added.items():
379 if not key in options:
380 options[key] = value
381 else:
382 options[key] = options[key] + value
383
384 def ConfigureObject(self, env, input, **kw):
385 if self.options['library'] == 'static':
386 return env.StaticObject(input, **kw)
387 else:
388 return env.SharedObject(input, **kw)
389
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000390 def ApplyEnvOverrides(self, env):
391 if not self.env_overrides:
392 return
393 if type(env['ENV']) == DictType:
394 env['ENV'].update(**self.env_overrides)
395 else:
396 env['ENV'] = self.env_overrides
397
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000398
399def PostprocessOptions(options):
400 # Adjust architecture if the simulator option has been set
401 if (options['simulator'] != 'none') and (options['arch'] != options['simulator']):
402 if 'arch' in ARGUMENTS:
403 # Print a warning if arch has explicitly been set
404 print "Warning: forcing architecture to match simulator (%s)" % options['simulator']
405 options['arch'] = options['simulator']
406
407
v8.team.kasperl727e9952008-09-02 14:56:44 +0000408def ParseEnvOverrides(arg):
409 # The environment overrides are in the format NAME1:value1,NAME2:value2
410 overrides = {}
411 for override in arg.split(','):
412 pos = override.find(':')
413 if pos == -1:
414 continue
415 overrides[override[:pos].strip()] = override[pos+1:].strip()
416 return overrides
417
418
419def BuildSpecific(env, mode, env_overrides):
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000420 options = {'mode': mode}
421 for option in SIMPLE_OPTIONS:
422 options[option] = env[option]
423 PostprocessOptions(options)
424
v8.team.kasperl727e9952008-09-02 14:56:44 +0000425 context = BuildContext(options, env_overrides, samples=SplitList(env['sample']))
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000426
427 library_flags = context.AddRelevantFlags(os.environ, LIBRARY_FLAGS)
428 v8_flags = context.AddRelevantFlags(library_flags, V8_EXTRA_FLAGS)
429 jscre_flags = context.AddRelevantFlags(library_flags, JSCRE_EXTRA_FLAGS)
430 dtoa_flags = context.AddRelevantFlags(library_flags, DTOA_EXTRA_FLAGS)
431 cctest_flags = context.AddRelevantFlags(v8_flags, CCTEST_EXTRA_FLAGS)
432 sample_flags = context.AddRelevantFlags(os.environ, SAMPLE_FLAGS)
433
434 context.flags = {
435 'v8': v8_flags,
436 'jscre': jscre_flags,
437 'dtoa': dtoa_flags,
438 'cctest': cctest_flags,
439 'sample': sample_flags
440 }
441
442 target_id = mode
443 suffix = SUFFIXES[target_id]
444 library_name = 'v8' + suffix
445 env['LIBRARY'] = library_name
446
447 # Build the object files by invoking SCons recursively.
448 object_files = env.SConscript(
449 join('src', 'SConscript'),
450 build_dir=join('obj', target_id),
451 exports='context',
452 duplicate=False
453 )
454
455 # Link the object files into a library.
kasperl@chromium.orgb9123622008-09-17 14:05:56 +0000456 env.Replace(**context.flags['v8'])
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000457 context.ApplyEnvOverrides(env)
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000458 if context.options['library'] == 'static':
459 library = env.StaticLibrary(library_name, object_files)
460 else:
461 # There seems to be a glitch in the way scons decides where to put
462 # PDB files when compiling using MSVC so we specify it manually.
463 # This should not affect any other platforms.
464 pdb_name = library_name + '.dll.pdb'
465 library = env.SharedLibrary(library_name, object_files, PDB=pdb_name)
466 context.library_targets.append(library)
467
468 for sample in context.samples:
469 sample_env = Environment(LIBRARY=library_name)
470 sample_env.Replace(**context.flags['sample'])
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000471 context.ApplyEnvOverrides(sample_env)
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000472 sample_object = sample_env.SConscript(
473 join('samples', 'SConscript'),
474 build_dir=join('obj', 'sample', sample, target_id),
475 exports='sample context',
476 duplicate=False
477 )
478 sample_name = sample + suffix
479 sample_program = sample_env.Program(sample_name, sample_object)
480 sample_env.Depends(sample_program, library)
481 context.sample_targets.append(sample_program)
482
483 cctest_program = env.SConscript(
484 join('test', 'cctest', 'SConscript'),
485 build_dir=join('obj', 'test', target_id),
486 exports='context object_files',
487 duplicate=False
488 )
489 context.cctest_targets.append(cctest_program)
490
491 return context
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000492
493
kasper.lund7276f142008-07-30 08:49:36 +0000494def Build():
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000495 opts = GetOptions()
496 env = Environment(options=opts)
497 Help(opts.GenerateHelpText(env))
498 VerifyOptions(env)
v8.team.kasperl727e9952008-09-02 14:56:44 +0000499 env_overrides = ParseEnvOverrides(env['env'])
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000500
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000501 SourceSignatures(env['sourcesignatures'])
502
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000503 libraries = []
504 cctests = []
505 samples = []
506 modes = SplitList(env['mode'])
507 for mode in modes:
v8.team.kasperl727e9952008-09-02 14:56:44 +0000508 context = BuildSpecific(env.Copy(), mode, env_overrides)
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000509 libraries += context.library_targets
510 cctests += context.cctest_targets
511 samples += context.sample_targets
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000512
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000513 env.Alias('library', libraries)
514 env.Alias('cctests', cctests)
515 env.Alias('sample', samples)
516
517 if env['sample']:
518 env.Default('sample')
kasper.lund7276f142008-07-30 08:49:36 +0000519 else:
mads.s.ager@gmail.com9a4089a2008-09-01 08:55:01 +0000520 env.Default('library')
kasper.lund7276f142008-07-30 08:49:36 +0000521
522
v8.team.kasperl727e9952008-09-02 14:56:44 +0000523# We disable deprecation warnings because we need to be able to use
524# env.Copy without getting warnings for compatibility with older
525# version of scons. Also, there's a bug in some revisions that
526# doesn't allow this flag to be set, so we swallow any exceptions.
527# Lovely.
528try:
529 SetOption('warn', 'no-deprecated')
530except:
531 pass
532
533
kasper.lund7276f142008-07-30 08:49:36 +0000534Build()