help-make
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

Re: non-recursive build question


From: Noel Yap
Subject: Re: non-recursive build question
Date: Mon, 03 May 2004 14:07:36 -0400
User-agent: Mozilla Thunderbird 0.5 (Windows/20040212)

I still contend that an order rule perfectly suits your needs:

.PHONY: default
default: executable

foo.exp: $(OBJECT_FILES)

foo.so: foo.exp

executable: | foo.so


The order rule /will/ put foo.so in the DAG but doesn't make it an explicit dependency of executable. For example, starting with a clean filesystem, everything from the .o's through to the executable will be rebuilt. Upon successive builds, any .o, .exp, and .so will be rebuilt if necessary. The executable will not be rebuilt (unless any of its direct sources are modified).

Note that the executable should /not/ be directly dependent upon the .exp.  
Otherwise, it /will/ be relinked each time any of the .so's object files are 
modified.

There is one assumption I'm making here:  The executable doesn't have to be 
relinked if symbols recently added to the .exp aren't used in the executable.  
If they are used, then presumably the source for the executables would've 
changed, too.

Noel

Sandy Currier wrote:

On Thursday 29 April 2004 10:43 am, Paul D. Smith wrote:
 > %% Noel Yap <address@hidden> writes:
 >
 >   ny> In the end, though, the OP should be able to do what he wanted
 >   ny> (assuming IIUC)
 >
 > That I don't know about; you could well be correct--I hope Sandy will
 > provide a list of explicit requirements that we can use to determine
 > that because I, at least, don't have any good feeling for exactly what
 > they are.

Ok - I think I finally have a reasonable understanding of the problem and at
least one solution.

From the beginning, there was at least one case of pilot error - I personally
thought that if a dependent's target rules fired, that gmake would then
automatically fire the target's rules as well. As explained and demonstrated,
this is not the case.  Whew.

Backgound:

When building shared libraries, it is sometimes the case that one does not want
to export (expose) all the symbols in the shared library.  In fact, it is
usually the case that one only wants to make available a limited number of the
symbols in a shared library.  One way to do this with gcc and ld is use the
following switch to gcc/g++:

-Wl,--version-script,<foo.exp>

where foo.exp is an ascii file containing the list of symbols to export.  I
believe that the .exp suffix is not a standard suffix. Here it is done with a script that scans, among other things, the (nm output of the) object files that will comprise the .so and some text files generated when the .o's were created.
The .exp file is a prerequisite of the .so and used by the linker.

With a valid .exp and .so for each shared library, one can then construct a DAG
such that, since it is a .so and not a .a, downstream targets that require
linking against the shared library need only depend (DAG wise) on the .exp. As mentioned, this saves large amount of developer time not waiting for gratuitous links of _everything_ that links against the .so. There are perhaps between a
1000 and 2000 unit tests just on the single product at hand, and there are
perhaps about 70 some odd products.  So even though something like
http://www.electric-cloud.com/ claims a 20x speed-up, it is still probably
better to not even try to distribute the 1000's of unnecessary links in the
first place.  Note that much of the time, particularly during the end-game
cycle of a codeline before FCS, last minute bug fixes rarely change the .exp's.
And this is the time when schedules are late and fast incremental build and
test cycles are important.

The Requirements:

1) the link of the .so requires the .exp
2) downstream targets that explicitly depend on the .exp actually also need the
correct .so to exist and for it to be valid
3) the targets of 2) above cannot depend on the .so and can only depend on the
.exp
4) both the .exp and the .so depend on the same list of .o's, namely those that
comprise the shared library
5) the link of the .so is not short but is not huge
6) the step of building the .exp is smart - the script knows that if the .exp
does not change, it will not update the file
7) vpath must work (target rules must be in terms of $@ etc.)

I now believe that it is 2) that more or less causes the curve ball.
Downstream targets only want to depend on the .exp but also must wait until the
.so is completely updated if it is going to be updated.

I do not think that an order dependency will work because something needs to add the creation of the .so into the DAG of the downstream target (and an order
rule will not do it).  Specifying a downstream unittest target for example
needs to add the .so to the DAG so that it gets rebuilt.

So, below is the solution that I tried and seems to work:

foo.so: $(ALL_FOO_OFILES) [plus various other stuff like the .exp generator] create_the_exp_file_smartly $(basename $@).exp $(filter-out [...],$^)
   create_the_so_file $@ $(filter-out [...],$^)

foo.exp: foo.so

The above uses the foo.so rule to build the foo.exp (smartly - it is only
actually updated if it changes) as a side effect. The second target specifies
no commands - it is there to just add the .so as a dependency.

With the above, I cannot seem to break the build by specifying any combination
of targets or any combination of -jN.

Interestingly, when a .c file is touched which causes the .o to be out of date (but the .exp does not change), gmake appears to never even print via '-d' the
fact that foo.exp must be rebuilt:

$ touch foo1.c
$ make -d
[...]
   Considering target file `../do_glnx86/productA/src/foo/foo.exp'.
Looking for an implicit rule for `../do_glnx86/productA/src/foo/foo.exp'.
    No implicit rule found for `../do_glnx86/productA/src/foo/foo.exp'.
     Pruning file `../do_glnx86/bin/libfoo.so'.
Finished prerequisites of target file `../do_glnx86/productA/src/foo/foo.exp'. Prerequisite `../do_glnx86/bin/libfoo.so' is newer than target `../do_glnx86/productA/src/foo/foo.exp'. No commands for `../do_glnx86/productA/src/foo/foo.exp' and no prerequisites actually changed.
   No need to remake target `../do_glnx86/productA/src/foo/foo.exp'.
[...]

Comments appreciated, and thanks in advance for any review!
-sandy







reply via email to

[Prev in Thread] Current Thread [Next in Thread]