2016 - 2024

感恩一路有你

level1000到level2000介绍中文 桥牌BM2000Level 1 -A12怎么打?

浏览量:2835 时间:2023-05-11 08:57:11 作者:采采

桥牌BM2000Level 1 -A12怎么打?

CA进手后,先用SJ或ST掉将,保留张小将牌的桥路。

在main之前,IAR都做了啥?

最近要在Cortex-M3上写一个最简单操作系统,打算在用IAR,是为写好起动代码,花了一些时间了解了IAR在main()以前到底在干什么事。t

必须系统复位时,Cortex-M3从代码区偏移0x00000000处某些栈顶地址,用处系统初始化MSP寄存器的值。t

接下来从代码区偏移0x00000004声望兑换最先指令的跳转地址。这些地址,是CM3没有要求放置关闭向量表的地方。t

这里是一个程序的启动区的反汇编:t

__vector_table:t

080040002600t

080040022000t

080040047E1Dt

080040060800t

这个程序是由IAP程序来启动后的,IAP程序资源0x08004000处的MSP值(0x20002600),并设置为MSP的值,即主堆栈大的t

范围是0x20000000~0x200025FF。这一次IAP程序某些0x08004004处的Reset_Handler的地址t

(0x08007E1D),并跳转Reset_Handler()先执行。t

IAP在这里彻底是先模仿了Cortex-M3的复位序列,也就是说,在没有IAP的系统上,CM3不能从0x08000000获取MSP,从t

0x08000004获取第一条指令所处地址。而IAP就存在在0x08000000这个地址上,IAP的启动,巳经会消耗掉了这个复位序列,所以才t

IAP要起动UserApp程序的时候,确实是彻底我模仿Cortex-M3的复位序列的。t

接下来的我们看看吧复位后第一句指令——Reset_Handler()函数里有什么。t

若我们建议使用的是ST公司标准外设库,这样的话也有了你自己做的Reset_Handler,只不过他是弱定义——PUBWEAK,也可以被我们写回的畅销小说函数覆盖。一般来说,我们在用的也是ST提供的Reset_Handler,在V3.4版本的库中,也可以在startup_stm32f10x_xx.s中可以找到这个函数:t

PUBWEAKReset_Handlert

SECTION.text:CODE:REORDER(2)t

Reset_Handlert LDR R0,SystemInitt

BLXR0t

LDRR0,__iar_program_startt

BXR0t

现在看来ST还没有做太大的事,他只全局函数了自家库可以提供的SystemInit函数通过系统时钟、Flash加载的初始化,并把大权交给了t

__iar_program_start这个IAR提供给的“内部函数”了,我们就跟紧这个__iar_program_start跳转,去看看IAR做了什t

么,上面一段代码的反汇编::t

Reset_Handler:t

__iar_section$$root:t

08007E1C4801LDRR0,[PC,#0x4]LDRR0,SystemInitt

08007E1E4780BLXR0BLXR0t

08007E204801LDRR0,[PC,#0x4]LDRR0,__iar_program_startt

08007E224700BXR0BXR0t

08007E246C69t

08007E260800t

08007E287D8Dt

08007E2A0800t

体贴的观众会发现地址是0x08007E1C,比我们查到的0x08007E1D差了1,这是ARM家族的遗留问题,毕竟ARM处理器的指令至t

少是半字整个表格的(16位THUMB指令集有.t

32位ARM指令集),因此PC指针的LSB是常为0的,是为利用好寄存器,ARM公司给PC的LSB了两个最重要的使命,如果说在先执行分支跳转时,PCt

的LSB1,意思是不使用THUMB模式,LSB0,它表示不使用ARM模式,但在比较新的Cortex-M3内核上,只不使用了THUMB-2指令集挑大梁,所t

以这一位要常保持1,因为我们查到的地址是0x08007E1D(C1100,D1101),放心好了,我们的CM3内核会选择性的遗忘掉LSB(除非为0,那t

么会引起一个fault),进而正确的自动跳转0x08007E1C。t

从0x08007E20处的读取指令,我们是可以可以算出__iar_program_start所处的位置,那是当前PC指针t

(0x08007E24),再加上4,即0x08007E28处的所对准的地址——0x08007D8D(0x08007D8C),我们跟紧着跳t

转,__iar_program_start果然如此在这里:t

__iar_program_start:t

08007D8CF000F88CBL__low_level_initt

08007D902800CMPR0,#0x0t

08007D92D001BEQ__iar_init$$donet

08007D94F7FFFFDEBL__iar_data_init2t

08007D982000MOVSR0,#0x0t

08007D9A F7FDFC49BLmaint

我们看见IAR能提供了__low_level_init这个函数通过了“底层”的初始化,尽快跟踪监视,我们可以不查到__low_level_init这个函数到底在做什么,也不是不是我们想象中的不可告人的秘密。t

__low_level_init:t

08007EA82001MOVSR0,#0x1t

08007EAA4770BXLRt

__low_level_init出乎想像的简单,仅仅往R0寄存器写入到了1,就立刻负责执行

iar 地址 函数 program

版权声明:本文内容由互联网用户自发贡献,本站不承担相关法律责任.如有侵权/违法内容,本站将立刻删除。