keil arm汇编如何指定一个绝对地址的代码区?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了keil arm汇编如何指定一个绝对地址的代码区?相关的知识,希望对你有一定的参考价值。

如何像51汇编一样用cseg at 在讲我想要的代码区(比如中断向量表)放入一个我想要的ROM下的绝对位置?

参考技术A 在要指定代码的存储空间不是一件特别简单的事情,尤其是你想为某个或某几个函数指定具体的地址。
1,编译器只有在最终的Link阶段才会为代码和数据分配内存地址,因此指定代码段的地址一般是通过写一个link脚本来进行的。Link阶段时,编译器的Linker会读取你写的Link脚本,并且按照脚本的规定给代码分配地址。
2,根据ARM开发工具的不同,link脚本的语法和形式也有所不同。ARM MDK,ARM ADS,Eclips+GCC,Linux GCC, ARM Realview等开发工具都支持Link脚本。
如果你英文还可以,建议你直接找到开发工具的Help手册去研究。如果你英语实在不行,也可以把开发工具名称和你代码的具体情况告诉我,我帮你看看。追问

你的意思是无论如何都不能向链接器指定一个绝对地址?51都可以,arm不行?

ARM汇编基础基于Keil创建STM32汇编程序的编写

一、新建项目

(1)工具介绍

使用工具是arm keil,可在官网下载—— 官网链接,点击下载:

(2)创建项目:

打开keil,点击project–>new μvision project:

选择文件路径,输入文件名,点击保存,建立项目:

二、配置环境

(1)配置芯片

这里选择STM32的芯片,F103系列:

选择STM32F103ZE,点击OK:

(2)配置运行环境

分别打开CMSIS的CORE,点击勾选,打开Device的startup,点击勾选,再点击OK:

到这里项目就创建和配置完成了,我们可以发现左边出现了项目文件,在Source Group 1中需要创建一个TEST的测试代码的文件:

右键鼠标,选择ADD NEW Item…:

选择Asm的file,输入文件名,再add:

三、调试代码(这里选择使用的是仿真调试)

(1)源代码

在TEST的文件中放入下面的测试代码:

AREA MYDATA, DATA
	
 AREA MYCODE, CODE
	ENTRY
	EXPORT __main

__main
	MOV R0, #10
	MOV R1, #11
	MOV R2, #12
	MOV R3, #13
	;LDR R0, =func01

	BL	func01
	;LDR R1, =func02
	BL	func02
	
	BL 	func03
	LDR LR, =func01
	LDR PC, =func03
	B .
		
func01
	MOV R5, #05
	BX LR
	
func02
	MOV R6, #06
	BX LR
	
func03
	MOV R7, #07
	MOV R8, #08	
	BX LR

(2)仿真设置

仿真是因为在没有芯片的情况下来调试运行代码的一种方式。
后面会要分析hex文件,所以打开project里的option for target…后选择Target:

再选择debug,选择use simulator,再将第二步里的两个参数分别改为DARMSTM.DLL和-pSTM32F103ZE,点击ok:

点击21行位置设置断点后build,再开始仿真调试:

运行后,点击F10,跳转到最后:

hex文件分析

(1)hex文件介绍

Intel HEX文件是由一行行符合Intel HEX文件格式的文本所构成的ASCII文本文件。在Intel HEX文件中,每一行包含一个HEX记录。这些记录由对应机器语言码和/或常量数据的十六进制编码数字组成。Intel HEX文件通常用于传输将被存于ROM或者EPROM中的程序和数据。大多数EPROM编程器或模拟器使用Intel HEX文件。

(2)hex文件数据格式

可以在创建项目的文件夹objects中找到hex文件,以文本格式打开:

:020000040800F2
:100000000004002031010008390100083B0100080C
:100010003D0100083F010008410100080000000008
:100020000000000000000000000000004301000884
:1000300045010008000000004701000849010008D0
:100040004B0100084B0100084B0100084B01000860
:100050004B0100084B0100084B0100084B01000850
:100060004B0100084B0100084B0100084B01000840
:100070004B0100084B0100084B0100084B01000830
:100080004B0100084B0100084B0100084B01000820
:100090004B0100084B0100084B0100084B01000810
:1000A0004B0100084B0100084B0100084B01000800
:1000B0004B0100084B0100084B0100084B010008F0
:1000C0004B0100084B0100084B0100084B010008E0
:1000D0004B0100084B0100084B0100084B010008D0
:1000E0004B0100084B0100084B0100084B010008C0
:1000F0004B0100084B0100084B0100084B010008B0
:100100004B0100084B0100084B0100084B0100089F
:100110004B0100084B0100084B0100084B0100088F
:100120004B0100084B0100084B0100084B0100087F
:100130000648804706480047FEE7FEE7FEE7FEE781
:10014000FEE7FEE7FEE7FEE7FEE7FEE781020008C6
:10015000550100084FF00A004FF00B014FF00C0260
:100160004FF00D0300F009F800F00AF800F00BF86A
:10017000DFF81CE0DFF81CF0FEE74FF005057047E4
:100180004FF0060670474FF007074FF0080870471A
:100190007B0100088701000810B500F001F810BDD0
:1001A0000CB50020019000903348006840F4803086
:1001B0003149086000BF3048006800F4003000900A
:1001C0000198401C0190009818B90198B0F5A06FF3
:1001D000F1D12948006800F4003010B101200090EE
:1001E00001E0002000900098012843D123480068D6
:1001F00040F01000214908600846006820F0030024
:1002000008600846006840F0020008601A4840682C
:10021000194948600846406848600846406840F40C
:10022000806048600846406820F47C1048600846BA
:10023000406840F4E81048600846006840F080706C
:10024000086000BF0C48006800F000700028F9D07A
:100250000948406820F003000749486008464068A4
:1002600040F00200486000BF0348406800F00C0006
:100270000828F9D10CBD0000001002400020024007
:1002800010B51348006840F00100114908600846A5
:100290004068104908400E494860084600680E4909
:1002A00008400B4908600846006820F48020086078
:1002B0000846406820F4FE0048604FF41F00886044
:1002C000FFF76AFF4FF000600449086010BD0000AE
:1002D000001002400000FFF8FFFFF6FE08ED00E00E
:0400000508000131BD
:00000001FF

其中第一行 :020000040800F2 中,可以看做是0x02 0x00 0x00 0x04 0x08 0x00 0xf2;
第一个0×02表示该行数据中有两个数据;
第二个,第三个0x00 0x00表示本行数据的起始地址位;
第四个字节 0x04 表示扩展线性地址记录,对应上述的TT域;
第五个、第六个 0x08 0x00表示数据字节,与**数据长度域(LL)**中对应,即第一个 0x02;
最后一个字节0xf2为校验和。校验和= 0x100 - 累加和。

参考资料

ARM汇编基础之基于MDK创建纯汇编语言的STM32工程
hex文件百科
基于 MDK 创建 STM32 汇编语言工程并分析 HEX 文件内容
STM32基于汇编方式创建工程文件和闪烁LED灯

以上是关于keil arm汇编如何指定一个绝对地址的代码区?的主要内容,如果未能解决你的问题,请参考以下文章

在ARM汇编编程中如何指定某段程序的存储地址

ARM汇编基础基于Keil创建STM32汇编程序的编写

keil c代码空间怎么指定分配,栈检测,常量代码数组指定存放地址?

汇编程序的运行方法

keil中 如何向指定地址写入数据

搭建并配置Keil嵌入式开发环境,完成一个基于STM32汇编程序的编写