📄 196570.html
字号:
<br>==========================================
<br>Linux各级内核源代码的子目录下都有Makefile,大多数Makefile要嵌入主目录下的Rule.make,Rule.make将识
别各个Makefile中所定义的一些变量。变量obj-y表示需要编绎到内核中的目标文件名集合,定义O_TARGET表示将obj-y连接为一个
O_TARGET名称的目标文件,定义L_TARGET表示将obj-y合并为一个L_TARGET名称的库文件。同样obj-m表示需要编绎成模块的目
标文件名集合。如果还需进行子目录make,则需要定义subdir-y和subdir-m。在Makefile中,用"obj-$
(CONFIG_BINFMT_ELF) += binfmt_elf.o"和"subdir-$(CONFIG_EXT2_FS)
+= ext2"这种形式自动为obj-y、obj-m、subdir-y、subdir-m添加文件名。有时,情况没有这么单
纯,还需要使用条件语句个别对待。Makefile中还有其它一些变量,如mod-subdirs定义了subdir-m以外的所有模块子目录。
<br>
<br>Rules.make是如何使make进入子目录的呢? 先来看subdir-y是如何处理的,在Rules.make中,先对
subdir-y中的每一个文件名加上前缀"_subdir_"再进行排序生成subdir-list集合,再以它作为目标集,对其中每一个目标产生一个
子make,同时将目标名的前缀去掉得到子目录名,作为子make的起始目录参数。subdir-m与subdir-y类似,但情况稍微复杂一些。由于
subdir-y中可能有模块定义,因此利用mod-subdirs变量将subdir-y中模块目录提取出来,再与subdir-m合成一个大的
MOD_SUB_DIRS集合。subdir-m的目标所用的前缀是"_modsubdir_"。
<br>
<br>一点说明,子目录中的Makefile与Rules.make都没有嵌入.config文件,它是通过主Makefile向下传递
MAKEFILES变量完成的。MAKEFILES是make自已识别的一个变量,在执行新的Makefile之前,make会首先加载
MAKEFILES所指的文件。在主Makefile中它即指向.config。
<br>
<br>
<br>==========================================
<br>Makefile解读之三: 模块的版本化处理
<br>==========================================
<br>模块的版本化是内核与模块接口之间进行严格类型匹配的一种方法。当内核配置了CONFIG_MODVERSIONS之后,make
dep操作会在include/linux/modules/目录下为各级Makefile中export-objs变量所对应的源文件生成扩展名为.
ver的文件。
<br>
<br>例如对于kernel/ksyms.c,make用以下命令生成对应的ksyms.ver:
<br>
<br>gcc -E -D__KERNEL__ -D__GENKSYMS__ ksyms.c | /sbin/genksyms -k 2.4.1 > ksyms.ver
<br>
<br>-D__GENKSYMS__的作用是使ksyms.c中的EXPORT_SYMBOL宏不进行扩展。genksyms命令识别EXPORT_SYMBOL()中的函数名和对应的原型,再根据其原型计算出该函数的版本号。
<br>
<br>例如ksyms.c中有一行:
<br>EXPORT_SYMBOL(kmalloc);
<br>kmalloc原型是:
<br>void *kmalloc(size_t, int);
<br>genksyms程序对应的输出为:
<br>#define __ver_kmalloc 93d4cfe6
<br>#define kmalloc _set_ver(kmalloc)
<br>在内核符号表和模块中,kmalloc将变成kmalloc_R93d4cfe6。
<br>
<br>在生成完所有的.ver文件后,make将重建include/linux/modversions.h文件,它包含一系列#include指令
行嵌入各个.ver文件。在编绎内核本身export-objs中的文件时,make会增加一个"-DEXPORT_SYMTAB"编绎标志,它使源文件
嵌入modversions.h文件,将EXPORT_SYMBOL宏展开中的函数名字符串进行版本名扩展;同时,它也定义_set_ver()宏为一空
操作,使代码中的函数名不受其影响。
<br>在编绎模块时,make会增加"-include=linux/modversion.h -DMODVERSIONS"编绎标志,使模块中代码的函数名得到相应版本扩展。
<br>
<br>由于生成.ver文件比较费时,make还为每个.ver创建了一个后缀为.stamp时戳文件。在make dep时,如果其.
stamp文件比源文件旧才重新生成.ver文件,否则只是更新.stamp文件时戳。另外,在生成.ver和modversions.h文件时,
make都会比较新文件和旧文件的内容,保持它们修改时间为最旧。
<br>
<br>
<br>
<br>==========================================
<br>Makefile解读之四: Rules.make的注释
<br>==========================================
<br>[code:1:974578564b]
<br>#
<br># This file contains rules which are shared between multiple Makefiles.
<br>#
<br>
<br>#
<br># False targets.
<br>#
<br>#
<br>.PHONY: dummy
<br>
<br>#
<br># Special variables which should not be exported
<br>#
<br># 取消这些变量通过环境向make子进程传递。
<br>unexport EXTRA_AFLAGS # as 的开关
<br>unexport EXTRA_CFLAGS # cc 的开关
<br>unexport EXTRA_LDFLAGS # ld 的开关
<br>unexport EXTRA_ARFLAGS # ar 的开关
<br>unexport SUBDIRS #
<br>unexport SUB_DIRS # 编绎内核需进入的子目录,等于subdir-y
<br>unexport ALL_SUB_DIRS # 所有的子目录
<br>unexport MOD_SUB_DIRS # 编绎模块需进入的子目录
<br>unexport O_TARGET # ld合并的输出对象
<br>unexport ALL_MOBJS # 所有的模块名
<br>
<br>unexport obj-y # 编绎成内核的文件集
<br>unexport obj-m # 编绎成模块的文件集
<br>unexport obj-n #
<br>unexport obj- #
<br>unexport export-objs # 需进行版本处理的文件集
<br>unexport subdir-y # 编绎内核所需进入的子目录
<br>unexport subdir-m # 编绎模块所需进入的子目录
<br>unexport subdir-n
<br>unexport subdir-
<br>
<br>#
<br># Get things started.
<br>#
<br>first_rule: sub_dirs
<br> $(MAKE) all_targets
<br>
<br># 在内核编绎子目录中过滤出可以作为模块的子目录。
<br>both-m := $(filter $(mod-subdirs), $(subdir-y))
<br>SUB_DIRS := $(subdir-y)
<br># 求出总模块子目录
<br>MOD_SUB_DIRS := $(sort $(subdir-m) $(both-m))
<br># 求出总子目录
<br>ALL_SUB_DIRS := $(sort $(subdir-y) $(subdir-m) $(subdir-n) $(subdir-))
<br>
<br>
<br>#
<br># Common rules
<br>#
<br># 将c文件编绎成汇编文件的规则,$@为目标对象。
<br>%.s: %.c
<br> $(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -S $< -o $@
<br># 将c文件生成预处理文件的规则。
<br>%.i: %.c
<br> $(CPP) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) $< > $@
<br># 将c文件编绎成目标文件的规则,$<为第一个所依赖的对象;
<br>#
<br>在目标文件的目录下生成flags文件,strip删除多余的空格,subst将逗号替换成冒号
<br>。
<br>%.o: %.c
<br> $(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -c -o $@ $<
<br> @ ( \
<br> echo 'ifeq ($(strip $(subst $(comma),:,$(CFLAGS) $(EXTRA_CFLAGS)
<br>$(CFLAGS_$@))),$$(strip $$(subst $$(comma),:,$$(CFLAGS) $$(EXTRA_CFLAGS)
<br>$$(CFLAGS_$@))))' ; \
<br> echo&nbs'''FILES_FLAGS_UP_TO_DATE += ''' ; \
<br> echo&nb;'''enf''' \
<br> ) > $(dir $@)/.$(notdir $@).flags
<br># 汇编文件生成目标文件的规则。
<br>%.o: %.s
<br> $(AS) $(AFLAGS) $(EXTRA_CFLAGS) -o $@ $<
<br>
<br># Old makefiles define their own rules for compiling .S files,
<br># but these standard rules are available for any Makefile that
<br># wants to use them. Our plan is to incrementally convert all
<br># the Makefiles to these standard rules. -- rmk, mec
<br>
<br>ifdef USE_STANDARD_AS_RULE
<br># 汇编文件生成预处理文件的标准规则。
<br>%.s: %.S
<br> $(CPP) $(AFLAGS) $(EXTRA_AFLAGS) $(AFLAGS_$@) $< > $@
<br># 汇编文件生成目标文件的标准规则。
<br>%.o: %.S
<br> $(CC) $(AFLAGS) $(EXTRA_AFLAGS) $(AFLAGS_$@) -c -o $@ $<
<br>
<br>endif
<br># c文件生成调试列表文件的规则,$*扩展为目标的主文件名。
<br>%.lst: %.c
<br> $(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -g -c -o $*.o $<
<br> $(TOPDIR)/scripts/makelst $* $(TOPDIR) $(OBJDUMP)
<br>#
<br>#
<br>#
<br>all_targets: $(O_TARGET) $(L_TARGET)
<br>
<br>#
<br># Rule to compile a set of .o files into one .o file
<br>#
<br>ifdef O_TARGET
<br>$(O_TARGET): $(obj-y)
<br> rm -f $@
<br># $^扩展为全部依赖对象,如果obj-y为空就生成一个同名空的库文件。
<br> ifneq "$(strip $(obj-y))" ""
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -