文章和代码已归档至【Github仓库:hardware-tutorial】,须要的敌人们自取。或者关注公众号【AIShareLab】,回复 嵌入式 也可获取。

一、试验目标

(1) 通过试验把握学会应用msr/mrs 指令实现ARM 处理器工作模式的切换,察看不同模式下的寄存器,加深对CPU 构造的了解;

(2) 通过试验把握ld 中如何应用命令行指定代码段起始地址。

二、试验环境

硬件:PC机。

软件:ADS1.2 集成开发环境

三、试验内容

通过 ARM 汇编指令,在各种处理器模式下切换并察看各种模式下寄存器的区别;把握ARM 不同模式的进入与退出。

四、试验要求

(1)依照2.3节介绍的办法, 在ADS下创立一个工程asmmodelab,实现各个模式下的堆栈初始化工作,并将R1-R12的内容存入以后模式下堆栈。通过AXD使用单步执行形式调试程序,验证工作模式的切换,留神察看CPSR寄存器中的变动。随着程序调试过程中在模式间的切换,应用寄存器观察器切换到不同的工作模式下察看SP(R13)的变动状况。

(2)试验过程中请记录并思考以下内容:

1)程序复位之后零碎处于什么模式?

2)记录每种模式下的初始堆栈指针,以及执行R1-R12内容压栈后本模式堆栈相干内存单元的数值。并剖析疾速中断FIQ模式与其余模式存入的R1-R12有什么不同。

3)切换成用户模式之后还是否从用户模式切换到其余模式(如零碎模式)?

4)用户模式下是否执行堆栈压栈操作?如果能得话,察看用户模式下压栈之前和压栈之后其堆栈区域的变动状况。

5)察看本程序模式切换过程中SPSR有无变动,并解释其起因。

五、试验状况

1、试验源代码(含正文):

usr_stack_legth equ 64 ;定义各个模式的栈空间长度svc_stack_legth equ 32fiq_stack_legth equ 16irq_stack_legth equ 64abt_stack_legth equ 16und_stack_legth equ 16                 area reset,code,readonly ;定义code片段reset只读  entry ;设置程序入口伪指令  code32 ;定义前面的指令为32位的ARM指令;设置各个寄存器中的内容start    mov r0,#0    mov r1,#1    mov r2,#2    mov r3,#3    mov r4,#4    mov r5,#5    mov r6,#6    mov r7,#7    mov r8,#8    mov r9,#9    mov r10,#10    mov r11,#11    mov r12,#12                 bl initstack  ;跳转至initstack,并且初始化各模式下的堆栈指针,关上IRQ中断(将cpsr寄存器的i位清0)                                  mrs r0,cpsr        ;r0<--cpsr    bic r0,r0,#0x80    ;cpsr的I地位0,开IRQ中断    msr cpsr_cxsf,r0   ;cpsr<--r0                                        ;切换到用户模式    msr cpsr_c,#0xd0  ;设置11010000,其中I,F地位1,禁止IRQ和FIQ中断,T=0,ARM执行,M[4:0]为10000,切换到用户模式    mrs r0,cpsr          ;r0<--cpsr    stmfd sp!,{r1-r12}   ;将R1-R12入栈     ;察看用户模式是否切换到其余模式    ;切换到管理模式    msr cpsr_c,#0xdf    ;设置11011111,其中I,F地位1,禁止IRQ和FIQ中断,T=0,ARM执行,M[4:0]为11111,切换到零碎模式    mrs r0,cpsr            ;r0<--cpsr    stmfd sp!,{r1-r12}  ;将寄存器列表中的r1-r12寄存器存入堆栈halt  b halt ;从halt跳转到halt循环initstack  mov r0,lr   ; r0<--lr,因为各种模式下r0是雷同的而各个模式不同                                              ;设置管理模式堆栈    msr cpsr_c,#0xd3   ; 设置11010011 切换到管理模式    ldr sp,stacksvc    ;设置管理模式堆栈地址    stmfd sp!,{r1-r12} ;R1-R12入栈,满递加模式    ;设置中断模式堆栈    msr cpsr_c,#0xd2   ;设置11010010  切换到中断模式    ldr sp,stackirq    ;设置中断模式堆栈地址    stmfd sp!,{r1-r12} ;R1-R12入栈,满递加模式    ;设置疾速中断模式堆栈    msr cpsr_c,#0xd1   ; 设置11010001  切换到疾速中断模式    ldr sp,stackfiq    ;设置疾速中断模式堆栈地址    stmfd sp!,{r1-r12} ;R1-R12入栈,满递加模式    ;设置停止模式堆栈       msr cpsr_c,#0xd7   ; 设置11010111  切换到停止模式    ldr sp,stackabt    ;设置停止模式堆栈地址    stmfd sp!,{r1-r12} ;R1-R12入栈,满递加模式    ;设置未定义模式堆栈       msr cpsr_c,#0xdb   ; 设置11011011  切换到未定义模式    ldr sp,stackund    ;设置未定义模式堆栈地址    stmfd sp!,{r1-r12} ;R1-R12入栈,满递加模式    ;设置零碎模式堆栈        msr cpsr_c,#0xdf   ; 设置11011111  切换到零碎模式    ldr sp,stackusr    ;设置零碎模式堆栈地址    stmfd sp!,{r1-r12} ;R1-R12入栈,满递加模式    mov pc,r0 ;返回        ;为各模式堆栈开拓一段间断的字存储空间stackusr    dcd  usrstackspace+(usr_stack_legth-1)*4stacksvc    dcd  svcstackspace+(svc_stack_legth-1)*4stackirq    dcd  irqstackspace+(irq_stack_legth-1)*4stackfiq    dcd  fiqstackspace+(fiq_stack_legth-1)*4stackabt    dcd  abtstackspace+(abt_stack_legth-1)*4stackund    dcd  undstackspace+(und_stack_legth-1)*4      ;定义data段并命名      area reset,data,noinit,align=2;为各模式堆栈调配存储区域usrstackspace space usr_stack_legth*4svcstackspace space svc_stack_legth*4irqstackspace space irq_stack_legth*4fiqstackspace space fiq_stack_legth*4abtstackspace space abt_stack_legth*4undstackspace space und_stack_legth*4    end

2、试验过程(含后果截图及相应文字解释):

试验过程中请记录并思考以下内容:

1)程序复位之后零碎处于什么模式?

由上可知,零碎复位后处于管理模式。

2)记录每种模式下的初始堆栈指针,以及执行R1-R12内容压栈后本模式堆栈相干内存单元的数值。并剖析疾速中断FIQ模式与其余模式存入的R1-R12有什么不同。

①管理模式

由上图可知,管理模式初始指针为0x8244。

执行R1-R12内容压栈后本模式堆栈相干内存单元的数值如上图所示,可知压栈后,堆栈指针变为0x8214,离初始的堆栈指针0x30字节,即12个字(32位零碎),从内存单元的数值能够看到别离与R1-R12存储的数值对应。

②中断模式

由上图可知,中断模式初始指针为0x8344。

执行R1-R12内容压栈后本模式堆栈相干内存单元的数值如上图所示,可知压栈后,堆栈指针变为0x8314,离初始的堆栈指针0x30字节,即12个字(32位零碎),从内存单元的数值能够看到别离与R1-R12存储的数值对应。

③疾速中断模式

由上图可知,疾速中断模式初始指针为0x8384。

执行R1-R12内容压栈后本模式堆栈相干内存单元的数值如上图所示,可知压栈后,堆栈指针变为0x8354,离初始的堆栈指针0x30字节,即12个字(32位零碎),从内存单元的数值能够看到别离与R1-R7存储的数值对应,阐明该模式下仅能压入R1-R7,因为疾速中断模式有本人的R8-R12。

④停止模式

由上图可知,停止模式初始指针为0x83C4。

执行R1-R12内容压栈后本模式堆栈相干内存单元的数值如上图所示,可知压栈后,堆栈指针变为0x8394,离初始的堆栈指针0x30字节,即12个字(32位零碎),从内存单元的数值能够看到别离与R1-R12存储的数值对应。

⑤未定义模式

由上图可知,未定义模式初始指针为0x8404。

执行R1-R12内容压栈后本模式堆栈相干内存单元的数值如上图所示,可知压栈后,堆栈指针变为0x83D4,离初始的堆栈指针0x30字节,即12个字(32位零碎),从内存单元的数值能够看到别离与R1-R12存储的数值对应。

⑥零碎模式

由上图可知,零碎模式初始指针为0x81C4。

执行R1-R12内容压栈后本模式堆栈相干内存单元的数值如上图所示,可知压栈后,堆栈指针变为0x8194,离初始的堆栈指针0x30字节,即12个字(32位零碎),从内存单元的数值能够看到别离与R1-R12存储的数值对应。

⑦用户模式:

由上图可知,用户模式初始指针为0x8194。

执行R1-R12内容压栈后本模式堆栈相干内存单元的数值如上图所示,可知压栈后,堆栈指针变为0x8164,离初始的堆栈指针0x30字节,即12个字(32位零碎),从内存单元的数值能够看到别离与R1-R12存储的数值对应。

3)切换成用户模式之后还是否从用户模式切换到其余模式(如零碎模式)?

由上图可知,当进行切换管理模式时,模式仍是用户模式,因而可知切换成用户模式之后,不能操作CPSR返回到其余模式。

4)用户模式下是否执行堆栈压栈操作?如果能得话,察看用户模式下压栈之前和压栈之后其堆栈区域的变动状况。

压栈前:

压栈后:

压栈后存储单元状况:

答:用户模式下能够执行堆栈压栈操作,且以4个字节(1个字)为单位进行压栈操作,压栈前堆栈区域状况如左图,压栈后如右图所示,对应的存储单元状况如上图。

5)察看本程序模式切换过程中SPSR有无变动,并解释其起因。

答:除了用户模式和零碎模式,其余模式下都有一个公有SPSR保留状态寄存器. 用来保留切换到该模式之前的执行状态,SPSR是异样模式的程序状态保留寄存器, 当特定的异常中断产生时,这个寄存器寄存CPSR的内容,在异常中断退出时,能够用SPSR来复原CPSR,然而通过观察可知,整个切换过程中没有异样的产生,因而SPSR没有变动。

六、总结

ARM处理器模式别离是usr(用户模式),fiq(疾速中断模式),irq(通用中断模式),svc(管理模式),abt(终止模式),sys(零碎模式)以及und(未定义模式)。也通过ARM指令,实现了ARM不同模式的进入与退出,切换各种处理器模式,并察看各种模式下寄存器的区别。当特定的异样呈现时,进入相应的模式。每种模式都有某些附加的寄存器,以防止异样呈现时用户模式的状态不牢靠。此外也应用状态寄存器到通用寄存器的传送指令(MRS)以及通用寄存器到状态寄存器的传送指令(MSR),批改状态寄存器通过“读取-批改-写回”三个步骤操作来实现。


欢送关注公众号【AIShareLab】,一起交换更多相干常识,前沿算法,Paper解读,我的项目源码,面经总结。