在编译 u-boot 之前都要执行”make XXX_config”命令,即配置 u-boot,使其编译出适合目标板的 bootloader。那么该命令都做了那些工作,具体的执行过程是怎样的?
我们首先从 u-boot 的 Makefile 文件看起,例如我们首先执行”make smdk2410_config”命令,则在 Makefile 中会执行:
MKCONFIG := $(SRCTREE)/mkconfig
export MKCONFIG
smdk2410_config : unconfig
@$(MKCONFIG) $(@:_config=) arm arm920t smdk2410 NULL s3c24x0
假定在 U-Boot-1.3.4 的根目录下编译,则其中的 MKCONFIG 就是根目录下的 mkconfig文件。
$(@:_config=):是一个参数,@代表smdk2410_config,然后将_config替换为空, 结果为 “smdk2410” 。所以“make smdk2410_config”实际上就是执行如下命令:
./mkconfig smdk2410 arm arm920t smdk2410 NULL s3c24x0
$0 $1 $2 $3 $4 $5 $6
再来看看 mkconfig 的作用,在 mkconfig 文件开头给出了它的用法:
# Parameters: Target Architecture CPU Board [VENDOR] [SOC]
也就是说执行” make XXX_config”实际上执行的是 mkconfig 脚本,下面是对mkconfig 文件的分析:
#!/bin/sh -e
APPEND=no # Default: Create new config file
BOARD_NAME="" # Name to print in make output
#如果命令行参数中有--,-a,-n *等参数,则执行以下循环。
#如果有-n XXX_config 或者-n XXX,则取出 XXX 作为目标板的名字
while [ $# -gt 0 ] ; do
case "$1" in
--) shift ; break ;; #break是跳出while而不是结束case
-a) shift ; APPEND=yes ;; #shift是对参数进行左移一个,即下一个参数
-n) shift ; BOARD_NAME="${1%%_config}" ; shift ;;
*) break ;; #*是任意匹配符
esac #注意最后是两个分号
done
#如果 BOARD_NAME 为空,则 BOARD_NAME 等于传递给该脚本的第一个参数
[ "${BOARD_NAME}" ] || BOARD_NAME="$1"
#如果传递给该脚本的参数小于 4 个或者大于 6 个,则退出
[ $# -lt 4 ] && exit 1
[ $# -gt 6 ] && exit 1
#配置成功后将打印出这一句话
echo "Configuring for ${BOARD_NAME} board..."
#OBJTREE 和 SRCTREE 都是在 Makefile 中导出的变量,分别为编译目录和源码目录
#如果编译目录和源码目录不为同一目录,则执行一下命令,创建$(OBJTREE)/include 等目录
if [ "$SRCTREE" != "$OBJTREE" ] ; then
mkdir -p ${OBJTREE}/include
mkdir -p ${OBJTREE}/include2
cd ${OBJTREE}/include2
rm -f asm
ln -s ${SRCTREE}/include/asm-$2 asm
LNPREFIX="../../include2/asm/"
cd ../include
rm -rf asm-$2
rm -f asm
mkdir asm-$2
ln -s asm-$2 asm
else
#如果编译目录和源码目录为同一个目录,则进入 include 目录,删除旧的
#asm 链接,创建目标板到 asm 的链接,例如如果目标体系结构为 arm,则创建 asm 软链接指向 asm-arm。
cd ./include
rm -f asm
ln -s asm-$2 asm
fi
#删除目标平台下的旧的 arch 链接
rm -f asm-$2/arch
#如果第 6 个参数长度为 0 或者其等于 NULL,则创建软链接asm-$2/arch 指向arch-$3,对于命令
#"./mkconfig smdk2410 arm arm920t smdk2410 NULL s3c24x0"
# $6=s3c24x0 不为空
# $3=arm920t $2=arm
#则最终执行的命令为"ln -s arch-s3c24x0 asm-arm/arch"
if [ -z "$6" -o "$6" = "NULL" ] ; then
ln -s ${LNPREFIX}arch-$3 asm-$2/arch
else
ln -s ${LNPREFIX}arch-$6 asm-$2/arch
fi
#如果目标平台为 ARM,则删除建立的 asm-arm 链接,重新建立从 proc-armv 到 asm-arm 的链接
if [ "$2" = "arm" ] ; then
rm -f asm-$2/proc
ln -s ${LNPREFIX}proc-armv asm-$2/proc
fi
# Create include file for Make
#单个>表示创建文件,两个>>表示追加
echo "ARCH = $2" > config.mk
echo "CPU = $3" >> config.mk
echo "BOARD = $4" >> config.mk
#如果参数 5 存在并且不为 NULL,则将 VENDOR = $5 追加在 config.mk 文件中
[ "$5" ] && [ "$5" != "NULL" ] && echo "VENDOR = $5" >> config.mk
#如果参数 6 存在并且不为 NULL,则将 SOC = $6 追加在 config.mk 文件中
[ "$6" ] && [ "$6" != "NULL" ] && echo "SOC = $6" >> config.mk
#创建 config.h 文件
if [ "$APPEND" = "yes" ] # Append to existing config file
then
echo >> config.h
else
> config.h # Create new config file
fi
echo "/* Automatically generated - do not edit */" >>config.h
echo "#include <configs/$1.h>" >>config.h
exit 0
总结一下,” make XXX_config”总共做了一下工作:
- 确定开发板名称为 BOARD_NAME = $1。
- 创建一些链接文件,将来在写代码过程中,头文件包含时非常有用。譬如一个头文件包含可能是:
#include <asm/xx.h>
:
ln -s asm-$2 asm
ln -s arch-$6 asm-$2/arch
ln -s proc-armv asm-$2/proc #仅在目标平台为 arm 的时候才执行。
- 创建顶层 Makefile 包含的文件 include/config.mk。
ARCH = $2
CPU = $3
BOARD = $4
VENDOR = $5
SOC = $6
- 创建开发板相关的头文件 include/config.h。
/* Automatically generated - do not edit */
include <configs/smdk2410 .h>