链接地址和运行地址

合集下载

DSP的CMD文件详解(整理版)

DSP的CMD文件详解(整理版)

DSP的CMD文件详解CMD是用来分配ROM和RAM空间用的,告诉链接程序怎样计算地址和分配空间。

所以不同的芯片就有不同大小的ROM和RAM,存放用户程序的地方也不尽相同。

所以要根据芯片进行修改,分为 MEMORY 和SECTIONS两个部分。

MEMORY{PAGE 0 ..........PAGE 1.........}SECTIONS{.vectors ..................reset .................................}MEMORY是用来指定芯片的ROM和RAM的大小和划分出几个区间。

PAGE 0对应ROM, PAGE 1对应RAM。

PAGE 里包含的区间名字与其后面的参数反映了该区间的起始地址和长度。

SECTIONS:(在程序里添加下面的段名,如.vectors。

用来指定该段名以下,另一个段名以上的程序(属于PAGE0)或数据(属于PAGE1)放到“>”符号后的空间名字所在的地方。

){.vectors : { } > VECS PAGE 0.reset : { } > VECS PAGE 0..................................}eg:MEMORY{PAGE 0:VECS :origin = 00000h, length = 00040h LOW :origin = 00040h, length = 03FC0h SARAM :origin = 04000h, length = 00800h B0 :origin = 0FF00h, length = 00100h PAGE 1:B0 :origin = 00200h, length = 00100h B1 :origin = 00300h, length = 00100h B2 :origin = 00060h, length = 00020h SARAM :origin = 08000h, length = 00800h }{.text : { } > LOW PAGE 0.cinit : { } > LOW PAGE 0.switch : { } > LOW PAGE 0.const : { } > SARAM PAGE 1.data : { } > SARAM PAGE 1.bss : { } > SARAM PAGE 1.stack : { } > SARAM PAGE 1.sysmem : { } > SARAM PAGE 1}由三部分组成:①输入/输出定义:这一部分,可以通过ccs的“BuildOption........”菜单设置: .obj(链接的目标文件)、.lib(链接的库文件)、.map(生成的交叉索引文件)、.out(生成的可执行代码)。

hello第四课(二)f2812存储器映射及cmd详解-日志-eyes417-

hello第四课(二)f2812存储器映射及cmd详解-日志-eyes417-

hello第四课(二):f2812存储器映射及cmd详解-日志-eyes417-...2812存储器映射2812具有32位的数据地址和22位的程序地址,总地址空间可以达到4M的数据空间和4M的程序空间。

32位的数据地址,就是能访问2的32次,是4G,而22位的程序地址,就是能访问2的22次,是4M。

其实,2812可寻址的数据空间最大是4G,但是实际线性地址能达到的只有4M,原因是2812的存储器分配采用的是分页机制,分页机制采用的是形如0xXXXXXXX的线性地址,所以数据空间能寻址的只有4M。

 2812的存储器被划分成了下面的几个部分:1. 程序空间和数据空间。

2812所具有的RAM、ROM和FLASH都被统一编址,映射到了程序空间和数据空间,这些空间的作用就是存放指令代码和数据变量。

2. 保留区。

数据空间里面某些地址被保留了,作为CPU的仿真寄存器使用,这些地址是不向用户开放的。

3. CPU中断向量。

在程序空间里也保留了64个地址作为CPU 的32个中断向量。

通过CPU的一个寄存器ST1中的VMAP位来将这一段地址映射到程序空间的底部或者顶部。

映射和空间的统一编址 F2812内部的映射空间 2812CMD详解CMD:command 命令,顾名思义就是命令文件指定存储区域的分配.2812的CMD采用的是分页制,其中PAGE0用于存放程序空间,而PAGE1用于存放数据空间。

1.)#pragma ,CODE_SECTION和DATA_SECTION伪指令#pragma DATA_SECTION(funcA,"dataA"); ------ 函数外声明将funcA数据块定位于用户自定义的段"dataA"中------ 需要在CMD中指定dataA段的物理地址2.)MEMORY和SECTIONS是命令文件中最常用的两伪指令。

电信光猫与无线路由器如何连接设置

电信光猫与无线路由器如何连接设置

电信光猫与无线路由器如何连接设置详细介绍首先说下光纤猫吧,它兼有路由器的功能,如果直接在没有接无线路由的情况下,在ie输入192.168.1.1,则会进入光纤猫(这个只是顺便提一下)言归正传,第一步,按照路由器与猫的链接方式,将线路链接好。

第二步,在ie192.168.1.1,进入无线路由设置界面账号跟密码都是admin由于光纤猫与无线路由的地址相同,会造成地址冲突,所以要对路由的地址进行修改,在网络参数中,对lan进行设置,修改ip地址如图,点击保存,重启路由器。

重启路由器之后,路由器的地址就变成192.168.2.1了,此时在浏览器中输入这个地址,然后输入账号和密码进入路由器设置界面,然后在打开的配置中心点击设置向导,然后点击下一步。

接着选择上网方式后,点击下一步后输入你的宽带或上网的账号和密码。

点击下一步。

设置无线路由器的密码,这样能防止别人盗用你的网络资源。

一般密码是6位的。

最后设置完成后,需要重启路由器使设置生效,就可以使用无线路由器上网了。

相关阅读:路由器安全特性关键点由于路由器是网络中比较关键的设备,针对网络存在的各种安全隐患,路由器必须具有如下的安全特性:(1)可靠性与线路安全可靠性要求是针对故障恢复和负载能力而提出来的。

对于路由器来说,可靠性主要体现在接口故障和网络流量增大两种情况下,为此,备份是路由器不可或缺的手段之一。

当主接口出现故障时,备份接口自动投入工作,保证网络的正常运行。

当网络流量增大时,备份接口又可承当负载分担的任务。

(2)身份认证路由器中的身份认证主要包括访问路由器时的身份认证、对端路由器的身份认证和路由信息的身份认证。

(3)访问控制对于路由器的访问控制,需要进行口令的分级保护。

有基于IP地址的访问控制和基于用户的访问控制。

(4)信息隐藏与对端通信时,不一定需要用真实身份进行通信。

通过地址转换,可以做到隐藏网内地址,只以公共地址的方式访问外部网络。

除了由内部网络首先发起的连接,网外用户不能通过地址转换直接访问网内资源。

IP地址的保留规则是什么的文章

IP地址的保留规则是什么的文章

IP地址的保留规则是什么的文章IP地址的保留规则是什么?随着互联网的快速发展和普及,IP地址作为互联网的基础架构之一,发挥着至关重要的作用。

为了有效管理和分配IP地址资源,保证网络的正常运行,国际互联网协会(ICANN)制定了一系列的IP地址保留规则。

本文将介绍IP地址的保留规则,包括IPv4和IPv6两个版本。

一、IPv4地址的保留规则IPv4地址是目前被广泛使用的地址格式,它由32位构成,通常以四个十进制数表示,每个数值范围为0至255。

在保留规则中,有三个特殊的IP地址范围被保留不分配给互联网上的公共网络,分别为以下三类:1. 保留IP地址段首先是保留IP地址段,也被称为专用IP地址段或者私有IP地址段。

这些地址范围被用于内部网络,例如家庭网络或公司的局域网(LAN),不会被分配给互联网上的公共网络。

以下是保留IP地址段的具体范围:- 10.0.0.0至10.255.255.255- 172.16.0.0至172.31.255.255- 192.168.0.0至192.168.255.2552. 回环IP地址回环IP地址范围是为了网络设备自我测试和互联网应用程序本地开发而保留的。

回环地址总是指向设备本身,用于在设备上测试网络功能。

IPv4的回环地址为127.0.0.0至127.255.255.255。

3. 保留IP地址特殊用途除了上述两类保留IP地址范围,IPv4还有一些特殊用途的保留IP 地址。

- 0.0.0.0: 表示本地网络的默认网关- 169.254.0.0至169.254.255.255: 用于自动配置IP地址时的本地链接- 192.0.0.0至192.0.0.255: 用于文档和协议中的特殊传输需求- 192.0.2.0至192.0.2.255: 用于文档和示例中的TEST-NET环境- 198.51.100.0至198.51.100.255: 用于文档和示例中的TEST-NET-2环境- 203.0.113.0至203.0.113.255: 用于文档和示例中的TEST-NET-3环境二、IPv6地址的保留规则随着IP地址资源的枯竭,IPv6被设计出来作为IPv4的替代方案,它采用128位地址,以8个分组的形式呈现,每个分组由四个十六进制数字组成。

操作系统第四章复习

操作系统第四章复习

页框号为21。 因为起始驻留集为空, 而0页对应的ห้องสมุดไป่ตู้框为空闲链表中的第三个空闲页框,其对应的页框号为21。
页框号为32。 因为11 > 10故发生第三轮扫描,页号为1的页框在第二轮已经处于空闲页框链表中,此刻该页又被重新访问,因此应被重新放回到驻留集中,其页框号为32。
页框号为41。 因为第2页从来没有被访问过,不在驻留集中。因此从空闲链表中取出链表头的页框,页框号为41。
4. 在虚拟内存管理中,地址变换机构将逻辑地址转换为物理地址,形成该逻辑地址的阶段是( )。 ① 编辑 ② 编译 ③ 链接 ④ 装载 5. 采用段式存储管理的系统中,若地址用24位表示,其中8位表示段号,则允许每段的最大长度是_______ A)224 B)28 C) 216 D) 232 6. 作业在执行中发生了缺页中断,经操作系统处理后,应让其执行______指令。 A)被中断的前一条 B)被中断的后一条 C)被中断的 D) 启动时的第一条
7、某基于动态分区存储管理的计算机,其主存容量为55MB(初始为空),采用最佳适配(Best fit)算法,分配和释放的顺序为:分配15MB,分配30MB,释放15MB,分配6MB,此时主存中最大空闲分区的大小是( ) A:7MB B:9MB C:10MB D:15MB
当该进程执行到时刻260时,要访问逻辑地址为17CAH的数据,请问答下列问题: (1)该逻辑地址对应的页号是多少? (2)若采用先进先出置换算法,该逻辑地址对应的物理地址是多少?要求给出计算过程。 (3)若采用时钟置换算法,该逻辑地址对应的物理地址是多少?(设搜索下一页的指针沿顺时针方向移动,且当前指向2号页框)
页号
存储块号
0 1 2 3
5 10 4 7

STM32学习:IAP简单的IAP例子

STM32学习:IAP简单的IAP例子

STM32学习:IAP简单的IAP例⼦章节概述:以⼀个最简单的例⼦⽰范IAP程序(没有⽂件通讯,没有跳转判断),需要借助IDE进⾏分区数据的划分以及下载。

准备IDE:keil-MDK 5MCU:STM32F103ZET6为例(Flash地址为0x08000000—0x0807ffff,共512KB)。

BSP:STM32-HAL启动⽅式:FLASH启动前32KB存放BootLoader程序(0x08000000 ~ 0x08007fff)剩余的空间存放APP程序(0x08008000 ~ 0x0807ffff)假定跳转的APP程序的实际物理地址为:0x08008000分别新建2个⼯程,名字可以为:bootLoader与app。

从Bootloader跳转到APPBootloader为了区分执⾏分区的不同,添加串⼝打印功能。

(略)例如:printf("Hello Bootloader\r\n");添加下⾯的代码以实现跳转功能:节选⾃CubeMX的例程:STM32Cube\Repository\xxx\Projects\Applications\IAP#define NVIC_VectTab_RAM ((u32)0x20000000)#define NVIC_VectTab_FLASH ((u32)0x08000000)#define PHYSICAL_ADDRESS_Flash (0x08000000) // 程序烧写的物理地址#define APPLICATION_POSADDR (0x0000C000) // APP 偏移量#define APPLICATION_ADDRESS ((PHYSICAL_ADDRESS_Flash) | (APPLICATION_POSADDR)) // 最终跳转的地址,实际上就是0x08008000typedef void (*pFunction)(void);pFunction JumpToApplication;uint32_t JumpAddress;void NVIC_SetVectorTable(uint32_t base, uint32_t offset){/* close interruption*/__set_FAULTMASK(1);/* set vector table*///NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0xffset);SCB->VTOR = base | offset;/* open interruption*/__set_FAULTMASK(0);}int main(void){// 在BootLoader程序的中断向量表指向设置中应有这么⼀句:NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0); //设置中断向量表指向/* Test if user code is programmed starting from address "APPLICATION_ADDRESS" */if (((*(__IO uint32_t*)APPLICATION_ADDRESS) & 0x2FFE0000 ) == 0x20000000) //①{/* Jump to user application */JumpAddress = *(__IO uint32_t*) (APPLICATION_ADDRESS + 4);// ②JumpToApplication = (pFunction) JumpAddress;//③/* Initialize user application's Stack Pointer */__set_MSP(*(__IO uint32_t*) APPLICATION_ADDRESS); // ④JumpToApplication(); // ⑤}while (1){}}解析:①因为⽤户程序开始位置(0x08008000处)的前4个字节存放的是堆栈的地址,堆栈地址必定是指向RAM空间的,⽽STM32的RAM空间起始地址为0x20000000,所以要进⾏判断。

相对地址和链接地址

相对地址和链接地址

关于相对地址和链接地址的问题问题提出:使用uboot版本:u-boot-1.3.4编译的board:smdk2410在顶层的Makefile中u-boot的连接sh指令:$(obj)u-boot: depend $(SUBDIRS) $(OBJS) $(LIBBOARD) $(LIBS) $(LDSCRIPT) UNDEF_SYM=`$(OBJDUMP) -x $(LIBBOARD) $(LIBS) | \sed -n -e 's/.*\($(SYM_PREFIX)__u_boot_cmd_.*\)/-u\1/p'|sort|uniq`;\cd $(LNDIR) && $(LD) $(LDFLAGS) $$UNDEF_SYM $(__OBJS) \--start-group $(__LIBS) --end-group $(PLATFORM_LIBS) \-Map u-boot.map -o u-boot其中LDFLAGS在顶层config.mk中定义为:LDFLAGS += -Bstatic -T $(LDSCRIPT) $(PLATFORM_LDFLAGS)ifneq ($(TEXT_BASE),)LDFLAGS += -Ttext $(TEXT_BASE)Endif所以u-boot连接sh指令的LDFLAGS会变成:-T board/smdk2410/U-Boot.lds –Ttext 0x33f80000其中Uboot.lds如下:OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")/*OUTPUT_FORMA T("elf32-arm", "elf32-arm", "elf32-arm")*/OUTPUT_ARCH(arm)ENTRY(_start)SECTIONS{. = 0x00000000;. = ALIGN(4);.text :{cpu/arm920t/start.o (.text)*(.text)}. = ALIGN(4);.rodata : { *(.rodata) }. = ALIGN(4);.data : { *(.data) }. = ALIGN(4);.got : { *(.got) }. = .;__u_boot_cmd_start = .;.u_boot_cmd : { *(.u_boot_cmd) }__u_boot_cmd_end = .;. = ALIGN(4);__bss_start = .;.bss (NOLOAD) : { *(.bss) }_end = .;}其中U-boot.lds中定义的当前运行地址为0:. = 0x00000000定义,但是后面又用-Ttext 0x3ff80000把运行地址定义成了0x3ff8000。

c语言运行流程

c语言运行流程

c语言运行流程C语言运行流程一、概述C语言是一种通用的高级编程语言,被广泛应用于系统软件开发、嵌入式系统以及科学计算等领域。

了解C语言的运行流程对于学习和使用C语言非常重要。

本文将介绍C语言程序的运行流程,包括编写、编译、链接和执行四个主要阶段。

二、编写在编写C语言程序之前,我们首先需要选择一个合适的集成开发环境(IDE)或者文本编辑器来编写代码。

常用的C语言开发工具有Visual Studio、Code::Blocks、Dev-C++等。

在编写程序时,我们需要使用C语言的语法规则来描述程序逻辑,包括变量、函数、控制结构等。

三、编译编写完C语言程序后,我们需要将其编译为机器语言,使计算机能够理解并执行程序。

编译器是将C语言代码转换为机器语言的工具。

常用的C语言编译器有GCC、Clang等。

编译过程主要包括以下几个步骤:1. 词法分析:将源代码分解为一个个词法单元,如关键字、标识符、运算符等。

2. 语法分析:根据语法规则检查词法单元是否符合语法要求,生成语法树。

3. 语义分析:对语法树进行类型检查和语义规则检查,生成中间代码。

4. 代码优化:对中间代码进行优化,提高程序执行效率。

5. 代码生成:将优化后的中间代码转换为目标机器代码。

四、链接在编译完成后,我们得到了目标机器代码。

然而,大型程序通常由多个源文件组成,这些源文件之间可能会有函数调用和变量引用的关系。

链接器负责将多个目标文件合并成一个可执行文件。

链接过程主要包括以下几个步骤:1. 符号解析:将函数和变量引用与其定义进行关联。

2. 地址重定位:将目标文件中的地址转换为最终的运行地址。

3. 符号决议:解决不同目标文件中相同符号的定义冲突。

4. 生成可执行文件:将链接后的目标代码生成可执行文件。

五、执行链接完成后,我们得到了一个可执行文件。

在运行程序时,操作系统加载可执行文件到内存中,并按照程序的入口点开始执行。

C语言程序的执行过程可以分为以下几个阶段:1. 初始化:为全局变量分配内存并进行初始化。

c语言链接原理

c语言链接原理

c语言链接原理C语言链接原理什么是链接链接是将多个源文件组合成一个可执行的程序的过程。

在C语言中,链接分为静态链接和动态链接两种方式。

静态链接静态链接是在编译时将所有源文件和依赖的库文件打包合并为一个可执行文件。

在静态链接的过程中,会将所有被调用的函数和符号解析成绝对地址,并进行地址重定位,以便在程序运行时能正确找到对应的函数或变量。

静态链接的好处是,可执行文件独立,不依赖于外部的库文件,可以方便地在不同操作系统或机器上进行传输和运行。

但同时,也会使得可执行文件的大小变大,并且无法共享已被其他程序加载的库文件。

动态链接动态链接是在程序运行时,将程序所需要的库文件动态加载到内存中,并建立起调用关系。

相比于静态链接,动态链接的主要优势在于节省了磁盘空间,同时可以方便地共享已加载的库文件。

在动态链接的过程中,程序除了需要链接器的支持外,还需要动态链接器或运行时链接器(如)的支持。

动态链接器会根据程序中对函数和符号的引用,到指定的共享库文件中查找对应的函数或变量地址,并进行重定位。

这种方式需要在程序运行时动态解析符号地址,因此速度可能比静态链接慢一些。

符号解析在链接的过程中,一个非常重要的步骤就是符号解析。

符号解析是将函数名或变量名与其对应的地址进行关联的过程。

在C语言中,通过extern关键字来声明外部变量或函数。

在链接时,链接器会根据这些声明找到对应的定义,并确定其地址。

符号解析的过程是由链接器完成的,它会先查找目标文件中是否存在该符号的定义,如果存在则将其地址记录下来,否则会继续在其他目标文件或库文件中进行查找。

如果所有的目标文件和库文件都没有找到符号的定义,链接器将会报链接错误。

链接顺序在进行静态链接时,链接器需要按照一定的顺序来合并多个目标文件和库文件。

常见的链接顺序是从左到右,从上到下。

这个顺序决定了符号解析的优先级,后面的文件中的符号会覆盖前面的文件中的同名符号。

如果出现了重复定义符号的情况,链接器会报重复定义的错误。

超链接的组成和功能

超链接的组成和功能

超链接的组成和功能超链接是互联网中常见的一种元素,它主要由两部分组成:链接文字和目标地址。

超链接的功能是可以将不同网页之间进行快速跳转和链接,从而实现信息的互通和流动。

一、超链接的组成1. 链接文字:链接文字是用户在网页上看到的可点击的文本,通常是蓝色并带有下划线。

它可以是任意文本,比如一个单词、一个短语、一个句子或者一个段落。

链接文字的作用是吸引用户点击,并告诉用户点击后将跳转到哪个页面。

2. 目标地址:目标地址是链接文字点击后要跳转到的网页的网址。

目标地址可以是同一个网站上的其他页面,也可以是其他网站的页面。

在HTML中,目标地址需要使用<a>标签来定义,并通过href属性来指定目标地址。

二、超链接的功能1. 跳转页面:超链接的最基本功能是跳转页面。

用户点击链接文字后,浏览器会根据链接中的目标地址加载对应的页面,并显示在当前窗口或新窗口中。

这样,用户可以方便地在不同页面之间进行跳转和浏览。

2. 下载文件:除了跳转页面,超链接还可以用于下载文件。

通过设置目标地址为文件的路径,用户点击链接后可以直接下载文件到本地。

常见的文件下载链接包括文档、图片、音频、视频等,用户可以根据需要选择下载。

3. 邮件发送:超链接还可以用于发送邮件。

通过设置目标地址为邮箱地址,并指定邮件主题和收件人等参数,用户点击链接后可以直接打开默认邮件客户端,并填写好收件人和主题等信息,方便用户发送邮件。

4. 跳转锚点:超链接还可以用于页面内的跳转。

通过设置目标地址为页面中的锚点,用户点击链接后可以直接跳转到页面中指定的位置。

这在长页面中特别有用,可以让用户快速定位到感兴趣的内容。

5. 打开新窗口:超链接还可以用于在新窗口中打开目标页面。

通过设置目标地址为"_blank",用户点击链接后可以在新窗口中加载目标页面,而不影响当前页面的浏览。

这在需要同时浏览多个页面或者避免离开当前页面时非常方便。

6. 运行脚本:超链接还可以用于运行脚本。

局域网怎样连接

局域网怎样连接

交换机共享上网1、花50多元买以太5口交换机(路由器也可,但要比交换机贵得多)。

2、二条直通线(就是咱们现在用的,线序两头一样)3、从猫出来的宽带信号线插入一个端口,二台电脑分别插入一个端口。

4、设置协议TCP/IP属性。

(1)本地连接右击本地连接—属性—常规—协议(TCP/IP)--属性—使用下面的IP地址A电脑:IP地址:192.168.0. 2 (这是我使用的)子网掩码:255.255.255.0默认网关:192.168.0.1B电脑:IP地址:192.168.0. 3(2—254之间除2以外的数)子网掩码:255.255.255.0默认网关:192.168.0.1(2)宽带连接右击宽带连接—属性—网络--协议(TCP/IP)--属性—使用下面的IP地址--两台电脑均设为“自动获得IP地址”。

5、这样二台电脑可以同时上网,也可以单独上网。

6、如果想互联共享要设置共享(1)启用来宾帐户。

开始-控制面版--管理工具--计算机管理-展开系统工具--本地用户和组--用户,在右边会见到一个GUEST用户,双击它,把帐号已停用前面的勾取消。

(2)用户权利指派。

“控制面板-管理工具-本地安全策略”,在“本地安全策略”对话框中,依次选择“本地策略-用户权利指派”,在右边的选项中依次对“从网络上访问这台计算机”和“拒绝从网络上访问这台计算机”这两个选项进行设置。

“从网络上访问这台计算机”选项需要将guest用户和everyone添加进去;“拒绝从网络上访问这台计算机”需要将被拒绝的所有用户删除掉,默认情况下guest是被拒绝访问的。

(3)安装协议开始→控制面板→网上邻居”→“网络连接”→“本地连接”(就是你目前使用中的局域网连接)上按右键→“属性”→“常规”→“安装”→“通讯协议”→“添加”,此时就可以看到“NetBEUI Protocol”这一项,此项原来是没有的,选取NetBEUI之后,按确定。

(4)建立工作组。

DSP的CMD文件写法综述

DSP的CMD文件写法综述

DSP的CMD文件写法综述DSP的存储器的地址范围,CMD是主要是根据那个来编的。

CMD 它是用来分配rom和ram空间用的,告诉链接程序怎样计算地址和分配空间。

不同的芯片就有不同大小的rom和ram.放用户程序的地方也不尽相同。

所以要根据芯片进行修改.分两部分.MEMORY和SECTIONS。

MEMORY{PAGE 0 ..........PAGE 1.........}SECTIONS{SECTIONS{.vectors ..................reset ................. ................}MEMORY是用来指定芯片的rom和ram的大小和划分出几个区间. PAGE 0 对应rom;PAGE 1对应ramPAGE 里包含的区间名字与其后面的参数反映了该区间的起始地址和长度.SECTIONS:(在程序里添加下面的段名如.vectors.用来指定该段名以下,另一个段名以上的程序(属于PAGE0)或数据(属于PAGE1)放到“>”符号后的空间名字所在的地方。

SECTIONS{.vectors : { } > VECS PAGE 0 /* Interrupt vector table */ .reset : { } > VECS PAGE 0 /* Reset code */ ..................................}eg:MEMORY{PAGE 0: VECS: origin = 00000h, length = 00040hLOW: origin = 00040h, length = 03FC0hSARAM: origin = 04000h, length = 00800hB0: origin = 0FF00h, length = 00100hPAGE 1: B0: origin = 00200h, length = 00100hB1: origin = 00300h, length = 00100hB2: origin = 00060h, length = 00020hSARAM: origin = 08000h, length = 00800h}SECTIONS{.text : { } > LOW PAGE 0.cinit : { } > LOW PAGE 0.switch : { } > LOW PAGE 0.const : { } > SARAM PAGE 1.data : { } > SARAM PAGE 1.bss : { } > SARAM PAGE 1.stack : { } > SARAM PAGE 1.sysmem : { } > SARAM PAGE 1}CMD的专业名称叫链接器配置文件,是存放链接器的配置信息的,我们简称为命令文件,其中比较关键的就是MEMORY和SECTIONS 两个伪指令的使用,常常令人困惑,系统出现的问题也经常与它们的不当使用有关。

运行地址和加载地址

运行地址和加载地址

U-Boot移植过程中的运行地址和装载地址的区别uboot移植涉及到底层硬件的设置,因此需要掌握UART、系统时钟频率、NOR FLASH、NAND FLASH、SDRAM、网卡、存储控制器等硬件的功能及配置,这些都可以参照相应开发板的芯片手册来完成,没有什么大的问题。

在移植过程中,一直困扰我的是PIC(代码无关性)问题,即运行地址和加载地址的区别,看过网上很多关于这两者的介绍,感觉懂一点,却一直不知所然。

在参考大量的文献下,算是得了一点心得。

首先来了解下运行地址及加载地址的区别运行地址:也叫链接地址,是程序定位的绝对地址,即在编译连接时确定的地址。

如果程序中有位置相关指令,程序在运行时,程序必须在运行地址上。

加载地址:程序放置的位置。

运行地址和加载地址的值有时相等,有时却不相等,所以这给初学者带来很大的困扰。

为了弄清楚这个问题,还得从NOR FLASH,NAND FLASH,S3C2440内部4KB RAM的映射说起。

左边表示从NOR FLASH启动时的映射,右边表示从NAND FLASH启动时的映射。

这里只讨论从NOR FLASH启动的情况,从图中可以看出NOR FLASH映射到了0X00000000的起始位置,假如UBOOT的代码存放在NOR FLASH上,即装载地址为0X00000000。

再来看看UBOOT的链接地址,代码在board/smdk2410/U-Boot.lds里。

连接脚本文件lds中没有设置LMA,只是设置了VMA。

VMA的设置是通过顶层目录下的config.mk文件中的LDFLAGS实现的在board/smdk2410/config.mk定义了TEXT_BASE = 0x33F80000(SDRAM),即程序的运行地址查看u-boot.map文件,代码的连接地址是从0x33F80000开始的。

167 .text 0x33f80000 0x232c8168 cpu/arm920t/start.o(.text)169 .text 0x33f80000 0x4a0cpu/arm920t/start.o170 0x33f80048 _bss_start171 0x33f8004c _bss_end172 0x33f80044_armboot_start173 0x33f80000_start174 board/samsung/fs2410/lowlevel_init.o(.text)175 .text 0x33f804a0 0x64board/samsung/fs2410/lowlevel_init.o176 0x33f804a4lowlevel_init177 board/samsung/fs2410/nand_read.o(.text)178 .text 0x33f80504 0xe8board/samsung/fs2410/nand_read.o179 0x33f80504 wait_idle180 0x33f80518nand_read_ll此时装载地址和运行地址明显不一样,为什么程序还能运行呢?这里就涉及到PIC----代码无关设计方面的知识了。

TI DSP中的CMD文件

TI DSP中的CMD文件

MEMORY
{
PAGE 0: /* PROGRAM MEMORY */
PM: ORIGIN=0h, LENGTH=08000h /* 32k on-chip flash memory */
SARAM_P:ORIGIN=08000h, LENGTH=0800h /* 2k saram in program space */
.bss: {所有.bss输入段名} load=加载地址 run =运行地址
.other: {所有.other输入段名} load=加载地址 run =运行地址
}
SECTIONS必须用大写字母,其后的大括号里是输出段的说明性语句,每一个输出段的说明都是从段名开始,段名之后是如何对输入段进行组织和给段分配存储器的参数说明。以.text段的属性语句为例,“{所有.text输入段名}”这段内容用来说明连接器输出段的.text段由哪些子目标文件的段组成。接下来的load和run,链接器为每个输出段都在目标存储器里分配两个地址:一个是加载地址,一个是运行地址。通常情况下两个地址是相同的,可以认为输出段只有一个地址,这时就可以不加“run =运行地址”这条语句了;但有时需要将两个地址分开,比如将程序加载到FLASH,然后放到RAM中高速运行,这就用到了运行地址和加载地址的分别配置了。
TI DSP中的CMD文件
来源: 刘礼亚的日志
CMD的专业名称叫链接器配置文件,是存放链接器的配置信息的,我们简称为命令文件。从其名称可以看出,该文件的作用是指明如何链接程序的。
那么我们知道,在编写TI DSP程序时,是可以将程序分为很多段,比如text、bss等,各段的作用均不相同。实际在片中运行时,所处的位置也不相同。比如text代码一般应该放在flash内,而bss的变量应该放在ram内。等等。但是对于不同的芯片,其各存储器的起止地址都是不一样的,而且,用户希望将某一段,尤其是自定义段,放在什么存储器的什么位置,这也是链接器不知道的。为了告诉链接器,即将使用的芯片其内部存储空间的分配和程序各段的具体存放位置,这就需要编写一个配置文件,即CMD文件了。

什么是重定位?为什么需要重定位?【转】

什么是重定位?为什么需要重定位?【转】

什么是重定位?为什么需要重定位?【转】⼀、必须知道的⼏个概念。

1、链接地址和运⾏地址。

①运⾏地址,顾名思义就是程序运⾏的时候的地址,也就是你⽤⼯具将代码下载到RAM的那个地址,也叫加载地址。

②链接地址,由链接脚本指定的地址。

为什么需要链接脚本指定地址呢?你想⼀下,在c语⾔编程中,当我们需要调⽤⼀个A函数的时候,编译器是怎么找到这个A函数?编译器肯定是知道它被放在哪⾥才可以找到它。

那就是链接脚本的作⽤,链接脚本其实在程序被执⾏之前都已经指定A函数⼀个地址编号,以后所有的函数调⽤我们都会去这个地址编号那⾥寻找A函数。

有点类似于c语⾔的指针变量。

2、位置有关码与位置⽆关码。

①位置有关码,就是这句代码的执⾏正确与否还需要取决于当前的地址,也就是说跟地址已经绑定了的,例如:ldr PC, _main,就是PC指针必须跳转到_main(函数名就是⼀个地址)这个地址去,代码执⾏成功与否就相当于受到了这个地址的约束,假如这个地址的内容不存放_main这个函数,就会出错了。

②位置⽆关码,就是这句代码在哪⾥运⾏都可以的,跟所处的地址⽆关,跟位置有关码相反。

⼆、重定位需要理解的⼀些问题。

1、链接地址跟运⾏地址不同的情况下会出现什么情况?答:以上⾯举的函数A为例,当链接地址跟运⾏地址不同的时候,假如链接地址是0x1000,运⾏地址(加载地址)是0x0000,链接脚本指定函数A将来是要存放到(基地址+偏移量)=0x1000+0x0001=0x1001地址的,但是程序在下载的时候却把这个程序下载到0x0000,所以函数A的地址实际上是存放在(基地址+偏移量)=0x0000+0x0001=0x0001这个地址的。

当程序运⾏到⼀⾏位置有关码例如:ldr PC, A ,编译器⾸先就会按照链接脚本指定的A的那个地址0x1001寻找A函数,但是因为加载地址跟链接地址不同的原因,实际上A函数已经被放到了0x0001,所以执⾏就会出错。

总结:代码重定位

总结:代码重定位

总结:代码重定位什么是重定位?为什么要代码重定位?要弄清楚上⾯的这两个问题,⾸先要理解下⾯这⼏个概念⼀、编码(1)位置⽆关编码:PIC,可执⾏程序运⾏时与代码在内存中的地址⽆关,代码中没有使⽤绝对地址,⽽是使⽤的相对地址。

(例如:B、BL、MOV等指令)(2)位置有关编码:可执⾏程序运⾏时与代码在内存中的地址有关系。

(例如:LDR PC, =MAIN等指令)⼆、地址(1)链接地址:程序编译链接时指定的地址(使⽤makefile或者链接脚本可以指定链接地址)(2)运⾏地址:程序在内存中实际运⾏的地址。

参考资料:三、S5PV210的启动在uboot中的启动⽅式,将整个uboot分为2个部分(BL1和BL2),是分别加载到内存中去运⾏的。

在这个过程中,程序的链接地址和运⾏地址就很可能不是相同的了,但是在代码中⼜有⼀些位置有关码,为了使得程序可以正常的运⾏,就必须使⽤重定位来解决。

(关于S5PV210的启动在另⼀篇⽂章⾥有介绍)四、如何重定位?1、链接脚本指定链接地址2、判断运⾏地址和链接地址是否相同3、复制代码到指定的链接地址处,使⽤长转移指令进⾏跳转具体的实现如下:【链接脚本代码】1 SECTIONS2 {3 . = 0xd0024000;45 .text : {6 start.o7 * (.text)8 }910 .data : {11 *(.data)12 }1314 bss_start = .;15 .bss : {16 * (.bss)17 }1819 bss_end = .;20 }【重定位代码】adr r0, _start // 短加载,获取_start的运⾏地址ldr r1, =_start // 长加载,获取_start的链接地址ldr r2, =bss_start // 获取bss链接地址,重定位代码的结束地址cmp r0, r1 // ⽐较_start的运⾏地址和链接地址是否相等beq clean_bss // 相等:不需要重定位;跳转到clean_bss// 不相等:需要重定位;继续往下执⾏// 汇编实现while完成代码赋值到重定位地址copy_loop:ldr r3, [r0], #4 // 源str r3, [r1], #4 // ⽬的先把r3中的内容放⼊r1的指向的地址;cmp r1, r2 // 然后r1 =r1 + 4bne copy_loop。

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

1、运行地址<--->链接地址:他们两个是等价的,只是两种不同的说法。

2、加载地址<--->存储地址:他们两个是等价的,也是两种不同的说法。

运行地址:程序在SRAM、SDRAM中执行时的地址。

就是执行这条指令时,PC应该等于这个地址,换句话说,PC等于这个地址时,这条指令应该保存在这个地址内。

加载地址:程序保存在Nand flash中的地址。

位置无关码:B、BL、MOV都是位置位置无关码。

位置有关码:LDR PC,=LABEL等类似的代码都是位置有关码。

下面我们来看看一个Makefile文件sdram.bin : head.S leds.carm-linux-gcc -c -o head.o head.Sarm-linux-gcc -c -o leds.o leds.carm-linux-ld -Ttext 0x30000000 head.o leds.o -o sdram_elfarm-linux-objcopy -O binary -S sdram_elf sdram.binarm-linux-objdump -D -m arm sdram_elf > sdram.disclean:rm -f sdram.dis sdram.bin sdram_elf *.o我们可以看到sdram_elf的代码段是从0x30000000地址开始存放,这个地址我们称之为运行地址。

为什么从这个地址开始存放,因为SDRAM的起始地址是0x30000000.下面来看看一个启动代码@*************************************************************************@ File:head.S@ 功能:设置SDRAM,将程序复制到SDRAM,然后跳到SDRAM继续执行@*************************************************************************.equ MEM_CTL_BASE, 0x48000000.equ SDRAM_BASE, 0x30000000.text.global _start_start:bl disable_watch_dog @ 关闭WATCHDOG,否则CPU会不断重启bl memsetup @ 设置存储控制器bl copy_steppingstone_to_sdram @ 复制代码到SDRAM中ldr pc, =on_sdram @ 跳到SDRAM中继续执行on_sdram:ldr sp, =0x34000000 @ 设置堆栈bl mainhalt_loop:b halt_loopdisable_watch_dog:@ 往WATCHDOG寄存器写0即可mov r1, #0x53000000mov r2, #0x0str r2, [r1]mov pc, lr @ 返回copy_steppingstone_to_sdram:@ 将Steppingstone的4K数据全部复制到SDRAM中去@ Steppingstone起始地址为0x00000000,SDRAM中起始地址为0x30000000mov r1, #0ldr r2, =SDRAM_BASEmov r3, #4*10241:ldr r4, [r1],#4 @ 从Steppingstone读取4字节的数据,并让源地址加4str r4, [r2],#4 @ 将此4字节的数据复制到SDRAM中,并让目地地址加4cmp r1, r3 @ 判断是否完成:源地址等于Steppingstone的未地址?bne 1b @ 若没有复制完,继续mov pc, lr @ 返回memsetup:@ 设置存储控制器以便使用SDRAM等外设mov r1, #MEM_CTL_BASE @ 存储控制器的13个寄存器的开始地址adrl r2, mem_cfg_val @ 这13个值的起始存储地址add r3, r1, #52 @ 13*4 = 541:ldr r4, [r2], #4 @ 读取设置值,并让r2加4 str r4, [r1], #4 @ 将此值写入寄存器,并让r1加4cmp r1, r3 @ 判断是否设置完所有13个寄存器bne 1b @ 若没有写成,继续mov pc, lr @ 返回.align 4mem_cfg_val:@ 存储控制器13个寄存器的设置值.long 0x22011110 @ BWSCON.long 0x00000700 @ BANKCON0.long 0x00000700 @ BANKCON1.long 0x00000700 @ BANKCON2.long 0x00000700 @ BANKCON3.long 0x00000700 @ BANKCON4.long 0x00000700 @ BANKCON5.long 0x00018005 @ BANKCON6.long 0x00018005 @ BANKCON7.long 0x008C07A3 @ REFRESH.long 0x000000B1 @ BANKSIZE.long 0x00000030 @ MRSRB6.long 0x00000030 @ MRSRB7下面来看看反汇编代码sdram_elf: file format elf32-littlearmDisassembly of section .text:30000000 <_start>:30000000: eb000005 bl 3000001c <disable_watch_dog> 30000004: eb000010 bl 3000004c <memsetup>30000008: eb000007 bl 3000002c <copy_steppingstone_to_sdram> 3000000c: e59ff090 ldr pc, [pc, #144] ; 300000a4<mem_cfg_val+0x34>30000010 <on_sdram>:30000010: e3a0d30d mov sp, #872415232 ; 0x34000000 30000014: eb000033 bl 300000e8 <main>30000018 <halt_loop>:30000018: eafffffe b 30000018 <halt_loop>3000001c <disable_watch_dog>:3000001c: e3a01453 mov r1, #1392508928 ; 0x53000000 30000020: e3a02000 mov r2, #030000024: e5812000 str r2, [r1]30000028: e1a0f00e mov pc, lr3000002c <copy_steppingstone_to_sdram>:3000002c: e3a01000 mov r1, #030000030: e3a02203 mov r2, #805306368 ; 0x30000000 30000034: e3a03a01 mov r3, #4096 ; 0x100030000038: e4914004 ldr r4, [r1], #43000003c: e4824004 str r4, [r2], #430000040: e1510003 cmp r1, r330000044: 1afffffb bne 30000038<copy_steppingstone_to_sdram+0xc>30000048: e1a0f00e mov pc, lr3000004c <memsetup>:3000004c: e3a01312 mov r1, #1207959552 ; 0x48000000 30000050: e28f2018 add r2, pc, #2430000054: e1a00000 nop ; (mov r0, r0) 30000058: e2813034 add r3, r1, #52 ; 0x343000005c: e4924004 ldr r4, [r2], #430000060: e4814004 str r4, [r1], #430000064: e1510003 cmp r1, r330000068: 1afffffb bne 3000005c <memsetup+0x10>3000006c: e1a0f00e mov pc, lr30000070 <mem_cfg_val>:30000070: 22011110 andcs r1, r1, #430000074: 00000700 andeq r0, r0, r0, lsl #1430000078: 00000700 andeq r0, r0, r0, lsl #143000007c: 00000700 andeq r0, r0, r0, lsl #1430000080: 00000700 andeq r0, r0, r0, lsl #1430000084: 00000700 andeq r0, r0, r0, lsl #1430000088: 00000700 andeq r0, r0, r0, lsl #143000008c: 00018005 andeq r8, r1, r530000090: 00018005 andeq r8, r1, r530000094: 008c07a3 addeq r0, ip, r3, lsr #1530000098: 000000b1 strheq r0, [r0], -r13000009c: 00000030 andeq r0, r0, r0, lsr r0300000a0: 00000030 andeq r0, r0, r0, lsr r0300000a4: 30000010 andcc r0, r0, r0, lsl r0300000a8: e1a00000 nop ; (mov r0, r0) 300000ac: e1a00000 nop ; (mov r0, r0)300000b0 <wait>:300000b0: e52db004 push {fp} ; (str fp, [sp, #-4]!)300000b4: e28db000 add fp, sp, #0300000b8: e24dd00c sub sp, sp, #12300000bc: e50b0008 str r0, [fp, #-8]300000c0: ea000002 b 300000d0 <wait+0x20>300000c4: e51b3008 ldr r3, [fp, #-8]300000c8: e2433001 sub r3, r3, #1300000cc: e50b3008 str r3, [fp, #-8]300000d0: e51b3008 ldr r3, [fp, #-8]300000d4: e3530000 cmp r3, #0300000d8: 1afffff9 bne 300000c4 <wait+0x14>300000dc: e28bd000 add sp, fp, #0300000e0: e8bd0800 pop {fp}300000e4: e12fff1e bx lr300000e8 <main>:300000e8: e92d4800 push {fp, lr}300000ec: e28db004 add fp, sp, #4300000f0: e24dd008 sub sp, sp, #8300000f4: e3a03000 mov r3, #0300000f8: e50b3008 str r3, [fp, #-8]300000fc: e59f3030 ldr r3, [pc, #48] ; 30000134 <main+0x4c>30000100: e3a02b55 mov r2, #87040 ; 0x15400 30000104: e5832000 str r2, [r3]30000108: e59f0028 ldr r0, [pc, #40] ; 30000138 <main+0x50>3000010c: ebffffe7 bl 300000b0 <wait>30000110: e59f3024 ldr r3, [pc, #36] ; 3000013c <main+0x54>30000114: e3a02000 mov r2, #030000118: e5832000 str r2, [r3]3000011c: e59f0014 ldr r0, [pc, #20] ; 30000138 <main+0x50>30000120: ebffffe2 bl 300000b0 <wait>30000124: e59f3010 ldr r3, [pc, #16] ; 3000013c <main+0x54>30000128: e3a02e1e mov r2, #480 ; 0x1e03000012c: e5832000 str r2, [r3]30000130: eafffff4 b 30000108 <main+0x20>30000134: 56000010 undefined instruction 0x5600001030000138: 00007530 andeq r7, r0, r0, lsr r53000013c: 56000014 undefined instruction 0x56000014Disassembly of section .ARM.attributes:00000000 <.ARM.attributes>:0: 00002541 andeq r2, r0, r1, asr #104: 61656100 cmnvs r5, r0, lsl #28: 01006962 tsteq r0, r2, ror #18c: 0000001b andeq r0, r0, fp, lsl r010: 00543405 subseq r3, r4, r5, lsl #814: 01080206 tsteq r8, r6, lsl #418: 04120109 ldreq r0, [r2], #-265 ; 0x1091c: 01150114 tsteq r5, r4, lsl r120: 01180317 tsteq r8, r7, lsl r324: Address 0x00000024 is out of bounds.Disassembly of section .comment:00000000 <.comment>:0: 3a434347 bcc 10d0d24 <SDRAM_BASE-0x2ef2f2dc>4: 74632820 strbtvc r2, [r3], #-2080 ; 0x8208: 312d676e teqcc sp, lr, ror #14c: 312e362e teqcc lr, lr, lsr #1210: 2e342029 cdpcs 0, 3, cr2, cr4, cr9, {1}14: 00332e34 eorseq r2, r3, r4, lsr lr当我们从Nand flash启动时,硬件会自动将Nand flash前4kB代码拷贝到片内SRAM中,然后CPU从SRAM的0x00000000地址处开始执行程序。

相关文档
最新文档