从头开始编译Linux时,我意识到在编译时会出现编译代码。
例如CC文件名,LD文件名,CC [M]文件名。
这些代码是什么意思?
不同的标记指定以下内容
让我们举一个具体的例子,并找出它在内核4.1中的作用,例如IHEX
。
找到一个代码做什么
赶紧跑:
make SHELL='sh -x'
这是如何工作的: https : //stackoverflow.com/a/32010960/895245
如果我们grep IHEX
的输出,我们找到这些行:
+ echo IHEX firmware/e100/d101s_ucode.bin IHEX firmware/e100/d101s_ucode.bin + objcopy -Iihex -Obinary /home/ciro/git/kernel/src/firmware/e100/d101s_ucode.bin.ihex firmware/e100/d101s_ucode.bin
所以我们得出结论IHEX
做了objcopy -Iihex
。
找到代码的定义
每个内核命令都必须使用如下的定义:
quiet_cmd_ihex = IHEX $@ cmd_ihex = $(OBJCOPY) -Iihex -Obinary $< $@ $(obj)/%: $(obj)/%.ihex $(call cmd,ihex)
对于详细设置(例如V=1
和make -s
)的工作。
所以一般来说,你只需要
git grep 'cmd.* = CODE'
找到CODE
。
我已经详细解释了这个系统是如何工作的: https : //stackoverflow.com/a/32023861/895245
获取所有代码的列表
make | grep -E '^ ' | sort -uk1,1
CC和CC [M]
在scripts/Makefile.build
定义:
quiet_cmd_cc_o_c = CC $(quiet_modtag) $@ cmd_cc_o_c = $(CC) $(c_flags) -c -o $@ $<
而[M]
来自目标特定变量 :
$(real-objs-m) : quiet_modtag := [M] $(real-objs-m:.o=.i) : quiet_modtag := [M] $(real-objs-m:.o=.s) : quiet_modtag := [M] $(real-objs-m:.o=.lst): quiet_modtag := [M] $(obj-m) : quiet_modtag := [M]
然后通过:
$(obj)/%.o: $(src)/%.c $(recordmcount_source) FORCE [...] $(call if_changed_rule,cc_o_c) define rule_cc_o_c [...] $(call echo-cmd,cc_o_c) $(cmd_cc_o_c); \
其中if_changed_rule
在scripts/Kbuild.include
定义为:
if_changed_rule = $(if $(strip $(any-prereq) $(arg-check) ), \ @set -e; \ $(rule_$(1)))
和Kbuild.include
包含在顶层的Makefile中。
LD
有几个版本,但最简单的似乎是:
quiet_cmd_link_o_target = LD $@ cmd_link_o_target = $(if $(strip $(obj-y)),\ $(LD) $(ld_flags) -r -o $@ $(filter $(obj-y), $^) \ $(cmd_secanalysis),\ rm -f $@; $(AR) rcs$(KBUILD_ARFLAGS) $@) $(builtin-target): $(obj-y) FORCE $(call if_changed,link_o_target)
并在scripts/Kbuild.include
:
# Execute command if command has changed or prerequisite(s) are updated. # if_changed = $(if $(strip $(any-prereq) $(arg-check)), \ @set -e; \ $(echo-cmd) $(cmd_$(1)); \ printf '%s\n' 'cmd_$@ := $(make-cmd)' > $(dot-target).cmd)
它应该显示:
CC
编译内核的核心部分 CC [M]
编译模块时 LD
链接时