+.\" 1. Builder's __call__ method can now take arbitrary keyword arguments.
+.\" These args are saved with the target node of the build, then passed along
.\" Copyright (c) 2001, 2002 Steven Knight
.\"
.\" Permission is hereby granted, free of charge, to any person obtaining
.RE
.fi
..
-.TH SCONS 1 "January 2002"
+.TH SCONS 1 "April 2002"
.SH NAME
-scons \- software constructor
+scons \- a software construction tool
.SH SYNOPSIS
.B scons
[
.TP
.RI --profile= file
-Run SCons under the Python profile
+Run SCons under the Python profiler
and save the results in the specified
.IR file .
The results may be analyzed using the Python
env.Object(target = 'bbb.o', source = 'bbb.c++')
env.Object(target = 'ccc.obj', source = 'ccc.f')
.EE
+.IP
+The
+.B Object
+builder accepts an optional "shared" keyword that, when non-zero,
+specifies that the object file should be built for
+inclusion in a shared library
+(that is, built with the '-fPIC' option when using gcc):
+
+.ES
+env.Object(target = 'ddd.obj', source = 'ddd.c', shared = 1)
+.EE
.IP Program
Builds an executable given one or more object files or C/C++ source
.EE
.IP Library
-Builds a library given one or more object files or C/C++ source
-files. If any C/C++ source files are given, then they will be automatically
+Builds a static or shared library given one or more object files
+or C, C++ or Fortran source files.
+If any source files are given,
+then they will be automatically
compiled to object files. The library prefix and suffix (if any) are
automatically added to the target. Example:
.ES
env.Library(target = 'bar', source = 'bar.c foo.o')
.EE
+.IP
+By default,
+.B Library
+builds a static library.
+A shared library (.so on a POSIX system, .dll on WIN32)
+may be specified by setting the
+.B shared
+keyword argument to non-zero:
+
+.ES
+env.Library(target = 'bar', source = 'bar.c foo.o', shared = 1)
+.EE
+.IP
+On WIN32 systems, the
+.B Library
+builder will always build an import (.lib) library
+in addition to the shared (.dll) library,
+adding a .lib library with the same basename
+if there is not already a .lib file explicitly
+listed in the targets.
+
+Any object files listed in the
+.B source
+list for a shared library
+must have been built for a shared library
+(that is, using a non-zero
+.B shared
+keyword argument).
+Conversely, object files built into a static library must
+.I not
+have been built for a shared library.
+.B scons
+will raise an error if there is any mismatch.
.IP CFile
Builds a C source file given a lex (.l) or yacc (.y) input file.
env.PostScript(target = 'bbb', source = 'bbb.dvi') # builds bbb.dvi
.EE
.LP
-.B scons automatically scans
+.B scons
+automatically scans
C, C++ and Fortran source files with .F, .fpp, or .FOR file extensions
for C preprocessor dependencies,
so the dependencies do not need to be specified explicitly.
.IP CC
The C compiler.
+.IP CCCOM
+The command line used to compile a C source file to a (static) object file.
+
.IP CCFLAGS
General options that are passed to the C compiler.
-.IP CCCOM
-The command line used to compile a C source file to an object file.
-
.IP CFILESUFFIX
The suffix for C source files.
This is used by the internal CFile builder
The default suffix is
.IR .cc .
-.IP CXXFLAGS
-General options that are passed to the C++ compiler.
-
.IP CXXCOM
The command line used to compile a C++ source file to an object file.
+.IP CXXFLAGS
+General options that are passed to the C++ compiler.
+
.IP DVIPDF
The TeX DVI file to PDF file converter.
.IP F77
The Fortran compiler.
-.IP F77FLAGS
-General options that are passed to the Fortran compiler.
-
.IP F77COM
The command line used to compile a Fortran source file to an object file.
+.IP F77FLAGS
+General options that are passed to the Fortran compiler.
+
.IP F77PPCOM
The command line used to compile a Fortran source file to an object file
after first running the file through the C preprocessor.
.EE
.IP LIBPREFIX
-The prefix used for library file names.
+The prefix used for (static) library file names.
+
+.IP LIBPREFIXES
+An array of legal prefixes for library file names.
.IP LIBS
A list of one or more libraries
created by this environment.
.IP LIBSUFFIX
-The suffix used for library file names.
+The suffix used for (static) library file names.
+
+.IP LIBSUFFIXES
+An array of legal suffixes for library file names.
.IP LINK
The linker.
.IP SCANNERS
A list of the available implicit dependency scanners. [CScan] by default.
+.IP SHCC
+The C compiler used for generating shared-library objects.
+
+.IP SHCCCOM
+The command line used to compile a C source file
+to a shared-library object file.
+
+.IP SHCCFLAGS
+Options that are passed to the C compiler
+to generate shared-library objects.
+
+.IP SHCXX
+The C++ compiler used for generating shared-library objects.
+
+.IP SHCXXCOM
+The command line used to compile a C++ source file
+to a shared-library object file.
+
+.IP SHCXXFLAGS
+Options that are passed to the C++ compiler
+to generate shared-library objects.
+
+.IP SHF77
+The Fortran compiler used for generating shared-library objects.
+
+.IP SHF77COM
+The command line used to compile a Fortran source file
+to a shared-library object file.
+
+.IP SHF77FLAGS
+Options that are passed to the Fortran compiler
+to generated shared-library objects.
+
+.IP SHF77PPCOM
+The command line used to compile a Fortran source file to a
+shared-library object file
+after first running the file through the C preprocessor.
+Any options specified in the $CPPFLAGS construction variable
+are included on this command line.
+
+.IP SHLIBPREFIX
+The prefix used for shared library file names.
+
+.IP SHLIBSUFFIX
+The suffix used for shared library file names.
+
+.IP SHLINK
+The linker for programs that use shared libraries.
+
+.IP SHLINKFLAGS
+General options passed to the linker for programs using shared libraries.
+
.IP TEX
The TeX formatter and typesetter.
.IP TEXFLAGS
General options passed to the TeX formatter and typesetter.
+.IP WIN32_INSERT_DEF
+When this is set to true,
+a library build of a WIN32 shared library (.dll file)
+will also build a corresponding .def file at the same time,
+if a .def file is not already listed as a build target.
+
+.IP WIN32DEFPREFIX
+The prefix used to build WIN32 .def files.
+
+.IP WIN32DEFSUFFIX
+The suffix used for WIN32 .def file names.
+
+.IP WIN32DLLPREFIX
+The prefix used to build WIN32 shared libraries (.dll files).
+
+.IP WIN32IMPLIBPREFIX
+The prefix used to build WIN32 import libraries.
+
.IP YACC
The parser generator.
configure (SConscript) file lives.
Note that you may enable and disable
this ability by calling
-.R SConscriptChdir()
+SConscriptChdir()
multiple times:
.ES
.ES
def commandhandler(cmd, args, env):
.EE
-
+.IP
.I cmd
is the path to the command to be executed.
.I args
can be extended by adding new builders to a construction
environment using the
.B Builder
-function. Builder accepts the following
-arguments:
+function.
+The
+.B Builder
+function accepts the following arguments:
.IP name
The name of the builder. This will be the
.IP action
The command line string used to build the target from the source.
.B action
-can also be a dictionary
+can also be:
+a list of strings representing the command
+to be executed and its arguments
+(suitable for enclosing white space in an argument),
+a dictionary
mapping source file name suffixes to
any combination of command line strings
(if the builder should accept multiple source file extensions),
-Python functions,
-or Action objects
-(see the next section).
+a Python function;
+an Action object
+(see the next section);
+or a list of any of the above.
.IP prefix
The prefix that will be prepended to the target file name.
+The value may also be a function call
+that returns the prefix.
+The function will be passed the environment
+and any extra keyword arguments
+supplied when the Builder is called.
.IP suffix
The suffix that will be appended to the target file name.
+The value may also be a function call
+that returns the suffix.
+The function will be passed the environment
+and any extra keyword arguments
+supplied when the Builder is called.
.IP src_suffix
The expected source file name suffix.
+The value may also be a function call
+that returns the source file name suffix.
+The function will be passed the environment
+and any extra keyword arguments
+supplied when the Builder is called.
.IP src_builder
Specifies a builder to use when a source file name suffix does not match
any of the suffixes of the builder. Using this argument produces a
multi-stage builder.
+.IP emitter
+A function that is passed the target, source, and environment,
+and which returns a tuple containing two lists,
+the list of targets to be built by this builder,
+and the list of sources for this builder.
+This allows the target and source lists to
+be manipulated before the target(s) are actually built.
+Example:
+
+.ES
+def e(target, source, env):
+ return (target + ['foo.foo'], source + ['foo.src'])
+
+b = Builder(name="Foo", emitter=e)
+.EE
+
.IP generator
-A function that returns a list of command lines that will be executed to build
-the target(s) from the source(s). The command lines must be returned as
-lists, where the first element of the list is the executable, and the other
-elements in the list are the arguments that will be passed to the
-executable. The
-.I generator
-and
-.I action
-arguments must not both be used for the same Builder. The generator function
-should take 3 arguments:
+A function that returns a list of actions that will be executed to build
+the target(s) from the source(s).
+The returned action(s) may be
+an Action object, or anything that
+can be converted into an Action object
+(see the next section).
+
+The generator function
+takes 3 arguments:
.I source
- a list of source nodes,
.I target
- the construction environment. Example:
.ES
-def g(env, source, target):
- return [["gcc", "-c", "-o"] + target + source]
+def g(source, target, env):
+ return [["gcc", "-c", "-o"] + target + source]
b = Builder(name="Object", generator=g)
.EE
+The
+.I generator
+and
+.I action
+arguments must not both be used for the same Builder.
+
+Any additional keyword arguments supplied
+when a Builder object is called
+will be associated with the target
+(and any other files built as a
+result of the call).
+
+.ES
+b = Builder(name='MyBuild', action="build < $SOURCE > $TARGET")
+env = Environment(BUILDERS = [b])
+env.MyBuild('foo.out', 'foo.in', my_arg = 'xyzzy')
+.EE
+
+These extra keyword arguments are passed to the
+following functions:
+command generator functions, funcion Actions,
+emitter functions,
+and functions that generate prefix, suffix or src_suffix.
+
.SS Action Objects
The Builder function will turn its
Action('$CC -c -o $TARGET $SOURCES')
.EE
+.IP List
+If the argument is a list,
+then a list of Action objects is returned.
+An Action object is created as necessary
+for each element in the list.
+If an element
+.I within
+the list is itself a list,
+the internal list is the
+command and arguments to be executed via
+the command line.
+This allows white space to be enclosed
+in an argument by defining
+a command in a list within a list:
+.ES
+Action([['cc', '-c', '-DWHITE SPACE', '-o', '$TARGET', '$SOURCES']])
+.EE
+
.IP Function
If the argument is a Python function,
a function Action is returned.
a = Action(build_it)
.EE
-
-.IP List
-If the argument is a list,
-then a list of Action objects is returned.
-An Action object is created as necessary
-for each element in the list.
.PP
If the action argument is not one of the above,
None is returned.
.LP
For example, given the construction variable CC='cc', targets=['foo'], and
sources=['foo.c', 'bar.c']:
+
.ES
action='$CC -c -o $TARGET $SOURCES'
.EE
+
would produce the command line:
+
.ES
cc -c -o foo foo.c bar.c
.EE
+
Variable names may be surrounded by curly braces ({})
to separate the name from the trailing characters.
Within the curly braces, a variable name may have
a Python slice subscript appended to select one
or more items from a list.
In the previous example, the string:
+
.ES
${SOURCES[1]}
.EE
+
would produce:
+
.ES
bar.c
.EE
+
Additionally, a variable name may
have the following special
modifiers appended within the enclosing curly braces
env.Program(target = 'foo', source = 'foo.c')
.EE
-.SS Building a Library
+.SS Building a Static Library
.ES
env = Environment()
-env.Library(target = 'mylib', source = 'l1.c l2.c')
+env.Library(target = 'static', source = 'l1.c l2.c')
+.EE
+
+.SS Building a Shared Library
+
+.ES
+env = Environment()
+env.Library(target = 'shared', source = 'l3.c l4.c',
+ shared = 1)
.EE
.SS Linking a Local Library Into a Program
- Internal cleanup of environment passing to function Actions.
+ - Builders can now take arbitrary keyword arguments to create
+ attributes to be passed to: command generator functions,
+ FunctionAction functions, Builder emitter functions (below),
+ and prefix/suffix generator functions (below).
+
+ - Command generator functions can now return ANYTHING that can be
+ converted into an Action (a function, a string, a CommandGenerator
+ instance, even an ActionBase instance).
+
+ - Actions now call get_contents() with the actual target and source
+ nodes used for the build.
+
+ - A new DictCmdGenerator class replaces CompositeBuilder to support
+ more flexible Builder behavior internally.
+
+ - Builders can now take an emitter= keyword argument. An emitter
+ is a function that takes target, source, and env argument, then
+ return a 2-tuple of (new sources, new targets). The emitter is
+ called when the Builder is __call__'ed, allowing a user to modify
+ source and target lists.
+
+ - The prefix, suffix and src_suffix Builder arguments now take a
+ callable as well a string. The callable is passed the Environment
+ and any extra Builder keyword arguments and is expected to return
+ the appropriate prefix or suffix.
+
+ - CommandActions can now be a string, a list of command + argument
+ strings, or a list of commands (strings or lists).
+
+ - Added shared library support. The Object and Library Builders now
+ take a "shared=1" keyword argument to specify that a shared object
+ or shared library should be built. It is an error to try to build
+ static objects into a shared library or vice versa.
+
+ - Win32 support for .def files has been added. Added the Win32-specific
+ construction variables $WIN32DEFPREFIX, $WIN32DEFSUFFIX,
+ $WIN32DLLPREFIX and $WIN32IMPLIBPREFIX. When building a .dll,
+ the new construction variable $WIN32_INSERT_DEF, controls whether
+ the appropriately-named .def file is inserted into the target
+ list (if not already present). A .lib file is always added to
+ a Library build if not present in the list of targets.
+
+ - ListBuilder now passes all targets to the action, not just the first.
+
From Steven Knight:
- Fix so that -c -n does *not* remove the targets!
import UserDict
import SCons.Util
+import SCons.Errors
print_actions = 1;
execute_actions = 1;
class CommandGenerator:
"""
- Wrappes a command generator function so the Action() factory
+ Wraps a command generator function so the Action() factory
function can tell a generator function from a function action.
"""
def __init__(self, generator):
self.generator = generator
+def _do_create_action(act):
+ """This is the actual "implementation" for the
+ Action factory method, below. This handles the
+ fact that passing lists to Action() itself has
+ different semantics than passing lists as elements
+ of lists.
+
+ The former will create a ListAction, the latter
+ will create a CommandAction by converting the inner
+ list elements to strings."""
-def Action(act):
- """A factory for action objects."""
if isinstance(act, ActionBase):
return act
+ elif SCons.Util.is_List(act):
+ return CommandAction(act)
elif isinstance(act, CommandGenerator):
return CommandGeneratorAction(act.generator)
elif callable(act):
return FunctionAction(act)
elif SCons.Util.is_String(act):
- return CommandAction(act)
- elif SCons.Util.is_List(act):
- return ListAction(act)
+ listCmds = map(lambda x: CommandAction(string.split(x)),
+ string.split(act, '\n'))
+ if len(listCmds) == 1:
+ return listCmds[0]
+ else:
+ return ListAction(listCmds)
else:
return None
+def Action(act):
+ """A factory for action objects."""
+ if SCons.Util.is_List(act):
+ acts = filter(lambda x: not x is None,
+ map(_do_create_action, act))
+ if len(acts) == 1:
+ return acts[0]
+ else:
+ return ListAction(acts)
+ else:
+ return _do_create_action(act)
+
class ActionBase:
"""Base class for actions that create output objects."""
def __cmp__(self, other):
t = [t]
try:
cwd = t[0].cwd
- except AttributeError:
+ except (IndexError, AttributeError):
pass
dict['TARGETS'] = SCons.Util.PathList(map(os.path.normpath, map(str, t)))
if dict['TARGETS']:
return dict
+def _string_from_cmd_list(cmd_list):
+ """Takes a list of command line arguments and returns a pretty
+ representation for printing."""
+ cl = []
+ for arg in cmd_list:
+ if ' ' in arg or '\t' in arg:
+ arg = '"' + arg + '"'
+ cl.append(arg)
+ return string.join(cl)
+
_rm = re.compile(r'\$[()]')
_remove = re.compile(r'\$\(([^\$]|\$[^\(])*?\$\)')
class CommandAction(ActionBase):
"""Class for command-execution actions."""
- def __init__(self, string):
- self.command = string
+ def __init__(self, cmd):
+ import SCons.Util
+
+ self.cmd_list = map(SCons.Util.to_String, cmd)
def execute(self, **kw):
dict = apply(self.subst_dict, (), kw)
import SCons.Util
- cmd_list = SCons.Util.scons_subst_list(self.command, dict, {}, _rm)
+ cmd_list = SCons.Util.scons_subst_list(self.cmd_list, dict, {}, _rm)
for cmd_line in cmd_list:
if len(cmd_line):
if print_actions:
- cl = []
- for arg in cmd_line:
- if ' ' in arg or '\t' in arg:
- arg = '"' + arg + '"'
- cl.append(arg)
- self.show(string.join(cl))
+ self.show(_string_from_cmd_list(cmd_line))
if execute_actions:
try:
ENV = kw['env']['ENV']
def get_raw_contents(self, **kw):
"""Return the complete contents of this action's command line.
"""
- return SCons.Util.scons_subst(self.command, self._sig_dict(kw), {})
+ return SCons.Util.scons_subst(string.join(self.cmd_list),
+ self._sig_dict(kw), {})
def get_contents(self, **kw):
"""Return the signature contents of this action's command line.
This strips $(-$) and everything in between the string,
since those parts don't affect signatures.
"""
- return SCons.Util.scons_subst(self.command, self._sig_dict(kw), {}, _remove)
+ return SCons.Util.scons_subst(string.join(self.cmd_list),
+ self._sig_dict(kw), {}, _remove)
class CommandGeneratorAction(ActionBase):
"""Class for command-generator actions."""
def __init__(self, generator):
self.generator = generator
- def execute(self, **kw):
- # ensure that target is a list, to make it easier to write
- # generator functions:
+ def __generate(self, kw):
import SCons.Util
- if kw.has_key("target") and not SCons.Util.is_List(kw["target"]):
- kw["target"] = [kw["target"]]
# Wrap the environment dictionary in an EnvDictProxy
# object to make variable interpolation easier for the
if args.has_key("env") and not isinstance(args["env"], EnvDictProxy):
args["env"] = EnvDictProxy(args["env"])
- gen_list = apply(self.generator, (), args)
- gen_list = map(lambda x: map(str, x), gen_list)
+ # ensure that target is a list, to make it easier to write
+ # generator functions:
+ if args.has_key("target") and not SCons.Util.is_List(args["target"]):
+ args["target"] = [args["target"]]
- # Do environment variable substitution on returned command list
- dict = apply(self.subst_dict, (), kw)
- cmd_list = [ ]
- for gen_line in gen_list:
- cmd_list.append([])
- curr_line = cmd_list[-1]
- for gen_arg in gen_line:
- arg_list = SCons.Util.scons_subst_list(gen_arg, dict, {})
- curr_line.extend(arg_list[0])
- if(len(arg_list) > 1):
- cmd_list.extend(arg_list[1:])
- curr_line = cmd_list[-1]
-
- for cmd_line in filter(lambda x: x, cmd_list):
- if print_actions:
- self.show(cmd_line)
- if execute_actions:
- try:
- ENV = kw['env']['ENV']
- except:
- import SCons.Defaults
- ENV = SCons.Defaults.ConstructionEnvironment['ENV']
- ret = spawn(cmd_line[0], cmd_line, ENV)
- if ret:
- return ret
+ ret = apply(self.generator, (), args)
+ gen_cmd = Action(ret)
+ if not gen_cmd:
+ raise SCons.Errors.UserError("Object returned from command generator: %s cannot be used to create an Action." % repr(ret))
+ return gen_cmd
- return 0
+ def execute(self, **kw):
+ return apply(self.__generate(kw).execute, (), kw)
def get_contents(self, **kw):
"""Return the signature contents of this action's command line.
This strips $(-$) and everything in between the string,
since those parts don't affect signatures.
"""
- kw['source'] = ["__s1__", "__s2__"]
- kw['target'] = ["__t1__", "__t2__"]
- cmd_list = apply(self.generator, (), kw)
- cmd_list = map(lambda x: map(str, x), cmd_list)
- cmd_list = map(lambda x: string.join(x, "\0"), cmd_list)
- cmd_list = map(lambda x: _remove.sub('', x), cmd_list)
- cmd_list = map(lambda x: filter(lambda y: y, string.split(x, "\0")), cmd_list)
- return cmd_list
+ return apply(self.__generate(kw).get_contents, (), kw)
class FunctionAction(ActionBase):
"""Class for Python function actions."""
exec "a3 = SCons.Action.Action(u'string')"
exec "assert isinstance(a3, SCons.Action.CommandAction), a3"
- a4 = SCons.Action.Action(["x", a2, "y"])
+ a4 = SCons.Action.Action(["x", "y", "z", [ "a", "b", "c"]])
assert isinstance(a4, SCons.Action.ListAction), a4
+ assert isinstance(a4.list[0], SCons.Action.CommandAction), a4.list[0]
+ assert isinstance(a4.list[1], SCons.Action.CommandAction), a4.list[1]
+ assert isinstance(a4.list[2], SCons.Action.CommandAction), a4.list[2]
+ assert isinstance(a4.list[3], SCons.Action.CommandAction), a4.list[3]
+ assert a4.list[3].cmd_list == [ "a", "b", "c" ], a4.list[3].cmd_list
a5 = SCons.Action.Action(1)
assert a5 is None, a5
a6 = SCons.Action.Action(a1)
- assert a6 is a1
+ assert a6 is a1, a6
+
+ a7 = SCons.Action.Action([[ "explicit", "command", "line" ]])
+ assert isinstance(a7, SCons.Action.CommandAction), a7
+ assert a7.cmd_list == [ "explicit", "command", "line" ], a7.cmd_list
+
+ a8 = SCons.Action.Action(["a8"])
+ assert isinstance(a8, SCons.Action.CommandAction), a8
+ assert a8.cmd_list == [ "a8" ], a8.cmd_list
class ActionBaseTestCase(unittest.TestCase):
def test_init(self):
"""Test creation of a command Action
"""
- a = SCons.Action.CommandAction("xyzzy")
- assert a.command == "xyzzy"
+ a = SCons.Action.CommandAction(["xyzzy"])
+ assert a.cmd_list == [ "xyzzy" ], a.cmd_list
def test_execute(self):
"""Test executing a command Action
return 0
SCons.Action.SetCommandHandler(func)
assert SCons.Action.spawn is func
- a = SCons.Action.CommandAction("xyzzy")
+ a = SCons.Action.CommandAction(["xyzzy"])
a.execute()
assert t.executed == 1
def test_get_raw_contents(self):
"""Test fetching the contents of a command Action
"""
- a = SCons.Action.CommandAction("| $( $foo | $bar $) |")
- c = a.get_contents(foo = 'FFF', bar = 'BBB')
+ a = SCons.Action.CommandAction(["|", "$(", "$foo", "|", "$bar",
+ "$)", "|"])
+ c = a.get_contents(target=[], source=[],
+ foo = 'FFF', bar = 'BBB')
assert c == "| $( FFF | BBB $) |"
def test_get_contents(self):
"""Test fetching the contents of a command Action
"""
- a = SCons.Action.CommandAction("| $foo $( | $) $bar |")
- c = a.get_contents(foo = 'FFF', bar = 'BBB')
- assert c == "| FFF BBB |"
+ a = SCons.Action.CommandAction(["|", "$(", "$foo", "|", "$bar",
+ "$)", "|"])
+ c = a.get_contents(target=[], source=[],
+ foo = 'FFF', bar = 'BBB')
+ assert c == "| |", c
class CommandGeneratorActionTestCase(unittest.TestCase):
assert env.subst('$FOO') == 'foo baz\nbar ack', env.subst('$FOO')
assert env.subst_list('$FOO') == [ [ 'foo', 'baz' ],
[ 'bar', 'ack' ] ], env.subst_list('$FOO')
- return [["$FOO"]]
+ return "$FOO"
+ def func_action(env, dummy, self=self):
+ assert env.subst('$foo') == 'bar', env.subst('$foo')
+ assert env.subst_list('$foo') == [ [ 'bar' ] ], env.subst_list('$foo')
+ assert env.subst_list([ '$foo', 'bar' ]) == [[ 'bar', 'bar' ]], env.subst_list([ [ '$foo', 'bar' ] ])
+ self.dummy=dummy
+ def f2(dummy, env, f=func_action):
+ return f
def ch(cmd, args, env, self=self):
self.cmd.append(cmd)
self.args.append(args)
assert self.dummy == 1
assert self.cmd == [ 'foo', 'bar'], self.cmd
assert self.args == [ [ 'foo', 'baz' ], [ 'bar', 'ack' ] ], self.args
+
+ b=SCons.Action.CommandGeneratorAction(f2)
+ self.dummy = 0
+ b.execute(dummy=2, env={ 'foo' : 'bar' })
+ assert self.dummy==2, self.dummy
del self.dummy
def test_get_contents(self):
return [["guux", foo, "$(", "ignore", "$)", bar]]
a = SCons.Action.CommandGeneratorAction(f)
- c = a.get_contents(foo = 'FFF', bar = 'BBB')
- assert c == [["guux", 'FFF', 'BBB']], c
+ c = a.get_contents(target=[], source=[],
+ foo = 'FFF', bar = 'BBB')
+ assert c == "guux FFF BBB", c
class FunctionActionTestCase(unittest.TestCase):
"""Test fetching the contents of a function Action
"""
a = SCons.Action.FunctionAction(Func)
- c = a.get_contents()
+ c = a.get_contents(target=[], source=[])
assert c == "\177\036\000\177\037\000d\000\000S", repr(c)
class ListActionTestCase(unittest.TestCase):
assert isinstance(a.list[0], SCons.Action.CommandAction)
assert isinstance(a.list[1], SCons.Action.FunctionAction)
assert isinstance(a.list[2], SCons.Action.ListAction)
- assert isinstance(a.list[2].list[0], SCons.Action.CommandAction)
- assert isinstance(a.list[2].list[1], SCons.Action.CommandAction)
+ assert a.list[2].list[0].cmd_list == [ 'y' ]
def test_execute(self):
"""Test executing a list of subsidiary Actions
"""Test fetching the contents of a list of subsidiary Actions
"""
a = SCons.Action.ListAction(["x", "y", "z"])
- c = a.get_contents()
+ c = a.get_contents(target=[], source=[])
assert c == "xyz", c
import os.path
import string
-from Errors import UserError
+import copy
+from SCons.Errors import UserError
import SCons.Action
import SCons.Node
import SCons.Node.FS
import SCons.Util
+class DictCmdGenerator:
+ """This is a callable class that can be used as a
+ command generator function. It holds on to a dictionary
+ mapping file suffixes to Actions. It uses that dictionary
+ to return the proper action based on the file suffix of
+ the source file."""
+
+ def __init__(self, action_dict):
+ self.action_dict = action_dict
+
+ def src_suffixes(self):
+ return self.action_dict.keys()
+
+ def __call__(self, source, target, env, **kw):
+ ext = None
+ for src in map(str, source):
+ my_ext = os.path.splitext(src)[1]
+ if ext and my_ext != ext:
+ raise UserError("Cannot build multiple sources with different extensions.")
+ ext = my_ext
+
+ if ext is None:
+ raise UserError("Cannot deduce file extension from source files: %s" % repr(map(str, source)))
+ try:
+ # XXX Do we need to perform Environment substitution
+ # on the keys of action_dict before looking it up?
+ return self.action_dict[ext]
+ except KeyError:
+ raise UserError("Don't know how to build a file with suffix %s." % ext)
def Builder(**kw):
"""A factory for builder objects."""
-
if kw.has_key('generator'):
if kw.has_key('action'):
raise UserError, "You must not specify both an action and a generator."
kw['action'] = SCons.Action.CommandGenerator(kw['generator'])
del kw['generator']
-
- if kw.has_key('action') and SCons.Util.is_Dict(kw['action']):
- return apply(CompositeBuilder, (), kw)
- elif kw.has_key('src_builder'):
+ elif kw.has_key('action') and SCons.Util.is_Dict(kw['action']):
+ action_dict = kw['action']
+ kw['action'] = SCons.Action.CommandGenerator(DictCmdGenerator(action_dict))
+ kw['src_suffix'] = action_dict.keys()
+
+ if kw.has_key('src_builder'):
return apply(MultiStepBuilder, (), kw)
else:
return apply(BuilderBase, (), kw)
-
-
def _init_nodes(builder, env, tlist, slist):
"""Initialize lists of target and source nodes with all of
the proper Builder information.
t.add_source(slist)
if builder.scanner:
t.target_scanner = builder.scanner
+
+class _callable_adaptor:
+ """When crteating a Builder, you can pass a string OR
+ a callable in for prefix, suffix, or src_suffix.
+ src_suffix even takes a list!
+
+ If a string or list is passed, we use this class to
+ adapt it to a callable."""
+ def __init__(self, static):
+ self.static = static
+
+ def __call__(self, **kw):
+ return self.static
+
+ def __cmp__(self, other):
+ if isinstance(other, _callable_adaptor):
+ return cmp(self.static, other.static)
+ return -1
+
+def _adjust_suffix(suff):
+ if suff and not suff[0] in [ '.', '$' ]:
+ return '.' + suff
+ return suff
class BuilderBase:
"""Base class for Builders, objects that create output
node_factory = SCons.Node.FS.default_fs.File,
target_factory = None,
source_factory = None,
- scanner = None):
+ scanner = None,
+ emitter = None):
if name is None:
raise UserError, "You must specify a name for the builder."
self.name = name
self.action = SCons.Action.Action(action)
- self.prefix = prefix
- self.suffix = suffix
- self.src_suffix = src_suffix
+ if callable(prefix):
+ self.prefix = prefix
+ else:
+ self.prefix = _callable_adaptor(str(prefix))
+
+ if callable(suffix):
+ self.suffix = suffix
+ else:
+ self.suffix = _callable_adaptor(str(suffix))
+
+ if callable(src_suffix):
+ self.src_suffix = src_suffix
+ elif SCons.Util.is_String(src_suffix):
+ self.src_suffix = _callable_adaptor([ str(src_suffix) ])
+ else:
+ self.src_suffix = _callable_adaptor(src_suffix)
+
self.target_factory = target_factory or node_factory
self.source_factory = source_factory or node_factory
self.scanner = scanner
- if self.suffix and self.suffix[0] not in '.$':
- self.suffix = '.' + self.suffix
- if self.src_suffix and self.src_suffix[0] not in '.$':
- self.src_suffix = '.' + self.src_suffix
+
+ self.emitter = emitter
def __cmp__(self, other):
return cmp(self.__dict__, other.__dict__)
- def _create_nodes(self, env, target = None, source = None):
+ def _create_nodes(self, env, args, target = None, source = None):
"""Create and return lists of target and source nodes.
"""
def adjustixes(files, pre, suf):
if pre and f[:len(pre)] != pre:
path, fn = os.path.split(os.path.normpath(f))
f = os.path.join(path, pre + fn)
- if suf:
- if f[-len(suf):] != suf:
- f = f + suf
- ret.append(f)
- return ret
-
+ # Only append a suffix if the file does not have one.
+ if suf and not os.path.splitext(f)[1]:
+ if f[-len(suf):] != suf:
+ f = f + suf
+ ret.append(f)
+ return ret
+
+ pre = self.get_prefix(env, args)
+ suf = self.get_suffix(env, args)
tlist = SCons.Node.arg2nodes(adjustixes(target,
- env.subst(self.prefix),
- env.subst(self.suffix)),
+ pre, suf),
self.target_factory)
-
+ src_suf = self.get_src_suffix(env, args)
slist = SCons.Node.arg2nodes(adjustixes(source,
None,
- env.subst(self.src_suffix)),
+ src_suf),
self.source_factory)
+ if self.emitter:
+ emit_args = { 'target' : tlist,
+ 'source' : slist,
+ 'env' : env }
+ emit_args.update(args)
+ target, source = apply(self.emitter, (), emit_args)
+
+ # Have to run it through again in case the
+ # function returns non-Node targets/sources.
+ tlist = SCons.Node.arg2nodes(adjustixes(target,
+ pre, suf),
+ self.target_factory)
+ slist = SCons.Node.arg2nodes(adjustixes(source,
+ None,
+ src_suf),
+ self.source_factory)
+
+ for t in tlist:
+ t.build_args = args
return tlist, slist
- def __call__(self, env, target = None, source = None):
- tlist, slist = self._create_nodes(env, target, source)
+ def __call__(self, env, target = None, source = None, **kw):
+ tlist, slist = self._create_nodes(env, kw, target, source)
if len(tlist) == 1:
_init_nodes(self, env, tlist, slist)
"""
return apply(self.action.get_contents, (), kw)
- def src_suffixes(self, env):
- if self.src_suffix != '':
- return [env.subst(self.src_suffix)]
- return []
+ def src_suffixes(self, env, args):
+ return map(lambda x, e=env: e.subst(_adjust_suffix(x)),
+ apply(self.src_suffix, (), args))
+
+ def get_src_suffix(self, env, args):
+ """Get the first src_suffix in the list of src_suffixes."""
+ ret = self.src_suffixes(env, args)
+ if not ret:
+ return ''
+ else:
+ return ret[0]
+
+ def get_suffix(self, env, args):
+ return env.subst(_adjust_suffix(apply(self.suffix, (), args)))
+
+ def get_prefix(self, env, args):
+ return env.subst(apply(self.prefix, (), args))
def targets(self, node):
"""Return the list of targets for this builder instance.
# unlink all targets and make all directories
# before building anything
t.prepare()
- kw['target'] = self.tlist[0]
+ kw['target'] = self.tlist
self.status = apply(self.builder.execute, (), kw)
for t in self.tlist:
if not t is kw['target']:
def get_contents(self, **kw):
return apply(self.builder.get_contents, (), kw)
- def src_suffixes(self, env):
- return self.builder.src_suffixes(env)
+ def src_suffixes(self, env, args):
+ return self.builder.src_suffixes(env, args)
def targets(self, node):
"""Return the list of targets for this builder instance.
node_factory = SCons.Node.FS.default_fs.File,
target_factory = None,
source_factory = None,
- scanner=None):
+ scanner=None,
+ emitter=None):
BuilderBase.__init__(self, name, action, prefix, suffix, src_suffix,
node_factory, target_factory, source_factory,
- scanner)
+ scanner, emitter)
+ if not SCons.Util.is_List(src_builder):
+ src_builder = [ src_builder ]
self.src_builder = src_builder
+ self.sdict = {}
- def __call__(self, env, target = None, source = None):
+ def __call__(self, env, target = None, source = None, **kw):
slist = SCons.Node.arg2nodes(source, self.source_factory)
final_sources = []
- src_suffix = env.subst(self.src_suffix)
- sdict = {}
- for suff in self.src_builder.src_suffixes(env):
- sdict[suff] = None
+
+ r=repr(env)
+ try:
+ sdict = self.sdict[r]
+ except KeyError:
+ sdict = {}
+ self.sdict[r] = sdict
+ for bld in self.src_builder:
+ for suf in bld.src_suffixes(env, kw):
+ sdict[suf] = bld
+
for snode in slist:
path, ext = os.path.splitext(snode.abspath)
if sdict.has_key(ext):
- tgt = self.src_builder(env, target = [ path ], source = snode)
+ src_bld = sdict[ext]
+
+ dictArgs = copy.copy(kw)
+ dictArgs['target'] = [path]
+ dictArgs['source'] = snode
+ dictArgs['env'] = env
+ tgt = apply(src_bld, (), dictArgs)
if not SCons.Util.is_List(tgt):
final_sources.append(tgt)
else:
final_sources.extend(tgt)
else:
final_sources.append(snode)
- return BuilderBase.__call__(self, env, target=target,
- source=final_sources)
-
- def src_suffixes(self, env):
- return BuilderBase.src_suffixes(self, env) + \
- self.src_builder.src_suffixes(env)
-
-class CompositeBuilder(BuilderBase):
- """This is a convenient Builder subclass that can build different
- files based on their suffixes. For each target, this builder
- will examine the target's sources. If they are all the same
- suffix, and that suffix is equal to one of the child builders'
- src_suffix, then that child builder will be used. Otherwise,
- UserError is thrown."""
- def __init__(self, name = None,
- prefix='',
- suffix='',
- action = {},
- src_builder = []):
- BuilderBase.__init__(self, name=name, prefix=prefix,
- suffix=suffix)
- if src_builder and not SCons.Util.is_List(src_builder):
- src_builder = [src_builder]
- self.src_builder = src_builder
- self.action_dict = action
- self.sdict = {}
- self.sbuild = {}
-
- def __call__(self, env, target = None, source = None):
- tlist, slist = BuilderBase._create_nodes(self, env,
- target=target, source=source)
-
- r = repr(env)
- if not self.sdict.has_key(r):
- self.sdict[r] = {}
- self.sbuild[r] = []
- for suff in self.src_suffixes(env):
- suff = env.subst(suff)
- self.sdict[r][suff] = suff
- self.sbuild[r].extend(filter(lambda x, e=env, s=suff:
- e.subst(x.suffix) == s,
- self.src_builder))
- for sb in self.sbuild[r]:
- suff = env.subst(sb.suffix)
- for s in sb.src_suffixes(env):
- self.sdict[r][env.subst(s)] = suff
-
- sufflist = map(lambda x, s=self.sdict[r]:
- s[os.path.splitext(x.path)[1]],
- slist)
- last_suffix = ''
- for suff in sufflist:
- if last_suffix and last_suffix != suff:
- raise UserError, "The builder for %s can only build source files of identical suffixes: %s." % \
- (tlist[0].path,
- str(map(lambda t: str(t.path), tlist[0].sources)))
- last_suffix = suff
-
- if last_suffix:
- kw = {
- 'name' : self.name,
- 'action' : self.action_dict[last_suffix],
- 'src_suffix' : last_suffix,
- }
- if self.sbuild[r]:
- sb = filter(lambda x, e=env, s=last_suffix:
- e.subst(x.suffix) == s,
- self.sbuild[r])
- if sb:
- kw['src_builder'] = sb[0]
- # XXX We should be able to cache this
- bld = apply(Builder, (), kw)
- for tnode in tlist:
- bld.__call__(env, target = tnode, source = slist)
-
- if len(tlist) == 1:
- tlist = tlist[0]
- return tlist
-
- def src_suffixes(self, env):
- suffixes = map(lambda k, e=env: e.subst(k), self.action_dict.keys()) + \
- reduce(lambda x, y: x + y,
- map(lambda b, e=env: b.src_suffixes(e),
- self.src_builder),
- [])
- return suffixes
+ dictKwArgs = kw
+ dictKwArgs['target'] = target
+ dictKwArgs['source'] = final_sources
+ return apply(BuilderBase.__call__,
+ (self, env), dictKwArgs)
+
+ def src_suffixes(self, env, args):
+ return BuilderBase.src_suffixes(self, env, args) + \
+ reduce(lambda x, y: x + y,
+ map(lambda b, e=env, args=args: b.src_suffixes(e, args),
+ self.src_builder),
+ [])
return self.d.get(s, s)
def get_scanner(self, ext):
return env_scanner
+ def Dictionary(self):
+ return {}
+ def autogenerate(self, dir=''):
+ return {}
+
env = Environment()
class BuilderTestCase(unittest.TestCase):
assert target.sources[0].name == 'n10'
assert target.sources[1].name == 'n11'
- if hasattr(types, 'UnicodeType'):
- code = """if 1:
- targets = builder(env, target = u'n12 n13', source = [u'n14 n15'])
- assert targets[0].name == u'n12'
- assert targets[0].sources[0].name == u'n14 n15'
- assert targets[1].name == u'n13'
- assert targets[1].sources[0].name == u'n14 n15'
-
- target = builder(env, target = [u'n16 n17'], source = u'n18 n19')
- assert target.name == u'n16 n17'
- assert target.sources[0].name == u'n18'
- assert target.sources[1].name == u'n19'
- \n"""
- exec code
+ if not hasattr(types, 'UnicodeType'):
+ uni = str
+ else:
+ uni = unicode
+
+ targets = builder(env, target = uni('n12 n13'),
+ source = [uni('n14 n15')])
+ assert targets[0].name == uni('n12')
+ assert targets[0].sources[0].name == uni('n14 n15')
+ assert targets[1].name == uni('n13')
+ assert targets[1].sources[0].name == uni('n14 n15')
+
+ target = builder(env, target = [uni('n16 n17')],
+ source = uni('n18 n19'))
+ assert target.name == uni('n16 n17')
+ assert target.sources[0].name == uni('n18')
+ assert target.sources[1].name == uni('n19')
def test_noname(self):
"""Test error reporting for missing name
Verify that we can retrieve the supplied action attribute.
"""
builder = SCons.Builder.Builder(name="builder", action="foo")
- assert builder.action.command == "foo"
+ assert builder.action.cmd_list == ["foo"]
def test_generator(self):
"""Test Builder creation given a generator function."""
def my_show(string):
global show_string
show_string = show_string + string + "\n"
- builder.action.show = my_show
+ for action in builder.action.list:
+ action.show = my_show
r = builder.execute()
assert r == 0
def __init__(self, **kw):
open(kw['out'], 'a').write("class2b\n")
- builder = MyBuilder(action = [cmd2, function2, class2a(), class2b], name = "clist")
+ builder = MyBuilder(action = SCons.Action.ListAction([cmd2, function2, class2a(), class2b]), name = "clist")
r = builder.execute(out = outfile)
assert r.__class__ == class2b
c = test.read(outfile, 'r')
contents = b2.get_contents()
assert contents == "\177\036\000\177\037\000d\000\000S", repr(contents)
- b3 = SCons.Builder.Builder(name = "b3", action = ["foo", Func, "bar"])
+ b3 = SCons.Builder.Builder(name = "b3", action = SCons.Action.ListAction(["foo", Func, "bar"]))
contents = b3.get_contents()
assert contents == "foo\177\036\000\177\037\000d\000\000Sbar", repr(contents)
Make sure that there is no '.' separator appended.
"""
builder = SCons.Builder.Builder(name = "builder", prefix = 'lib.')
- assert builder.prefix == 'lib.'
+ assert builder.get_prefix(env,{}) == 'lib.'
builder = SCons.Builder.Builder(name = "builder", prefix = 'lib')
- assert builder.prefix == 'lib'
+ assert builder.get_prefix(env,{}) == 'lib'
tgt = builder(env, target = 'tgt1', source = 'src1')
assert tgt.path == 'libtgt1', \
"Target has unexpected name: %s" % tgt.path
env = Environment(XSUFFIX = '.x', YSUFFIX = '.y')
b1 = SCons.Builder.Builder(name = "builder", src_suffix = '.c')
- assert b1.src_suffixes(env) == ['.c'], b1.src_suffixes(env)
+ assert b1.src_suffixes(env,{}) == ['.c'], b1.src_suffixes(env,{})
tgt = b1(env, target = 'tgt2', source = 'src2')
assert tgt.sources[0].path == 'src2.c', \
b2 = SCons.Builder.Builder(name = "b2",
src_suffix = '.2',
src_builder = b1)
- assert b2.src_suffixes(env) == ['.2', '.c'], b2.src_suffixes(env)
+ assert b2.src_suffixes(env,{}) == ['.2', '.c'], b2.src_suffixes(env,{})
b3 = SCons.Builder.Builder(name = "b3",
action = {'.3a' : '', '.3b' : ''})
- s = b3.src_suffixes(env)
+ s = b3.src_suffixes(env,{})
s.sort()
assert s == ['.3a', '.3b'], s
b4 = SCons.Builder.Builder(name = "b4", src_suffix = '$XSUFFIX')
- assert b4.src_suffixes(env) == ['.x'], b4.src_suffixes(env)
+ assert b4.src_suffixes(env,{}) == ['.x'], b4.src_suffixes(env,{})
b5 = SCons.Builder.Builder(name = "b5", action = {'$YSUFFIX' : ''})
- assert b5.src_suffixes(env) == ['.y'], b5.src_suffixes(env)
+ assert b5.src_suffixes(env,{}) == ['.y'], b5.src_suffixes(env,{})
def test_suffix(self):
"""Test Builder creation with a specified target suffix
beginning if it isn't already present.
"""
builder = SCons.Builder.Builder(name = "builder", suffix = '.o')
- assert builder.suffix == '.o'
+ assert builder.get_suffix(env,{}) == '.o', builder.get_suffix(env,{})
builder = SCons.Builder.Builder(name = "builder", suffix = 'o')
- assert builder.suffix == '.o'
+ assert builder.get_suffix(env,{}) == '.o', builder.get_suffix(env,{})
tgt = builder(env, target = 'tgt3', source = 'src3')
assert tgt.path == 'tgt3.o', \
"Target has unexpected name: %s" % tgt[0].path
def test_CompositeBuilder(self):
"""Testing CompositeBuilder class."""
+ def func_action(target, source, env):
+ return 0
+
builder = SCons.Builder.Builder(name = "builder",
- action={ '.foo' : 'foo',
- '.bar' : 'bar' })
+ action={ '.foo' : func_action,
+ '.bar' : func_action })
- assert isinstance(builder, SCons.Builder.CompositeBuilder)
+ assert isinstance(builder, SCons.Builder.BuilderBase)
+ assert isinstance(builder.action, SCons.Action.CommandGeneratorAction)
tgt = builder(env, target='test1', source='test1.foo')
assert isinstance(tgt.builder, SCons.Builder.BuilderBase)
- assert tgt.builder.action.command == 'foo'
- tgt = builder(env, target='test2', source='test2.bar')
- assert tgt.builder.action.command == 'bar'
+ assert isinstance(tgt.builder.action.generator, SCons.Builder.DictCmdGenerator)
flag = 0
+ tgt = builder(env, target='test2', source='test2.bar test1.foo')
try:
- tgt = builder(env, target='test2', source='test2.bar test1.foo')
- except SCons.Errors.UserError:
+ tgt.build()
+ except SCons.Errors.BuildError, e:
+ assert e.args[0] == SCons.Errors.UserError
flag = 1
assert flag, "UserError should be thrown when we build targets with files of different suffixes."
action = { '.foo' : 'foo',
'.bar' : 'bar' },
src_builder = foo_bld)
- assert isinstance(builder, SCons.Builder.CompositeBuilder)
+ assert isinstance(builder, SCons.Builder.MultiStepBuilder)
+ assert isinstance(builder.action, SCons.Action.CommandGeneratorAction)
tgt = builder(env, target='t1', source='t1a.ina t1b.ina')
assert isinstance(tgt.builder, SCons.Builder.BuilderBase)
action = { '.foo' : 'foo',
'.bar' : 'bar' },
src_builder = [foo_bld, bar_bld])
- assert isinstance(builder, SCons.Builder.CompositeBuilder)
+ assert isinstance(builder, SCons.Builder.MultiStepBuilder)
+ assert isinstance(builder.action, SCons.Action.CommandGeneratorAction)
tgt = builder(env, target='t3-foo', source='t3a.foo t3b.ina')
assert isinstance(tgt.builder, SCons.Builder.MultiStepBuilder)
assert isinstance(tgt.builder, SCons.Builder.MultiStepBuilder)
flag = 0
+ tgt = builder(env, target='t5', source='test5a.foo test5b.inb')
try:
- tgt = builder(env, target='t5', source='test5a.foo test5b.inb')
- except SCons.Errors.UserError:
+ tgt.build()
+ except SCons.Errors.BuildError, e:
+ assert e.args[0] == SCons.Errors.UserError
flag = 1
assert flag, "UserError should be thrown when we build targets with files of different suffixes."
flag = 0
+ tgt = builder(env, target='t6', source='test6a.bar test6b.ina')
try:
- tgt = builder(env, target='t6', source='test6a.bar test6b.ina')
- except SCons.Errors.UserError:
+ tgt.build()
+ except SCons.Errors.BuildError, e:
+ assert e.args[0] == SCons.Errors.UserError
flag = 1
assert flag, "UserError should be thrown when we build targets with files of different suffixes."
flag = 0
+ tgt = builder(env, target='t4', source='test4a.ina test4b.inb')
try:
- tgt = builder(env, target='t4', source='test4a.ina test4b.inb')
- except SCons.Errors.UserError:
+ tgt.build()
+ except SCons.Errors.BuildError, e:
+ assert e.args[0] == SCons.Errors.UserError
flag = 1
assert flag, "UserError should be thrown when we build targets with files of different suffixes."
+
def test_build_scanner(self):
"""Testing ability to set a target scanner through a builder."""
global instanced
assert tgt.target_scanner != env_scanner, tgt.target_scanner
assert src.source_scanner == env_scanner
+ def test_Builder_Args(self):
+ """Testing passing extra agrs to a builder."""
+ def buildFunc(target, source, env, foo, bar, s=self):
+ s.foo=foo
+ s.bar=bar
+
+ builder = SCons.Builder.Builder(name="builder", action=buildFunc)
+ tgt = builder(env, target='foo', source='bar', foo=1, bar=2)
+ tgt.build()
+ assert self.foo == 1, self.foo
+ assert self.bar == 2, self.bar
+
+ def test_emitter(self):
+ """Test emitter functions."""
+ def emit(target, source, env, foo=0, bar=0):
+ if foo:
+ target.append("bar")
+ if bar:
+ source.append("foo")
+ return ( target, source )
+
+ builder = SCons.Builder.Builder(name="builder", action='foo',
+ emitter=emit)
+ tgt = builder(env, target='foo', source='bar')
+ assert str(tgt) == 'foo', str(tgt)
+ assert str(tgt.sources[0]) == 'bar', str(tgt.sources[0])
+
+ tgt = builder(env, target='foo', source='bar', foo=1)
+ assert len(tgt) == 2, len(tgt)
+ assert 'foo' in map(str, tgt), map(str, tgt)
+ assert 'bar' in map(str, tgt), map(str, tgt)
+
+ tgt = builder(env, target='foo', source='bar', bar=1)
+ assert str(tgt) == 'foo', str(tgt)
+ assert len(tgt.sources) == 2, len(tgt.sources)
+ assert 'foo' in map(str, tgt.sources), map(str, tgt.sources)
+ assert 'bar' in map(str, tgt.sources), map(str, tgt.sources)
+
if __name__ == "__main__":
suite = unittest.makeSuite(BuilderTestCase, 'test_')
if not unittest.TextTestRunner().run(suite).wasSuccessful():
import stat
import string
import sys
+import os.path
import SCons.Action
import SCons.Builder
import SCons.Scanner.Prog
import SCons.Util
-
-
+class SharedCmdGenerator:
+ """A callable class that acts as a command generator.
+ It is designed to hold on to 2 actions, and return
+ one if the shared=1 keyword arg is supplied to the
+ Builder method, and the other if not.
+
+ Also, all target nodes will have the shared attribute
+ set to match the vaue of the shared keyword argument,
+ zero by default."""
+ def __init__(self, static, shared):
+ self.action_static = static
+ self.action_shared = shared
+
+ def __call__(self, target, source, env, shared=0):
+ for src in source:
+ try:
+ if src.attributes.shared != shared:
+ raise UserError("Source file: %s must be built with shared=%s in order to be compatible with the selected target." % (src, str(shared)))
+ except AttributeError:
+ pass
+ for t in target:
+ t.attributes.shared = shared
+ if shared:
+ return self.action_shared
+ else:
+ return self.action_static
+
CFile = SCons.Builder.Builder(name = 'CFile',
action = { '.l' : '$LEXCOM',
'.y' : '$YACCCOM',
},
suffix = '$CXXFILESUFFIX')
-CPlusPlusAction = SCons.Action.Action('$CXXCOM')
-
-FortranAction = SCons.Action.Action('$F77COM')
-
-FortranPPAction = SCons.Action.Action('$F77PPCOM')
-
+CXXAction = SCons.Action.Action("$CXXCOM")
+ShCXXAction = SCons.Action.Action("$SHCXXCOM")
+F77Action = SCons.Action.Action("$F77COM")
+ShF77Action = SCons.Action.Action("$SHF77COM")
+F77PPAction = SCons.Action.Action("$F77PPCOM")
+ShF77PPAction = SCons.Action.Action("$SHF77PPCOM")
+
+shared_obj = SCons.Builder.DictCmdGenerator({ ".C" : ShCXXAction,
+ ".cc" : ShCXXAction,
+ ".cpp" : ShCXXAction,
+ ".cxx" : ShCXXAction,
+ ".c++" : ShCXXAction,
+ ".C++" : ShCXXAction,
+ ".c" : "$SHCCCOM",
+ ".f" : ShF77Action,
+ ".for" : ShF77Action,
+ ".FOR" : ShF77Action,
+ ".F" : ShF77PPAction,
+ ".fpp" : ShF77PPAction,
+ ".FPP" : ShF77PPAction })
+
+static_obj = SCons.Builder.DictCmdGenerator({ ".C" : CXXAction,
+ ".cc" : CXXAction,
+ ".cpp" : CXXAction,
+ ".cxx" : CXXAction,
+ ".c++" : CXXAction,
+ ".C++" : CXXAction,
+ ".c" : "$CCCOM",
+ ".f" : F77Action,
+ ".for" : F77Action,
+ ".F" : F77PPAction,
+ ".FOR" : F77Action,
+ ".fpp" : F77PPAction,
+ ".FPP" : F77PPAction })
+
Object = SCons.Builder.Builder(name = 'Object',
- action = { '.c' : '$CCCOM',
- '.C' : CPlusPlusAction,
- '.cc' : CPlusPlusAction,
- '.cpp' : CPlusPlusAction,
- '.cxx' : CPlusPlusAction,
- '.c++' : CPlusPlusAction,
- '.C++' : CPlusPlusAction,
- '.f' : FortranAction,
- '.for' : FortranAction,
- '.FOR' : FortranAction,
- '.F' : FortranPPAction,
- '.fpp' : FortranPPAction,
- '.FPP' : FortranPPAction,
- },
+ generator = \
+ SharedCmdGenerator(static=SCons.Action.CommandGeneratorAction(static_obj),
+ shared=SCons.Action.CommandGeneratorAction(shared_obj)),
prefix = '$OBJPREFIX',
suffix = '$OBJSUFFIX',
+ src_suffix = static_obj.src_suffixes(),
src_builder = [CFile, CXXFile])
Program = SCons.Builder.Builder(name = 'Program',
src_builder = Object,
scanner = SCons.Scanner.Prog.ProgScan())
-Library = SCons.Builder.Builder(name = 'Library',
- action = '$ARCOM',
- prefix = '$LIBPREFIX',
- suffix = '$LIBSUFFIX',
- src_suffix = '$OBJSUFFIX',
- src_builder = Object)
+class LibAffixGenerator:
+ def __init__(self, static, shared):
+ self.static_affix = static
+ self.shared_affix = shared
+
+ def __call__(self, shared=0, win32=0):
+ if shared:
+ return self.shared_affix
+ return self.static_affix
+
+def win32LibGenerator(target, source, env, shared=1):
+ listCmd = [ "$SHLINK", "$SHLINKFLAGS" ]
+
+ for tgt in target:
+ ext = os.path.splitext(str(tgt))[1]
+ if ext == env.subst("$LIBSUFFIX"):
+ # Put it on the command line as an import library.
+ listCmd.append("${WIN32IMPLIBPREFIX}%s" % tgt)
+ else:
+ listCmd.append("${WIN32DLLPREFIX}%s" % tgt)
+
+ listCmd.extend([ '$_LIBDIRFLAGS', '$_LIBFLAGS' ])
+ for src in source:
+ ext = os.path.splitext(str(src))[1]
+ if ext == env.subst("$WIN32DEFSUFFIX"):
+ # Treat this source as a .def file.
+ listCmd.append("${WIN32DEFPREFIX}%s" % src)
+ else:
+ # Just treat it as a generic source file.
+ listCmd.append(str(src))
+ return listCmd
+
+def win32LibEmitter(target, source, env, shared=0):
+ if shared:
+ dll = None
+ for tgt in target:
+ ext = os.path.splitext(str(tgt))[1]
+ if ext == env.subst("$SHLIBSUFFIX"):
+ dll = tgt
+ break
+ if not dll:
+ raise UserError("A shared library should have exactly one target with the suffix: %s" % env.subst("$SHLIBSUFFIX"))
+
+ if env.has_key("WIN32_INSERT_DEF") and \
+ env["WIN32_INSERT_DEF"] and \
+ not '.def' in map(lambda x: os.path.split(str(x))[1],
+ source):
+
+ # append a def file to the list of sources
+ source.append("%s%s" % (os.path.splitext(str(dll))[0],
+ env.subst("$WIN32DEFSUFFIX")))
+ if not env.subst("$LIBSUFFIX") in \
+ map(lambda x: os.path.split(str(x))[1], target):
+ # Append an import library to the list of targets.
+ target.append("%s%s%s" % (env.subst("$LIBPREFIX"),
+ os.path.splitext(str(dll))[0],
+ env.subst("$LIBSUFFIX")))
+ return (target, source)
+
+PosixLibrary = SCons.Builder.Builder(name = 'Library',
+ generator = \
+ SharedCmdGenerator(shared="$SHLINKCOM",
+ static="$ARCOM"),
+ prefix = \
+ LibAffixGenerator(static='$LIBPREFIX',
+ shared='$SHLIBPREFIX'),
+ suffix = \
+ LibAffixGenerator(static='$LIBSUFFIX',
+ shared='$SHLIBSUFFIX'),
+ src_suffix = '$OBJSUFFIX',
+ src_builder = Object)
+
+Win32Library = SCons.Builder.Builder(name = 'Library',
+ generator = \
+ SharedCmdGenerator(shared=SCons.Action.CommandGeneratorAction(win32LibGenerator),
+ static="$ARCOM"),
+ emitter = win32LibEmitter,
+ prefix = \
+ LibAffixGenerator(static='$LIBPREFIX',
+ shared='$SHLIBPREFIX'),
+ suffix = \
+ LibAffixGenerator(static='$LIBSUFFIX',
+ shared='$SHLIBSUFFIX'),
+ src_suffix = '$OBJSUFFIX',
+ src_builder = Object)
LaTeXAction = SCons.Action.Action('$LATEXCOM')
'CC' : 'cl',
'CCFLAGS' : '/nologo',
'CCCOM' : '$CC $CCFLAGS $CPPFLAGS $_INCFLAGS /c $SOURCES /Fo$TARGET',
+ 'SHCC' : '$CC',
+ 'SHCCFLAGS' : '$CCFLAGS',
+ 'SHCCCOM' : '$SHCC $SHCCFLAGS $CPPFLAGS $_INCFLAGS /c $SOURCES /Fo$TARGET',
'CFILESUFFIX' : '.c',
'CXX' : '$CC',
'CXXFLAGS' : '$CCFLAGS',
'CXXCOM' : '$CXX $CXXFLAGS $CPPFLAGS $_INCFLAGS /c $SOURCES /Fo$TARGET',
+ 'SHCXX' : '$CXX',
+ 'SHCXXFLAGS' : '$CXXFLAGS',
+ 'SHCXXCOM' : '$SHCXX $SHCXXFLAGS $CPPFLAGS $_INCFLAGS /c $SOURCES /Fo$TARGET',
'CXXFILESUFFIX' : '.cc',
'F77' : 'g77',
'F77FLAGS' : '',
'F77COM' : '$F77 $F77FLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
'F77PPCOM' : '$F77 $F77FLAGS $CPPFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
+ 'SHF77' : '$F77',
+ 'SHF77FLAGS' : '$F77FLAGS',
+ 'SHF77COM' : '$SHF77 $SHF77FLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
+ 'SHF77PPCOM' : '$SHF77 $SHF77FLAGS $CPPFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
'LINK' : 'link',
'LINKFLAGS' : '/nologo',
'LINKCOM' : '$LINK $LINKFLAGS /OUT:$TARGET $_LIBDIRFLAGS $_LIBFLAGS $SOURCES',
+ 'SHLINK' : '$LINK',
+ 'SHLINKFLAGS': '$LINKFLAGS /dll',
+ 'SHLINKCOM' : '$SHLINK $SHLINKFLAGS /OUT:$TARGET $_LIBDIRFLAGS $_LIBFLAGS $SOURCES',
'AR' : 'lib',
'ARFLAGS' : '/nologo',
'ARCOM' : '$AR $ARFLAGS /OUT:$TARGET $SOURCES',
+ 'SHLIBPREFIX': '',
+ 'SHLIBSUFFIX': '.dll',
'LEX' : 'lex',
'LEXFLAGS' : '',
'LEXCOM' : '$LEX $LEXFLAGS -t $SOURCES > $TARGET',
'PSCOM' : '$DVIPS $DVIPSFLAGS -o $TARGET $SOURCES',
'PSPREFIX' : '',
'PSSUFFIX' : '.ps',
- 'BUILDERS' : [Alias, CFile, CXXFile, DVI, Library, Object,
+ 'BUILDERS' : [Alias, CFile, CXXFile, DVI, Win32Library, Object,
PDF, PostScript, Program],
'SCANNERS' : [CScan],
'OBJPREFIX' : '',
'PROGPREFIX' : '',
'PROGSUFFIX' : '.exe',
'LIBPREFIX' : '',
+ 'LIBPREFIXES': '$LIBPREFIX',
'LIBSUFFIX' : '.lib',
+ 'LIBSUFFIXES': '$LIBSUFFIX',
'LIBDIRPREFIX' : '/LIBPATH:',
'LIBDIRSUFFIX' : '',
'LIBLINKPREFIX' : '',
'LIBLINKSUFFIX' : '$LIBSUFFIX',
'INCPREFIX' : '/I',
'INCSUFFIX' : '',
+ 'WIN32DEFPREFIX' : '/def:',
+ 'WIN32DEFSUFFIX' : '.def',
+ 'WIN32DLLPREFIX' : '/out:',
+ 'WIN32IMPLIBPREFIX' : '/implib:',
+ 'WIN32_INSERT_DEF' : 1,
'ENV' : {
'INCLUDE' : include,
'LIB' : lib,
if os.name == 'posix':
-
+ Library = PosixLibrary
+
arcom = '$AR $ARFLAGS $TARGET $SOURCES'
ranlib = 'ranlib'
if SCons.Util.WhereIs(ranlib):
'CC' : 'cc',
'CCFLAGS' : '',
'CCCOM' : '$CC $CCFLAGS $CPPFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
+ 'SHCC' : '$CC',
+ 'SHCCFLAGS' : '$CCFLAGS -fPIC',
+ 'SHCCCOM' : '$SHCC $SHCCFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
'CFILESUFFIX' : '.c',
'CXX' : 'c++',
'CXXFLAGS' : '$CCFLAGS',
'CXXCOM' : '$CXX $CXXFLAGS $CPPFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
'CXXFILESUFFIX' : '.cc',
+ 'SHCXX' : '$CXX',
+ 'SHCXXFLAGS' : '$CXXFLAGS -fPIC',
+ 'SHCXXCOM' : '$SHCXX $SHCXXFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
'F77' : 'g77',
'F77FLAGS' : '',
'F77COM' : '$F77 $F77FLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
'F77PPCOM' : '$F77 $F77FLAGS $CPPFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
+ 'SHF77FLAGS' : '$F77FLAGS -fPIC',
+ 'SHF77COM' : '$F77 $SHF77FLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
+ 'SHF77PPCOM' : '$F77 $SHF77FLAGS $CPPFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
+ 'SHF77' : '$F77',
+ 'SHF77FLAGS' : '$F77FLAGS -fPIC',
+ 'SHF77COM' : '$SHF77 $SHF77FLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
+ 'SHF77PPCOM' : '$SHF77 $SHF77FLAGS $CPPFLAGS $_INCFLAGS -c -o $TARGET $SOURCES',
'LINK' : '$CXX',
'LINKFLAGS' : '',
'LINKCOM' : '$LINK $LINKFLAGS -o $TARGET $SOURCES $_LIBDIRFLAGS $_LIBFLAGS',
+ 'SHLINK' : '$LINK',
+ 'SHLINKFLAGS': '$LINKFLAGS -shared',
+ 'SHLINKCOM' : '$SHLINK $SHLINKFLAGS -o $TARGET $SOURCES $_LIBDIRFLAGS $_LIBFLAGS',
'AR' : 'ar',
'ARFLAGS' : 'r',
'RANLIB' : ranlib,
'RANLIBFLAGS' : '',
'ARCOM' : arcom,
+ 'SHLIBPREFIX': '$LIBPREFIX',
+ 'SHLIBSUFFIX': '.so',
'LEX' : 'lex',
'LEXFLAGS' : '',
'LEXCOM' : '$LEX $LEXFLAGS -t $SOURCES > $TARGET',
'PSCOM' : '$DVIPS $DVIPSFLAGS -o $TARGET $SOURCES',
'PSPREFIX' : '',
'PSSUFFIX' : '.ps',
- 'BUILDERS' : [Alias, CFile, CXXFile, DVI, Library, Object,
+ 'BUILDERS' : [Alias, CFile, CXXFile, DVI, PosixLibrary, Object,
PDF, PostScript, Program],
'SCANNERS' : [CScan],
'OBJPREFIX' : '',
'PROGPREFIX' : '',
'PROGSUFFIX' : (sys.platform == 'cygwin') and '.exe' or '',
'LIBPREFIX' : 'lib',
+ 'LIBPREFIXES': '$LIBPREFIX',
'LIBSUFFIX' : '.a',
+ 'LIBSUFFIXES': [ '$LIBSUFFIX', '$SHLIBSUFFIX' ],
'LIBDIRPREFIX' : '-L',
'LIBDIRSUFFIX' : '',
'LIBLINKPREFIX' : '-l',
}
elif os.name == 'nt':
+ Library = Win32Library
+
versions = None
try:
versions = get_devstudio_versions()
self.env = env
self.builder = builder
- def __call__(self, target = None, source = None):
- return self.builder(self.env, target, source)
+ def __call__(self, target = None, source = None, **kw):
+ return apply(self.builder, (self.env, target, source),
+ kw)
# This allows a Builder to be executed directly
# through the Environment to which it's attached.
def __delitem__(self, key):
del self._dict[key]
+ def has_key(self, key):
+ return self._dict.has_key(key)
+
def Command(self, target, source, action):
"""Builds the supplied target files from the supplied
source files using the supplied action. Action may
"""
return SCons.Util.scons_subst(string, self._dict, {})
+ def subst_list(self, string):
+ """Calls through to SCons.Util.scons_subst_list(). See
+ the documentation for that function."""
+ return SCons.Util.scons_subst_list(string, self._dict, {})
+
def get_scanner(self, skey):
"""Find the appropriate scanner given a key (usually a file suffix).
Does a linear search. Could be sped up by creating a dictionary if
action='buildfoo $target $source')
assert t.builder
assert t.builder.action.__class__.__name__ == 'CommandAction'
- assert t.builder.action.command == 'buildfoo $target $source'
+ assert t.builder.action.cmd_list == ['buildfoo', '$target', '$source']
assert 'foo1.in' in map(lambda x: x.path, t.sources)
assert 'foo2.in' in map(lambda x: x.path, t.sources)
str = env.subst("$AAA ${AAA}A ${AAA}B $BBB")
assert str == "c c", str
+ env = Environment(AAA = '$BBB', BBB = '$CCC', CCC = [ 'a', 'b\nc' ])
+ lst = env.subst_list([ "$AAA", "B $CCC" ])
+ assert lst == [ [ "a", "b" ], [ "c", "B a", "b" ], [ "c" ] ], lst
+
def test_autogenerate(dict):
"""Test autogenerating variables in a dictionary."""
env = Environment(LIBS = [ 'foo', 'bar', 'baz' ],
class Builder:
def execute(self, **kw):
- global built_it, built_target, built_source
- built_it = 1
+ global built_it, built_target, built_source, built_args
+ built_it = 1
built_target = kw['target']
built_source = kw['source']
+ built_args = kw
return 0
- def get_contents(self, env):
+ def get_contents(self, env, target, source):
return 7
class NoneBuilder(Builder):
node.env_set(Environment())
node.path = "qqq"
node.sources = ["rrr", "sss"]
+ node.build_args = { "foo" : 1, "bar" : 2 }
node.build()
assert built_it
assert type(built_target) == type(MyNode()), type(built_target)
assert str(built_target) == "qqq", str(built_target)
assert built_source == ["rrr", "sss"], built_source
+ assert built_args["foo"] == 1, built_args
+ assert built_args["bar"] == 2, built_args
fff = MyNode()
ggg = MyNode()
build, or use to build other Nodes.
"""
+ class Attrs:
+ pass
+
def __init__(self):
self.sources = [] # source files used to build node
self.depends = [] # explicit dependencies (from Depends)
self.precious = None
self.found_includes = {}
self.includes = None
+ self.build_args = {}
+ self.attributes = self.Attrs() # Generic place to stick information about the Node.
+
+ def generate_build_args(self):
+ dict = copy.copy(self.env.Dictionary())
+ if hasattr(self, 'dir'):
+ auto = self.env.autogenerate(dir = self.dir)
+ else:
+ auto = self.env.autogenerate()
+ dict.update(auto)
+
+ dictArgs = { 'env' : dict,
+ 'target' : self,
+ 'source' : self.sources }
+ dictArgs.update(self.build_args)
+ return dictArgs
def build(self):
"""Actually build the node. Return the status from the build."""
stat = self.builder.status
except AttributeError:
try:
- dict = copy.copy(self.env.Dictionary())
- if hasattr(self, 'dir'):
- auto = self.env.autogenerate(dir = self.dir)
- else:
- auto = self.env.autogenerate()
- dict.update(auto)
- stat = self.builder.execute(env = dict,
- target = self,
- source = self.sources)
+ stat = apply(self.builder.execute, (),
+ self.generate_build_args())
except:
raise BuildError(self, "Exception",
sys.exc_type,
def __init__(self, node):
self.node = node
def get_contents(self):
- dict = self.node.env.Dictionary()
- dict.update(self.node.env.autogenerate())
- try:
- dir = self.node.getcwd()
- except AttributeError:
- dir = None
- return self.node.builder.get_contents(env = dict)
+ return apply(self.node.builder.get_contents, (),
+ self.node.generate_build_args())
return Adapter(self)
def get_implicit_deps(self, env, scanner, target):
libs = string.split(libs)
try:
- prefix = env.Dictionary('LIBPREFIX')
+ prefix = env.Dictionary('LIBPREFIXES')
+ if not SCons.Util.is_List(prefix):
+ prefix = [ prefix ]
except KeyError:
- prefix = ''
+ prefix = [ '' ]
try:
- suffix = env.Dictionary('LIBSUFFIX')
+ suffix = env.Dictionary('LIBSUFFIXES')
+ if not SCons.Util.is_List(suffix):
+ suffix = [ suffix ]
except KeyError:
- suffix = ''
+ suffix = [ '' ]
- libs = map(lambda x, s=suffix, p=prefix: p + x + s, libs)
- return SCons.Node.FS.find_files(libs, libpath, fs.File)
+ ret = []
+ for suf in map(env.subst, suffix):
+ for pref in map(env.subst, prefix):
+ ret.extend(map(lambda x, s=suf, p=pref: p + x + s, libs))
+ return SCons.Node.FS.find_files(ret, libpath, fs.File)
class DummyEnvironment:
def __init__(self, **kw):
self._dict = kw
- self._dict['LIBSUFFIX'] = '.lib'
+ self._dict['LIBSUFFIXES'] = '.lib'
def Dictionary(self, *args):
if not args:
def __delitem__(self,key):
del self.Dictionary()[key]
+ def subst(self, s):
+ return s
+
def deps_match(deps, libs):
deps=map(str, deps)
deps.sort()
def signature(obj):
"""Generate a signature for an object
"""
- try:
- contents = str(obj.get_contents())
- except AttributeError, e:
- raise AttributeError, \
- "unable to fetch contents of '%s': %s" % (str(obj), e)
- return hexdigest(md5.new(contents).digest())
+ if not hasattr(obj, 'get_contents'):
+ raise AttributeError, "unable to fetch contents of '%s'" % str(obj)
+ return hexdigest(md5.new(str(obj.get_contents())).digest())
def to_string(signature):
"""Convert a signature to a string"""
try:
signature('string')
except AttributeError, e:
- # the error string should begin with "unable to fetch contents of 'string': "
- assert string.find(str(e), "unable to fetch contents of 'string':") == 0
+ assert string.find(str(e), "unable to fetch contents") == 0, str(e)
else:
raise AttributeError, "unexpected get_contents() attribute"
the argv array that should be passed to a spawn or exec
function.
+ Also, this method can accept a list of strings as input
+ to strSubst, which explicitly denotes the command line
+ arguments. This is useful if you want to pass in
+ command line arguments with spaces or newlines in them.
+ Otheriwise, if you just passed in a string, they would
+ get split along the spaces and newlines.
+
One important thing this guy does is preserve environment
variables that are lists. For instance, if you have
an environment variable that is a Python list (or UserList-
if e is None:
s = ''
elif is_List(e):
- s = string.join(map(str, e), '\0')
+ s = string.join(map(to_String, e), '\0')
else:
- s = _space_sep.sub('\0', str(e))
+ s = _space_sep.sub('\0', to_String(e))
except NameError:
s = ''
return s
n = 1
- # Tokenize the original string...
- strSubst = _space_sep.sub('\0', str(strSubst))
+ if is_List(strSubst):
+ # This looks like our input is a list of strings,
+ # as explained in the docstring above. Munge
+ # it into a tokenized string by concatenating
+ # the list with nulls.
+ strSubst = string.join(strSubst, '\0')
+ else:
+ # Tokenize the original string...
+ strSubst = _space_sep.sub('\0', to_String(strSubst))
# Now, do the substitution
while n != 0:
def is_List(e):
return type(e) is types.ListType or isinstance(e, UserList.UserList)
+def to_String(s):
+ """Better than str() because it will preserve a unicode
+ object without converting it to ASCII."""
+ if is_String(s):
+ return s
+ else:
+ return str(s)
+
def argmunge(arg):
"""This function converts a string or list into a list of strings or Nodes.
It follows the rules outlined in the SCons design document by accepting
assert newcom == cvt("test %s/foo/bar.exe"%os.getcwd()), newcom
newcom = scons_subst("test ${SOURCES.abspath}", loc, {})
- assert newcom == cvt("test %s/foo/blah.cpp /bar/ack.cpp %s/foo/ack.c"%(os.getcwd(),os.path.normpath(os.getcwd()+"/.."))), newcom
+ assert newcom == cvt("test %s/foo/blah.cpp %s %s/foo/ack.c"%(os.getcwd(),
+ os.path.abspath(os.path.normpath("/bar/ack.cpp")),
+ os.path.normpath(os.getcwd()+"/.."))), newcom
newcom = scons_subst("test ${SOURCE.abspath}", loc, {})
assert newcom == cvt("test %s/foo/blah.cpp"%os.getcwd()), newcom
assert cmd_list[1][0] == 'after', cmd_list[1][0]
assert cmd_list[0][2] == cvt('../foo/ack.cbefore'), cmd_list[0][2]
+ # Test inputting a list to scons_subst_list()
+ cmd_list = scons_subst_list([ "$SOURCES$NEWLINE", "$TARGETS",
+ "This is a test" ],
+ loc, {})
+ assert len(cmd_list) == 2, len(cmd_list)
+ assert cmd_list[0][0] == cvt('foo/blah with spaces.cpp'), cmd_list[0][0]
+ assert cmd_list[1][0] == cvt("after"), cmd_list[1]
+ assert cmd_list[1][4] == "This is a test", cmd_list[1]
+
glob = { 'a' : 1, 'b' : 2 }
loc = {'a' : 3, 'c' : 4 }
cmd_list = scons_subst_list("test $a $b $c $d test", glob, loc)
assert len(cmd_list) == 1, cmd_list
assert cmd_list[0] == ['test', '3', '2', '4', 'test'], cmd_list
+
+
+
def test_render_tree(self):
class Node:
def __init__(self, name, children=[]):
assert not is_String({})
assert not is_String([])
+ def test_to_String(self):
+ """Test the to_String() method."""
+ assert to_String(1) == "1", to_String(1)
+ assert to_String([ 1, 2, 3]) == str([1, 2, 3]), to_String([1,2,3])
+ assert to_String("foo") == "foo", to_String("foo")
+
+ try:
+ import UserString
+
+ s1=UserString.UserString('blah')
+ assert to_String(s1) is s1, s1
+ assert to_String(s1) == 'blah', s1
+
+ class Derived(UserString.UserString):
+ pass
+ s2 = Derived('foo')
+ assert to_String(s2) is s2, s2
+ assert to_String(s2) == 'foo', s2
+
+ if hasattr(types, 'UnicodeType'):
+ s3=UserString.UserString(unicode('bar'))
+ assert to_String(s3) is s3, s3
+ assert to_String(s3) == unicode('bar'), s3
+ except ImportError:
+ pass
+
+ if hasattr(types, 'UnicodeType'):
+ s4 = unicode('baz')
+ assert to_String(s4) == unicode('baz'), to_String(s4)
+
def test_WhereIs(self):
test = TestCmd.TestCmd(workdir = '')
env.Command(target = 'f2.out', source = 'f2.in',
action = r'%s' + " build.py temp2 $SOURCES\\n" + r'%s' + " build.py $TARGET temp2")
env.Command(target = 'f3.out', source = 'f3.in',
- action = [r'%s build.py temp3 $SOURCES',
- r'%s build.py $TARGET temp3'])
+ action = [ [ r'%s', 'build.py', 'temp3', '$SOURCES' ],
+ [ r'%s', 'build.py', '$TARGET', 'temp3'] ])
""" % (python, python, python, python))
test.write('f1.in', "f1.in\n")
stdout = "f1.c 1\nprog.c\n")
test.run(program = prog2,
stdout = "f1.c 1\nprog.c\n")
-
-test.up_to_date(arguments = '.')
-
+#test.up_to_date(arguments = '.')
# Change LIBPATH and make sure we don't rebuild because of it.
test.write('SConstruct', """
env1 = Environment(LIBS = [ 'foo1' ],
env2.Library(target = 'foo2', source = 'f1.c')
""")
-test.up_to_date(arguments = '.', stderr = None)
+test.up_to_date(arguments = '.', stderr=None)
test.write('f1.c', r"""
void
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import sys
+import TestSCons
+
+if sys.platform == 'win32':
+ _lib = '.lib'
+else:
+ _lib = '.a'
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+env = Environment(LIBPREFIX = 'xxx-',
+ LIBPREFIXES = ['xxx-'])
+lib = env.Library(target = 'foo', source = 'foo.c')
+env.Program(target = 'prog', source = ['prog.c', lib])
+""")
+
+test.write('foo.c', r"""
+void
+foo(void)
+{
+ printf("foo.c\n");
+}
+""")
+
+test.write('prog.c', r"""
+void foo(void);
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ foo();
+ printf("prog.c\n");
+ return 0;
+}
+""")
+
+test.run(arguments = '.')
+
+test.fail_test(not os.path.exists(test.workpath('xxx-foo' + _lib)))
+
+test.run(program = test.workpath('prog'), stdout = "foo.c\nprog.c\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import sys
+import TestSCons
+
+if sys.platform == 'win32':
+ lib_ = ''
+else:
+ lib_ = 'lib'
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+env = Environment(LIBSUFFIX = '.xxx',
+ LIBSUFFIXES = ['.xxx'])
+lib = env.Library(target = 'foo', source = 'foo.c')
+env.Program(target = 'prog', source = ['prog.c', lib])
+""")
+
+test.write('foo.c', r"""
+void
+foo(void)
+{
+ printf("foo.c\n");
+}
+""")
+
+test.write('prog.c', r"""
+void foo(void);
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ foo();
+ printf("prog.c\n");
+ return 0;
+}
+""")
+
+test.run(arguments = '.')
+
+test.fail_test(not os.path.exists(test.workpath(lib_ + 'foo.xxx')))
+
+test.run(program = test.workpath('prog'), stdout = "foo.c\nprog.c\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import string
+import sys
+import TestSCons
+
+python = sys.executable
+
+if sys.platform == 'win32':
+ _exe = '.exe'
+else:
+ _exe = ''
+
+test = TestSCons.TestSCons()
+
+test.write("wrapper.py",
+"""import os
+import string
+import sys
+open('%s', 'wb').write("wrapper.py\\n")
+os.system(string.join(sys.argv[1:], " "))
+""" % string.replace(test.workpath('wrapper.out'), '\\', '\\\\'))
+
+test.write('SConstruct', """
+foo = Environment()
+shcc = foo.Dictionary('SHCC')
+bar = Environment(SHCC = r'%s wrapper.py ' + shcc)
+foo.Program(target = 'foo', source = 'foo.c', shared = 1)
+bar.Program(target = 'bar', source = 'bar.c', shared = 1)
+""" % python)
+
+test.write('foo.c', r"""
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("foo.c\n");
+ exit (0);
+}
+""")
+
+test.write('bar.c', r"""
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("foo.c\n");
+ exit (0);
+}
+""")
+
+
+test.run(arguments = 'foo' + _exe)
+
+test.fail_test(os.path.exists(test.workpath('wrapper.out')))
+
+test.run(arguments = 'bar' + _exe)
+
+test.fail_test(test.read('wrapper.out') != "wrapper.py\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import sys
+import TestSCons
+
+if sys.platform == 'win32':
+ _obj = '.obj'
+ fooflags = '/nologo -DFOO'
+ barflags = '/nologo -DBAR'
+else:
+ _obj = '.o'
+ fooflags = '-DFOO'
+ barflags = '-DBAR'
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+foo = Environment(SHCCFLAGS = '%s')
+bar = Environment(SHCCFLAGS = '%s')
+foo.Object(target = 'foo%s', source = 'prog.c', shared = 1)
+bar.Object(target = 'bar%s', source = 'prog.c', shared = 1)
+foo.Program(target = 'foo', source = 'foo%s', shared = 1)
+bar.Program(target = 'bar', source = 'bar%s', shared = 1)
+""" % (fooflags, barflags, _obj, _obj, _obj, _obj))
+
+test.write('prog.c', r"""
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+#ifdef FOO
+ printf("prog.c: FOO\n");
+#endif
+#ifdef BAR
+ printf("prog.c: BAR\n");
+#endif
+ exit (0);
+}
+""")
+
+
+test.run(arguments = '.')
+
+test.run(program = test.workpath('foo'), stdout = "prog.c: FOO\n")
+test.run(program = test.workpath('bar'), stdout = "prog.c: BAR\n")
+
+test.write('SConstruct', """
+bar = Environment(SHCCFLAGS = '%s')
+bar.Object(target = 'foo%s', source = 'prog.c', shared = 1)
+bar.Object(target = 'bar%s', source = 'prog.c', shared = 1)
+bar.Program(target = 'foo', source = 'foo%s', shared = 1)
+bar.Program(target = 'bar', source = 'bar%s', shared = 1)
+""" % (barflags, _obj, _obj, _obj, _obj))
+
+test.run(arguments = '.')
+
+test.run(program = test.workpath('foo'), stdout = "prog.c: BAR\n")
+test.run(program = test.workpath('bar'), stdout = "prog.c: BAR\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import string
+import sys
+import TestSCons
+
+python = sys.executable
+
+if sys.platform == 'win32':
+ _exe = '.exe'
+else:
+ _exe = ''
+
+test = TestSCons.TestSCons()
+
+test.write("wrapper.py",
+"""import os
+import string
+import sys
+open('%s', 'wb').write("wrapper.py\\n")
+os.system(string.join(sys.argv[1:], " "))
+""" % string.replace(test.workpath('wrapper.out'), '\\', '\\\\'))
+
+test.write('SConstruct', """
+foo = Environment()
+shcxx = foo.Dictionary('SHCXX')
+bar = Environment(SHCXX = r'%s wrapper.py ' + shcxx)
+foo.Program(target = 'foo', source = 'foo.cc', shared = 1)
+bar.Program(target = 'bar', source = 'bar.cc', shared = 1)
+""" % python)
+
+test.write('foo.cc', r"""
+#include <stdio.h>
+#include <stdlib.h>
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("foo.c\n");
+ exit (0);
+}
+""")
+
+test.write('bar.cc', r"""
+#include <stdio.h>
+#include <stdlib.h>
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("foo.c\n");
+ exit (0);
+}
+""")
+
+
+test.run(arguments = 'foo' + _exe)
+
+test.fail_test(os.path.exists(test.workpath('wrapper.out')))
+
+test.run(arguments = 'bar' + _exe)
+
+test.fail_test(test.read('wrapper.out') != "wrapper.py\n")
+
+test.pass_test()
--- /dev/null
+
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import sys
+import TestSCons
+
+if sys.platform == 'win32':
+ _obj = '.obj'
+ fooflags = '/nologo -DFOO'
+ barflags = '/nologo -DBAR'
+else:
+ _obj = '.o'
+ fooflags = '-DFOO'
+ barflags = '-DBAR'
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+foo = Environment(SHCXXFLAGS = '%s')
+bar = Environment(SHCXXFLAGS = '%s')
+foo.Object(target = 'foo%s', source = 'prog.cc', shared = 1)
+bar.Object(target = 'bar%s', source = 'prog.cc', shared = 1)
+foo.Program(target = 'foo', source = 'foo%s', shared = 1)
+bar.Program(target = 'bar', source = 'bar%s', shared = 1)
+""" % (fooflags, barflags, _obj, _obj, _obj, _obj))
+
+test.write('prog.cc', r"""
+#include <stdio.h>
+#include <stdlib.h>
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+#ifdef FOO
+ printf("prog.c: FOO\n");
+#endif
+#ifdef BAR
+ printf("prog.c: BAR\n");
+#endif
+ exit (0);
+}
+""")
+
+
+test.run(arguments = '.')
+
+test.run(program = test.workpath('foo'), stdout = "prog.c: FOO\n")
+test.run(program = test.workpath('bar'), stdout = "prog.c: BAR\n")
+
+test.write('SConstruct', """
+bar = Environment(SHCXXFLAGS = '%s')
+bar.Object(target = 'foo%s', source = 'prog.cc', shared = 1)
+bar.Object(target = 'bar%s', source = 'prog.cc', shared = 1)
+bar.Program(target = 'foo', source = 'foo%s', shared = 1)
+bar.Program(target = 'bar', source = 'bar%s', shared = 1)
+""" % (barflags, _obj, _obj, _obj, _obj))
+
+test.run(arguments = '.')
+
+test.run(program = test.workpath('foo'), stdout = "prog.c: BAR\n")
+test.run(program = test.workpath('bar'), stdout = "prog.c: BAR\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import string
+import sys
+import TestSCons
+
+python = sys.executable
+
+if sys.platform == 'win32':
+ _exe = '.exe'
+else:
+ _exe = ''
+
+test = TestSCons.TestSCons()
+
+
+
+if sys.platform == 'win32':
+
+ test.write('mylink.py', r"""
+import getopt
+import os
+import sys
+args = sys.argv[1:]
+while args:
+ a = args[0]
+ if a[0] != '/':
+ break
+ args.pop(0)
+ if a[:5] == '/OUT:': out = a[5:]
+infile = open(args[0], 'rb')
+outfile = open(out, 'wb')
+for l in infile.readlines():
+ if l[:5] != '#link':
+ outfile.write(l)
+sys.exit(0)
+""")
+
+else:
+
+ test.write('mylink.py', r"""
+import getopt
+import os
+import sys
+opts, args = getopt.getopt(sys.argv[1:], 'o:')
+for opt, arg in opts:
+ if opt == '-o': out = arg
+infile = open(args[0], 'rb')
+outfile = open(out, 'wb')
+for l in infile.readlines():
+ if l[:5] != '#link':
+ outfile.write(l)
+sys.exit(0)
+""")
+
+test.write('myg77.py', r"""
+import getopt
+import os
+import sys
+opts, args = getopt.getopt(sys.argv[1:], 'cf:o:')
+for opt, arg in opts:
+ if opt == '-o': out = arg
+infile = open(args[0], 'rb')
+outfile = open(out, 'wb')
+for l in infile.readlines():
+ if l[:4] != '#g77':
+ outfile.write(l)
+sys.exit(0)
+""")
+
+test.write('SConstruct', """
+env = Environment(LINK = r'%s mylink.py',
+ SHF77 = r'%s myg77.py')
+env.Program(target = 'test1', source = 'test1.f', shared = 1)
+env.Program(target = 'test2', source = 'test2.for', shared = 1)
+env.Program(target = 'test3', source = 'test3.FOR', shared = 1)
+env.Program(target = 'test4', source = 'test4.F', shared = 1)
+env.Program(target = 'test5', source = 'test5.fpp', shared = 1)
+env.Program(target = 'test6', source = 'test6.FPP', shared = 1)
+""" % (python, python))
+
+test.write('test1.f', r"""This is a .f file.
+#g77
+#link
+""")
+
+test.write('test2.for', r"""This is a .for file.
+#g77
+#link
+""")
+
+test.write('test3.FOR', r"""This is a .FOR file.
+#g77
+#link
+""")
+
+test.write('test4.F', r"""This is a .F file.
+#g77
+#link
+""")
+
+test.write('test5.fpp', r"""This is a .fpp file.
+#g77
+#link
+""")
+
+test.write('test6.FPP', r"""This is a .FPP file.
+#g77
+#link
+""")
+
+test.run(arguments = '.', stderr = None)
+
+test.fail_test(test.read('test1' + _exe) != "This is a .f file.\n")
+
+test.fail_test(test.read('test2' + _exe) != "This is a .for file.\n")
+
+test.fail_test(test.read('test3' + _exe) != "This is a .FOR file.\n")
+
+test.fail_test(test.read('test4' + _exe) != "This is a .F file.\n")
+
+test.fail_test(test.read('test5' + _exe) != "This is a .fpp file.\n")
+
+test.fail_test(test.read('test6' + _exe) != "This is a .FPP file.\n")
+
+
+
+g77 = None
+for dir in string.split(os.environ['PATH'], os.pathsep):
+ g = os.path.join(dir, 'g77' + _exe)
+ if os.path.exists(g):
+ g77 = g
+ break
+
+if g77:
+
+ test.write("wrapper.py",
+"""import os
+import string
+import sys
+open('%s', 'wb').write("wrapper.py\\n")
+os.system(string.join(sys.argv[1:], " "))
+""" % string.replace(test.workpath('wrapper.out'), '\\', '\\\\'))
+
+ test.write('SConstruct', """
+foo = Environment(LIBS = 'g2c')
+shf77 = foo.Dictionary('SHF77')
+bar = foo.Copy(SHF77 = r'%s wrapper.py ' + shf77)
+foo.Program(target = 'foo', source = 'foo.f', shared = 1)
+bar.Program(target = 'bar', source = 'bar.f', shared = 1)
+""" % python)
+
+ test.write('foo.f', r"""
+ PROGRAM FOO
+ PRINT *,'foo.f'
+ STOP
+ END
+""")
+
+ test.write('bar.f', r"""
+ PROGRAM BAR
+ PRINT *,'bar.f'
+ STOP
+ END
+""")
+
+
+ test.run(arguments = 'foo' + _exe, stderr = None)
+
+ test.run(program = test.workpath('foo'), stdout = " foo.f\n")
+
+ test.fail_test(os.path.exists(test.workpath('wrapper.out')))
+
+ test.run(arguments = 'bar' + _exe)
+
+ test.run(program = test.workpath('bar'), stdout = " bar.f\n")
+
+ test.fail_test(test.read('wrapper.out') != "wrapper.py\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import string
+import sys
+import TestSCons
+
+python = sys.executable
+
+if sys.platform == 'win32':
+ _exe = '.exe'
+else:
+ _exe = ''
+
+test = TestSCons.TestSCons()
+
+
+
+if sys.platform == 'win32':
+
+ test.write('mylink.py', r"""
+import getopt
+import os
+import sys
+args = sys.argv[1:]
+while args:
+ a = args[0]
+ if a[0] != '/':
+ break
+ args.pop(0)
+ if a[:5] == '/OUT:': out = a[5:]
+infile = open(args[0], 'rb')
+outfile = open(out, 'wb')
+for l in infile.readlines():
+ if l[:5] != '#link':
+ outfile.write(l)
+sys.exit(0)
+""")
+
+else:
+
+ test.write('mylink.py', r"""
+import getopt
+import os
+import sys
+opts, args = getopt.getopt(sys.argv[1:], 'o:')
+for opt, arg in opts:
+ if opt == '-o': out = arg
+infile = open(args[0], 'rb')
+outfile = open(out, 'wb')
+for l in infile.readlines():
+ if l[:5] != '#link':
+ outfile.write(l)
+sys.exit(0)
+""")
+
+test.write('myg77.py', r"""
+import getopt
+import os
+import sys
+opts, args = getopt.getopt(sys.argv[1:], 'co:x')
+optstring = ''
+for opt, arg in opts:
+ if opt == '-o': out = arg
+ else: optstring = optstring + ' ' + opt
+infile = open(args[0], 'rb')
+outfile = open(out, 'wb')
+outfile.write(optstring + "\n")
+for l in infile.readlines():
+ if l[:4] != '#g77':
+ outfile.write(l)
+sys.exit(0)
+""")
+
+test.write('SConstruct', """
+env = Environment(LINK = r'%s mylink.py',
+ SHF77 = r'%s myg77.py', SHF77FLAGS = '-x')
+env.Program(target = 'test1', source = 'test1.f', shared = 1)
+env.Program(target = 'test2', source = 'test2.for', shared = 1)
+env.Program(target = 'test3', source = 'test3.FOR', shared = 1)
+env.Program(target = 'test4', source = 'test4.F', shared = 1)
+env.Program(target = 'test5', source = 'test5.fpp', shared = 1)
+env.Program(target = 'test6', source = 'test6.FPP', shared = 1)
+""" % (python, python))
+
+test.write('test1.f', r"""This is a .f file.
+#g77
+#link
+""")
+
+test.write('test2.for', r"""This is a .for file.
+#g77
+#link
+""")
+
+test.write('test3.FOR', r"""This is a .FOR file.
+#g77
+#link
+""")
+
+test.write('test4.F', r"""This is a .F file.
+#g77
+#link
+""")
+
+test.write('test5.fpp', r"""This is a .fpp file.
+#g77
+#link
+""")
+
+test.write('test6.FPP', r"""This is a .FPP file.
+#g77
+#link
+""")
+
+test.run(arguments = '.', stderr = None)
+
+test.fail_test(test.read('test1' + _exe) != " -x -c\nThis is a .f file.\n")
+
+test.fail_test(test.read('test2' + _exe) != " -x -c\nThis is a .for file.\n")
+
+test.fail_test(test.read('test3' + _exe) != " -x -c\nThis is a .FOR file.\n")
+
+test.fail_test(test.read('test4' + _exe) != " -x -c\nThis is a .F file.\n")
+
+test.fail_test(test.read('test5' + _exe) != " -x -c\nThis is a .fpp file.\n")
+
+test.fail_test(test.read('test6' + _exe) != " -x -c\nThis is a .FPP file.\n")
+
+
+
+g77 = None
+for dir in string.split(os.environ['PATH'], os.pathsep):
+ g = os.path.join(dir, 'g77' + _exe)
+ if os.path.exists(g):
+ g77 = g
+ break
+
+if g77:
+
+ test.write("wrapper.py",
+"""import os
+import string
+import sys
+open('%s', 'wb').write("wrapper.py\\n")
+os.system(string.join(sys.argv[1:], " "))
+""" % string.replace(test.workpath('wrapper.out'), '\\', '\\\\'))
+
+ test.write('SConstruct', """
+foo = Environment(LIBS = 'g2c')
+shf77 = foo.Dictionary('SHF77')
+bar = foo.Copy(SHF77 = r'%s wrapper.py ' + shf77, SHF77FLAGS = '-Ix')
+foo.Program(target = 'foo', source = 'foo.f', shared = 1)
+bar.Program(target = 'bar', source = 'bar.f', shared = 1)
+""" % python)
+
+ test.write('foo.f', r"""
+ PROGRAM FOO
+ PRINT *,'foo.f'
+ STOP
+ END
+""")
+
+ test.write('bar.f', r"""
+ PROGRAM BAR
+ PRINT *,'bar.f'
+ STOP
+ END
+""")
+
+
+ test.run(arguments = 'foo' + _exe, stderr = None)
+
+ test.run(program = test.workpath('foo'), stdout = " foo.f\n")
+
+ test.fail_test(os.path.exists(test.workpath('wrapper.out')))
+
+ test.run(arguments = 'bar' + _exe)
+
+ test.run(program = test.workpath('bar'), stdout = " bar.f\n")
+
+ test.fail_test(test.read('wrapper.out') != "wrapper.py\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import sys
+import TestSCons
+
+if sys.platform == 'win32':
+ _lib = '.dll'
+else:
+ _lib = '.so'
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+env = Environment(SHLIBPREFIX = 'shlib-')
+env.Library(target = 'foo', source = 'foo.c', shared = 1)
+""")
+
+test.write('foo.c', r"""
+void
+foo(void)
+{
+ printf("foo.c\n");
+}
+""")
+
+test.run(arguments = '.')
+
+test.fail_test(not os.path.exists(test.workpath('shlib-foo' + _lib)))
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import sys
+import TestSCons
+
+if sys.platform == 'win32':
+ lib_ = ''
+else:
+ lib_ = 'lib'
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+env = Environment(SHLIBSUFFIX = '.shlib')
+env.Library (target = 'foo', source = 'foo.c', shared = 1)
+""")
+
+test.write('foo.c', r"""
+void
+foo(void)
+{
+ printf("foo.c\n");
+}
+""")
+
+test.run(arguments = '.')
+
+test.fail_test(not os.path.exists(test.workpath(lib_ + 'foo.shlib')))
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import string
+import sys
+import TestSCons
+
+python = sys.executable
+
+if sys.platform == 'win32':
+ _exe = '.exe'
+else:
+ _exe = ''
+
+test = TestSCons.TestSCons()
+
+test.pass_test() #XXX Until someone can take a look and fix this.
+
+test.write("wrapper.py",
+"""import os
+import string
+import sys
+open('%s', 'wb').write("wrapper.py\\n")
+os.system(string.join(sys.argv[1:], " "))
+""" % string.replace(test.workpath('wrapper.out'), '\\', '\\\\'))
+
+test.write('SConstruct', """
+foo = Environment()
+shlink = foo.Dictionary('SHLINK')
+bar = Environment(SHLINK = r'%s wrapper.py ' + shlink)
+foo.Program(target = 'foo', source = 'foo.c', shared = 1)
+bar.Program(target = 'bar', source = 'bar.c', shared = 1)
+""" % python)
+
+test.write('foo.c', r"""
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("foo.c\n");
+ exit (0);
+}
+""")
+
+test.write('bar.c', r"""
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("foo.c\n");
+ exit (0);
+}
+""")
+
+
+test.run(arguments = 'foo' + _exe)
+
+test.fail_test(os.path.exists(test.workpath('wrapper.out')))
+
+test.run(arguments = 'bar' + _exe)
+
+test.fail_test(test.read('wrapper.out') != "wrapper.py\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import os
+import string
+import sys
+import TestSCons
+
+python = sys.executable
+
+if sys.platform == 'win32':
+ _exe = '.exe'
+else:
+ _exe = ''
+
+test = TestSCons.TestSCons()
+
+test.pass_test() #XXX Until someone can take a look and fix this.
+
+test.write("wrapper.py",
+"""import os
+import string
+import sys
+open('%s', 'wb').write("wrapper.py\\n")
+os.system(string.join(sys.argv[1:], " "))
+""" % string.replace(test.workpath('wrapper.out'), '\\', '\\\\'))
+
+test.write('SConstruct', """
+foo = Environment()
+shlink = foo.Dictionary('SHLINK')
+bar = Environment(SHLINK = '', SHLINKFLAGS = r'%s wrapper.py ' + shlink)
+foo.Program(target = 'foo', source = 'foo.c', shared = 1)
+bar.Program(target = 'bar', source = 'bar.c', shared = 1)
+""" % python)
+
+test.write('foo.c', r"""
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("foo.c\n");
+ exit (0);
+}
+""")
+
+test.write('bar.c', r"""
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ printf("foo.c\n");
+ exit (0);
+}
+""")
+
+
+test.run(arguments = 'foo' + _exe)
+
+test.fail_test(os.path.exists(test.workpath('wrapper.out')))
+
+test.run(arguments = 'bar' + _exe)
+
+test.fail_test(test.read('wrapper.out') != "wrapper.py\n")
+
+test.pass_test()
--- /dev/null
+#!/usr/bin/env python
+#
+# Copyright (c) 2001, 2002 Steven Knight
+#
+# Permission is hereby granted, free of charge, to any person obtaining
+# a copy of this software and associated documentation files (the
+# "Software"), to deal in the Software without restriction, including
+# without limitation the rights to use, copy, modify, merge, publish,
+# distribute, sublicense, and/or sell copies of the Software, and to
+# permit persons to whom the Software is furnished to do so, subject to
+# the following conditions:
+#
+# The above copyright notice and this permission notice shall be included
+# in all copies or substantial portions of the Software.
+#
+# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
+# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
+# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+#
+
+__revision__ = "__FILE__ __REVISION__ __DATE__ __DEVELOPER__"
+
+import TestSCons
+import os
+
+test = TestSCons.TestSCons()
+
+test.write('SConstruct', """
+env=Environment()
+env2 = Environment(LIBS = [ 'foo1', 'foo2', 'foo3' ],
+ LIBPATH = [ '.' ])
+env.Library(target = 'foo1', source = 'f1.c', shared=1)
+env.Library(target = 'foo2', source = 'f2a.c f2b.c f2c.c', shared=1)
+env.Library(target = 'foo3', source = ['f3a.c', 'f3b.c', 'f3c.c'], shared=1)
+env2.Program(target = 'prog', source = 'prog.c')
+""")
+
+test.write('f1.c', r"""
+#include <stdio.h>
+
+void
+f1(void)
+{
+ printf("f1.c\n");
+ fflush(stdout);
+}
+""")
+
+test.write("foo1.def", r"""
+LIBRARY "foo1"
+DESCRIPTION "Foo1 Shared Library"
+
+EXPORTS
+ f1
+""")
+
+test.write('f2a.c', r"""
+void
+f2a(void)
+{
+ printf("f2a.c\n");
+}
+""")
+
+test.write('f2b.c', r"""
+void
+f2b(void)
+{
+ printf("f2b.c\n");
+}
+""")
+
+test.write('f2c.c', r"""
+#include <stdio.h>
+
+void
+f2c(void)
+{
+ printf("f2c.c\n");
+ fflush(stdout);
+}
+""")
+
+test.write("foo2.def", r"""
+LIBRARY "foo2"
+DESCRIPTION "Foo2 Shared Library"
+
+EXPORTS
+ f2a
+ f2b
+ f2c
+""")
+
+test.write('f3a.c', r"""
+void
+f3a(void)
+{
+ printf("f3a.c\n");
+}
+""")
+
+test.write('f3b.c', r"""
+void
+f3b(void)
+{
+ printf("f3b.c\n");
+}
+""")
+
+test.write('f3c.c', r"""
+#include <stdio.h>
+
+f3c(void)
+{
+ printf("f3c.c\n");
+ fflush(stdout);
+}
+""")
+
+test.write("foo3.def", r"""
+LIBRARY "foo3"
+DESCRIPTION "Foo3 Shared Library"
+
+EXPORTS
+ f3a
+ f3b
+ f3c
+""")
+
+test.write('prog.c', r"""
+void f1(void);
+void f2a(void);
+void f2b(void);
+void f2c(void);
+void f3a(void);
+void f3b(void);
+void f3c(void);
+int
+main(int argc, char *argv[])
+{
+ argv[argc++] = "--";
+ f1();
+ f2a();
+ f2b();
+ f2c();
+ f3a();
+ f3b();
+ f3c();
+ printf("prog.c\n");
+ return 0;
+}
+""")
+
+test.run(arguments = '.')
+
+if os.name == 'posix':
+ os.environ['LD_LIBRARY_PATH'] = '.'
+test.run(program = test.workpath('prog'),
+ stdout = "f1.c\nf2a.c\nf2b.c\nf2c.c\nf3a.c\nf3b.c\nf3c.c\nprog.c\n")
+
+test.pass_test()
""")
test.write('SConstruct', """
-B1 = Builder(name = 'B1', action = [r'%s build.py .temp $SOURCES',
- r'%s build.py $TARGETS .temp'])
+B1 = Builder(name = 'B1', action = [ [ r'%s', 'build.py', '.temp', '$SOURCES' ],
+ [ r'%s', 'build.py', '$TARGETS', '.temp'] ])
B2 = Builder(name = 'B2', action = r'%s' + " build.py .temp $SOURCES\\n" + r'%s' + " build.py $TARGETS .temp")
env = Environment(BUILDERS = [B1, B2])
env.B1(target = 'foo1.out', source = 'foo1.in')