Linux系统移植:U-Boot 顶层 Makefile 分析(下)

Posted JeckXu666

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Linux系统移植:U-Boot 顶层 Makefile 分析(下)相关的知识,希望对你有一定的参考价值。

Linux系统移植:U-Boot 顶层 Makefile 分析(下)

继续沿着上的代码内容往下研读

一、调用 scripts/Kbuild.include

下面的代码就是在 Makefile 会调用文件 scripts/Kbuild.include 这个文件

“include” 包含了文件 scripts/Kbuild.include,此文件里面定义了很多变量,我们打开看一下

这些变量在编译工过程中会被调用

二、导出交叉编译工具变量设置

下面的代码就是设置交叉编译器的各种工具

三、导出其他变量

之后的代码导出各种变量

export VERSION PATCHLEVEL SUBLEVEL UBOOTRELEASE UBOOTVERSION
export ARCH CPU BOARD VENDOR SOC CPUDIR BOARDDIR
export CONFIG_SHELL HOSTCC HOSTCFLAGS HOSTLDFLAGS CROSS_COMPILE AS LD CC
export CPP AR NM LDR STRIP OBJCOPY OBJDUMP
export MAKE AWK PERL PYTHON
export HOSTCXX HOSTCXXFLAGS DTC CHECK CHECKFLAGS

export KBUILD_CPPFLAGS NOSTDINC_FLAGS UBOOTINCLUDE OBJCOPYFLAGS LDFLAGS
export KBUILD_CFLAGS KBUILD_AFLAGS

重点是第二行的几个变量:

export ARCH CPU BOARD VENDOR SOC CPUDIR BOARDDIR

我们在程序中加入自己的代码,把这些变量打印出来:

var_test:
	@echo 'ARCH=' $(ARCH)
	@echo 'CPU=' $(CPU)
	@echo 'BOARD=' $(BOARD)
	@echo 'VENDOR=' $(VENDOR)
	@echo 'SOC=' $(SOC)
	@echo 'CPUDIR=' $(CPUDIR)
	@echo 'BOARDDIR=' $(BOARDDIR)

使用 make var_test 执行后的结果如下:

  • ARCH :架构信息
  • CPU :具体CPU架构
  • BOARD :开发板信息
  • VENDOR :厂商
  • SOC :芯片型号
  • CPUDIR :CPU 架构文件目录
  • BOARDDIR:开发板文件目录

这几个变量在 uboot 根目录下 config.mk 里面定义

四、 make xxx_config过程

uboot 可以使用 “make xxx_defconfig” 命令来配置 uboot,配置代码如下:

# To make sure we do not include .config for any of the *config targets
# catch them early, and hand them over to scripts/kconfig/Makefile
# It is allowed to specify more targets when calling make, including
# mixing *config targets and build targets.
# For example 'make oldconfig all'.
# Detect when mixed targets is specified, and make a second invocation
# of make so .config is not included in this case either (for *config).

version_h := include/generated/version_autogenerated.h
timestamp_h := include/generated/timestamp_autogenerated.h

no-dot-config-targets := clean clobber mrproper distclean \\
			 help %docs check% coccicheck \\
			 ubootversion backup

config-targets := 0
mixed-targets  := 0
dot-config     := 1

开头定义了变量 version_h,这变量保存版本号文件(此文件是自动生成的),内容如下:

后面定义了变量 timestamp_h,保存时间戳文件,同时定义了其他的一些变量 no-dot-config-targets、config-targets、mixed-targets 并赋予了初值

下面的代码将 MAKECMDGOALS 中不符合 no-dot-config-targets 的部分过滤掉,剩下的如果不为空的话条件就成立

ifneq ($(filter $(no-dot-config-targets), $(MAKECMDGOALS)),)
	ifeq ($(filter-out $(no-dot-config-targets), $(MAKECMDGOALS)),)
		dot-config := 0
	endif
endif

判断KBUILD_EXTMOD 是否为空,如果KBUILD_EXTMOD为空的话条件成立,将 MAKECMDGOALS 中不符合“config”和“%config”的部分过滤掉,如果剩下的部分不为空条件就成立,然后统计 MAKECMDGOALS 中的单词个数,如果不为 1 的话条件成立

ifeq ($(KBUILD_EXTMOD),)
        ifneq ($(filter config %config,$(MAKECMDGOALS)),)
                config-targets := 1
                ifneq ($(words $(MAKECMDGOALS)),1)
                        mixed-targets := 1
                endif
        endif
endif

words 匹配函数格式如下:

$(words <text>)

判断完成后,几个变量值如下:

config-targets = 1
mixed-targets = 0
dot-config = 1

后面的代码根据值进行进一步执行,下面的代码分析直接写在注释内:

ifeq ($(mixed-targets),1)
# ===========================================================================
# We're called with mixed targets (*config and build targets).
# Handle them one by one.

PHONY += $(MAKECMDGOALS) __build_one_by_one

$(filter-out __build_one_by_one, $(MAKECMDGOALS)): __build_one_by_one
	@:

__build_one_by_one:
	$(Q)set -e; \\
	for i in $(MAKECMDGOALS); do \\
		$(MAKE) -f $(srctree)/Makefile $$i; \\
	done

else
ifeq ($(config-targets),1)
# ===========================================================================
# *config targets only - make sure prerequisites are updated, and descend
# in scripts/kconfig to make the *config target

KBUILD_DEFCONFIG := sandbox_defconfig
export KBUILD_DEFCONFIG KBUILD_KCONFIG

#最终要执行的脚本
config: scripts_basic outputmakefile FORCE
	$(Q)$(MAKE) $(build)=scripts/kconfig $@
%config: scripts_basic outputmakefile FORCE
	$(Q)$(MAKE) $(build)=scripts/kconfig $@

else
# ===========================================================================
# Build targets only - this includes vmlinux, arch specific targets, clean
# targets and others. In general all targets except *config targets.

ifeq ($(dot-config),1)
# Read in config
-include include/config/auto.conf

输入的指令最终执行下面的两句

①、scripts_basic 目标对应的命令

make -f ./scripts/Makefile.build obj=scripts/basic

②、%config 目标对应的命令

make -f ./scripts/Makefile.build obj=scripts/kconfig xxx_defconfig

其具体的执行流程如下:

五、make过程

先直接看 make 的执行流程示意图

流程解释如下:

使用 make 命令后,先使用默认目标 _all,如下:

目标_all 又依赖于 all,然后加载 all 依赖

在主 Makefile 中 all 目标规则如下

具体 All-y 规则:

ALL-y 主要包含 u-boot.srec、u-boot.bin、u-boot.sym、System.map、u-boot.cfg 和 binary_size_check 这几个文件。根据 uboot 的配置情况也可能包含其他的文件

ALL-y 里面有个 u-boot.bin,这个就是我们最终需要的 uboot 二进制可执行文件,目标 u-boot.bin 依赖于 u-boot-nodtb.bin 文件,所以又先生成 u-boot-nodtb.bin 文件,在 Makefile 中代码如下:

而目标 u-boot-nodtb.bin 又依赖于 u-boot,u-boot 相关规则如下:

规则就相当于将以 u-boot.lds 为链接脚本,将 arch/arm/cpu/armv7/start.o 和各个子目录下的 built-in.o 链接在一起生成 u-boot

表现在程序上最终是用 arm-linux-gnueabihf-ld.bfd 命令将 arch/arm/cpu/armv7/start.o 和其他众多的 built_in.o 链接在一起,形成 u-boot,以上就是 make 的大致流程,其 Makefile 主要就下面两个功能

  • make xxx_defconfig:配置 uboot,生成.config 文件
  • make:编译 uboot,生成二进制的 u-boot.bin 文件和其他的一些与 uboot 有关的文件,比如 u-boot.imx 等等

以上是关于Linux系统移植:U-Boot 顶层 Makefile 分析(下)的主要内容,如果未能解决你的问题,请参考以下文章

Linux系统移植:U-Boot 工程创建

U-BOOT-2016.07移植 (第一篇) 初步分析

Linux系统移植:正点原子 U-Boot 移植

04.移植u-boot

Linux系统移植:U-Boot 链接脚本

Linux系统移植:NXP 官板 uboot 移植