We've already seen that SCons will scan the contents of
a source file for #include file names
and realize that targets built from that source file
also depend on the #include file(s).
For each directory in the $CPPPATH
list,
SCons will actually search the corresponding directories
in any repository trees and establish the
correct dependencies on any
#include files that it finds
in repository directory.
Unless the C compiler also knows about these directories in the repository trees, though, it will be unable to find the #include files. If, for example, the hello.c file in our previous example includes the hello.h; in its current directory, and the hello.h; only exists in the repository:
% scons -Q cc -o hello.o -c hello.c hello.c:1: hello.h: No such file or directory
In order to inform the C compiler about the repositories,
SCons will add appropriate
-I flags to the compilation commands
for each directory in the $CPPPATH
list.
So if we add the current directory to the
construction environment $CPPPATH
like so:
env = Environment(CPPPATH = ['.']) env.Program('hello.c') Repository('/usr/repository1')
Then re-executing SCons yields:
% scons -Q cc -o hello.o -c -I. -I/usr/repository1 hello.c cc -o hello hello.o
The order of the -I options replicates,
for the C preprocessor,
the same repository-directory search path
that SCons uses for its own dependency analysis.
If there are multiple repositories and multiple $CPPPATH
directories, SCons will add the repository directories
to the beginning of each $CPPPATH
directory,
rapidly multiplying the number of -I flags.
If, for example, the $CPPPATH
contains three directories
(and shorter repository path names!):
env = Environment(CPPPATH = ['dir1', 'dir2', 'dir3']) env.Program('hello.c') Repository('/r1', '/r2')
Then we'll end up with nine -I options
on the command line,
three (for each of the $CPPPATH
directories)
times three (for the local directory plus the two repositories):
% scons -Q cc -o hello.o -c -Idir1 -I/r1/dir1 -I/r2/dir1 -Idir2 -I/r1/dir2 -I/r2/dir2 -Idir3 -I/r1/dir3 -I/r2/dir3 hello.c cc -o hello hello.o
SCons relies on the C compiler's -I options to control the order in which the preprocessor will search the repository directories for #include files. This causes a problem, however, with how the C preprocessor handles #include lines with the file name included in double-quotes.
As we've seen, SCons will compile the hello.c file from the repository if it doesn't exist in the local directory. If, however, the hello.c file in the repository contains a #include line with the file name in double quotes:
#include "hello.h" int main(int argc, char *argv[]) { printf(HELLO_MESSAGE); return (0); }
Then the C preprocessor will always use a hello.h file from the repository directory first, even if there is a hello.h file in the local directory, despite the fact that the command line specifies -I as the first option:
% scons -Q cc -o hello.o -c -I. -I/usr/repository1 /usr/repository1/hello.c cc -o hello hello.o
This behavior of the C preprocessor--always search for a #include file in double-quotes first in the same directory as the source file, and only then search the -I--can not, in general, be changed. In other words, it's a limitation that must be lived with if you want to use code repositories in this way. There are three ways you can possibly work around this C preprocessor behavior:
Some modern versions of C compilers do have an option
to disable or control this behavior.
If so, add that option to $CFLAGS
(or $CXXFLAGS
or both) in your construction environment(s).
Make sure the option is used for all construction
environments that use C preprocessing!
Change all occurrences of #include "file.h" to #include <file.h>. Use of #include with angle brackets does not have the same behavior--the -I directories are searched first for #include files--which gives SCons direct control over the list of directories the C preprocessor will search.
Require that everyone working with compilation from repositories check out and work on entire directories of files, not individual files. (If you use local wrapper scripts around your source code control system's command, you could add logic to enforce this restriction there.