SConscript revision 9850
1# -*- mode:python -*-
2
3# Copyright (c) 2004-2005 The Regents of The University of Michigan
4# All rights reserved.
5#
6# Redistribution and use in source and binary forms, with or without
7# modification, are permitted provided that the following conditions are
8# met: redistributions of source code must retain the above copyright
9# notice, this list of conditions and the following disclaimer;
10# redistributions in binary form must reproduce the above copyright
11# notice, this list of conditions and the following disclaimer in the
12# documentation and/or other materials provided with the distribution;
13# neither the name of the copyright holders nor the names of its
14# contributors may be used to endorse or promote products derived from
15# this software without specific prior written permission.
16#
17# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
20# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
21# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28#
29# Authors: Nathan Binkert
30
31import array
32import bisect
33import imp
34import marshal
35import os
36import re
37import sys
38import zlib
39
40from os.path import basename, dirname, exists, isdir, isfile, join as joinpath
41
42import SCons
43
44# This file defines how to build a particular configuration of gem5
45# based on variable settings in the 'env' build environment.
46
47Import('*')
48
49# Children need to see the environment
50Export('env')
51
52build_env = [(opt, env[opt]) for opt in export_vars]
53
54from m5.util import code_formatter, compareVersions
55
56########################################################################
57# Code for adding source files of various types
58#
59# When specifying a source file of some type, a set of guards can be
60# specified for that file.  When get() is used to find the files, if
61# get specifies a set of filters, only files that match those filters
62# will be accepted (unspecified filters on files are assumed to be
63# false).  Current filters are:
64#     main -- specifies the gem5 main() function
65#     skip_lib -- do not put this file into the gem5 library
66#     <unittest> -- unit tests use filters based on the unit test name
67#
68# A parent can now be specified for a source file and default filter
69# values will be retrieved recursively from parents (children override
70# parents).
71#
72class SourceMeta(type):
73    '''Meta class for source files that keeps track of all files of a
74    particular type and has a get function for finding all functions
75    of a certain type that match a set of guards'''
76    def __init__(cls, name, bases, dict):
77        super(SourceMeta, cls).__init__(name, bases, dict)
78        cls.all = []
79        
80    def get(cls, **guards):
81        '''Find all files that match the specified guards.  If a source
82        file does not specify a flag, the default is False'''
83        for src in cls.all:
84            for flag,value in guards.iteritems():
85                # if the flag is found and has a different value, skip
86                # this file
87                if src.all_guards.get(flag, False) != value:
88                    break
89            else:
90                yield src
91
92class SourceFile(object):
93    '''Base object that encapsulates the notion of a source file.
94    This includes, the source node, target node, various manipulations
95    of those.  A source file also specifies a set of guards which
96    describing which builds the source file applies to.  A parent can
97    also be specified to get default guards from'''
98    __metaclass__ = SourceMeta
99    def __init__(self, source, parent=None, **guards):
100        self.guards = guards
101        self.parent = parent
102
103        tnode = source
104        if not isinstance(source, SCons.Node.FS.File):
105            tnode = File(source)
106
107        self.tnode = tnode
108        self.snode = tnode.srcnode()
109
110        for base in type(self).__mro__:
111            if issubclass(base, SourceFile):
112                base.all.append(self)
113
114    @property
115    def filename(self):
116        return str(self.tnode)
117
118    @property
119    def dirname(self):
120        return dirname(self.filename)
121
122    @property
123    def basename(self):
124        return basename(self.filename)
125
126    @property
127    def extname(self):
128        index = self.basename.rfind('.')
129        if index <= 0:
130            # dot files aren't extensions
131            return self.basename, None
132
133        return self.basename[:index], self.basename[index+1:]
134
135    @property
136    def all_guards(self):
137        '''find all guards for this object getting default values
138        recursively from its parents'''
139        guards = {}
140        if self.parent:
141            guards.update(self.parent.guards)
142        guards.update(self.guards)
143        return guards
144
145    def __lt__(self, other): return self.filename < other.filename
146    def __le__(self, other): return self.filename <= other.filename
147    def __gt__(self, other): return self.filename > other.filename
148    def __ge__(self, other): return self.filename >= other.filename
149    def __eq__(self, other): return self.filename == other.filename
150    def __ne__(self, other): return self.filename != other.filename
151        
152class Source(SourceFile):
153    '''Add a c/c++ source file to the build'''
154    def __init__(self, source, Werror=True, swig=False, **guards):
155        '''specify the source file, and any guards'''
156        super(Source, self).__init__(source, **guards)
157
158        self.Werror = Werror
159        self.swig = swig
160
161class PySource(SourceFile):
162    '''Add a python source file to the named package'''
163    invalid_sym_char = re.compile('[^A-z0-9_]')
164    modules = {}
165    tnodes = {}
166    symnames = {}
167    
168    def __init__(self, package, source, **guards):
169        '''specify the python package, the source file, and any guards'''
170        super(PySource, self).__init__(source, **guards)
171
172        modname,ext = self.extname
173        assert ext == 'py'
174
175        if package:
176            path = package.split('.')
177        else:
178            path = []
179
180        modpath = path[:]
181        if modname != '__init__':
182            modpath += [ modname ]
183        modpath = '.'.join(modpath)
184
185        arcpath = path + [ self.basename ]
186        abspath = self.snode.abspath
187        if not exists(abspath):
188            abspath = self.tnode.abspath
189
190        self.package = package
191        self.modname = modname
192        self.modpath = modpath
193        self.arcname = joinpath(*arcpath)
194        self.abspath = abspath
195        self.compiled = File(self.filename + 'c')
196        self.cpp = File(self.filename + '.cc')
197        self.symname = PySource.invalid_sym_char.sub('_', modpath)
198
199        PySource.modules[modpath] = self
200        PySource.tnodes[self.tnode] = self
201        PySource.symnames[self.symname] = self
202
203class SimObject(PySource):
204    '''Add a SimObject python file as a python source object and add
205    it to a list of sim object modules'''
206
207    fixed = False
208    modnames = []
209
210    def __init__(self, source, **guards):
211        '''Specify the source file and any guards (automatically in
212        the m5.objects package)'''
213        super(SimObject, self).__init__('m5.objects', source, **guards)
214        if self.fixed:
215            raise AttributeError, "Too late to call SimObject now."
216
217        bisect.insort_right(SimObject.modnames, self.modname)
218
219class SwigSource(SourceFile):
220    '''Add a swig file to build'''
221
222    def __init__(self, package, source, **guards):
223        '''Specify the python package, the source file, and any guards'''
224        super(SwigSource, self).__init__(source, **guards)
225
226        modname,ext = self.extname
227        assert ext == 'i'
228
229        self.module = modname
230        cc_file = joinpath(self.dirname, modname + '_wrap.cc')
231        py_file = joinpath(self.dirname, modname + '.py')
232
233        self.cc_source = Source(cc_file, swig=True, parent=self)
234        self.py_source = PySource(package, py_file, parent=self)
235
236class ProtoBuf(SourceFile):
237    '''Add a Protocol Buffer to build'''
238
239    def __init__(self, source, **guards):
240        '''Specify the source file, and any guards'''
241        super(ProtoBuf, self).__init__(source, **guards)
242
243        # Get the file name and the extension
244        modname,ext = self.extname
245        assert ext == 'proto'
246
247        # Currently, we stick to generating the C++ headers, so we
248        # only need to track the source and header.
249        self.cc_file = File(joinpath(self.dirname, modname + '.pb.cc'))
250        self.hh_file = File(joinpath(self.dirname, modname + '.pb.h'))
251
252class UnitTest(object):
253    '''Create a UnitTest'''
254
255    all = []
256    def __init__(self, target, *sources, **kwargs):
257        '''Specify the target name and any sources.  Sources that are
258        not SourceFiles are evalued with Source().  All files are
259        guarded with a guard of the same name as the UnitTest
260        target.'''
261
262        srcs = []
263        for src in sources:
264            if not isinstance(src, SourceFile):
265                src = Source(src, skip_lib=True)
266            src.guards[target] = True
267            srcs.append(src)
268
269        self.sources = srcs
270        self.target = target
271        self.main = kwargs.get('main', False)
272        UnitTest.all.append(self)
273
274# Children should have access
275Export('Source')
276Export('PySource')
277Export('SimObject')
278Export('SwigSource')
279Export('ProtoBuf')
280Export('UnitTest')
281
282########################################################################
283#
284# Debug Flags
285#
286debug_flags = {}
287def DebugFlag(name, desc=None):
288    if name in debug_flags:
289        raise AttributeError, "Flag %s already specified" % name
290    debug_flags[name] = (name, (), desc)
291
292def CompoundFlag(name, flags, desc=None):
293    if name in debug_flags:
294        raise AttributeError, "Flag %s already specified" % name
295
296    compound = tuple(flags)
297    debug_flags[name] = (name, compound, desc)
298
299Export('DebugFlag')
300Export('CompoundFlag')
301
302########################################################################
303#
304# Set some compiler variables
305#
306
307# Include file paths are rooted in this directory.  SCons will
308# automatically expand '.' to refer to both the source directory and
309# the corresponding build directory to pick up generated include
310# files.
311env.Append(CPPPATH=Dir('.'))
312
313for extra_dir in extras_dir_list:
314    env.Append(CPPPATH=Dir(extra_dir))
315
316# Workaround for bug in SCons version > 0.97d20071212
317# Scons bug id: 2006 gem5 Bug id: 308
318for root, dirs, files in os.walk(base_dir, topdown=True):
319    Dir(root[len(base_dir) + 1:])
320
321########################################################################
322#
323# Walk the tree and execute all SConscripts in subdirectories
324#
325
326here = Dir('.').srcnode().abspath
327for root, dirs, files in os.walk(base_dir, topdown=True):
328    if root == here:
329        # we don't want to recurse back into this SConscript
330        continue
331
332    if 'SConscript' in files:
333        build_dir = joinpath(env['BUILDDIR'], root[len(base_dir) + 1:])
334        SConscript(joinpath(root, 'SConscript'), variant_dir=build_dir)
335
336for extra_dir in extras_dir_list:
337    prefix_len = len(dirname(extra_dir)) + 1
338    for root, dirs, files in os.walk(extra_dir, topdown=True):
339        # if build lives in the extras directory, don't walk down it
340        if 'build' in dirs:
341            dirs.remove('build')
342
343        if 'SConscript' in files:
344            build_dir = joinpath(env['BUILDDIR'], root[prefix_len:])
345            SConscript(joinpath(root, 'SConscript'), variant_dir=build_dir)
346
347for opt in export_vars:
348    env.ConfigFile(opt)
349
350def makeTheISA(source, target, env):
351    isas = [ src.get_contents() for src in source ]
352    target_isa = env['TARGET_ISA']
353    def define(isa):
354        return isa.upper() + '_ISA'
355    
356    def namespace(isa):
357        return isa[0].upper() + isa[1:].lower() + 'ISA' 
358
359
360    code = code_formatter()
361    code('''\
362#ifndef __CONFIG_THE_ISA_HH__
363#define __CONFIG_THE_ISA_HH__
364
365''')
366
367    for i,isa in enumerate(isas):
368        code('#define $0 $1', define(isa), i + 1)
369
370    code('''
371
372#define THE_ISA ${{define(target_isa)}}
373#define TheISA ${{namespace(target_isa)}}
374#define THE_ISA_STR "${{target_isa}}"
375
376#endif // __CONFIG_THE_ISA_HH__''')
377
378    code.write(str(target[0]))
379
380env.Command('config/the_isa.hh', map(Value, all_isa_list),
381            MakeAction(makeTheISA, Transform("CFG ISA", 0)))
382
383########################################################################
384#
385# Prevent any SimObjects from being added after this point, they
386# should all have been added in the SConscripts above
387#
388SimObject.fixed = True
389
390class DictImporter(object):
391    '''This importer takes a dictionary of arbitrary module names that
392    map to arbitrary filenames.'''
393    def __init__(self, modules):
394        self.modules = modules
395        self.installed = set()
396
397    def __del__(self):
398        self.unload()
399
400    def unload(self):
401        import sys
402        for module in self.installed:
403            del sys.modules[module]
404        self.installed = set()
405
406    def find_module(self, fullname, path):
407        if fullname == 'm5.defines':
408            return self
409
410        if fullname == 'm5.objects':
411            return self
412
413        if fullname.startswith('m5.internal'):
414            return None
415
416        source = self.modules.get(fullname, None)
417        if source is not None and fullname.startswith('m5.objects'):
418            return self
419
420        return None
421
422    def load_module(self, fullname):
423        mod = imp.new_module(fullname)
424        sys.modules[fullname] = mod
425        self.installed.add(fullname)
426
427        mod.__loader__ = self
428        if fullname == 'm5.objects':
429            mod.__path__ = fullname.split('.')
430            return mod
431
432        if fullname == 'm5.defines':
433            mod.__dict__['buildEnv'] = m5.util.SmartDict(build_env)
434            return mod
435
436        source = self.modules[fullname]
437        if source.modname == '__init__':
438            mod.__path__ = source.modpath
439        mod.__file__ = source.abspath
440
441        exec file(source.abspath, 'r') in mod.__dict__
442
443        return mod
444
445import m5.SimObject
446import m5.params
447from m5.util import code_formatter
448
449m5.SimObject.clear()
450m5.params.clear()
451
452# install the python importer so we can grab stuff from the source
453# tree itself.  We can't have SimObjects added after this point or
454# else we won't know about them for the rest of the stuff.
455importer = DictImporter(PySource.modules)
456sys.meta_path[0:0] = [ importer ]
457
458# import all sim objects so we can populate the all_objects list
459# make sure that we're working with a list, then let's sort it
460for modname in SimObject.modnames:
461    exec('from m5.objects import %s' % modname)
462
463# we need to unload all of the currently imported modules so that they
464# will be re-imported the next time the sconscript is run
465importer.unload()
466sys.meta_path.remove(importer)
467
468sim_objects = m5.SimObject.allClasses
469all_enums = m5.params.allEnums
470
471if m5.SimObject.noCxxHeader:
472    print >> sys.stderr, \
473        "warning: At least one SimObject lacks a header specification. " \
474        "This can cause unexpected results in the generated SWIG " \
475        "wrappers."
476
477# Find param types that need to be explicitly wrapped with swig.
478# These will be recognized because the ParamDesc will have a
479# swig_decl() method.  Most param types are based on types that don't
480# need this, either because they're based on native types (like Int)
481# or because they're SimObjects (which get swigged independently).
482# For now the only things handled here are VectorParam types.
483params_to_swig = {}
484for name,obj in sorted(sim_objects.iteritems()):
485    for param in obj._params.local.values():
486        # load the ptype attribute now because it depends on the
487        # current version of SimObject.allClasses, but when scons
488        # actually uses the value, all versions of
489        # SimObject.allClasses will have been loaded
490        param.ptype
491
492        if not hasattr(param, 'swig_decl'):
493            continue
494        pname = param.ptype_str
495        if pname not in params_to_swig:
496            params_to_swig[pname] = param
497
498########################################################################
499#
500# calculate extra dependencies
501#
502module_depends = ["m5", "m5.SimObject", "m5.params"]
503depends = [ PySource.modules[dep].snode for dep in module_depends ]
504
505########################################################################
506#
507# Commands for the basic automatically generated python files
508#
509
510# Generate Python file containing a dict specifying the current
511# buildEnv flags.
512def makeDefinesPyFile(target, source, env):
513    build_env = source[0].get_contents()
514
515    code = code_formatter()
516    code("""
517import m5.internal
518import m5.util
519
520buildEnv = m5.util.SmartDict($build_env)
521
522compileDate = m5.internal.core.compileDate
523_globals = globals()
524for key,val in m5.internal.core.__dict__.iteritems():
525    if key.startswith('flag_'):
526        flag = key[5:]
527        _globals[flag] = val
528del _globals
529""")
530    code.write(target[0].abspath)
531
532defines_info = Value(build_env)
533# Generate a file with all of the compile options in it
534env.Command('python/m5/defines.py', defines_info,
535            MakeAction(makeDefinesPyFile, Transform("DEFINES", 0)))
536PySource('m5', 'python/m5/defines.py')
537
538# Generate python file containing info about the M5 source code
539def makeInfoPyFile(target, source, env):
540    code = code_formatter()
541    for src in source:
542        data = ''.join(file(src.srcnode().abspath, 'r').xreadlines())
543        code('$src = ${{repr(data)}}')
544    code.write(str(target[0]))
545
546# Generate a file that wraps the basic top level files
547env.Command('python/m5/info.py',
548            [ '#/COPYING', '#/LICENSE', '#/README', ],
549            MakeAction(makeInfoPyFile, Transform("INFO")))
550PySource('m5', 'python/m5/info.py')
551
552########################################################################
553#
554# Create all of the SimObject param headers and enum headers
555#
556
557def createSimObjectParamStruct(target, source, env):
558    assert len(target) == 1 and len(source) == 1
559
560    name = str(source[0].get_contents())
561    obj = sim_objects[name]
562
563    code = code_formatter()
564    obj.cxx_param_decl(code)
565    code.write(target[0].abspath)
566
567def createParamSwigWrapper(target, source, env):
568    assert len(target) == 1 and len(source) == 1
569
570    name = str(source[0].get_contents())
571    param = params_to_swig[name]
572
573    code = code_formatter()
574    param.swig_decl(code)
575    code.write(target[0].abspath)
576
577def createEnumStrings(target, source, env):
578    assert len(target) == 1 and len(source) == 1
579
580    name = str(source[0].get_contents())
581    obj = all_enums[name]
582
583    code = code_formatter()
584    obj.cxx_def(code)
585    code.write(target[0].abspath)
586
587def createEnumDecls(target, source, env):
588    assert len(target) == 1 and len(source) == 1
589
590    name = str(source[0].get_contents())
591    obj = all_enums[name]
592
593    code = code_formatter()
594    obj.cxx_decl(code)
595    code.write(target[0].abspath)
596
597def createEnumSwigWrapper(target, source, env):
598    assert len(target) == 1 and len(source) == 1
599
600    name = str(source[0].get_contents())
601    obj = all_enums[name]
602
603    code = code_formatter()
604    obj.swig_decl(code)
605    code.write(target[0].abspath)
606
607def createSimObjectSwigWrapper(target, source, env):
608    name = source[0].get_contents()
609    obj = sim_objects[name]
610
611    code = code_formatter()
612    obj.swig_decl(code)
613    code.write(target[0].abspath)
614
615# Generate all of the SimObject param C++ struct header files
616params_hh_files = []
617for name,simobj in sorted(sim_objects.iteritems()):
618    py_source = PySource.modules[simobj.__module__]
619    extra_deps = [ py_source.tnode ]
620
621    hh_file = File('params/%s.hh' % name)
622    params_hh_files.append(hh_file)
623    env.Command(hh_file, Value(name),
624                MakeAction(createSimObjectParamStruct, Transform("SO PARAM")))
625    env.Depends(hh_file, depends + extra_deps)
626
627# Generate any needed param SWIG wrapper files
628params_i_files = []
629for name,param in params_to_swig.iteritems():
630    i_file = File('python/m5/internal/%s.i' % (param.swig_module_name()))
631    params_i_files.append(i_file)
632    env.Command(i_file, Value(name),
633                MakeAction(createParamSwigWrapper, Transform("SW PARAM")))
634    env.Depends(i_file, depends)
635    SwigSource('m5.internal', i_file)
636
637# Generate all enum header files
638for name,enum in sorted(all_enums.iteritems()):
639    py_source = PySource.modules[enum.__module__]
640    extra_deps = [ py_source.tnode ]
641
642    cc_file = File('enums/%s.cc' % name)
643    env.Command(cc_file, Value(name),
644                MakeAction(createEnumStrings, Transform("ENUM STR")))
645    env.Depends(cc_file, depends + extra_deps)
646    Source(cc_file)
647
648    hh_file = File('enums/%s.hh' % name)
649    env.Command(hh_file, Value(name),
650                MakeAction(createEnumDecls, Transform("ENUMDECL")))
651    env.Depends(hh_file, depends + extra_deps)
652
653    i_file = File('python/m5/internal/enum_%s.i' % name)
654    env.Command(i_file, Value(name),
655                MakeAction(createEnumSwigWrapper, Transform("ENUMSWIG")))
656    env.Depends(i_file, depends + extra_deps)
657    SwigSource('m5.internal', i_file)
658
659# Generate SimObject SWIG wrapper files
660for name,simobj in sim_objects.iteritems():
661    py_source = PySource.modules[simobj.__module__]
662    extra_deps = [ py_source.tnode ]
663
664    i_file = File('python/m5/internal/param_%s.i' % name)
665    env.Command(i_file, Value(name),
666                MakeAction(createSimObjectSwigWrapper, Transform("SO SWIG")))
667    env.Depends(i_file, depends + extra_deps)
668    SwigSource('m5.internal', i_file)
669
670# Generate the main swig init file
671def makeEmbeddedSwigInit(target, source, env):
672    code = code_formatter()
673    module = source[0].get_contents()
674    code('''\
675#include "sim/init.hh"
676
677extern "C" {
678    void init_${module}();
679}
680
681EmbeddedSwig embed_swig_${module}(init_${module});
682''')
683    code.write(str(target[0]))
684    
685# Build all swig modules
686for swig in SwigSource.all:
687    env.Command([swig.cc_source.tnode, swig.py_source.tnode], swig.tnode,
688                MakeAction('$SWIG $SWIGFLAGS -outdir ${TARGETS[1].dir} '
689                '-o ${TARGETS[0]} $SOURCES', Transform("SWIG")))
690    cc_file = str(swig.tnode)
691    init_file = '%s/%s_init.cc' % (dirname(cc_file), basename(cc_file))
692    env.Command(init_file, Value(swig.module),
693                MakeAction(makeEmbeddedSwigInit, Transform("EMBED SW")))
694    Source(init_file, **swig.guards)
695
696# Build all protocol buffers if we have got protoc and protobuf available
697if env['HAVE_PROTOBUF']:
698    for proto in ProtoBuf.all:
699        # Use both the source and header as the target, and the .proto
700        # file as the source. When executing the protoc compiler, also
701        # specify the proto_path to avoid having the generated files
702        # include the path.
703        env.Command([proto.cc_file, proto.hh_file], proto.tnode,
704                    MakeAction('$PROTOC --cpp_out ${TARGET.dir} '
705                               '--proto_path ${SOURCE.dir} $SOURCE',
706                               Transform("PROTOC")))
707
708        # Add the C++ source file
709        Source(proto.cc_file, **proto.guards)
710elif ProtoBuf.all:
711    print 'Got protobuf to build, but lacks support!'
712    Exit(1)
713
714#
715# Handle debug flags
716#
717def makeDebugFlagCC(target, source, env):
718    assert(len(target) == 1 and len(source) == 1)
719
720    val = eval(source[0].get_contents())
721    name, compound, desc = val
722    compound = list(sorted(compound))
723
724    code = code_formatter()
725
726    # file header
727    code('''
728/*
729 * DO NOT EDIT THIS FILE! Automatically generated
730 */
731
732#include "base/debug.hh"
733''')
734
735    for flag in compound:
736        code('#include "debug/$flag.hh"')
737    code()
738    code('namespace Debug {')
739    code()
740
741    if not compound:
742        code('SimpleFlag $name("$name", "$desc");')
743    else:
744        code('CompoundFlag $name("$name", "$desc",')
745        code.indent()
746        last = len(compound) - 1
747        for i,flag in enumerate(compound):
748            if i != last:
749                code('$flag,')
750            else:
751                code('$flag);')
752        code.dedent()
753
754    code()
755    code('} // namespace Debug')
756
757    code.write(str(target[0]))
758
759def makeDebugFlagHH(target, source, env):
760    assert(len(target) == 1 and len(source) == 1)
761
762    val = eval(source[0].get_contents())
763    name, compound, desc = val
764
765    code = code_formatter()
766
767    # file header boilerplate
768    code('''\
769/*
770 * DO NOT EDIT THIS FILE!
771 *
772 * Automatically generated by SCons
773 */
774
775#ifndef __DEBUG_${name}_HH__
776#define __DEBUG_${name}_HH__
777
778namespace Debug {
779''')
780
781    if compound:
782        code('class CompoundFlag;')
783    code('class SimpleFlag;')
784
785    if compound:
786        code('extern CompoundFlag $name;')
787        for flag in compound:
788            code('extern SimpleFlag $flag;')
789    else:
790        code('extern SimpleFlag $name;')
791
792    code('''
793}
794
795#endif // __DEBUG_${name}_HH__
796''')
797
798    code.write(str(target[0]))
799
800for name,flag in sorted(debug_flags.iteritems()):
801    n, compound, desc = flag
802    assert n == name
803
804    env.Command('debug/%s.hh' % name, Value(flag),
805                MakeAction(makeDebugFlagHH, Transform("TRACING", 0)))
806    env.Command('debug/%s.cc' % name, Value(flag),
807                MakeAction(makeDebugFlagCC, Transform("TRACING", 0)))
808    Source('debug/%s.cc' % name)
809
810# Embed python files.  All .py files that have been indicated by a
811# PySource() call in a SConscript need to be embedded into the M5
812# library.  To do that, we compile the file to byte code, marshal the
813# byte code, compress it, and then generate a c++ file that
814# inserts the result into an array.
815def embedPyFile(target, source, env):
816    def c_str(string):
817        if string is None:
818            return "0"
819        return '"%s"' % string
820
821    '''Action function to compile a .py into a code object, marshal
822    it, compress it, and stick it into an asm file so the code appears
823    as just bytes with a label in the data section'''
824
825    src = file(str(source[0]), 'r').read()
826
827    pysource = PySource.tnodes[source[0]]
828    compiled = compile(src, pysource.abspath, 'exec')
829    marshalled = marshal.dumps(compiled)
830    compressed = zlib.compress(marshalled)
831    data = compressed
832    sym = pysource.symname
833
834    code = code_formatter()
835    code('''\
836#include "sim/init.hh"
837
838namespace {
839
840const uint8_t data_${sym}[] = {
841''')
842    code.indent()
843    step = 16
844    for i in xrange(0, len(data), step):
845        x = array.array('B', data[i:i+step])
846        code(''.join('%d,' % d for d in x))
847    code.dedent()
848    
849    code('''};
850
851EmbeddedPython embedded_${sym}(
852    ${{c_str(pysource.arcname)}},
853    ${{c_str(pysource.abspath)}},
854    ${{c_str(pysource.modpath)}},
855    data_${sym},
856    ${{len(data)}},
857    ${{len(marshalled)}});
858
859} // anonymous namespace
860''')
861    code.write(str(target[0]))
862
863for source in PySource.all:
864    env.Command(source.cpp, source.tnode, 
865                MakeAction(embedPyFile, Transform("EMBED PY")))
866    Source(source.cpp)
867
868########################################################################
869#
870# Define binaries.  Each different build type (debug, opt, etc.) gets
871# a slightly different build environment.
872#
873
874# List of constructed environments to pass back to SConstruct
875envList = []
876
877date_source = Source('base/date.cc', skip_lib=True)
878
879# Function to create a new build environment as clone of current
880# environment 'env' with modified object suffix and optional stripped
881# binary.  Additional keyword arguments are appended to corresponding
882# build environment vars.
883def makeEnv(label, objsfx, strip = False, **kwargs):
884    # SCons doesn't know to append a library suffix when there is a '.' in the
885    # name.  Use '_' instead.
886    libname = 'gem5_' + label
887    exename = 'gem5.' + label
888    secondary_exename = 'm5.' + label
889
890    new_env = env.Clone(OBJSUFFIX=objsfx, SHOBJSUFFIX=objsfx + 's')
891    new_env.Label = label
892    new_env.Append(**kwargs)
893
894    swig_env = new_env.Clone()
895
896    # Both gcc and clang have issues with unused labels and values in
897    # the SWIG generated code
898    swig_env.Append(CCFLAGS=['-Wno-unused-label', '-Wno-unused-value'])
899
900    # Add additional warnings here that should not be applied to
901    # the SWIG generated code
902    new_env.Append(CXXFLAGS='-Wmissing-declarations')
903
904    if env['GCC']:
905        # Depending on the SWIG version, we also need to supress
906        # warnings about uninitialized variables and missing field
907        # initializers.
908        swig_env.Append(CCFLAGS=['-Wno-uninitialized',
909                                 '-Wno-missing-field-initializers'])
910
911        if compareVersions(env['GCC_VERSION'], '4.6') >= 0:
912            swig_env.Append(CCFLAGS='-Wno-unused-but-set-variable')
913
914        # If gcc supports it, also warn for deletion of derived
915        # classes with non-virtual desctructors. For gcc >= 4.7 we
916        # also have to disable warnings about the SWIG code having
917        # potentially uninitialized variables.
918        if compareVersions(env['GCC_VERSION'], '4.7') >= 0:
919            new_env.Append(CXXFLAGS='-Wdelete-non-virtual-dtor')
920            swig_env.Append(CCFLAGS='-Wno-maybe-uninitialized')
921    if env['CLANG']:
922        # Always enable the warning for deletion of derived classes
923        # with non-virtual destructors
924        new_env.Append(CXXFLAGS=['-Wdelete-non-virtual-dtor'])
925
926    werror_env = new_env.Clone()
927    werror_env.Append(CCFLAGS='-Werror')
928
929    def make_obj(source, static, extra_deps = None):
930        '''This function adds the specified source to the correct
931        build environment, and returns the corresponding SCons Object
932        nodes'''
933
934        if source.swig:
935            env = swig_env
936        elif source.Werror:
937            env = werror_env
938        else:
939            env = new_env
940
941        if static:
942            obj = env.StaticObject(source.tnode)
943        else:
944            obj = env.SharedObject(source.tnode)
945
946        if extra_deps:
947            env.Depends(obj, extra_deps)
948
949        return obj
950
951    static_objs = \
952        [ make_obj(s, True) for s in Source.get(main=False, skip_lib=False) ]
953    shared_objs = \
954        [ make_obj(s, False) for s in Source.get(main=False, skip_lib=False) ]
955
956    static_date = make_obj(date_source, static=True, extra_deps=static_objs)
957    static_objs.append(static_date)
958    
959    shared_date = make_obj(date_source, static=False, extra_deps=shared_objs)
960    shared_objs.append(shared_date)
961
962    # First make a library of everything but main() so other programs can
963    # link against m5.
964    static_lib = new_env.StaticLibrary(libname, static_objs)
965    shared_lib = new_env.SharedLibrary(libname, shared_objs)
966
967    # Now link a stub with main() and the static library.
968    main_objs = [ make_obj(s, True) for s in Source.get(main=True) ]
969
970    for test in UnitTest.all:
971        flags = { test.target : True }
972        test_sources = Source.get(**flags)
973        test_objs = [ make_obj(s, static=True) for s in test_sources ]
974        if test.main:
975            test_objs += main_objs
976        testname = "unittest/%s.%s" % (test.target, label)
977        new_env.Program(testname, test_objs + static_objs)
978
979    progname = exename
980    if strip:
981        progname += '.unstripped'
982
983    targets = new_env.Program(progname, main_objs + static_objs)
984
985    if strip:
986        if sys.platform == 'sunos5':
987            cmd = 'cp $SOURCE $TARGET; strip $TARGET'
988        else:
989            cmd = 'strip $SOURCE -o $TARGET'
990        targets = new_env.Command(exename, progname,
991                    MakeAction(cmd, Transform("STRIP")))
992
993    new_env.Command(secondary_exename, exename,
994            MakeAction('ln $SOURCE $TARGET', Transform("HARDLINK")))
995
996    new_env.M5Binary = targets[0]
997    envList.append(new_env)
998
999# Start out with the compiler flags common to all compilers,
1000# i.e. they all use -g for opt and -g -pg for prof
1001ccflags = {'debug' : [], 'opt' : ['-g'], 'fast' : [], 'prof' : ['-g', '-pg'],
1002           'perf' : ['-g']}
1003
1004# Start out with the linker flags common to all linkers, i.e. -pg for
1005# prof, and -lprofiler for perf. The -lprofile flag is surrounded by
1006# no-as-needed and as-needed as the binutils linker is too clever and
1007# simply doesn't link to the library otherwise.
1008ldflags = {'debug' : [], 'opt' : [], 'fast' : [], 'prof' : ['-pg'],
1009           'perf' : ['-Wl,--no-as-needed', '-lprofiler', '-Wl,--as-needed']}
1010
1011# For Link Time Optimization, the optimisation flags used to compile
1012# individual files are decoupled from those used at link time
1013# (i.e. you can compile with -O3 and perform LTO with -O0), so we need
1014# to also update the linker flags based on the target.
1015if env['GCC']:
1016    if sys.platform == 'sunos5':
1017        ccflags['debug'] += ['-gstabs+']
1018    else:
1019        ccflags['debug'] += ['-ggdb3']
1020    ldflags['debug'] += ['-O0']
1021    # opt, fast, prof and perf all share the same cc flags, also add
1022    # the optimization to the ldflags as LTO defers the optimization
1023    # to link time
1024    for target in ['opt', 'fast', 'prof', 'perf']:
1025        ccflags[target] += ['-O3']
1026        ldflags[target] += ['-O3']
1027
1028    ccflags['fast'] += env['LTO_CCFLAGS']
1029    ldflags['fast'] += env['LTO_LDFLAGS']
1030elif env['CLANG']:
1031    ccflags['debug'] += ['-g', '-O0']
1032    # opt, fast, prof and perf all share the same cc flags
1033    for target in ['opt', 'fast', 'prof', 'perf']:
1034        ccflags[target] += ['-O3']
1035else:
1036    print 'Unknown compiler, please fix compiler options'
1037    Exit(1)
1038
1039
1040# To speed things up, we only instantiate the build environments we
1041# need.  We try to identify the needed environment for each target; if
1042# we can't, we fall back on instantiating all the environments just to
1043# be safe.
1044target_types = ['debug', 'opt', 'fast', 'prof', 'perf']
1045obj2target = {'do': 'debug', 'o': 'opt', 'fo': 'fast', 'po': 'prof',
1046              'gpo' : 'perf'}
1047
1048def identifyTarget(t):
1049    ext = t.split('.')[-1]
1050    if ext in target_types:
1051        return ext
1052    if obj2target.has_key(ext):
1053        return obj2target[ext]
1054    match = re.search(r'/tests/([^/]+)/', t)
1055    if match and match.group(1) in target_types:
1056        return match.group(1)
1057    return 'all'
1058
1059needed_envs = [identifyTarget(target) for target in BUILD_TARGETS]
1060if 'all' in needed_envs:
1061    needed_envs += target_types
1062
1063# Debug binary
1064if 'debug' in needed_envs:
1065    makeEnv('debug', '.do',
1066            CCFLAGS = Split(ccflags['debug']),
1067            CPPDEFINES = ['DEBUG', 'TRACING_ON=1'],
1068            LINKFLAGS = Split(ldflags['debug']))
1069
1070# Optimized binary
1071if 'opt' in needed_envs:
1072    makeEnv('opt', '.o',
1073            CCFLAGS = Split(ccflags['opt']),
1074            CPPDEFINES = ['TRACING_ON=1'],
1075            LINKFLAGS = Split(ldflags['opt']))
1076
1077# "Fast" binary
1078if 'fast' in needed_envs:
1079    makeEnv('fast', '.fo', strip = True,
1080            CCFLAGS = Split(ccflags['fast']),
1081            CPPDEFINES = ['NDEBUG', 'TRACING_ON=0'],
1082            LINKFLAGS = Split(ldflags['fast']))
1083
1084# Profiled binary using gprof
1085if 'prof' in needed_envs:
1086    makeEnv('prof', '.po',
1087            CCFLAGS = Split(ccflags['prof']),
1088            CPPDEFINES = ['NDEBUG', 'TRACING_ON=0'],
1089            LINKFLAGS = Split(ldflags['prof']))
1090
1091# Profiled binary using google-pprof
1092if 'perf' in needed_envs:
1093    makeEnv('perf', '.gpo',
1094            CCFLAGS = Split(ccflags['perf']),
1095            CPPDEFINES = ['NDEBUG', 'TRACING_ON=0'],
1096            LINKFLAGS = Split(ldflags['perf']))
1097
1098Return('envList')
1099