From: stevenknight Date: Thu, 6 Feb 2003 05:17:25 +0000 (+0000) Subject: Checkin of in-progress work on the User Guide. X-Git-Url: http://git.tremily.us/?a=commitdiff_plain;h=e5d338c1695b959891c35f90ed4e10baf1b24c2b;p=scons.git Checkin of in-progress work on the User Guide. git-svn-id: http://scons.tigris.org/svn/scons/trunk@575 fdb21ef1-2011-0410-befe-b5e4ea1792b1 --- diff --git a/doc/scons.mod b/doc/scons.mod index b156e4ce..642f979a 100644 --- a/doc/scons.mod +++ b/doc/scons.mod @@ -170,3 +170,14 @@ construction variables"> Dictionary"> + + + +hello"> +hello.c"> +hello.h"> +stdio.h"> diff --git a/doc/user/actions.sgml b/doc/user/actions.sgml new file mode 100644 index 00000000..447b331e --- /dev/null +++ b/doc/user/actions.sgml @@ -0,0 +1,240 @@ + + + + + + + X + + + +
+ X + + + + X + + + +
diff --git a/doc/user/alias.sgml b/doc/user/alias.sgml new file mode 100644 index 00000000..33c236ac --- /dev/null +++ b/doc/user/alias.sgml @@ -0,0 +1,45 @@ + + + + + + + X + + + +
+ Alias Targets + + + + X + + + +
diff --git a/doc/user/builders.sgml b/doc/user/builders.sgml index 2cba10ed..ac5f1937 100644 --- a/doc/user/builders.sgml +++ b/doc/user/builders.sgml @@ -22,6 +22,79 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + X diff --git a/doc/user/caching.sgml b/doc/user/caching.sgml index 3ddfd32b..5403c43e 100644 --- a/doc/user/caching.sgml +++ b/doc/user/caching.sgml @@ -21,6 +21,17 @@ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +--> + + diff --git a/doc/user/command.sgml b/doc/user/command.sgml new file mode 100644 index 00000000..abb3a581 --- /dev/null +++ b/doc/user/command.sgml @@ -0,0 +1,73 @@ + + + + + + + X + + + +
+ The &Command; Method + + + + X + + + +
diff --git a/doc/user/cons.pl b/doc/user/cons.pl new file mode 100644 index 00000000..8afbfecd --- /dev/null +++ b/doc/user/cons.pl @@ -0,0 +1,720 @@ +=head1 Introduction + +B is a system for constructing, primarily, software, but is quite +different from previous software construction systems. Cons was designed +from the ground up to deal easily with the construction of software spread +over multiple source directories. Cons makes it easy to create build scripts +that are simple, understandable and maintainable. Cons ensures that complex +software is easily and accurately reproducible. + +Cons uses a number of techniques to accomplish all of this. Construction +scripts are just Perl scripts, making them both easy to comprehend and very +flexible. Global scoping of variables is replaced with an import/export +mechanism for sharing information between scripts, significantly improving +the readability and maintainability of each script. B are introduced: these are Perl objects that capture the +information required for controlling the build process. Multiple +environments are used when different semantics are required for generating +products in the build tree. Cons implements automatic dependency analysis +and uses this to globally sequence the entire build. Variant builds are +easily produced from a single source tree. Intelligent build subsetting is +possible, when working on localized changes. Overrides can be setup to +easily override build instructions without modifying any scripts. MD5 +cryptographic B are associated with derived files, and are used +to accurately determine whether a given file needs to be rebuilt. + +While offering all of the above, and more, Cons remains simple and easy to +use. This will, hopefully, become clear as you read the remainder of this +document. + + + +=head2 Automatic global build sequencing + +Because Cons does full and accurate dependency analysis, and does this +globally, for the entire build, Cons is able to use this information to take +full control of the B of the build. This sequencing is evident +in the above examples, and is equivalent to what you would expect for make, +given a full set of dependencies. With Cons, this extends trivially to +larger, multi-directory builds. As a result, all of the complexity involved +in making sure that a build is organized correctly--including multi-pass +hierarchical builds--is eliminated. We'll discuss this further in the next +sections. + + + +=head1 A Model for sharing files + + +=head2 Some simple conventions + +In any complex software system, a method for sharing build products needs to +be established. We propose a simple set of conventions which are trivial to +implement with Cons, but very effective. + +The basic rule is to require that all build products which need to be shared +between directories are shared via an intermediate directory. We have +typically called this F, and, in a C environment, provided +conventional sub-directories of this directory, such as F, F, +F, etc. + +These directories are defined by the top-level F file. A simple +F file for a B application, organized using +multiple directories, might look like this: + + # Construct file for Hello, World! + + # Where to put all our shared products. + $EXPORT = '#export'; + + Export qw( CONS INCLUDE LIB BIN ); + + # Standard directories for sharing products. + $INCLUDE = "$EXPORT/include"; + $LIB = "$EXPORT/lib"; + $BIN = "$EXPORT/bin"; + + # A standard construction environment. + $CONS = new cons ( + CPPPATH => $INCLUDE, # Include path for C Compilations + LIBPATH => $LIB, # Library path for linking programs + LIBS => '-lworld', # List of standard libraries + ); + + Build qw( + hello/Conscript + world/Conscript + ); + +The F directory's F file looks like this: + + # Conscript file for directory world + Import qw( CONS INCLUDE LIB ); + + # Install the products of this directory + Install $CONS $LIB, 'libworld.a'; + Install $CONS $INCLUDE, 'world.h'; + + # Internal products + Library $CONS 'libworld.a', 'world.c'; + +and the F directory's F file looks like this: + + # Conscript file for directory hello + Import qw( CONS BIN ); + + # Exported products + Install $CONS $BIN, 'hello'; + + # Internal products + Program $CONS 'hello', 'hello.c'; + +To construct a B program with this directory structure, go to +the top-level directory, and invoke C with the appropriate +arguments. In the following example, we tell Cons to build the directory +F. To build a directory, Cons recursively builds all known products +within that directory (only if they need rebuilding, of course). If any of +those products depend upon other products in other directories, then those +will be built, too. + + % cons export + Install world/world.h as export/include/world.h + cc -Iexport/include -c hello/hello.c -o hello/hello.o + cc -Iexport/include -c world/world.c -o world/world.o + ar r world/libworld.a world/world.o + ar: creating world/libworld.a + ranlib world/libworld.a + Install world/libworld.a as export/lib/libworld.a + cc -o hello/hello hello/hello.o -Lexport/lib -lworld + Install hello/hello as export/bin/hello + + +=head2 Clean, understandable, location-independent scripts + +You'll note that the two F files are very clean and +to-the-point. They simply specify products of the directory and how to build +those products. The build instructions are minimal: they specify which +construction environment to use, the name of the product, and the name of +the inputs. Note also that the scripts are location-independent: if you wish +to reorganize your source tree, you are free to do so: you only have to +change the F file (in this example), to specify the new locations +of the F files. The use of an export tree makes this goal easy. + +Note, too, how Cons takes care of little details for you. All the F +directories, for example, were made automatically. And the installed files +were really hard-linked into the respective export directories, to save +space and time. This attention to detail saves considerable work, and makes +it even easier to produce simple, maintainable scripts. + + + +=head1 Signatures + +Cons uses file B to decide if a derived file is out-of-date +and needs rebuilding. In essence, if the contents of a file change, +or the manner in which the file is built changes, the file's signature +changes as well. This allows Cons to decide with certainty when a file +needs rebuilding, because Cons can detect, quickly and reliably, whether +any of its dependency files have been changed. + + +=head2 MD5 content and build signatures + +Cons uses the B (B) algorithm to compute file +signatures. The MD5 algorithm computes a strong cryptographic checksum +for any given input string. Cons can, based on configuration, use two +different MD5 signatures for a given file: + +The B of a file is an MD5 checksum of the file's +contents. Consequently, when the contents of a file change, its content +signature changes as well. + +The B of a file is a combined MD5 checksum of: + +=over 4 + +the signatures of all the input files used to build the file + +the signatures of all dependency files discovered by source scanners +(for example, C<.h> files) + +the signatures of all dependency files specified explicitly via the +C method) + +the command-line string used to build the file + +=back + +The build signature is, in effect, a digest of all the dependency +information for the specified file. Consequently, a file's build +signature changes whenever any part of its dependency information +changes: a new file is added, the contents of a file on which it depends +change, there's a change to the command line used to build the file (or +any of its dependency files), etc. + +For example, in the previous section, the build signature of the +F file will include: + +=over 4 + +the signature of the F file + +the signatures of any header files that Cons detects are included, +directly or indirectly, by F + +the text of the actual command line was used to generate F + +=back + +Similarly, the build signature of the F file will include +all the signatures of its constituents (and hence, transitively, the +signatures of B constituents), as well as the command line that +created the file. + +Note that there is no need for a derived file to depend upon any +particular F or F file. If changes to these files +affect a file, then this will be automatically reflected in its build +signature, since relevant parts of the command line are included in the +signature. Unrelated F or F changes will have no +effect. + + +=head2 Storing signatures in .consign files + +Before Cons exits, it stores the calculated signatures for all of the +files it built or examined in F<.consign> files, one per directory. +Cons uses this stored information on later invocations to decide if +derived files need to be rebuilt. + +After the previous example was compiled, the F<.consign> file in the +F directory looked like this: + + world.h:985533370 - d181712f2fdc07c1f05d97b16bfad904 + world.o:985533372 2a0f71e0766927c0532977b0d2158981 + world.c:985533370 - c712f77189307907f4189b5a7ab62ff3 + libworld.a:985533374 69e568fc5241d7d25be86d581e1fb6aa + +After the file name and colon, the first number is a timestamp of the +file's modification time (on UNIX systems, this is typically the number +of seconds since January 1st, 1970). The second value is the build +signature of the file (or ``-'' in the case of files with no build +signature--that is, source files). The third value, if any, is the +content signature of the file. + + +=head2 Using build signatures to decide when to rebuild files + +When Cons is deciding whether to build or rebuild a derived file, it +first computes the file's current build signature. If the file doesn't +exist, it must obviously be built. + +If, however, the file already exists, Cons next compares the +modification timestamp of the file against the timestamp value in +the F<.consign> file. If the timestamps match, Cons compares the +newly-computed build signature against the build signature in the +F<.consign> file. If the timestamps do not match or the build +signatures do not match, the derived file is rebuilt. + +After the file is built or rebuilt, Cons arranges to store the +newly-computed build signature in the F<.consign> file when it exits. + + +=head2 Signature example + +The use of these signatures is an extremely simple, efficient, and +effective method of improving--dramatically--the reproducibility of a +system. + +We'll demonstrate this with a simple example: + + # Simple "Hello, World!" Construct file + $CFLAGS = '-g' if $ARG{DEBUG} eq 'on'; + $CONS = new cons(CFLAGS => $CFLAGS); + Program $CONS 'hello', 'hello.c'; + +Notice how Cons recompiles at the appropriate times: + + % cons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % cons hello + cons: "hello" is up-to-date. + % cons DEBUG=on hello + cc -g -c hello.c -o hello.o + cc -o hello hello.o + % cons DEBUG=on hello + cons: "hello" is up-to-date. + % cons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + + +=head2 Source-file signature configuration + +Cons provides a C method that allows you to configure +how the signature should be calculated for any source file when its +signature is being used to decide if a dependent file is up-to-date. +The arguments to the C method consist of one or more +pairs of strings: + + SourceSignature 'auto/*.c' => 'content', + '*' => 'stored-content'; + +The first string in each pair is a pattern to match against derived file +path names. The pattern is a file-globbing pattern, not a Perl regular +expression; the pattern <*.l> will match all Lex source files. The C<*> +wildcard will match across directory separators; the pattern C +would match all C source files in any subdirectory underneath the C +subdirectory. + +The second string in each pair contains one of the following keywords to +specify how signatures should be calculated for source files that match +the pattern. The available keywords are: + +=over 4 + +=item content + +Use the content signature of the source file when calculating signatures +of files that depend on it. This guarantees correct calculation of the +file's signature for all builds, by telling Cons to read the contents of +a source file to calculate its content signature each time it is run. + +=item stored-content + +Use the source file's content signature as stored in the F<.consign> +file, provided the file's timestamp matches the cached timestamp value +in the F<.consign> file. This optimizes performance, with the slight +risk of an incorrect build if a source file's contents have been changed +so quickly after its previous update that the timestamp still matches +the stored timestamp in the F<.consign> file even though the contents +have changed. + +=back + +The Cons default behavior of always calculating a source file's +signature from the file's contents is equivalent to specifying: + + SourceSignature '*' => 'content'; + +The C<*> will match all source files. The C keyword +specifies that Cons will read the contents of a source file to calculate +its signature each time it is run. + +A useful global performance optimization is: + + SourceSignature '*' => 'stored-content'; + +This specifies that Cons will use pre-computed content signatures +from F<.consign> files, when available, rather than re-calculating a +signature from the the source file's contents each time Cons is run. In +practice, this is safe for most build situations, and only a problem +when source files are changed automatically (by scripts, for example). +The Cons default, however, errs on the side of guaranteeing a correct +build in all situations. + +Cons tries to match source file path names against the patterns in the +order they are specified in the C arguments: + + SourceSignature '/usr/repository/objects/*' => 'stored-content', + '/usr/repository/*' => 'content', + '*.y' => 'content', + '*' => 'stored-content'; + +In this example, all source files under the F +directory will use F<.consign> file content signatures, source files +anywhere else underneath F will not use F<.consign> +signature values, all Yacc source files (C<*.y>) anywhere else will not +use F<.consign> signature values, and any other source file will use +F<.consign> signature values. + + +=head2 Derived-file signature configuration + +Cons provides a C construction variable that allows you to +configure how signatures are calculated for any derived file when its +signature is being used to decide if a dependent file is up-to-date. +The value of the C construction variable is a Perl array +reference that holds one or more pairs of strings, like the arguments to +the C method. + +The first string in each pair is a pattern to match against derived file +path names. The pattern is a file-globbing pattern, not a Perl regular +expression; the pattern `*.obj' will match all (Win32) object files. +The C<*> wildcard will match across directory separators; the pattern +`foo/*.a' would match all (UNIX) library archives in any subdirectory +underneath the foo subdirectory. + +The second string in each pair contains one of the following keywords +to specify how signatures should be calculated for derived files that +match the pattern. The available keywords are the same as for the +C method, with an additional keyword: + +=over 4 + +=item build + +Use the build signature of the derived file when calculating signatures +of files that depend on it. This guarantees correct builds by forcing +Cons to rebuild any and all files that depend on the derived file. + +=item content + +Use the content signature of the derived file when calculating signatures +of files that depend on it. This guarantees correct calculation of the +file's signature for all builds, by telling Cons to read the contents of +a derived file to calculate its content signature each time it is run. + +=item stored-content + +Use the derived file's content signature as stored in the F<.consign> +file, provided the file's timestamp matches the cached timestamp value +in the F<.consign> file. This optimizes performance, with the slight +risk of an incorrect build if a derived file's contents have been +changed so quickly after a Cons build that the file's timestamp still +matches the stored timestamp in the F<.consign> file. + +=back + +The Cons default behavior (as previously described) for using +derived-file signatures is equivalent to: + + $env = new cons(SIGNATURE => ['*' => 'build']); + +The C<*> will match all derived files. The C keyword specifies +that all derived files' build signatures will be used when calculating +whether a dependent file is up-to-date. + +A useful alternative default C configuration for many sites: + + $env = new cons(SIGNATURE => ['*' => 'content']); + +In this configuration, derived files have their signatures calculated +from the file contents. This adds slightly to Cons' workload, but has +the useful effect of "stopping" further rebuilds if a derived file is +rebuilt to exactly the same file contents as before, which usually +outweighs the additional computation Cons must perform. + +For example, changing a comment in a C file and recompiling should +generate the exact same object file (assuming the compiler doesn't +insert a timestamp in the object file's header). In that case, +specifying C or C for the signature calculation +will cause Cons to recognize that the object file did not actually +change as a result of being rebuilt, and libraries or programs that +include the object file will not be rebuilt. When C is +specified, however, Cons will only "know" that the object file was +rebuilt, and proceed to rebuild any additional files that include the +object file. + +Note that Cons tries to match derived file path names against the +patterns in the order they are specified in the C array +reference: + + $env = new cons(SIGNATURE => ['foo/*.o' => 'build', + '*.o' => 'content', + '*.a' => 'stored-content', + '*' => 'content']); + +In this example, all object files underneath the F subdirectory +will use build signatures, all other object files (including object +files underneath other subdirectories!) will use F<.consign> file +content signatures, libraries will use F<.consign> file build +signatures, and all other derived files will use content signatures. + + +=head2 Debugging signature calculation + +Cons provides a C<-S> option that can be used to specify what internal +Perl package Cons should use to calculate signatures. The default Cons +behavior is equivalent to specifying C<-S md5> on the command line. + +The only other package (currently) available is an C +package that prints out detailed information about the MD5 signature +calculations performed by Cons: + + % cons -S md5::debug hello + sig::md5::srcsig(hello.c) + => |52d891204c62fe93ecb95281e1571938| + sig::md5::collect(52d891204c62fe93ecb95281e1571938) + => |fb0660af4002c40461a2f01fbb5ffd03| + sig::md5::collect(52d891204c62fe93ecb95281e1571938, + fb0660af4002c40461a2f01fbb5ffd03, + cc -c %< -o %>) + => |f7128da6c3fe3c377dc22ade70647b39| + sig::md5::current(|| + eq |f7128da6c3fe3c377dc22ade70647b39|) + cc -c hello.c -o hello.o + sig::md5::collect() + => |d41d8cd98f00b204e9800998ecf8427e| + sig::md5::collect(f7128da6c3fe3c377dc22ade70647b39, + d41d8cd98f00b204e9800998ecf8427e, + cc -o %> %< ) + => |a0bdce7fd09e0350e7efbbdb043a00b0| + sig::md5::current(|| + eq |a0bdce7fd09e0350e7efbbdb043a00b0|) + cc -o hello, hello.o + + + + + + + +=head1 Temporary overrides + +Cons provides a very simple mechanism for overriding aspects of a build. The +essence is that you write an override file containing one or more +C commands, and you specify this on the command line, when you run +C: + + % cons -o over export + +will build the F directory, with all derived files subject to the +overrides present in the F file. If you leave out the C<-o> option, +then everything necessary to remove all overrides will be rebuilt. + + +=head2 Overriding environment variables + +The override file can contain two types of overrides. The first is incoming +environment variables. These are normally accessible by the F +file from the C<%ENV> hash variable. These can trivially be overridden in +the override file by setting the appropriate elements of C<%ENV> (these +could also be overridden in the user's environment, of course). + + +=head2 The Override command + +The second type of override is accomplished with the C command, +which looks like this: + + Override , => , => , ...; + +The regular expression I is matched against every derived file that +is a candidate for the build. If the derived file matches, then the +variable/value pairs are used to override the values in the construction +environment associated with the derived file. + +Let's suppose that we have a construction environment like this: + + $CONS = new cons( + COPT => '', + CDBG => '-g', + CFLAGS => '%COPT %CDBG', + ); + +Then if we have an override file F containing this command: + + Override '\.o$', COPT => '-O', CDBG => ''; + +then any C invocation with C<-o over> that creates F<.o> files via +this environment will cause them to be compiled with C<-O >and no C<-g>. The +override could, of course, be restricted to a single directory by the +appropriate selection of a regular expression. + +Here's the original version of the Hello, World! program, built with this +environment. Note that Cons rebuilds the appropriate pieces when the +override is applied or removed: + + % cons hello + cc -g -c hello.c -o hello.o + cc -o hello hello.o + % cons -o over hello + cc -O -c hello.c -o hello.o + cc -o hello hello.o + % cons -o over hello + cons: "hello" is up-to-date. + % cons hello + cc -g -c hello.c -o hello.o + cc -o hello hello.o + +It's important that the C command only be used for temporary, +on-the-fly overrides necessary for development because the overrides are not +platform independent and because they rely too much on intimate knowledge of +the workings of the scripts. For temporary use, however, they are exactly +what you want. + +Note that it is still useful to provide, say, the ability to create a fully +optimized version of a system for production use--from the F and +F files. This way you can tailor the optimized system to the +platform. Where optimizer trade-offs need to be made (particular files may +not be compiled with full optimization, for example), then these can be +recorded for posterity (and reproducibility) directly in the scripts. + + + +=head2 The C method + +The C method is a combination of the C and C +methods. Rather than generating an executable program directly, this command +allows you to specify your own command to actually generate a module. The +method is invoked as follows: + + Module $env , , ; + +This command is useful in instances where you wish to create, for example, +dynamically loaded modules, or statically linked code libraries. + + + + +=head2 The C method + +The C method returns the construction variables for building +various components with one of the rule sets supported by Cons. The +currently supported rule sets are: + +=over 4 + +=item msvc + +Rules for the Microsoft Visual C++ compiler suite. + +=item unix + +Generic rules for most UNIX-like compiler suites. + +=back + +On systems with more than one available compiler suite, this allows you +to easily create side-by-side environments for building software with +multiple tools: + + $msvcenv = new cons(RuleSet("msvc")); + $cygnusenv = new cons(RuleSet("unix")); + +In the future, this could also be extended to other platforms that +have different default rule sets. + + +=head2 The C method + +The C method sets the default construction variables that +will be returned by the C method to the specified arguments: + + DefaultRules(CC => 'gcc', + CFLAGS => '', + CCCOM => '%CC %CFLAGS %_IFLAGS -c %< -o %>'); + $env = new cons(); + # $env now contains *only* the CC, CFLAGS, + # and CCCOM construction variables + +Combined with the C method, this also provides an easy way +to set explicitly the default build environment to use some supported +toolset other than the Cons defaults: + + # use a UNIX-like tool suite (like cygwin) on Win32 + DefaultRules(RuleSet('unix')); + $env = new cons(); + +Note that the C method completely replaces the default +construction environment with the specified arguments, it does not +simply override the existing defaults. To override one or more +variables in a supported C, append the variables and values: + + DefaultRules(RuleSet('unix'), CFLAGS => '-O3'); + $env1 = new cons(); + $env2 = new cons(); + # both $env1 and $env2 have 'unix' defaults + # with CFLAGS set to '-O3' + + + + + + + + +=head2 The C method + +The C mathod returns the real source path name of a file, +as opposed to the path name within a build directory. It is invoked +as follows: + + $path = SourcePath ; + + +=head2 The C method + +The C method returns true if the supplied path is a derivable +file, and returns undef (false) otherwise. +It is invoked as follows: + + $result = ConsPath ; + + +=head2 The C method + +The C method looks up multiple path names in a string separated +by the default path separator for the operating system (':' on UNIX +systems, ';' on Windows NT), and returns the fully-qualified names. +It is invoked as follows: + + @paths = SplitPath ; + +The C method will convert names prefixed '#' to the +appropriate top-level build name (without the '#') and will convert +relative names to top-level names. + + +=head2 The C method + +The C method returns the build path name(s) of a directory or +list of directories. It is invoked as follows: + + $cwd = DirPath ; + +The most common use for the C method is: + + $cwd = DirPath '.'; + +to fetch the path to the current directory of a subsidiary F +file. + + +=head2 The C method + +The C method returns the build path name(s) of a file or +list of files. It is invoked as follows: + + $file = FilePath ; diff --git a/doc/user/cons.sgml b/doc/user/cons.sgml index 3024c023..448777f1 100644 --- a/doc/user/cons.sgml +++ b/doc/user/cons.sgml @@ -22,6 +22,7 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + X diff --git a/doc/user/copyright.sgml b/doc/user/copyright.sgml index 500b81c1..7f6059ce 100644 --- a/doc/user/copyright.sgml +++ b/doc/user/copyright.sgml @@ -26,7 +26,7 @@
- Copyright (c) 2002, 2003 Steven Knight + SCons User's Guide Copyright (c) 2003 Steven Knight
diff --git a/doc/user/default.sgml b/doc/user/default.sgml new file mode 100644 index 00000000..bb81a3a8 --- /dev/null +++ b/doc/user/default.sgml @@ -0,0 +1,78 @@ + + + + + + + X + + + +
+ The &Default; Method + + + + X + + + +
diff --git a/doc/user/depends.sgml b/doc/user/depends.sgml index d95ae797..c94ac0d1 100644 --- a/doc/user/depends.sgml +++ b/doc/user/depends.sgml @@ -22,9 +22,76 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + - X + So far we've seen how &SCons; handles one-time builds. + But the real point of a build tool like &SCons; + is to rebuild the necessary things when source files change. + The other side of rebuilding all the necessary things + is that &SCons; should not + unnecessarily rebuild things that have already + been built. + You can see this at work simply be re-invoking &SCons; + after building our simple &hello; example: + + + + + % scons . + cc -c hello.c -o hello.o + cc -o hello hello.o + % scons . + % + + + + + &SCons; has remembered that the &hello; program + is up-to-date with respect to the current &hello_c; source file, + and avoids rebuilding it. + You can see this more clearly by naming + the &hello; program explicitly on the command line: + + + + + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % scons hello + scons: `hello' is up to date. + % + + + + + (&SCons; only reports "...is up to date" + for target files named explicitly on the command line, + to avoid cluttering the output.) @@ -33,14 +100,75 @@ + &SCons; keeps track of whether a file has changed + based on the file's contents, + not the modification time. + This means that you may be surprised by the + behavior of &SCons; if you are used to the + &Make; convention of forcing + a rebuild by updating the file's + modification time like so: + + + + + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % touch hello.c + % scons hello + scons: `hello' is up to date. + % + + + + + This saves time by avoiding unnecessary rebuilds when, + for example, someone rewrites the + contents of a file without making a change. + But if the contents of the file change, + then &SCons; detects the change + and rebuilds the program as required: + + + + + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % edit hello.c + [CHANGE THE CONTENTS OF hello.c] + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % + + + + X + +
- Time Stamps + Implicit Dependencies @@ -48,6 +176,135 @@ + + #define string "world" + + + + #include "hello.h" + int + main() + { + printf("Hello, %s!\n", string); + } + + + + env = Environment(CPPPATH = '.') XXX IS CPPPATH NECESSARY? + hello = env.Program('hello.c') + + + + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % scons hello + scons: `hello' is up to date. + % edit hello.h + [CHANGE THE CONTENTS OF hello.h] + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % + + + + + &SCons; knows that the &hello; + program must be rebuilt + because it scans the contents of + the &hello_c; + for the #include lines that indicate + another file is being included in the compilation. + &SCons; records these as + implicit dependencies + of the target file, + Consequently, + when the &hello_h; file changes, + &SCons; realizes that the &hello_c; file includes it, + and rebuilds the resulting &hello; program + that depends on both the &hello_c; and &hello_h; files. + + + + + + X + + + + + +
+ +
+ The &Ignore; Method + + + + Sometimes it makes sense + to not rebuild a program, + even if a dependency file changes. + In this case, + you would tell &SCons; specifically + to ignore a dependency as follows: + + + + + env = Environment() + hello = env.Program('hello.c') + env.Ignore(hello, 'hello.h') + + + + + + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % scons hello + scons: `hello' is up to date. + % edit hello.h + [CHANGE THE CONTENTS OF hello.h] + % scons hello + scons: `hello' is up to date. + + + + + Now, the above example is a little contrived, + because it's hard to imagine a real-world situation + where you wouldn't to rebuild &hello; + if the &hello_h; file changed. + A more realistic example + might be if the &hello; + program is being built in a + directory that is shared between multiple systems + that have different copies of the + &stdio_h; include file. + In that case, + &SCons; would notice the differences between + the different systems' copies of &stdio_h; + and would rebuild &hello; + each time you change systems. + You could avoid these rebuilds as follows: + + + + + env = Environment() + hello = env.Program('hello.c') + env.Ignore(hello, '/usr/include/stdio.h') + +
@@ -55,14 +312,44 @@ - X + On the other hand, + sometimes a file depends on another + file that the scanner(s) in &SCons; will not detect. + For this situation, + &SCons; allows you to specific that one file explicitly + depends on another file, + and must be rebuilt whenever that file changes, + using the &Depends; method: + + env = Environment() + hello = env.Program('hello.c') + env.Depends(hello, 'other_file') + + + + + + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + % scons hello + scons: `hello' is up to date. + % edit other_file + [CHANGE THE CONTENTS OF other_file] + % scons hello + cc -c hello.c -o hello.o + cc -o hello hello.o + +
+ +
- The &Ignore; Method + The &Salt; Method @@ -72,8 +359,10 @@
+ --> +
- The &Salt; Method + Time Stamps diff --git a/doc/user/environments.sgml b/doc/user/environments.sgml index e0ea8ee8..b06ade11 100644 --- a/doc/user/environments.sgml +++ b/doc/user/environments.sgml @@ -22,25 +22,488 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + X - + + + + + + optimize = Environment(CCFLAGS = '-O2') + debug = Environment(CCFLAGS = '-g') + + o = optimize.Object('foo-opt', 'foo.c') + optimize.Program(o) + + d = debug.Object('foo-debug', 'foo.c') + debug.Program(d) + + + + % scons . + cc -c -g foo.c -o foo-debug.o + cc -o foo-debug foo-debug.o + cc -c -O2 foo.c -o foo-opt.o + cc -o foo-opt foo-opt.o +
- The &Clone; Method + The &Environment; Constructor diff --git a/doc/user/errors.sgml b/doc/user/errors.sgml index 3024c023..448777f1 100644 --- a/doc/user/errors.sgml +++ b/doc/user/errors.sgml @@ -22,6 +22,7 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + X diff --git a/doc/user/example.sgml b/doc/user/example.sgml index 3024c023..448777f1 100644 --- a/doc/user/example.sgml +++ b/doc/user/example.sgml @@ -22,6 +22,7 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + X diff --git a/doc/user/help.sgml b/doc/user/help.sgml new file mode 100644 index 00000000..7c1732e7 --- /dev/null +++ b/doc/user/help.sgml @@ -0,0 +1,88 @@ + + + + + X + + + +
+ Providing build-specific help instructions + + + + It's often very useful to be able to give + users some help that describes the + specific targets, build options, etc., + that can be used for the build. + &SCons; provides the &Help; function + to allow you to specify this help text: + + + + + Help(""" + Type: 'scons program' to build the production program, + 'scons debug' to build the debug version. + """) + + + + + (Note the above use of the Python triple-quote syntax, + which comes in very handy for + specifying multi-line strings like help text.) + + + + + + When the &SConstruct; or &SConscript; files + contain such a call to the &Help; function, + the specified help text will be displayed in response to + the &SCons; -h option: + + + + + % scons -h + Type: 'scons program' to build the production program, + 'scons debug' to build the debug version. + + + + + If there is no &Help; text in the &SConstruct; or + &SConscript; files, + &SCons; will revert to displaying its + standard list and description of available command-line + options. + This list is also always displayed whenever + the -H option is used. + + + +
diff --git a/doc/user/hierarchy.sgml b/doc/user/hierarchy.sgml index 3f65aebb..aa56c8d4 100644 --- a/doc/user/hierarchy.sgml +++ b/doc/user/hierarchy.sgml @@ -22,6 +22,204 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + X diff --git a/doc/user/install.sgml b/doc/user/install.sgml new file mode 100644 index 00000000..530fa7ad --- /dev/null +++ b/doc/user/install.sgml @@ -0,0 +1,97 @@ + + + + + + + X + + + +
+ The &Install; Builder + + + + X + + + +
+ +
+ The &InstallAs; Method + + + + X + + + +
diff --git a/doc/user/library.sgml b/doc/user/library.sgml new file mode 100644 index 00000000..0c9292ca --- /dev/null +++ b/doc/user/library.sgml @@ -0,0 +1,81 @@ + + + + + + + X + + + +
+ The &Library; Builder + + + + X + + + +
diff --git a/doc/user/main.sgml b/doc/user/main.sgml index ca07ce47..bb64b725 100644 --- a/doc/user/main.sgml +++ b/doc/user/main.sgml @@ -32,22 +32,33 @@ %scons; + + + + + + + + + + + @@ -100,17 +111,64 @@ &depends; + + + + Default Targets + &default; + + + + Providing Build Help + &help; + + + + Installing Files in Other Directories + &install; + + + + The Program Builder + &program; + + + + The Library Builder + &library; + + More Things to Do With Builds &more; + + The Command Builder + &command; + + + + The Object Builder + &object; + + + + Preventing Removal of Targets + &precious; + + - Hierarchial Builds + Hierarchical Builds &hierarchy; - + + Separating Source and Build Directories + &separate; + + + Variant Builds &variants; @@ -120,6 +178,11 @@ &builders; + + SCons Actions + &actions; + + Writing Scanners &scanners; @@ -140,6 +203,11 @@ &run; + + Alias Targets + &alias; + + Troubleshooting &troubleshoot; diff --git a/doc/user/make.sgml b/doc/user/make.sgml index 3024c023..5666ea64 100644 --- a/doc/user/make.sgml +++ b/doc/user/make.sgml @@ -22,6 +22,76 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + X diff --git a/doc/user/more.sgml b/doc/user/more.sgml index 1c298a80..7defa414 100644 --- a/doc/user/more.sgml +++ b/doc/user/more.sgml @@ -22,78 +22,13 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + X -
- The &InstallAs; Method - - - - X - - - -
- -
- The &Precious; Method - - - - X - - - -
- -
- The &Command; Method - - - - X - - - -
- -
- The &Objects; Method - - - - X - - - -
- -
- The &Program; Method - - - - X - - - -
- -
- The &Library; Method - - - - X - - - -
-
The &Module; Method diff --git a/doc/user/object.sgml b/doc/user/object.sgml new file mode 100644 index 00000000..9e887d8c --- /dev/null +++ b/doc/user/object.sgml @@ -0,0 +1,71 @@ + + + + + + + X + + + +
+ The &Object; Method + + + + X + + + +
diff --git a/doc/user/precious.sgml b/doc/user/precious.sgml new file mode 100644 index 00000000..9cd2a431 --- /dev/null +++ b/doc/user/precious.sgml @@ -0,0 +1,69 @@ + + + + + + + X + + + +
+ The &Precious; Method + + + + X + + + +
diff --git a/doc/user/preface.sgml b/doc/user/preface.sgml index 34217021..82ea44a1 100644 --- a/doc/user/preface.sgml +++ b/doc/user/preface.sgml @@ -22,6 +22,7 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + X diff --git a/doc/user/program.sgml b/doc/user/program.sgml new file mode 100644 index 00000000..30f90d29 --- /dev/null +++ b/doc/user/program.sgml @@ -0,0 +1,77 @@ + + + + + + + X + + + +
+ The &Program; Builder + + + + X + + + +
diff --git a/doc/user/reference.sgml b/doc/user/reference.sgml index 57f58d82..ef8e5a94 100644 --- a/doc/user/reference.sgml +++ b/doc/user/reference.sgml @@ -22,6 +22,7 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + X diff --git a/doc/user/repositories.sgml b/doc/user/repositories.sgml index 3f0a597c..4acc9432 100644 --- a/doc/user/repositories.sgml +++ b/doc/user/repositories.sgml @@ -22,29 +22,426 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + - X + Often, a software project will have + one or more central repositories, + directory trees that contain + source code, or derived files, or both. + You can eliminate additional unnecessary + rebuilds of files by having &SCons;, + use files from one or more code repositories + to build files in your local build tree.
The &Repository; Method + + + + + You use the &Repository; method + to tell &SCons; to search one or more + central code repositories (in order) + for any source files and derived files + that are not present in the local build tree: + + + + + env = Environment() + env.Program('hello.c') + Repository('/usr/repository1', '/usr/repository2') + + + + + (Note that multiple calls to the &Repository; method + will simply add repositories to the global list + that &SCons; maintains, + with the exception that &SCons; will automatically eliminate + the current directory and any non-existent + directories from the list.) + + + +
+ +
+ Finding source files in repositories + + + + The above example + specifies that &SCons; + will first search for files under + the /usr/repository1 tree + and next under the /usr/repository2 tree. + &SCons; expects that any files it searches + for will be found in the same position + relative to the top-level directory XXX + In the above example, if the &hello_c; file is not + found in the local build tree, + &SCons; will search first for + a /usr/repository1/hello.c file + and then for a /usr/repository1/hello.c file + to use in its place. + + + + + + So given the &SConstruct; file above, + if the &hello_c; file exists in the local + build directory, + &SCons; will rebuild the &hello; program + as normal: + + + + + % scons . + gcc -c hello.c -o hello.o + gcc -o hello hello.o + + + + + If, however, there is no local &hello_c; file, + but one exists in /usr/repository1, + &SCons; will recompile the &hello; program + from the source file it finds in the repository: + + + + + % scons . + gcc -c /usr/repository1/hello.c -o hello.o + gcc -o hello hello.o + + + + + And similarly, if there is no local &hello_c; file + and no /usr/repository1/hello.c, + but one exists in /usr/repository2: + + + + + % scons . + gcc -c /usr/repository2/hello.c -o hello.o + gcc -o hello hello.o + + + + + + +
+ +
+ Finding the &SConstruct; file in repositories + + + + &SCons; will also search in repositories + for the &SConstruct; file and any specified &SConscript; files. + This poses a problem, though: how can &SCons; search a + repository tree for an &SConstruct; file + if the &SConstruct; file itself contains the information + about the pathname of the repository? + To solve this problem, &SCons; allows you + to specify repository directories + on the command line using the -Y: + + + + + % scons -Y /usr/repository1 -Y /usr/repository2 . + + + + + When looking for source or derived files, + &SCons; will first search the repositories + specified on the command line, + and then search the repositories + specified in the &SConstruct; or &SConscript; files. + + + +
+ +
+ Finding derived files in repositories + + + + If a repository contains not only source files, + but also derived files (such as object files, + libraries, or exectuables), &SCons; will perform + its normal MD5 signature calculation to + decide if a derived file in a repository is up-to-date, + or the derived file must be rebuilt in the local build directory. + For the &SCons; signature calculation to work correctly, + a repository tree must contain the &sconsign; files + that &SCons; uses to keep track of signature information. + + + + + + Usually, this would be done by a build integrator + who would run &SCons; in the repository + to create all of its derived files and &sconsign; files, + or who would &SCons; in a separate build directory + and copying the resulting tree to the desired repository: + + + + + % cd /usr/repository1 + % scons . + gcc -c hello.c -o hello.o + gcc -o hello hello.o + + + + + (Note that this is safe even if the &SConstruct; file + lists /usr/repository1 as a repository, + because &SCons; will remove the current build directory + from its repository list for that invocation.) + + + + + + Now, with the repository populated, + we only need to create the one local source file + we're interested in working with at the moment, + and use the -Y option to + tell &SCons; to fetch any other files it needs + from the repository: + + + + + % cd $HOME/build + % edit hello.c + % scons -Y /usr/repository1 . + gcc -c hello.c -o hello.o + gcc -o hello hello.o + XXXXXXX + + - X + Notice that &SCons; realizes that it does not need to + rebuild a local XXX.o file, + but instead uses the already-compiled XXX.o file + from the repository.
- X + Guaranteeing local copies of files + + + + If the repository tree contains the complete results of a build, + and we try to build from the repository + without any files in our local tree, + something moderately surprising happens: + + + + + % mkdir $HOME/build2 + % cd $HOME/build2 + % cons -Y /usr/all/repository hello + cons: "hello" is up-to-date. + + + + + Why does &SCons; say that the &hello; program + is up-to-date when there is no &hello; program + in the local build directory? + Because the repository (not the local directory) + contains the up-to-date &hello; program, + and &SCons; correctly determines that nothing + needs to be done to rebuild that + up-to-date copy of the file. + + + + + + There are, however, many times when you want to ensure that a + local copy of a file always exists. + A packaging or testing script, for example, + may assume that certain generated files exist locally. + To tell &SCons; to make a copy of any up-to-date repository + file in the local build directory, + use the &Local; function: + + + + + env = Environment() + hello = env.Program('hello.c') + Local(hello) + + + + + If we then run the same command, + &SCons; will make a local copy of the program + from the repository copy, + and tell you that it is doing so: + + + + + % cons -R /usr/all/repository hello + Local copy of hello from /usr/all/repository/hello + scons: `hello' is up-to-date. + XXXXXX DO WE REALLY REPORT up-to-date, TOO? + - X + (Notice that, because the act of making the local copy + is not considered a "build" of the &hello; file, + &SCons; still reports that it is up-to-date.) + XXXXXX DO WE REALLY REPORT up-to-date, TOO? diff --git a/doc/user/run.sgml b/doc/user/run.sgml index ba802ff0..30b61ca5 100644 --- a/doc/user/run.sgml +++ b/doc/user/run.sgml @@ -22,22 +22,340 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> - - X + + + + + X + +
Selective Builds diff --git a/doc/user/scanners.sgml b/doc/user/scanners.sgml index 3024c023..8d13b0c1 100644 --- a/doc/user/scanners.sgml +++ b/doc/user/scanners.sgml @@ -22,6 +22,105 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + X diff --git a/doc/user/separate.sgml b/doc/user/separate.sgml new file mode 100644 index 00000000..e1c0e49b --- /dev/null +++ b/doc/user/separate.sgml @@ -0,0 +1,128 @@ + + + + + + + X + + + +
+ X + + + + X + + + +
diff --git a/doc/user/simple.sgml b/doc/user/simple.sgml index 5de9d3c6..90711608 100644 --- a/doc/user/simple.sgml +++ b/doc/user/simple.sgml @@ -22,85 +22,223 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + - X + Here's how to build the famous "Hello, World!" example using &SCons;. + Enter the following into a file name &SConstruct;: -
- The &SConstruct; File + + int + main() + { + printf("Hello, world!\n"); + } + - + + env = Environment() + env.Program('hello.c') + - X + - + That's it. Now run the &scons; command to build the program. + On a POSIX-compliant system like Linux or UNIX, + you'll see something like: -
+
-
- The &Program; Builder + + % scons . + cc -c hello.c -o hello.o + cc -o hello hello.o + - + - X + On a Windows system with the Microsoft Visual C++ compiler, + you'll see something like: - + -
+ + C:\>scons . + cl /Fohello.obj hello.c + link /Fohello.exe hello.obj + + + + + First, notice that you need to supply a '.' on the command line. + This tells &SCons; to build everything in the current directory + or its subdirectories. + We'll tell you later how to avoid having to use the '.' + + + + + + Next, notice that the same input &SConstruct; file + without any special input, + generates the right output file names on both systems: + hello.o and hello + on POSIX systems, + hello.obj and hello.exe + on Windows systems. + (We won't provide side-by-side examples of POSIX + and Windows runs for all future examples; + just know that XXX.) + +
- The &Library; Builder + The &SConstruct; File - X + If you're used to build systems like &Make; + you've already figured out that the &SConstruct; file + is the &SCons; equivalent of a &Makefile;. + That is, the &SConstruct; file is the input file + that &SCons; reads to control the build. -
+ -
- The &Install; Builder + What may not be obvious, though, is that + there's an important difference between + an &SConstruct; file and a &Makefile: + the &SConstruct; file is actually a Python script. + If you're not already familiar with Python, don't worry; + you don't really need to know Python to be able to use + &SCons; effectively. + But you'll see that being able to use the power of a + real scripting language + can greatly simplify the solutions + to complex requirements of real-world builds. + + - X + For now, one ramification of using Python as the + scripting language means that you can put comments + in your &SConstruct; file using Python's commenting convention; + that is, everything between a '#' and the end of the line + will be ignored: + + env = Environment() # Create an environment. + # Arrange to build the "hello" program. + env.Program('hello.c') + +
- Running &SCons; + Compiling Multiple Source Files - X + If you want + + env = Environment() + env.Program('program', ['main.c', 'file1.c', 'file2.']) + + + + env = Environment() + env.Program('program', Split('main.c file1.c file2.')) + + + + % scons . + cc -c file1.c -o file1.o + cc -c file2.c -o file2.o + cc -c main.c -o main.o + cc -o program main.o file1.o file2.o + +
- The &Default; Method + Compiling Multiple Programs - X + If you want + + env = Environment() + env.Program('foo.c') + env.Program('bar', ['bar1.c', 'bar2.c']) + + + + % scons . + cc -c bar1.c -o bar1.o + cc -c bar2.c -o bar2.o + cc -o bar bar1.o bar2.o + cc -c foo.c -o foo.o + cc -o foo foo.o + +
- The &Help; Method + Sharing Files Between Multiple Programs - X + If you want + + common = ['common1.c', 'common2.c'] + env = Environment() + env.Program(['foo.c'] + common) + env.Program('bar', ['bar1.c', 'bar2.c'] + common) + + + + % scons . + cc -c bar1.c -o bar1.o + cc -c bar2.c -o bar2.o + cc -c common1.c -o common1.o + cc -c common2.c -o common2.o + cc -o bar bar1.o bar2.o common1.o common2.o + cc -c foo.c -o foo.o + cc -o foo foo.o common1.o common2.o + +
diff --git a/doc/user/troubleshoot.sgml b/doc/user/troubleshoot.sgml index 3024c023..448777f1 100644 --- a/doc/user/troubleshoot.sgml +++ b/doc/user/troubleshoot.sgml @@ -22,6 +22,7 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + X diff --git a/doc/user/variants.sgml b/doc/user/variants.sgml index 3024c023..fdd209ee 100644 --- a/doc/user/variants.sgml +++ b/doc/user/variants.sgml @@ -22,6 +22,91 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. --> + + + X