# ----------------------------------------------------------------------------- # # (c) 2009 The University of Glasgow # # This file is part of the GHC build system. # # To understand how the build system works and how to modify it, see # https://gitlab.haskell.org/ghc/ghc/wikis/building/architecture # https://gitlab.haskell.org/ghc/ghc/wikis/building/modifying # # ----------------------------------------------------------------------------- # Here's an interesting rule! # The .hi file may or may not change when we compile the corresponding # .hs file. If GHC figures out that the .hi file has not changed, it # doesn't touch it. This is a useful optimisation, because it means # some modules may not get recompiled if the .hi files of the modules # they depend on have not changed. # # See: # https://gitlab.haskell.org/ghc/ghc/wikis/commentary/compiler/recompilation-avoidance # # So how do we express this dependency to make? The exact form of # this rule is quite fragile. Here are some versions that don't work # very well: # # %.hi : %.o # @if [ ! -f $@ ] ; then \ # echo Panic! $< exists, but $@ does not.; \ # exit 1; \ # fi # # This version adds a useful sanity check, and is a good solution, # except that it means spawning a shell. This can be expensive, # especially on Windows where spawning a shell takes about 0.3s. # We'd like to avoid the shell if necessary. This also hides the # message "nothing to be done for 'all'", since make thinks it has # actually done something. Therefore we only use this version # if ExtraMakefileSanityChecks is enabled. # # %.hi : %.o # # This version doesn't work: GNU make knows it has't done anything to # update the .hi file, so even if the .o file has been updated, it # won't rebuild anything that depends on the .hi file. So you might # think a more correct way is to change the .hs rule: # # %.hi %.o : %.hs # $(HC) ... # # this says "compiling %.hs updates both %.hi and %.o", but that's not # true, since compiling the .hs file might not update the .hi file, if # the .hi file didn't change. And if we use this version, then make # will keep trying to rebuild %.hi if it is out of date with respect # to %.hs. # # Using this form seems to be the best compromise: # # %.hi : %.o ; # # the ';' at the end signifies an "empty command" (see the GNU make # documentation). An empty command is enough to get GNU make to think # it has updated %.hi, but without actually spawning a shell to do so. # # However, given that rule, make thinks that it can make .hi files # for any object file, even if the object file was created from e.g. # a C source file. We therefore also add a dependency on the .hs/.lhs # source file, which means we end up with rules like: # # a/%.hi : a/%.o b/%.hs ; # # But! If a file is not explicitly mentioned in a makefile, then if # make needs to build it using such a %-rule then it treats it as an # 'intermediate file', and deletes it when it is finished. Most .hi # files are mentioned in .depend* files, as some other module depends on # them, but there are some library modules that aren't imported by # anything in the tree. # # We could stop make from deleting the .hi files by declaring # ".SECONDARY:", but if we do that then make takes a pathologically long # time with our build system. So we now generate (by calling hi-rule # from .depend* files) rules that look like # # a/B.hi a/B.dyn_hi : %hi : %o x/B.hs # # Now all the .hi files are explicitly mentioned in the makefiles, so # make doesn't think they are merely intermediate files, and doesn't # delete them. ifeq "$(ExtraMakefileSanityChecks)" "NO" define hi-rule # $1 = rule header $1 ; endef else define hi-rule # $1 = rule header $1 @if [ ! -f $$@ ] ; then \ echo "Panic! $$< exists, but $$@ does not."; \ exit 1; \ fi endef endif