uboot的编译和链接过程分析(一)

时间:2021-07-20 16:36:00

1.make xc2440_config

2.make all

从这两个步骤来理解,中间好多gcc 参数和shell脚本搞不清楚,先放一放,整个流程搞完了,我还有回头学习shell脚本.

make xc2440_config 执行的结果是:

1.include下面生成了config.mk  config.h. 

2.在include下面建立一个连接asm到arch/arm/include/asm

uboot的编译和链接过程分析(一)

#!/bin/sh -e

#
#			   xc2440_config       
#	@$(MKCONFIG) $(@:_config=) arm arm920t xc2440 NULL s3c24x0

APPEND=no	# Default: Create new config file
BOARD_NAME=""	# Name to print in make output
TARGETS=""


#BOARD_NAME = xc2440
[ "${BOARD_NAME}" ] || BOARD_NAME="$1"				

#s#我们用的是6,这里都不满足条件都可以删除掉.
#[ $# -lt 4 ] && exit 1
#[ $# -gt 6 ] && exit 1

#-a与的意思
#没看到,实际应该没有执行.
if [ "${ARCH}" -a "${ARCH}" != "$2" ]; then
	echo "Failed: \$ARCH=${ARCH}, should be '$2' for ${BOARD_NAME}" 1>&2
	exit 1
fi

echo "============================================================"
echo "执行./mkconifg $*"
echo "Configuring for ${BOARD_NAME} board...by zhangquan\n"

echo "ln -s /arch/arm/include/asm		include/asm"
echo "ln -s asm/arch-s3c2440			incluce/asm/arch"
echo "ln -s asm/proc-armv       		include/asm/proc"

echo "\nIt will create incluce/config.h include/config.mk"

echo "ARCH	= $2		ARCH----->arch/arm"
echo "CPU	= $3		CPU------>arch/arm/cpu/arm920t"
echo "BOARD	= $4		BOARD---->board/xc2440"
echo "SOC	= $6		SOC------>arch/arm/cpu/arm920t/s3c24x0"
echo "============================================================"

#
# Create link to architecture specific headers
#

#在Makefile里面我们知道 OBJTREE == OBJTREE 
#执行else部分
if [ "$OBJTREE" != "$OBJTREE" ] ; then
	mkdir -p ${OBJTREE}/include
	mkdir -p ${OBJTREE}/include2
	cd ${OBJTREE}/include2
	rm -f asm
	ln -s ${SRCTREE}/arch/$2/include/asm asm
	LNPREFIX=${SRCTREE}/arch/$2/include/asm/
	cd ../include
	rm -f asm
	ln -s ${SRCTREE}/arch/$2/include/asm asm
else
	cd ./include
	rm -fr asm
	ln -s ../arch/$2/include/asm asm
	#ln -s ../arch/arm/include/asm asm	  反正在uboot/include下面多了一个asm  
fi

#删除asm/arch ---->arch/arm/include/arch-$6,
#上一步决定的.
rm -fr asm/arch

if [ -z "$6" -o "$6" = "NULL" ] ; then
	ln -s ${LNPREFIX}arch-$3 asm/arch		#arch-arm920t这个不存在,所以$6不能为空,否则编译错误.
else
	ln -s ${LNPREFIX}arch-$6 asm/arch		#所以arch-s3c24x0  是asm/arch真实的路径.  在include/asm/arch下面就是arch-s3c24x0的内容.		
fi											#include <asm/arch/s3c2410.h> 我看到这样的一个包含关系,实际还是arch/arm/asm/arch-s3c2410/s3c2410.h
										    #<asm/arch/s3c2410.h>这样写,少写了好多绝对路径.方便头文件的包含.
										    #注释到这儿了,我想起了好多头文件的包含,是不是都是在这儿?
										    #以后的6410相关的头文件,都可以放到include/arch下面.

if [ "$2" = "arm" ] ; then
	rm -f asm/proc							#asm/proc下面的内容是proc-armv的内容.
	ln -s ${LNPREFIX}proc-armv asm/proc		#这个用到哪儿去了,我还没有体会到.
fi

#
# Create include file for Make				#因为上一步是cd ./include ,所以现在在include路径下.config.mk 也在include这个路径下生成.
#
echo "ARCH   = $2" >  config.mk
echo "CPU    = $3" >> config.mk
echo "BOARD  = $4" >> config.mk


#我们的$5 = NULL
[ "$5" ] && [ "$5" != "NULL" ] && echo "VENDOR = $5" >> config.mk

[ "$6" ] && [ "$6" != "NULL" ] && echo "SOC    = $6" >> config.mk

# Assign board directory to BOARDIR variable
if [ -z "$5" -o "$5" = "NULL" ] ; then
    BOARDDIR=$4							   #我在这里$5厂商写了NUll 所以BOARDIR = xc2440 如果默认的就是samsung/xc2440
else
    BOARDDIR=$5/$4
fi

#
# Create board specific header file
#
if [ "$APPEND" = "yes" ]	# Append to existing config file
then
	echo >> config.h
else
	> config.h		# Create new config file		#在mkconfig的开头,APPEND=no, 所以在这里创建了config.h
fi
echo "/* Automatically generated - do not edit */" >>config.h

#开头可以看见TARGETS为空.这里不执行的.
for i in ${TARGETS} ; do
	echo "#define CONFIG_MK_${i} 1" >>config.h ;
done

#这里的“cat << EOF >> config.h”表示将输入的内容追加到config.h中,直到出现“EOF”这样的标识为止。 不明白
cat << EOF >> config.h
#define CONFIG_BOARDDIR board/$BOARDDIR	
#include <config_defaults.h>
#include <configs/$1.h>							    
#include <asm/config.h>						        
EOF

exit 0

#最终的产生的变化就是
#1.在uboot/include下面多了一个链接文asm。  
#	asm下面多了一个链接文件arch 一个链接文件proc ,
#		arch指向asm-s3c24x0(里面是cpu相关的文件,比如寄存器地址)
#		proc指向pro-armv(里面很多汇编的东西)
#   这个asm是arch/arm/include/asm的链接.
#
#	ln -s /arch/arm/include/asm   include/asm
#	ln -s asm/arch-s3c2440 		  incluce/asm/arch
#	ln -s asm/proc-armv 		  include/asm/proc

#在incluce 下面生成config.mk  这个文件然后被Makefile包含进来了了,要用到这些参数。
#	ARCH   = arm			ARCH----->对应arch/arm目录.
#	CPU    = arm920t	        CPU------>对应arch/arm/cpu/arm920t目录
#	BOARD  = xc2440			BOARD---->对应board/xc2440目录
#	SOC    = s3c24x0		SOC------>对应arch/arm/cpu/arm920t/s3c24x0
#

#在include 下面生成一个config.h
	/* Automatically generated - do not edit */
	#define CONFIG_BOARDDIR board/xc2440		#所以要在board路径下要创建一个xc2440的文件夹.
	#include <config_defaults.h>			
	#include <configs/xc2440.h>					#这个是配置文件.比如内存的大小, 网卡的ip ,最后make根据这个生成automake.mk.就像<span style="white-space:pre">									</span>        #linux根据配置选CONFIG_XXX项生成.config
									#所以要在include/configs/要创建一个xc2440.h这个配置文件.			    
	#include <asm/config.h>						#这个文件里面就一句话,#define CONFIG_RELOC_FIXUP_WORKS 用于代码重定向用的
	
#我要做的就是 
#	1.在board下面创建xc2440这个文件夹.
#   2.在include/configs/下面创建配置文件xc2440.h
#	3.剩下就是修改2410成2440
能力不够好多东西分析不清楚,先贴个图知道个大概的.细节好多不清楚.

uboot的编译和链接过程分析(一)

现在简单的理解大致编译过程.

1.进入totls  example/standalone  example/api     arch/arm/cpu/arm920t(CPUDIR)   链接脚本的路径下执行  make _depend  ,  看了下makefile编译的打印,没做什么.

     然后进入tool  example/standalon  example/api 执行  make all   。 tool下面都是生成一些.o    example/standalone下面生成.a 文件   example/api  没做什么.

     具体里面makefiel没有看,根据结果就是这个样子的.

2.进入arch/arm/cpu/arm920t  下面执行   make start.o 。 由start.S 变成start.o, 原来最先编译的是这个文件.

3.根据LIB这个变量,决定进入那些目录下面编译成静态库*.a   .

   默认是:   lib下面的lib**.a    

     arch/arm/cpu/libarm920t.a   (这个是cpu.a  下面是 soc.a 再接着就是 arch.a, 后面还有个个board.a      前面三个都arch/arm目录下面.  后面在board目录下)

                      arch/arm/cpu/arm920t/libs3c24x0.a (这里有个疑惑,我们@$(MKCONFIG) $(@:_config=) arm arm920t xc2440 NULL s3c24x0   这里s3c24x0的s小写,而                                                                                                   arm920t下面的只有S3c24x0,难道这里大小写不区分)

                     arch/arm/lib/libarm.a     arch/arm/下面只有3个目录:cpu  lib  include   

                                                                cpu下面有好多架构arm920t  arm926e等. 每个架构下面又有对应的soc.  arm920t下面有s3c24x0  at91   arm1176下面有s3c64xx

                                                                lib下可能都是公用的库.

                                                                asm下面都是素有架构的相关的代码,比如这个架构的处理器的寄存器定义.

    接着 fs.net  .disk.drivers  drives下面的东西最多,什么i2c  input misc mtd(nand   onenand  spi)   net  pci  rtc  serial  usb  video

                    接着common   api  post

4.编译单板 相关的  board/xc2440/libxc2440.a

    反正到这里感觉主要的还是arch/arm/cpu/arm920t/start.S uboot.lds (start.o)    arm020t/s3c24x0 (libs3c24x0.a)     board/xc2440/libxc2440.a   drivers/lib***.a   这几个                才是主要 要关心的.

5.链接脚本本来在arch/arm/arm920t/下面,编译完后再根目录, 不是拷贝过去的,而已一些其他的操作在顶层生成的.

6.剩下的操作就是根据链接脚本把生成的start.o  *.a  根据链接脚本变成uboot.bin

   根据编译后-Ttext 后面的第一个文件确实是  start.o,但后面其他的都是*.a  这个样子搞的话,那么我们之前改变链接脚本把 lowlevel_init.o  clock.o nand_read.o放在前面起作      用 么

7.到这一步就生成了uboot.现在用objdcopy  把uboot变成了uboot.bin文件.

8.为什么要包lowleve_init.o  clock.o  nand_read.o要放到前面,因为bootloader的前4k上电被拷贝到内部的sram中执行,然后再拷贝剩下的在外置sdram中执行.

   在前面的4k里面,我们肯定要配置时钟clock.c,第二部要用sdram,时钟时序图时间要精确.   还要从nand里面拷贝剩下的代码到sdram。

   这些和时钟.sdram  .nand的代码都必须在前4k里面,不然如如果前4k不包含这些代码的,第二部就做有可能不能成功的执行. 所以要放到前面。