基于ARM的硬件启动程序设计-初始化堆栈
2009-12-30 21:40
218 查看
ARM有7种运行状态,每一种状态的堆栈指针寄存器(SP)都是独立的。所以,对于程序中需要用的每一种处理器模式,都要给SP定义一个堆栈地址。流程为:修改状态寄存器内的状态位,使处理器切换到需要的模式,然后给SP赋值。需要注意的是:不要切换到User模式进行该模式下的堆栈设置,因为进入User模式后就不能再操作CPSR返回到其他模式了。
先定义各种模式对应的CPSR寄存器M[4:0]的值,该值决定了进入何种模式,可参考相关数据手册。
Mode_USR EQU 0x10
Mode_FIQ EQU 0x11
Mode_IRQ EQU 0x12
Mode_SVC EQU 0x13
Mode_ABT EQU 0x17
Mode_UND EQU 0x1B
Mode_SYS EQU 0x1F
I_BIT EQU 0x80 ; when I bit is set (1), IRQ is disabled
F_BIT EQU 0x40 ; when F bit is set (1), FIQ is disabled
下面是初始化堆栈的代码段:
InitStack
MOV R0, LR
; /* 设置管理模式堆栈 */
MSR CPSR_c, #(Mode_SVC | I_BIT | F_BIT) ; 0xd3
LDR SP, StackSvc
; /* 设置中断模式堆栈 */
MSR CPSR_c, #(Mode_IRQ | I_BIT | F_BIT) ; 0xd2
LDR SP, StackIrq
; /* 设置快速中断模式堆栈 */
MSR CPSR_c, #(Mode_FIQ | I_BIT | F_BIT) ; 0xd1
LDR SP, StackFiq
; /* 设置中止模式堆栈 */
MSR CPSR_c, #(Mode_ABT | I_BIT | F_BIT) ; 0xd7
LDR SP, StackAbt
; /* 设置未定义模式堆栈 */
MSR CPSR_c, #(Mode_UND | I_BIT | F_BIT) ; 0xdb
LDR SP, StackUnd
; /* 设置系统模式堆栈 */
MSR CPSR_c, #(Mode_SYS | I_BIT | F_BIT) ; 0xdf
LDR SP, StackUsr
MOV PC, R0
StackUsr DCD UsrStackSpace + (USR_STACK_LEGTH - 1) * 4
StackSvc DCD SvcStackSpace + (SVC_STACK_LEGTH - 1)* 4
StackIrq DCD IrqStackSpace + (IRQ_STACK_LEGTH - 1)* 4
StackFiq DCD FiqStackSpace + (FIQ_STACK_LEGTH - 1)* 4
StackAbt DCD AbtStackSpace + (ABT_STACK_LEGTH - 1)* 4
StackUnd DCD UndtStackSpace + (UND_STACK_LEGTH - 1)* 4
; /* 分配堆栈空间 */
AREA MyStacks, DATA, NOINIT, ALIGN=2
UsrStackSpace SPACE USR_STACK_LEGTH * 4 ;用户(系统)模式堆栈空间
SvcStackSpace SPACE SVC_STACK_LEGTH * 4 ;管理模式堆栈空间
IrqStackSpace SPACE IRQ_STACK_LEGTH * 4 ;中断模式堆栈空间
FiqStackSpace SPACE FIQ_STACK_LEGTH * 4 ;快速中断模式堆栈空间
AbtStackSpace SPACE ABT_STACK_LEGTH * 4 ;中止义模式堆栈空间
UndtStackSpace SPACE UND_STACK_LEGTH * 4 ;未定义模式堆栈
以管理模式堆栈空间为例说明一下。
SVC_STACK_LEGTH 定义为 16 ,分配结果为:分配16个4字节的存储空间,把该存储空间初始化为0,并且把SP指向堆栈底部(内存高位)。
从上面memory的内容上面可以看出。管理模式堆栈分配了从0x00008080~0x00008044 一共16*4字节空间。其中0x00008040地址存储器上放置的是0x00008080,也就是该堆栈底部的指针的值。
就是说,StackSvc DCD SvcStackSpace + (SVC_STACK_LEGTH - 1)* 4 这条32位的伪指令,经过编译后,是在0x00008040(StackSvc)地址上放置SvcStackSpace地址值(0x00008080)。也就是把0x00008080赋给了SP了。
再看看地址0x00008080~0x00008044 上的内容,全部是andeq r0,r0,r0。从汇编指令格式来翻译,andeq r0,r0,r0翻译成16禁止代码后,刚好是0x00000000。
先定义各种模式对应的CPSR寄存器M[4:0]的值,该值决定了进入何种模式,可参考相关数据手册。
Mode_USR EQU 0x10
Mode_FIQ EQU 0x11
Mode_IRQ EQU 0x12
Mode_SVC EQU 0x13
Mode_ABT EQU 0x17
Mode_UND EQU 0x1B
Mode_SYS EQU 0x1F
I_BIT EQU 0x80 ; when I bit is set (1), IRQ is disabled
F_BIT EQU 0x40 ; when F bit is set (1), FIQ is disabled
下面是初始化堆栈的代码段:
InitStack
MOV R0, LR
; /* 设置管理模式堆栈 */
MSR CPSR_c, #(Mode_SVC | I_BIT | F_BIT) ; 0xd3
LDR SP, StackSvc
; /* 设置中断模式堆栈 */
MSR CPSR_c, #(Mode_IRQ | I_BIT | F_BIT) ; 0xd2
LDR SP, StackIrq
; /* 设置快速中断模式堆栈 */
MSR CPSR_c, #(Mode_FIQ | I_BIT | F_BIT) ; 0xd1
LDR SP, StackFiq
; /* 设置中止模式堆栈 */
MSR CPSR_c, #(Mode_ABT | I_BIT | F_BIT) ; 0xd7
LDR SP, StackAbt
; /* 设置未定义模式堆栈 */
MSR CPSR_c, #(Mode_UND | I_BIT | F_BIT) ; 0xdb
LDR SP, StackUnd
; /* 设置系统模式堆栈 */
MSR CPSR_c, #(Mode_SYS | I_BIT | F_BIT) ; 0xdf
LDR SP, StackUsr
MOV PC, R0
StackUsr DCD UsrStackSpace + (USR_STACK_LEGTH - 1) * 4
StackSvc DCD SvcStackSpace + (SVC_STACK_LEGTH - 1)* 4
StackIrq DCD IrqStackSpace + (IRQ_STACK_LEGTH - 1)* 4
StackFiq DCD FiqStackSpace + (FIQ_STACK_LEGTH - 1)* 4
StackAbt DCD AbtStackSpace + (ABT_STACK_LEGTH - 1)* 4
StackUnd DCD UndtStackSpace + (UND_STACK_LEGTH - 1)* 4
; /* 分配堆栈空间 */
AREA MyStacks, DATA, NOINIT, ALIGN=2
UsrStackSpace SPACE USR_STACK_LEGTH * 4 ;用户(系统)模式堆栈空间
SvcStackSpace SPACE SVC_STACK_LEGTH * 4 ;管理模式堆栈空间
IrqStackSpace SPACE IRQ_STACK_LEGTH * 4 ;中断模式堆栈空间
FiqStackSpace SPACE FIQ_STACK_LEGTH * 4 ;快速中断模式堆栈空间
AbtStackSpace SPACE ABT_STACK_LEGTH * 4 ;中止义模式堆栈空间
UndtStackSpace SPACE UND_STACK_LEGTH * 4 ;未定义模式堆栈
以管理模式堆栈空间为例说明一下。
SVC_STACK_LEGTH 定义为 16 ,分配结果为:分配16个4字节的存储空间,把该存储空间初始化为0,并且把SP指向堆栈底部(内存高位)。
从上面memory的内容上面可以看出。管理模式堆栈分配了从0x00008080~0x00008044 一共16*4字节空间。其中0x00008040地址存储器上放置的是0x00008080,也就是该堆栈底部的指针的值。
就是说,StackSvc DCD SvcStackSpace + (SVC_STACK_LEGTH - 1)* 4 这条32位的伪指令,经过编译后,是在0x00008040(StackSvc)地址上放置SvcStackSpace地址值(0x00008080)。也就是把0x00008080赋给了SP了。
再看看地址0x00008080~0x00008044 上的内容,全部是andeq r0,r0,r0。从汇编指令格式来翻译,andeq r0,r0,r0翻译成16禁止代码后,刚好是0x00000000。
相关文章推荐
- 基于ARM的硬件启动程序设计-初始化堆栈(转载)
- 基于ARM的硬件启动程序设计-初始化堆栈
- 基于ARM的硬件启动程序设计-分配中断向量表
- ARM OS启动时的 堆栈初始化
- 基于ARM硬件初始化的一些细节问题的解释
- AP核MAILBOX地址的初始化和启动过程(基于ARM64 的 APCI Parking protocol)
- 2010年1月16日,基于ARM的嵌入式系统软硬件开发讲座(北京)
- 基于ARM的嵌入式系统Bootloader启动流程分析
- 基于IAR集成开发平台的ARM程序设计方法(转)
- 基于ARM 的Linux 的启动分析报告——ARM+Linux的启动分析(3)
- 嵌入式ARM平台基于Linux配置应用程序开机自启动
- 基于ARM的嵌入式系统Bootloader启动流程分析----- 转!!
- linux-3.2.36内核启动3-setup_arch中的内存初始化2(arm平台 分析建立页表)
- ARM学习笔记012之mini2440上电的启动和初始化程序分析(即Bootloader分析)
- 基于Arm 的linux 的启动分析
- ARM硬件平台上基于UCOS移植Lwip网络协议栈 分类: 嵌入式开发学习 2015-06-14 10:33 55人阅读 评论(1) 收藏
- 软件、硬件的初始化(启动文件初始化)
- Uboot启动过程 之 硬件初始化
- 基于 ARM 体系的嵌入式系统 BSP 的程序设计
- 基于ARM-uCLinux嵌入式系统启动引导的实现