18

在从头开始编译 Linux 时,我意识到编译时会出现编译代码。

例如 CC 文件名、LD 文件名、CC[M] 文件名。

这些代码是什么意思?

4

3 回答 3

20

不同的标记指定以下内容

  • [CC] - 将 C 文件编译成指定的目标文件。目标文件包含该 .c 文件的架构汇编代码。因为它也可能引用其范围之外的部分。例如在另一个 .c 文件中调用另一个函数。函数调用在目标文件中保持打开状态,稍后将包含在链接器中。所以
  • [LD] 是将编译后的对象链接在一起的过程,并将编译器保持打开的函数调用连接起来。但是,许多部分作为内核的核心部分链接在一起,而有些部分则被遗漏了。因此你看到
  • [CC (M)] 用于那些在运行时被编译为点加载到内核中的部分。但是它们在内核的整体部分中没有链接在一起。但是可以在内核启动时插入。
于 2012-07-28T06:07:21.867 回答
9

让我们举一个具体的例子,看看它在内核 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

所以我们得出的结论是IHEXa objcopy -Iihex

查找代码的定义位置

每个内核命令都必须定义为:

quiet_cmd_ihex  = IHEX    $@
      cmd_ihex  = $(OBJCOPY) -Iihex -Obinary $< $@

$(obj)/%: $(obj)/%.ihex
        $(call cmd,ihex)

使详细设置(例如V=1make -s)起作用。

所以一般来说,你只需要

git grep 'cmd.* = CODE'

找到CODE

我已经详细解释了这个系统是如何工作的:https ://stackoverflow.com/a/32023861/895245

获取所有代码列表

make | grep -E '^  ' | sort -uk1,1

抄送和抄送 [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)
于 2015-08-15T11:16:37.993 回答
3

它应该显示:

  • CC编译内核的核心部分时
  • CC [M]编译模块时
  • LD链接时
于 2012-07-28T05:14:31.997 回答