The Linux kernel makefile is anexcellent example of usingmake in a complex build environment. While it is beyond the scope of this book to explain how the Linux kernel is structured and built, we can examine several interesting uses of make employed by the kernel build system. Seehttp://macarchive./ols2003/Proceedings/All-Reprints/Reprint-Germaschewski-OLS2003.pdffor a more complete discussion of the 2.5/2.6 kernel build processand its evolution from the 2.4 approach. Since the makefile has so many facets, we will discuss just a few features that are applicable to a variety of applications. First, we'll look at how single-lettermake variables are used to simulate single-letter command-line options. We'll see how the source and binary trees are separated in a way that allows users to invokemake from the source tree. Next,we'll examine the way themakefile controls the verboseness of the output.Then we'll review the most interesting user-definedfunctions and see how they reduce code duplication, improvereadability, and provide encapsulation. Finally,we'll look at the way themakefile implements a simple help facility. The Linux kernel build follows the familiar configure, build, installpattern used by my most free software. While many free and opensoftware packages use a separateconfigurescript (typically built byautoconf), the Linuxkernel makefile implements configuration withmake, invoking scripts and helper programsindirectly. When the configuration phase is complete, a simplemake ormakeall will build the bare kernel, all the modules,and produce a compressed kernel image (these are thevmlinux,modules, andbzImage targets, respectively). Each kernel buildis given a unique version number in the fileversion.o linked into the kernel. This number(and theversion.o file) are updated by themakefile itself. Some makefile features you might want to adaptto your ownmakefile are: the handling ofcommand line options, analyzing command-line goals, saving buildstatus between builds, and managing the output ofmake. 11.2.1 Command-Line OptionsThe first part of the makefile contains code for setting common build options from the command line. Here is an excerpt that controls theverbose flag:
The nested(嵌套) ifdef/ifeq pair ensures that theKBUILD_VERBOSE variable is set only ifV is set on the command line. SettingV in the environment ormakefile has no effect. The followingifndef conditional will then turn off the verboseoption ifKBUILD_VERBOSE has not yet been set. Toset the verbose option from either the environment ormakefile, you must setKBUILD_VERBOSE and notV. Notice, however, that setting KBUILD_VERBOSEdirectly on the command line is allowed and works as expected. Thiscan be useful when writing shell scripts (or aliases) to invoke themakefile. These scripts would then be moreself-documenting, similar to using GNU longoptions. The other command-line options, sparse checking (C)and external modules (M), both use the same carefulchecking to avoid accidentally setting them from within themakefile. The next section of the makefile handles the output directory option (O).This is a fairly involved piece of code. To high light its structure,we've replaced some parts of this excerpt with ellipses:
Essentially, this says that if KBUILD_OUTPUT is set, invokemake recursively(遞歸) in the output directory defined byKBUILD_OUTPUT. Set KBUILD_SRC to the directory where make was originally executed, and grab the makefilefrom there as well. The rest of the makefile will not be seen by make, sinceskip-makefile will be set. The recursivemake will reread this same makefile again, only this time KBUILD_SRC will be set, soskip-makefile will be undefined, and the rest ofthemakefile will be read and processed. This concludes the processing of command-line options. The bulk o fthe makefile follows in the ifeq($(skip-makefile),) section. 11.2.2 Configuration Versus BuildingThe makefile contains configuration targets and build targets.The configuration targets have the formmenuconfig,defconfig, etc. Maintenance targets likeclean are treated as configuration targets aswell. Other targets such asall,vmlinux, andmodules are build targets. The primary result of invoking a configuration target is two files:.config and.config.cmd. These two files are included by themakefile for build targets but are not included for configuration targets (since the configuration target creates them). It is also possible to mix both configuration targets and build targets on a singlemake invocation, suchas: $ make oldconfig all In this case, the makefile invokes itself recursively handling each target individually, thus handling configuration targets separately from build targets. The code controlling configuration, build, and mixed targets begin swith:
The variable no-dot-config-targets lists additional targets that do not require a.configfile. The code then initializes theconfig-targets, mixed-targets,and dot-config variables. Theconfig-targets variable is 1 if there are any configuration targets on the command line.The dot-config variable is 1 if there are build targets on the command line.Finally,mixed-targets is 1 if there are both configuration and build targets. The code to set dot-config is:
The filter expression is non-empty if there are configuration targets inMAKECMDGOALS. The ifneq part is true if thefilter expression is not empty. The code is hard to follow partly because it contains a double negative.Theifeq expression is true if MAKECMDGOALS contains only configuration targets.So,dot-config will be set to 0 if there are configuration targets and only configuration targets inMAKECMDGOALS. A more verbose implementation might make the meaning of these two conditionals more clear:
The ifdef form can be used instead of ifneq, becauseempty variables are treated as undefined, but care must be taken to ensure a variable does not contain merely a string of blanks (which would cause it to be defined). The config-targets and mixed-targets variables are set in the next code block:
KBUILD_EXTMOD 在makefile 開始部分的賦值過程 ifeq ("$(origin M)", "command line") KBUILD_EXTMOD := $(M)endif 在編譯模塊時,M=選項(xiàng)讓該makefile在構(gòu)造目標(biāo)之前返回到模塊源代碼目錄??梢詤⒖糒DD第三版 29頁 KBUILD_EXTMOD will be non-empty when external modules are being built, but not during normal builds.The firstifneq will be true when MAKECMDGOALS contains a goal with theconfig suffix. The second ifneq will be true when MAKECMDGOALS contains nonconfigtargets, too. Once the variables are set, they are used in an if-else chain with four branches. The code has been condensed and indented to hig hlight its structure:
The first branch, ifeq ($(mixed-targets),1),handles mixed command-line arguments. The only target in this branch is a completely generic pattern rule. Since there are no specific rules to handle targets (those rules are in another conditional branch), each target invokes the pattern rule once. This is how a command line with both configuration targets and build targets is separated into a simpler command line. The command script for the generic pattern rule invokes make recursively for each target, causing this same logic to be applied, only this time with no mixed command-line targets. TheFORCEprerequisite(依賴) is used instead of .PHONY, because pattern rules like: %:: FORCE cannot be declared .PHONY. So it seems reasonable to useFORCE consistently everywhere. The second branch of theif-else chain,ifeq($(config-targets),1), is invoked when there are only configuration targets on the commandline. Here the primary target in the branch is the pattern rule %config(other targets have been omitted). The command script invokesmake recursively in thescripts/kconfigsubdirectory and passes along the target.The curious$(build) construct is defined at the end of themakefile:
If KBUILD_SRC is set, the-foption is given a full path to the scripts makefile, otherwise a simple relative path is used. Next, theobj variable is set to the righthand side of the equals sign. The third branch, ifeq($(dot-config),1),handles build targets that require including the two generated configuration files,.config and.config.cmd.The final branch merely includes a dummy target for autoconf.h to allow it to be used as aprerequisite, even if it doesn't exist. Most of the remainder of the makefile follows the third and fourth branches. It contains the code for building thekernel and modules. 11.2.3 Managing Command EchoThe kernel makefiles use a noveltechnique for managing the level of detailechoed by commands. Each significant task is represented in both averbose and a quiet version. The verbose version is simply thecommand to be executed in its natural form and is stored in avariable namedcmd_action. The briefversion is a short message describing the action and is stored in avariable namedquiet_cmd_action. Forexample, the command to produce emacs tags is:
A command is executed by calling the cmdfunction:
To invoke the code for building emacs tags, themakefile would contain:
Notice the cmd function begins with an@, so the only text echoed by the function is textfrom theecho command. In normal mode, thevariablequiet is empty, and the test in theif,$($(quiet)cmd_$(1)),expands to$(cmd_TAGS). Since this variable is notempty, the entire function expands to:
If the quiet version is desired, the variablequiet contains the valuequiet_and the function expands to:
The variable can also be set to silent_. Sincethere is no commandsilent_cmd_TAGS, this valuecauses thecmdfunction to echo nothing at all. Echoing the command sometimes becomes more complex, particularly ifcommands contain single quotes. In these cases, themakefile contains this code:
Here the echo command contains a substitution thatreplaces single quotes with escaped single quotes to allow them to beproperly echoed. Minor commands that do not warrant the trouble of writingcmd_ andquiet_cmd_ variablesare prefixed with$(Q), which contains eithernothing or@:
11.2.4 User-Defined FunctionsThe kernel makefile defines anumber of functions. Here wecover the most interesting ones. The code has been reformatted toimprove readability. The check_gcc function is used to select agcc command-line option.
The function works by invoking gcc on a null inputfile with the preferred command-line option. The output file,standard output, and standard error files are discarded. If thegcc command succeeds, it means the preferredcommand-line option is valid for this architecture and is returned bythe function. Otherwise, the option is invalid and the alternateoption is returned. An example use can be found inarch/i386/Makefile:
The if_changed_dep function generates dependencyinformation using a remarkable technique.
The function consists of a single if clause. Thedetails of the test are pretty obscure, but it is clear the intent isto be non-empty if the dependency file should be regenerated. Normaldependency information is concerned with the modification timestampson files. The kernel build system adds another wrinkle to this task.The kernel build uses a wide variety of compiler options to controlthe construction and behavior of components. To ensure thatcommand-line options are properly accounted for during a build, themakefile is implemented so that if command-lineoptions used for a particular target change, the file is recompiled.Let's see how this is accomplished. In essence, the command used to compile each file in the kernel issaved in a.cmd file. When a subsequent build isexecuted,make reads the .cmdfiles and compares the current compile command with the last command.If they are different, the.cmd dependency fileis regenerated causing the object file to be rebuilt. The.cmd file usually contains two items: thedependencies that represent actual files for the target file and asingle variable recording the command-line options. For example, thefilearch/i386/kernel/cpu/mtrr/if.c yields this(abbreviated) file:
Getting back to the if_changed_dep function, thefirst argument to thestrip is simply theprerequisites that are newer than the target, if any. The secondargument tostrip is all the prerequisites otherthan files and the empty target FORCE. The reallyobscure bit is the last two filter-out calls:
One or both of these calls will expand to a non-empty string if thecommand-line options have changed. The macro$(cmd_$(1)) is the current command and$(cmd_$@) will be the previous command, forinstance the variablecmd_arch/i386/kernel/cpu/mtrr/if.o just shown. Ifthe new command contains additional options, the firstfilter-out will be empty, and the second willexpand to the new options. If the new command contains fewer options,the first command will contain the deleted options and the secondwill be empty. Interestingly, sincefilter-outaccepts a list of words (each treated as an independent pattern), theorder of options can change and thefilter-outwill still accurately identify added or removed options. Prettynifty. The first statement in the command script sets a shell option to exitimmediately on error. This prevents the multiline script fromcorrupting files in the event of problems. For simple scripts anotherway to achieve this effect is to connect statements with&& rather than semicolons. The next statement is an echo command writtenusing the techniques described inSection 11.2.3 earlier in this chapter,followed by the dependency generating command itself. The commandwrites$(depfile), which is then transformed byscripts/basic/fixdep. The nestedsubst function in thefixdepcommand line first escapes single quotes, then escapes occurrences of$$ (the current process number in shell syntax). Finally, if no errors have occurred, the intermediate file$(depfile) is removed and the generated dependencyfile (with its.cmd suffix) is moved into place. The next function, if_changed_rule, uses thesame comparison technique asif_changed_dep tocontrol the execution of a command:
In the topmost makefile, this function is usedto link the kernel with these macros:
The if_changed_rule function is used to invokerule_vmlinux, which performs the link and buildsthe finalSystem.map. As the comment in themakefile notes, therule_vmlinux__ function must regenerate the kernel version file andrelinkinit.o before relinkingvmlinux. This is controlled by the firstif inrule_vmlinux_ _. Thesecondif controls the echoing of the linkcommand,$(cmd_vmlinux_ _). After the linkcommand, the actual command executed is recorded in a.cmd file for comparison in the next build. |
|