- 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
- 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
- 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
- 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们。
- 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
- 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
ARM下的位置无关位置无关代码,即该段代码无论放在内存的哪个地址,都能正确运行。究其原因,是因为代码里没有使用绝对地址,都是相对地址。位置无关的写法:
(1) B指令B指令接受一个相对地址,因此在汇编里用B跳转到一个标号时,实际编译的结果是一个相对跳转。相对地址有个范围限制,即目标不能太远,一般目标放在同一个文件里是肯定可以的。 O_start:
?? ?b? _reset
_reset:
? ?? ?...
(2) BL
BL用于调用函数,也是一个相对跳转(3) ADR
获取标号的地址,在编译时会使用PC+偏移的方式得到该位置的地址。例如,当TEXT_BASE是0时SMRDATA可能被放在0x100的位置,当TEXT_BASE为0放在0位置。使用ADR
总能获取正确的位置,与程序的加载地址无关。
?? ?ADR R0, SMRDATA
SMRDATA:
?? ?.word? 0
?? ?.word? 0x00002F50?
?? ?.word? 0
(相应的, LDR Rn, =LABEL是位置相关的)
(4) LDR
当加标号时,LDR可以用于伪指令,也可以真指令。
真指令: (标号前不加=号,表示取标号处的值)
?? ?LDR R0,? SDRDATA
实际被编译为LDR R0, [PC, #NN],其中NN是目标的相对距离
伪指令: (标号前加=号,取标号的地址)
?? ?LDR R0, = SDRDATA
实际编译的时候的时候,会在某位置存处SDRDATA的值,然后用一个LDR取出来。
显然,用LDR时,加不加=号有很大区别。
无=号:取该标号处的值,位置无关
有=号:取该标号的地址,位置相关
举例分析
例1:中断向量跳转
_start:?? ?
?? ?b?????? reset
?? ?ldr?? ?pc, _undefined_instruction
?? ?ldr?? ?pc, _software_interrupt
?? ?ldr?? ?pc, _prefetch_abort
?? ?ldr?? ?pc, _data_abort
?? ?ldr?? ?pc, _not_used
?? ?ldr?? ?pc, _irq
?? ?ldr?? ?pc, _fiq
_undefined_instruction:?? ?.word undefined_instruction
_software_interrupt:?? ?.word software_interrupt
_prefetch_abort:?? ?.word prefetch_abort
_data_abort:?? ??? ?.word data_abort
_not_used:?? ??? ?.word not_used
_irq:?? ??? ??? ?.word irq
_fiq:?? ??? ??? ?.word fiq
其中,
ldr pc, _irq,由于没加=号,表示取值_irq处的值放在pc里 (位置无关)
_irq:? .word irq ,表示_irq存放的值是irq的绝对地址(位置有关)
例2:
bl? main ; 位置无关
ldr pc, =main; 把main的地址放在pc,位置相关
例3: 静态变量
_MAGIC_NUM:
?? ?.word 0取值
?? ?LDR? R0, _MAGIC_NUM? ; 位置无关
例4: 存放标号绝对地址(绝对地址是编译的时候已经固定)
_OS_Running_p:
?? ?.word? OS_Runing
则_OS_Running_p存放的是标号OS_Running的绝对地址
例5: 显式LDR和隐式LDR
以给某C中的变量的g_num赋值为例
(1) 使用伪指令LDR,即为隐式
?? ?LDR? R0, =g_num??? @取g_num的地址到R0
?? ?MOV R1, #10
??????????? STR? R1, R0
(2) 显式赋值
先定义一个变量p_g_num,用于保存g_num的地址
p_g_num:
?? ?.word?? g_num?? @ g_num的绝对地址
然后赋值
?? ?LDR R0, p_g_num
?? ?MOV R1, #10
?? ?STR R1,? R0
显然,两者其实一样,伪指令被展开后其实就是(2)的样子。
不同点在于:在多次引用的时候,如果使用伪指令,则会有多个临时定义。所以,
在多次引用的时候应该使用显式定义。
例6: 使用LinkScript中的变量
这种情形和例5相同
1) LinkScript中定义了两个位置
{
?? ?__bs
文档评论(0)