armv7a中的SMC指令应用

2024-06-03 17:58
文章标签 应用 指令 smc armv7a

本文主要是介绍armv7a中的SMC指令应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

工作中用到了smc指令,感觉除了arm的手册介绍这个指令的资料并不是太多,所以写这篇文章记录一下。

SMC是TrustZone的一部分。Non-Secure world要切换到Secure-World的时候需要进入Monitor模式才能进行操作,而在armv7a中就是通过SMC指令进入到TrustZone。本文中只介绍跟SMC指令相关的东西,并不介绍TrustZone。本文中的所以指令的运行都是运行在Secure world下, 所以在SMC指令使用中没有涉及Secure world跟Non secure world切换,并且本文是armv7a的架构下使用qemu进行模拟。

SMC指令格式

这里写图片描述

SMC有Thumb编码和ARM编码,T1是Thumb,A1是ARM。可以看到不管是Thumb还是ARM都有一个4位的立即数imm4,这个imm4在armv7a的架构没有定义是干什么,用户可以自己选择如何使用它,通常就跟使用SVC后面的立即数差不多。需要注意的是上图中Thumb指令的低16位在右边,所有imm4其实跟ARM指令的位置是一样的,都是在32位中最低4位,这样在处理获取imm4的时候,我们就不需要去判断到底是ARM指令还是THUMB指令。

设置SMC异常向量表

SMC跟SVC指令很类似,都会进入一种软件异常模式。所以使用SMC必须要提供一个SMC的异常向量表。需要注意的是Monitor模式拥有自己的一套异常向量表,它并不与其他的异常/中断模式共享一套异常向量。Monitor模式所需要的异常向量入口保存在这个MVBAR(Monitor Vector Base Address)寄存器中,需要注意的是MVBAR必须在Secure world的PL1级别下才能够进行读写,也就是说MVBAR在系统中必须由底层软件去设置。

MVBAR描述

下面看一下如何设置SMC异常向量表,需要注意的是Monitor模式的不仅异常向量是自己用的一套,它的栈也是自己的,与其他模式如SVC模式使用的不是同一个栈,所以在系统初始化的时候需要指定好Monitor模式的栈

    mrs r0, cpsrbic r0, r0, #0x1forr r0, r0, #0xd6  ;cpsr.M=d6=monitro modemsr cpsr, r0ldr sp, =smc_stack_top/*Set the cpu to SVC32 mode*//*Only secure world pl1 can change mvbar reg*/mrs r0, cpsrbic r0, r0, #0x1forr r0, r0, #0xd3msr cpsr, r0ldr sp, =svc_stack_top/*Set up monitor mode vectors by MVBAR*/ldr r0, =__monitor_vectormcr p15, 0, r0, c12, c0, 1  @Set MVBAR
    .align 4
__monitor_vector:@ Monitor-mode exception vector.word 0        /* reset */.word 0        /* undef */ldr pc, =monitor_mode.word 0 /* pabt */.word 0 /* dabt */.word 0.word 0 /* irq */.word 0 /* fiq */.align

可以看到基本上与普通的异常向量表一样,只是第三个异常入口不是Supervisor Mode而是Monitor Mode。当系统已经进入Monitor Mode时,这时候假设有一个IRQ发生,那么触发的Monitor Mode下的IRQ中断处理函数。

进入SMC异常

进入SMC很简单,只需要调用SMC指令即可,就跟SVC差不多。SMC后面可以跟一个4位的立即数,armv7a中并没有定义这个立即数是干什么的,用户可以自己使用它。

smc #0

退出SMC异常

退出SMC异常操作跟SVC差不多,armv7a的手册给出一些退出异常的操作,如下图所示,需要主要的是调用MOV PC XX指令的时候,需要使用MOVS才能退出Monitor Mode。
这里写图片描述

调用SMC指令时,CPU做了些什么事?

官方的手册给出了CPU在处理SMC指令时候的伪代码如下。

   Take SMCException()ITAdvanced(); lr = cpsr.T == 1 ? pc : pc - 4; //Thumb relatedspsr = cpsr;cpsr.M = "10110";                     //Change to Monitor Modecpsr.j = '0';cpsr.T = sctlr.TEcpsr.E = sctlr.EEcpsr.A = '1';cpsr.F = '1';cpsr.I = '1';cpsr.IT = '0000_0000'pc = monitor_vetors + 8;

ITAdavance()我也不知道干了什么,有高手知道的可以指出来,反正我在用SMC指令的时候没遇到这个操作的问题。

lr = cpsr.T == 1 ? pc : pc - 4; //Thumb related

这个操作确定了SMC的返回地址,也就是lr寄存器。当CPU运行在Thumb模式下,ld = pc,为什么是pc呢?CPU流水线取址比执行多了两级,所以此时的pc应该是 smc的下两条指令的地址,而Thumb模式下是2字节对齐,所以正好是pc = smc_adr + 2*2,但smc指令即使在thumb模式下也是4个字节,所以此时lr = pc = smc_adr + 4,正好指向了smc指令的一下条指令。而在ARM模式下,指令时4字节对齐,所以取指流水线上的pc = smc_adr + 4*2, 而smc指令的下一条指令的地址是smc_adr + 4,所以当cpsr.T = 0时,lr = pc - 4 = smc_adr + 4,指向了smc的下一条指令。

spsr = cpsr;
cpsr.M = "10110";

然后CPU就备份了当前模式下的CPSR到SPSR,并切换到Monitor模式。可以从armv7a手册中查到cpsr.M都对应什么模式,10110就是Monitor。

       cpsr.j = '0';cpsr.T = sctlr.TEcpsr.E = sctlr.EEcpsr.A = '1';cpsr.F = '1';cpsr.I = '1';cpsr.IT = '0000_0000'

接着CPU就设置了一些CPSR的状态标志位。

       pc = monitor_vetors + 8;

然后CPU就把当前PC切换到了Monitor异常的入口地址,硬件做的动作到此为止,接下来做的事就是软件该做的事了。

一个非典型的Monitor异常处理程序

这个例子是我自己写的,然而我也没有自信说它是一个典型的处理程序。这里就抛砖引玉,给出一个简单例子来说明Monitor异常处理。

monitor_mode:/*Save register according to ATPCS*/stmfd sp!, {r0-r3, r12, lr}/*Real SMC Handler code*/ldr r0, [lr, #-4]bic r0, r0, #0xfffffff0bl smc_handler/*restore the spsr and regs of ATCPS*/ldmfd sp!, {r0-r3, r12, pc}^

其实很简单,手写保存一些寄存器,当然把所有的寄存器都保存起来,这取决于你的英语。然后从SMC指令中拿出这个4位的立即数,从上一个段落分析可知,lr-4就是调用SMC指令的地址,这里存放着SMC指令的机器码。然后跳转到smc_handler程序中进行处理,我这里把smc_handler写成了C程序,完全可以用汇编实现,这个函数很简单,我就打印了一下这个SMC的操作数。最后当处理完之后使用ldmfd 指令将寄存器弹出,需要注意的是一定要加^,否则SPSR就没办法恢复,那么Monitor模式就退出不了,会导致一些不是期望的行为。

smc_handler非典型实现, 很简单,仅仅只是打印了一下相关函数的信息。

static void smc_handle0()
{board_puts("######maybe rom patch can be here222####\n");
}static void smc_handle1()
{board_puts("smc_handle1\n");
}static void default_smc_handle()
{board_puts("default_smc_handle\n");
}smc_handle smc_handle_arr[] = {smc_handle0,smc_handle1,NULL,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,default_smc_handle,
};void smc_handler(int smc_num)
{if (smc_num >= SMC_HANDLE_MAX || smc_num < 0) {return;}board_puts("smc_handler! smc_num is ");board_putc(smc_num + '0');board_putc('\n');if (smc_handle_arr[smc_num] != NULL) {(smc_handle_arr[smc_num])();} else {board_puts("smc handle not exist\n");}board_puts("\n");
}

源码

https://github.com/JiaminMa/Misc_demo/tree/master/smc_demo
这个源码可以直接make run或者make debug,使用qemu进行模拟,如果不知道怎么用qemu进行armv7a裸机程序模拟可以看我这篇文章。
http://blog.csdn.net/u011280717/article/details/71915669。

这篇关于armv7a中的SMC指令应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1027705

相关文章

亮相WOT全球技术创新大会,揭秘火山引擎边缘容器技术在泛CDN场景的应用与实践

2024年6月21日-22日,51CTO“WOT全球技术创新大会2024”在北京举办。火山引擎边缘计算架构师李志明受邀参与,以“边缘容器技术在泛CDN场景的应用和实践”为主题,与多位行业资深专家,共同探讨泛CDN行业技术架构以及云原生与边缘计算的发展和展望。 火山引擎边缘计算架构师李志明表示:为更好地解决传统泛CDN类业务运行中的问题,火山引擎边缘容器团队参考行业做法,结合实践经验,打造火山

自制的浏览器主页,可以是最简单的桌面应用,可以把它当成备忘录桌面应用

自制的浏览器主页,可以是最简单的桌面应用,可以把它当成备忘录桌面应用。如果你看不懂,请留言。 完整代码: <!DOCTYPE html><html lang="zh-CN"><head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><ti

Python应用开发——30天学习Streamlit Python包进行APP的构建(9)

st.area_chart 显示区域图。 这是围绕 st.altair_chart 的语法糖。主要区别在于该命令使用数据自身的列和指数来计算图表的 Altair 规格。因此,在许多 "只需绘制此图 "的情况下,该命令更易于使用,但可定制性较差。 如果 st.area_chart 无法正确猜测数据规格,请尝试使用 st.altair_chart 指定所需的图表。 Function signa

气象站的种类和应用范围可以根据不同的分类标准进行详细的划分和描述

气象站的种类和应用范围可以根据不同的分类标准进行详细的划分和描述。以下是从不同角度对气象站的种类和应用范围的介绍: 一、气象站的种类 根据用途和安装环境分类: 农业气象站:专为农业生产服务,监测土壤温度、湿度等参数,为农业生产提供科学依据。交通气象站:用于公路、铁路、机场等交通场所的气象监测,提供实时气象数据以支持交通运营和调度。林业气象站:监测林区风速、湿度、温度等气象要素,为林区保护和

PyTorch模型_trace实战:深入理解与应用

pytorch使用trace模型 1、使用trace生成torchscript模型2、使用trace的模型预测 1、使用trace生成torchscript模型 def save_trace(model, input, save_path):traced_script_model = torch.jit.trace(model, input)<

哺乳细胞重组表达人鼠嵌合抗体:制备与应用

重组抗体是一类具有广泛应用价值的蛋白质,在药物研发和生物医学研究中发挥着重要作用。本文将介绍重组抗体的表达方式,重点关注嵌合抗体制备和哺乳细胞重组表达人鼠嵌合抗体的技术原理和应用。 重组抗体表达的原理和方法 重组抗体表达是通过将人或动物源的免疫球蛋白基因导入表达宿主细胞,并使其表达出特异性抗体蛋白质。常用的表达系统包括细菌、哺乳细胞和真核微生物等。 嵌合抗体制备的步骤和优势 选择适当的抗原

【Qt6.3 基础教程 16】 掌握Qt中的时间和日期:QTimer和QDateTime的高效应用

文章目录 前言QTimer:定时任务的强大工具QTimer的基本用法高级特性:单次定时器 QDateTime:处理日期和时间获取当前日期和时间日期和时间的格式化输出日期和时间计算 用例:创建一个倒计时应用结论 前言 在开发桌面应用程序时,处理时间和日期是一个常见且重要的任务。Qt框架提供了强大的工具来处理与时间相关的功能,其中QTimer和QDateTime是最核心的类。本

基于Spring Boot的企业级应用架构设计

基于Spring Boot的企业级应用架构设计 大家好,我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编,也是冬天不穿秋裤,天冷也要风度的程序猿!今天,我想和大家分享一下基于Spring Boot的企业级应用架构设计,希望对大家有所帮助。 一、Spring Boot概述 Spring Boot是由Pivotal团队提供的全新框架,它简化了Spring应用程序的创建和开发过程。

LoRaWAN在嵌入式网络通信中的应用:打造高效远程监控系统(附代码示例)

引言 随着物联网(IoT)技术的发展,远程监控系统在各个领域的应用越来越广泛。LoRaWAN(Long Range Wide Area Network)作为一种低功耗广域网通信协议,因其长距离传输、低功耗和高可靠性等特点,成为实现远程监控的理想选择。本文将详细介绍LoRaWAN的基本原理、应用场景,并通过一个具体的项目展示如何使用LoRaWAN实现远程监控系统。希望通过图文并茂的讲解,帮助读

一二三应用开发平台应用开发示例(4)——视图类型介绍以及新增、修改、查看视图配置

调整上级属性类型 前面为了快速展示平台的低代码配置功能,将实体文件夹的数据模型上级属性的数据类型暂时配置为文本类型,现在我们调整下,将其数据类型调整为实体,如下图所示: 数据类型需要选择实体,并在实体选择框中选择自身“文件夹” 这时候,再点击生成代码,平台会报错,提示“实体【文件夹】未设置主参照视图”。这是因为文件夹选择的功能页面,同样是基于配置产生的,因为视图我们还没有配置,所以会报错。