分析整个uboot首先分析makefile。。
整个uboot包括两个阶段,1、配置 2、编译
一、 配置
我们在配置的时候 执行make smdk2410_config
我们在makefile中找到这条配置命令:smdk2400_config : unconfig
@$(MKCONFIG) $(@:_config=) arm arm920t smdk2400 NULL s3c24x0
而MKCONFIG这个变量在makefile开始部分有定义::MKCONFIG := $(SRCTREE)/mkconfig 表示当前目录下的mkconfig文件
# mkconfig smdk2410 arm arm920t smdk2410 NULL s3c24x0
# $0 $1 $2 $3 $4 $5 $6
APPEND=no # Default: Create new config file
BOARD_NAME="" # Name to print in make output
#$# 表示存入参数个数 显然下面的while不成立
while [ $# -gt 0 ] ; do
case "$1" in
--) shift ; break ;;
-a) shift ; APPEND=yes ;;
-n) shift ; BOARD_NAME="${1%%_config}" ; shift ;;
*) break ;;
esac
done
#由上面定义知BOARD_NAME为空 所以执行后面的BOARD_NAME = smdk2410
[ "${BOARD_NAME}" ] || BOARD_NAME="$1"
#$# 参数的个数 如果小于4退出 大于6也退出 这里参数个数为6 不会退出
[ $# -lt 4 ] && exit 1
[ $# -gt 6 ] && exit 1
#输出下面的Configuring for smdk2410 board... 在执行make smdk2410_config时会显示
echo "Configuring for ${BOARD_NAME} board..."
#
# Create link to architecture specific headers
#我们的源码与OBJTREE是相等的 makefile里面可以知道 所以执行下面的else
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
# 进入当前目录的下级 include目录
# 删除 asm
# 建立一个链接 ln -s asm-arm asm
else
cd ./include
rm -f asm
ln -s asm-$2 asm
fi
rm -f asm-$2/arch #删除 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
if [ "$2" = "arm" ] ; then
rm -f asm-$2/proc
ln -s ${LNPREFIX}proc-armv asm-$2/proc
fi
#
# Create include file for Make 看注释名字就知道 下面是生成一个make配置文件
#
echo "ARCH = $2" > config.mk #>表示新建一个config。mk文件
echo "CPU = $3" >> config.mk #》表示追加文件到config。mk
echo "BOARD = $4" >> config.mk
[ "$5" ] && [ "$5" != "NULL" ] && echo "VENDOR = $5" >> config.mk
[ "$6" ] && [ "$6" != "NULL" ] && echo "SOC = $6" >> config.mk
#
# Create board specific header file 创建一个板子相关的头文件
#
if [ "$APPEND" = "yes" ] # Append to existing config file
then
echo >> config.h
else
> config.h # Create new config file 新建一个头文件 config。h
fi
echo "/* Automatically generated - do not edit */" >>config.h #里面增加左边的注释
echo "#include <configs/$1.h>" >>config.h #同时包含另一个头文件 configs/smdk2410.h 以后我们的配置文件都在smdk2410.h里面说明 支持什么什么命令 定义什么什么宏
exit 0
二、编译过程
由上往下看 首先看到下面一段
ifeq ($(OBJTREE)/include/config.mk,$(wildcard $(OBJTREE)/include/config.mk))
# load ARCH, BOARD, and CPU configuration
# 分析 make编译过程 由此行开始看 首先 包含当前目录/include/config.mk文件 里面定义了架构 arm cpu board啊。。。。 ARCH =ARM
include $(OBJTREE)/include/config.mk
export ARCH CPU BOARD VENDOR SOC
ifeq ($(ARCH),arm) #接下来指定我们的编译工具
CROSS_COMPILE = arm-linux-
endif
#########################################################################
# U-Boot objects....order is important (i.e. start must be first)
#接下来指定源文件 OBJS = cpu/arm920T/start.o
OBJS = cpu/$(CPU)/start.o
ifeq ($(CPU),i386)
OBJS += cpu/$(CPU)/start16.o
OBJS += cpu/$(CPU)/reset.o
endif
ifeq ($(CPU),ppc4xx)
OBJS += cpu/$(CPU)/resetvec.o
endif
ifeq ($(CPU),mpc83xx)
OBJS += cpu/$(CPU)/resetvec.o
endif
ifeq ($(CPU),mpc85xx)
OBJS += cpu/$(CPU)/resetvec.o
endif
ifeq ($(CPU),mpc86xx)
OBJS += cpu/$(CPU)/resetvec.o
endif
ifeq ($(CPU),bf533)
OBJS += cpu/$(CPU)/start1.o cpu/$(CPU)/interrupt.o cpu/$(CPU)/cache.o
OBJS += cpu/$(CPU)/cplbhdlr.o cpu/$(CPU)/cplbmgr.o cpu/$(CPU)/flush.o
endif
OBJS := $(addprefix $(obj),$(OBJS))
上面的$(CPU),在开始的config。mk里指定了 假设为arm920T,所以源文件为cpu/arm920T/start。o
#下面指定lib路径
LIBS = lib_generic/libgeneric.a
LIBS += board/$(BOARDDIR)/lib$(BOARD).a #等效于 board/smdk2410/libsmdk2410.a
LIBS += cpu/$(CPU)/lib$(CPU).a #等效于 cpu/arm920T/libarm920T.a
ifdef SOC
LIBS += cpu/$(CPU)/$(SOC)/lib$(SOC).a
endif
LIBS += lib_$(ARCH)/lib$(ARCH).a
LIBS += fs/cramfs/libcramfs.a fs/fat/libfat.a fs/fdos/libfdos.a fs/jffs2/libjffs2.a \
fs/reiserfs/libreiserfs.a fs/ext2/libext2fs.a
LIBS += net/libnet.a
LIBS += disk/libdisk.a
LIBS += rtc/librtc.a
LIBS += dtt/libdtt.a
LIBS += drivers/libdrivers.a
LIBS += drivers/nand/libnand.a
LIBS += drivers/nand_legacy/libnand_legacy.a
LIBS += drivers/sk98lin/libsk98lin.a
LIBS += post/libpost.a post/cpu/libcpu.a
LIBS += common/libcommon.a
LIBS += $(BOARDLIBS)
ALL = $(obj)u-boot.srec $(obj)u-boot.bin $(obj)System.map $(U_BOOT_NAND)
#生成目标 all 后面是依赖 防止没有指定 于是生成第一个目标 makefile规定
# 这里如果我们想看反汇编文件 可以在上面的ALL里加入$(obj)u-boot.dis
all: $(ALL)
$(obj)u-boot.hex: $(obj)u-boot
$(OBJCOPY) ${OBJCFLAGS} -O ihex $< $@
$(obj)u-boot.srec: $(obj)u-boot
$(OBJCOPY) ${OBJCFLAGS} -O srec $< $@
$(obj)u-boot.bin: $(obj)u-boot #这里的u-boot为elf格式 使他生成开发板可以执行的bin格式
$(OBJCOPY) ${OBJCFLAGS} -O binary $< $@
$(obj)u-boot.img: $(obj)u-boot.bin
./tools/mkimage -A $(ARCH) -T firmware -C none \
-a $(TEXT_BASE) -e 0 \
-n $(shell sed -n -e 's/.*U_BOOT_VERSION//p' $(VERSION_FILE) | \
sed -e 's/"[ ]*$$/ for $(BOARD) board"/') \
-d $< $@
$(obj)u-boot.dis: $(obj)u-boot
$(OBJDUMP) -d $< > $@
#这里找到了我们生成u-boot_elf格式文件所需要的所有源文件 包括.s .c .h
$(obj)u-boot: depend version $(SUBDIRS) $(OBJS) $(LIBS) $(LDSCRIPT)
UNDEF_SYM=`$(OBJDUMP) -x $(LIBS) |sed -n -e 's/.*\(__u_boot_cmd_.*\)/-u\1/p'|sort|uniq`;\
cd $(LNDIR) && $(LD) $(LDFLAGS) $$UNDEF_SYM $(__OBJS) \
--start-group $(__LIBS) --end-group $(PLATFORM_LIBS) \