uboot 链接地址与运行地址的区别

对于ARM架构的CPU,上电后PC寄存器是指向0地址处的,从这个地址开始运行程序,那么运行了启动代码后会把程序搬移到内存中去运行,这样就是产生程序会在运行时有个两地址,而由源码编译为可执行文件时只会指定一个链接地址,指定的这个地址通常是在内存中运行时的运行地址,那么刚上电启动时的程序运行地址怎么办呐?这里要先介绍汇编语言中的位置无关码。

使用C/C++或者其他高级语言编程,最后会被编译器工具转换为汇编代码,最后再翻译成机器码存储在内存、硬盘或者其他存储器上。机器码的构造不同的CPU有不同的规则,在这里就不再讨论,我们去关注汇编代码。在嵌入式开发中汇编程序通常用于非常关键的地方,比如系统启动阶段,中断向量表等。在启动阶段会经常使用到b、bl两条跳转指令。这两条语句到时完成跳转的工作,但是bl在跳转时会将跳转处的地址存放在ARM的LR寄存器中,方便在后面跳转回来。在使用b或bl跳转时,下一条指令的地址是这样计算的:将指令中24位带符号的补码扩展为32位;将此32位数左移两位;将得到的值加入到PC寄存器中,即可得到跳转的目标地址。这样可以发现,b跳转指令依赖于当前PC寄存器的值,这个特性使得使用b指令的程序不依赖于代码存储的位置---即不管这条代码放在什么位置,b指令都可以跳到正确的位置,这类指令被称为位置无关码。

所以u-boot程序刚开始执行时,他所处的地址不等于运行地址,不管是norflash启动还是nandflash启动,刚开始运行都是从0地址,那么在开头就要先使用b、bl、mov等“位置无关”的指令完成将代码从Flash等设备中复制到内存的“运行地址”处,然后再跳到“运行地址去执行”。 
那么程序的运行地址是哪,u-boot的程序是怎么组织的呢?由源码到可执行程序通常会经过这样几个步骤:预处理、编译、汇编、链接

最后的链接阶段通常使用arm-linux-ld选项用于将多个目标文件、库文件链接成可执行文件,其中“-T”选项,可以直接用它来指定代码段、数据段、bss段的起始地址,也可以用来指定一个链接脚本,在连接脚本里进行更复杂的地址设置。

1.直接指定代码段,数据段,BSS段的起始地址: 
-Ttest startaddr     -Tdata startaddr       -Tbss startaddr 
示例:arm-linux-ld –Ttext 0x0000000 led_on.o -o led_on_elf 
他表示代码段的运行地址为0x0000000,由于没有定义数据段、bss段的起始地址他们依次被放在代码段的后面 
  2..使用连接脚本设置地址:arm-linux-ld -T u-boot.lds -o u-boot  
这里的链接脚本就是u-boot.lds这个文件,它定义了整个程序编译之后的连接过程,决定了一个可执行程序的各个段的存储位置。 
先看一下 GNU官方网站上 对.lds文件形式的完整描述:

SECTIONS {
...
secname start BLOCK(align) (NOLOAD) : AT ( ldadr )
  { contents } >region :phdr =fill
...
}

secname和contents是必须的,其他的都是可选的。下面挑几个常用的看看: 
1、secname:段名 
2、contents:决定哪些内容放在本段,可以是整个目标文件,也可以是目标文件中的某段(代码段、数据段等) 
3、start:本段连接(运行)的地址,如果没有使用AT(ldadr),本段存储的地址也是start。GNU网站上说start可以用任意一种描述地址的符号来描述。 
4、AT(ldadr):定义本段存储(加载)的地址。 
  下面,结合u-boot.lds看看一个正式的连接脚本文件。

OUTPUT_FORMAT("elf32­littlearm", "elf32­littlearm", "elf32­littlearm")
;指定输出可执行文件是elf格式,32位ARM指令,小端
OUTPUT_ARCH(arm)
;指定输出可执行文件的平台为ARM
ENTRY(_start)
;指定输出可执行文件的起始代码段为_start.
SECTIONS
{
    . = 0x00000000 ; 从0x0位置开始
    . = ALIGN(4) ; 代码以4字节对齐
    .text : ;指定代码段
    {
       cpu/arm920t/start.o (.text) ; 代码的第一个代码部分
       *(.text) ;其它代码部分
    }
    . = ALIGN(4) 
    .rodata : { *(.rodata) } ;指定只读数据段
    . = ALIGN(4);
    .data : { *(.data) } ;指定读/写数据段
    . = ALIGN(4);
    .got : { *(.got) } ;指定got段, got段式是uboot自定义的一个段, 非标准段
    __u_boot_cmd_start = . ;把__u_boot_cmd_start赋值为当前位置, 即起始位置
    .u_boot_cmd : { *(.u_boot_cmd) } ;指定u_boot_cmd段, uboot把所有的uboot命令放在该段.
    __u_boot_cmd_end = .;把__u_boot_cmd_end赋值为当前位置,即结束位置
    . = ALIGN(4);
    __bss_start = .; 把__bss_start赋值为当前位置,即bss段的开始位置
    .bss : { *(.bss) }; 指定bss段
    _end = .; 把_end赋值为当前位置,即bss段的结束位置
}

以上就是u-boot整个工程的链接脚本,它来指定生成的u-boot可执行程序的组成架构。下面列出在u-boot工程编译过程的的最后一步生成u-boot.bin的过程: 
arm-linux-ld -Bstatic -T /u-boot-1.3.4/board/mini2440/u-boot.lds  -Ttext 0x33F80000 $UNDEF_SYM cpu/arm920t/start.o --start-group lib_generic/libgeneric.a cpu/arm920t/libarm920t.a cpu/arm920t/s3c24x0/libs3c24x0.a lib_arm/libarm.a fs/cramfs/libcramfs.a fs/fat/libfat.a fs/fdos/libfdos.a fs/jffs2/libjffs2.a fs/reiserfs/libreiserfs.a ……….(省略若干内容) board/mini2440/libmini2440.a --end-group -L /usr/local/arm/4.3.2/bin/../lib/gcc/arm-none-linux-gnueabi/4.3.2/armv4t -lgcc   -Map u-boot.map -o u-boot 
arm-linux-objcopy --gap-fill=0xff -O srec u-boot u-boot.srec 
arm-linux-objcopy --gap-fill=0xff -O binary u-boot u-boot.bin 
  上面可以看到在最后的链接阶段参考了链接脚本的规划,指定了程序的运行地址为0x33f80000,上电启动后,第一阶段的代码就会把u-boot自身搬运到内存的0x33f80000位置去运行。

时间: 2024-12-11 02:28:54

uboot 链接地址与运行地址的区别的相关文章

编译地址与运行地址

编译地址: 32位的处理器,它的每一条指令是4个字节,以4个字节存储顺序,进行顺序执行,CPU是顺序执行的,只要没发生什么跳转,它会顺序进行执行行, 编译器会对每一条指令分配一个编译地址,这是编译器分配的,在编译过程中分配的地址,我们称之为编译地址.运行地址:是指程序指令真正运行的地址,是由用户指定的,用户将运行地址烧录到哪里,哪里就是运行的地址.比如有一个指令的编译地址是0x5,实际运行的地址是0x200,如果用户将指令烧到0x200上,那么这条指令的运行地址就是0x200, 当编译地址和运行

u-boot中链接地址和加载地址的相关知识

以zc702开发板的u-boot为例 链接地址(运行地址):链接地址是在程序编译链接阶段就确定好的地址. u-boot的链接脚本由CONFIG_SYS_LDSCRIPT宏定义来指定,如在zynq_common.h当中有如下代码: #define CONFIG_SYS_LDSCRIPT "arch/arm/cpu/armv7/zynq/u-boot.lds" 在该链接脚本中指定了u-boot中各部分的链接顺序.同时zynq_common.h中的CONFIG_SYS_TEXT_BASE则指

uboot load address、entry point、 bootm address以及kernel运行地址的意义及联系

按各地址起作用的顺序,uboot引导linux内核启动涉及到以下地址: load address: entry point: 这两个地址是mkimage时指定的 bootm address:bootm为uboot的一个命令,以此从address启动kernel kernel运行地址:在具体mach目录中的Makefile.boot中指定,为kernel启动后实际运行的物理地址 mkimage -n 'linux-3.2.1' -A arm -O linux -T kernel -C none -

[转]IP地址和MAV地址——区别和联系

[转载]http://wenda.tianya.cn/question/27f9476d1e86f6b6 一.IP地址  对于IP地址,相信大家都很熟悉,即指使用TCP/IP协议指定给主机的32位地址.IP地址由用点分隔开的4个8八位组构成,如192.168.0.1就是一个IP地址,这种写法叫点分十进制格式.IP地址由网络地址和主机地址两部分组成,分配给这两部分的位数随地址类(A类.B类.C类等)的不同而不同.网络地址用于路由选择,而主机地址用于在网络或子网内部寻找一个单独的主机.一个IP地址使

uboot生成随机的MAC地址

转载:http://blog.chinaunix.net/uid-25885064-id-3303132.html 在使用U-boot时,有个问题就是MAC地址的设置,如果MAC地址相同的两块开发板在同一局域网中,会互相影响.由 于设备量产需要每台机器的MAC地址都不一样,所以我们给u-boot使用随即生成MAC地址,我们这里的随机生成的变量只保存到RAM空间中,其实也可 以保存到flash或MMC中,但是如果存储设备为nand的话,每次启动都会擦写一些nand ,众所周知,nand的擦写次数是

一个链接打开两个地址

<html><head><title>一个链接打开两个地址</title><script LANGUAGE="JavaScript">function hrefClick(newWin, locationWin) {   window.open(newWin);          //打开新的窗口   window.location = locationWin;//当前位置窗口也改变}</script></h

MAC地址和IP地址的区别与联系(计算机网络篇)

最近参加一个考试,总是遇到mac地址的相关问题,于是在网上做一个知识整理,记得这些知识还是大二的时候学的,已经基本还回给黄沛杰老师了,不过很感谢他耐心又负责地教导我们计算机组成原理的知识,起码还是记得他曾经给我们带来很多美好的大学记忆. MAC地址和IP地址的区别与联系 简单地说:ip地址是服务商给你的,mac地址是你的网卡物理地址. 一.IP地址 对于IP地址,相信大家都很熟悉,即指使用TCP/IP协议指定给主机的32位地址.IP地址由用点分隔开的4个8八位组构成,如192.168.0.1就是

swift禁用webView对H5中数字,链接,日期,地址,电话号码做解析

showWebView.dataDetectorTypes = .None //swift禁用webView对H5中数字,链接,日期,地址,电话号码做解析 其UIDataDetectorTypes属性: static var PhoneNumber: UIDataDetectorTypes { get } // Phone number detection static var Link: UIDataDetectorTypes { get } // URL detection static v

Uboot 引导内核时加载地址与入口地址问题

如果使用 mkimage 生成内核镜像文件的话,会在内核的前头加上了 64 bytes 的信息头,供建立 tag 之用.bootm 命令会首先判断 bootm xxx 这个指定的地址 xxx 与 -a 指定的加载地址是否相同. 如果不同的话会从这个地址开始提取出这个 64 bytes 的头部,对其进行分析,然后把去掉头部的内核复制到 -a 指定的加载地址去运行; 如果相同的话那就让其原封不同的放在那,但 -e 指定的入口地址会推后 64 bytes,以跳过这 64 bytes 的头部. 我们来看