嵌入式ARM设计编程(三) 处理器工作模式

这篇具有很好参考价值的文章主要介绍了嵌入式ARM设计编程(三) 处理器工作模式。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

文章和代码已归档至【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 32
fiq_stack_legth equ 16
irq_stack_legth equ 64
abt_stack_legth equ 16
und_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)*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  undstackspace+(und_stack_legth-1)*4
	  ;定义data段并命名
      area reset,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
undstackspace space und_stack_legth*4
    end

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

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

1)程序复位之后系统处于什么模式?

timerring 嵌入式arm设计编程,ARM,arm开发

由上可知,系统复位后处于管理模式。

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

①管理模式

timerring 嵌入式arm设计编程,ARM,arm开发

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

timerring 嵌入式arm设计编程,ARM,arm开发

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

②中断模式

timerring 嵌入式arm设计编程,ARM,arm开发

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

timerring 嵌入式arm设计编程,ARM,arm开发

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

③快速中断模式

timerring 嵌入式arm设计编程,ARM,arm开发

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

timerring 嵌入式arm设计编程,ARM,arm开发

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

④中止模式

timerring 嵌入式arm设计编程,ARM,arm开发

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

timerring 嵌入式arm设计编程,ARM,arm开发

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

⑤未定义模式

timerring 嵌入式arm设计编程,ARM,arm开发

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

timerring 嵌入式arm设计编程,ARM,arm开发

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

⑥系统模式

timerring 嵌入式arm设计编程,ARM,arm开发

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

timerring 嵌入式arm设计编程,ARM,arm开发

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

⑦用户模式:

timerring 嵌入式arm设计编程,ARM,arm开发

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

timerring 嵌入式arm设计编程,ARM,arm开发

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

3)切换成用户模式之后还能否从用户模式切换到其他模式(如系统模式)?

timerring 嵌入式arm设计编程,ARM,arm开发

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

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

压栈前:

timerring 嵌入式arm设计编程,ARM,arm开发

压栈后:

timerring 嵌入式arm设计编程,ARM,arm开发

压栈后存储单元情况:

timerring 嵌入式arm设计编程,ARM,arm开发

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

5)观察本程序模式切换过程中SPSR有无变化,并解释其原因。

答:除了用户模式和系统模式,其余模式下都有一个私有SPSR保存状态寄存器. 用来保存切换到该模式之前的执行状态,SPSR是异常模式的程序状态保存寄存器, 当特定的异常中断发生时,这个寄存器存放CPSR的内容,在异常中断退出时,可以用SPSR来恢复CPSR,但是通过观察可知,整个切换过程中没有异常的发生,因此SPSR没有变化。

六、总结

ARM处理器模式分别是usr(用户模式),fiq(快速中断模式),irq(通用中断模式),svc(管理模式),abt(终止模式),sys(系统模式)以及und(未定义模式)。也通过ARM指令,实现了ARM不同模式的进入与退出,切换各种处理器模式,并观察各种模式下寄存器的区别。当特定的异常出现时,进入相应的模式。每种模式都有某些附加的寄存器,以避免异常出现时用户模式的状态不可靠。此外也使用状态寄存器到通用寄存器的传送指令(MRS)以及通用寄存器到状态寄存器的传送指令(MSR),修改状态寄存器通过“读取-修改-写回”三个步骤操作来实现。文章来源地址https://www.toymoban.com/news/detail-775933.html

到了这里,关于嵌入式ARM设计编程(三) 处理器工作模式的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • [ARM 汇编]高级部分—ARM汇编编程实战—3.3.3 嵌入式应用程序设计实例

    在本章节中,我们将学习如何使用ARM汇编编写一个简单的嵌入式应用程序。我们将以STM32F103微控制器为例,编写一个程序,实现按下按钮时点亮LED的功能。 硬件连接 首先,我们需要将STM32F103微控制器的一个GPIO引脚连接到LED(通过一个合适的电阻),另一个GPIO引脚连接到按钮

    2024年02月10日
    浏览(42)
  • [ARM 汇编]高级部分—ARM汇编编程实战—3.3.2 嵌入式开发环境搭建

    搭建一个嵌入式开发环境主要包括以下几个部分: 安装交叉编译器 配置集成开发环境(IDE) 安装调试工具 下载和烧录程序 接下来,我们将详细介绍每个部分,并提供相应的实例。 安装交叉编译器 交叉编译器是用于将您编写的ARM汇编代码编译成可执行文件的工具。在本教程

    2024年02月11日
    浏览(54)
  • 嵌入式学习笔记(3)ARM的异常处理方式介绍

    什么是异常   正常工作之外的流程都叫异常   异常会打断正在执行的工作,并且一般我们希望异常处理完后继续回来执行原工作   中断是异常的一种 异常向量表   所有的CPU都有异常向量表,这是CPU设计时就设定好的,是硬件决定的。   当异常发生时,CPU会自动动作(

    2024年02月11日
    浏览(48)
  • [ARM 汇编]高级部分—ARM汇编编程实战—3.3.1 嵌入式系统的基本概念

    嵌入式系统是一种特殊的计算机系统,通常用于执行特定的任务。它通常包含一个或多个微处理器、存储器和外围设备。与通用计算机系统相比,嵌入式系统具有体积小、功耗低、成本低和实时性强等特点。在这一部分,我们将介绍嵌入式系统的基本概念,并通过实例来展示

    2024年02月11日
    浏览(48)
  • 嵌入式学习笔记(1)ARM的编程模式和7种工作模式

        ARM态-ARM指令集(32-bit)     Thumb态-Thumb指令集(16-bit)     Thumb2态-Thumb2指令集(16 32 bit) Thumb指令集是对ARM指令集的一个子集重新编码得到的,指令长度为16位。通常在处理器执行ARM程式时,称处理器处于ARM状态;当处理器执行Thumb程式时,称处理器处于Thumb状态。

    2024年02月10日
    浏览(48)
  • ARM简单程序设计【嵌入式系统】

    2023-4-6 20:26:54 以下内容源自《【嵌入式系统】》 仅供学习交流使用 Keil 4 安装教程及简单使用【嵌入式系统】 新建工程xxx 芯片:ARM7 (Little Endian) 设置工程属性 Build结果必须是0Error的 如果是下图看配置是否正确 注意这个: 1.每一次修改代码就需要重写Build 2.READWRITE区变量初始

    2023年04月20日
    浏览(69)
  • ARM嵌入式编程优化之在C/C++中栈的使用

    栈在C/C++中使用得非常频繁,比如,栈中可以保存: 在调用函数时,保存函数的返回地址。 保存一些寄存器的值,这取决于ARM 架构:the Procedure Call Standard for the Arm Architecture (AAPCS) or the Procedure Call Standard for the Arm 64-bit Architecture (AAPCS64),比如在进入子程序时,将一些寄存器里

    2024年02月06日
    浏览(51)
  • 嵌入式:ARM汇编语言程序设计基础教程

    ① 合理地分配存储器资源,将前述的目标系统‘数据结构模型’表示到各存储器单元。 ② CPU寄存器数量有限,在程序中,大多数操作都要使用寄存器;并且有的操作使用特定的寄存器(如堆栈操作使用SP/R13等),程序中要合理分配各寄存器的用途。 用计算机语言,对数据结

    2023年04月23日
    浏览(88)
  • ARM处理器概述

    RISC处理器和CISC处理器 首先了解一下两种处理器名字: RISC(Reduced Instruction Set Computer): 精简指令集 处理器 与之相对应的是: CISC(Complex Instruction Set Computer): 复杂指令集 处理器 顾名思义,RISC比CISC更加简单,那么对于处理器来说什么能称为简单呢?可以联想为生活中可

    2024年02月13日
    浏览(56)
  • 【ARM 嵌入式 编译系列 2.1 -- GCC 预处理命令 #error 和 #warning 详细介绍 】

    在C语言中, #error 和 #warning 预处理指令可以用于在编译时生成错误或警告信息,通常用于调试或当代码中某些条件未满足时提醒开发者。当这些指令被编译器处理时,会自动包含出现这些指令的文件名和行号,所以你可以清楚地看到问题出现的位置。 #error 当编译器遇到 #e

    2024年01月22日
    浏览(53)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包