一、本文说明

  本文为linux内核Makefile整体分析的续篇,是依据Linux内核Makefile体系的主要内容编写一个简要的测试工程。Linux内核Makefile体系就好像一只“大鸟”,而这篇测试算是“麻雀”,主要是为了通过动手实战进一步理解Linux内核Makefile体系的原理和特点。

二、源码结构

Linux Kernel Makefile Test-LMLPHP

1、测试源码包括1个顶层目录以及5个子目录。

2、Makefile体系的构成是顶层1个Makefile文件、scripts目录中的4个脚本文件以及add、sub、main三个目录中的Makefile文件。

Linux Kernel Makefile Test-LMLPHP

3、C程序源码是add/add.c、sub/sub.c、main/main.c

三、Makefile重要源码展示

1、顶层Makefile

PHONY :=
MAKE := make srctree := $(if $(KBUILD_SRC),$(KBUILD_SRC),$(CURDIR))
export srctree AS = $(CROSS_COMPILE)as
LD = $(CROSS_COMPILE)ld
CC = $(CROSS_COMPILE)gcc
CPP = $(CC) -E
AR = $(CROSS_COMPILE)ar
NM = $(CROSS_COMPILE)nm export AS LD CC CPP AR NM include $(srctree)/scripts/Kbuild.include //该文件中定义有build变量 all: target srcdir := add sub main
PHONY += $(srcdir) objdir := $(srcdir)
PHONY += $(objdir) target: $(srcdir) //总目标的依赖是srcdir
gcc -o target add/built-in.o sub/built-in.o main/built-in.o //定义总目标的生成规则 $(srcdir): //通过srcdir依次生成各个子目录中的*/built-in.o
$(MAKE) $(build)=$@ clean-dirs := $(addprefix _clean_,$(objdir)) //clean-dirs其实与objdir中的目录是一样的,只是为了执行不同的命令(clean-dirs要删除目标文件),所以要clean-dirs的前边要加上前缀以与srcdir=objdir的目标区别
PHONY += $(clean-dirs) clean $(clean-dirs):
$(MAKE) $(clean)=$(patsubst _clean_%,%,$@) clean: $(clean-dirs)
rm -f target clean := -f scripts/Makefile.clean obj .PHONY: $(PHONY)

2、srcdirs/Makefile.build

# ==========================================================================
# Building
# ========================================================================== src := $(obj) PHONY := __build
__build: //Makefile.build中的总目标 include scripts/Kbuild.include # The filename Kbuild has precedence over Makefile
kbuild-dir := $(if $(filter /%,$(src)),$(src),$(srctree)/$(src))
kbuild-file := $(if $(wildcard $(kbuild-dir)/Kbuild),$(kbuild-dir)/Kbuild,$(kbuild-dir)/Makefile)
include $(kbuild-file) //包含子目录中的Makefile include scripts/Makefile.lib ifneq ($(strip $(obj-y)),)
builtin-target := $(obj)/built-in.o
endif __build: $(builtin-target) # If the list of objects to link is empty, just create an empty built-in.o
cmd_link_o_target = $(if $(strip $(obj-y)),\
$(LD) $(ld_flags) -r -o $@ $(filter $(obj-y), $^)) $(builtin-target): $(obj-y)
$(call if_changed,link_o_target) //调用生成库built-in.o的函数 cmd_cc_o_c = $(CC) $(c_flags) -c -o $@ $< define rule_cc_o_c
$(cmd_cc_o_c)
endef # Built-in and composite module parts
$(obj)/%.o: $(src)/%.c
$(call if_changed_rule,cc_o_c) //调用生成目标文件.o的函数 PHONY += FORCE
FORCE: .PHONY: $(PHONY)

3、srcdirs/Kbuild.include

build := -f scripts/Makefile.build obj

if_changed = $(cmd_$())//定义函数if_changed

# Usage: $(call if_changed_rule,foo)
# Will check if $(cmd_foo) or any of the prerequisites changed,
# and if so will execute $(rule_foo).
if_changed_rule = $(rule_$()) //定义函数if_changed_rule

4、srcdirs/Makefile.lib

c_flags        =        //定义gcc编译器选项

ld_flags       = $(LDFLAGS) $(ldflags-y) //定义ld连接器选项
obj-y := $(patsubst %/, %/built-in.o, $(obj-y)) //对obj-y中的目录添加built-in.o
obj-y := $(addprefix $(obj)/,$(obj-y))//再添加obj前缀

5、srcdirs/Makefile.clean

# ==========================================================================
# Cleaning up
# ========================================================================== src := $(obj) PHONY := __clean //此为Makefile.clean中的总目标
__clean:   
rm -f $(obj)/*.o //删除目标文件.o

6、add/Makefile

obj-y                += add.o

 四、当前源码缺陷

1、没有完成依赖的自动生成

2、没有完成生成目标的目录与源码不在同一目录的功能

附:LinuxKernelMakefileTest.zip

05-11 15:14