痞子衡嵌入式:关于恩智浦SDK2.0里事务型中断处理函数(DriverIRQHandler)的重定向注意事项

news/2025/1/15 17:22:49/文章来源:https://www.cnblogs.com/henjay724/p/18542788

  大家好,我是痞子衡,是正经搞技术的痞子。今天痞子衡给大家介绍的是SDK2.0里事务型中断处理函数(DriverIRQHandler)的重定向注意事项

  最近有一个 i.MXRT 客户在使用官方 SDK 外设驱动里的中断处理函数时遇到了代码重定向失效问题,客户用得是一个 XIP Flash 工程,想把程序中断向量表以及相关外设的驱动函数全部重定向到 RAM 中以提高系统性能,但实测发现中断发生时,仍然存在 Flash 访问行为。这本来不是个大问题,因为 SDK 在设计时已经从中断处理函数命名上就做了明确提醒,但是很多客户并没有意识到,今天痞子衡就来聊聊这个话题:

一、事务型驱动函数简介

  恩智浦 SDK 软件包里的外设驱动(HAL级)正常来说提供的 API 都是面对外设配置(init、deinit、set_feature、get_status) 的通用功能函数。此外对于通信接口类外设,一般还会有阻塞式(blocking)的数据传输功能函数。以 LPUART 外设为例,其数据传输有以下四个 API:

// 写入(发送)一个 Byte 数据(需在 FIFO 没满的情况下)
static inline void LPUART_WriteByte(LPUART_Type *base, uint8_t data);
// 读取(接收)一个 Byte 数据(需在 FIFO 非空的情况下)
static inline uint8_t LPUART_ReadByte(LPUART_Type *base);
// 阻塞式写入(发送)多个 Byte 数据
status_t LPUART_WriteBlocking(LPUART_Type *base, const uint8_t *data, size_t length);
// 阻塞式读取(接收)多个 Byte 数据
status_t LPUART_ReadBlocking(LPUART_Type *base, uint8_t *data, size_t length);

  阻塞式数据传输 API 本质上就是独占 CPU 时间进行查询式传输,API 一旦调用,必须等到数据收发结束才会返回,这样会导致 CPU 利用率不高,其一般不利用外设中断。为了结合外设中断进行高效数据传输(non-blocking),SDK2.0 中额外提供了如下事务型相关函数(仅列出了部分):

// 创建事务型数据传输句柄
void LPUART_TransferCreateHandle(LPUART_Type *base,lpuart_handle_t *handle,lpuart_transfer_callback_t callback,void *userData);
// 非阻塞式写入(发送)多个 Byte 数据
status_t LPUART_TransferSendNonBlocking(LPUART_Type *base, lpuart_handle_t *handle, lpuart_transfer_t *xfer)
// 非阻塞式读取(接收)多个 Byte 数据
status_t LPUART_TransferReceiveNonBlocking(LPUART_Type *base,lpuart_handle_t *handle,lpuart_transfer_t *xfer,size_t *receivedBytes);
// 事务型数据传输中断处理函数
void LPUART_TransferHandleIRQ(LPUART_Type *base, void *irqHandle);

  非阻塞式数据传输 API 显然就是结合了外设中断来做数据传输,API 调用后填入一些配置后会立刻返回,没有过多消耗 CPU 时间,等外设中断发生时再进一步处理数据。这类型 API 常常和应用设计紧相关,所以也称为事务型函数(transactional API)。

  SDK 里并不是所有外设驱动里包含事务性函数,这类 API 常出现在传输接口类外设上。对于 i.MXRT 来说,支持此类 API 的外设有:DMA、LPUART、LPSPI、LPI2C、SAI、FLEXIO、FLEXSPI、USDHC、ENET、CAN、MIPI_DSI/CSI、SPDIF、ASRC、PDM 等。

二、事务型中断处理函数设计

  这里继续以 LPUART 外设来具体介绍。如下 i.MXRT1011 SDK 里提供的 8 个 LPUART 例程中有 5 个是基于事务型驱动函数的,我们就以 interrupt_transfer 的 IAR 工程为例。

  打开这个 lpuart_interrupt_transfer 工程,找到芯片启动文件 startup_MIMXRT1011.s,在里面我们能找到 PUBWEAK 型的 LPUART1_IRQHandler() 函数定义,这个是大家比较常见的中断处理函数名,其代码里面就是简单跳转到另一个 PUBWEAK 型 LPUART1_DriverIRQHandler 函数。

  在 fsl_lpuart.c/.h 驱动里,找不到 LPUART1_IRQHandler() 定义,但是有 LPUART1_DriverIRQHandler() 定义。这意味着 SDK 驱动设计时,将默认的 LPUART1_IRQHandler() 函数重写的权利留给了用户,而重新设计了 LPUART1_DriverIRQHandler() 函数来存放事务性中断处理代码,从而避免因用户自己重写中断处理函数时发生函数名重定义而去修改 fsl_lpuart.c 驱动文件的麻烦。

三、重定向事务型中断处理函数

  现在我们尝试重定向 lpuart_interrupt_transfer 工程,可以按照 《IAR下代码重定向的三种方法》 一文里的方法,将 fsl_lpuart.o 和 lpuart_interrupt_transfer.o 两个目标文件都重定向到 RAM 中,并且在 main 里加上拷贝 0x60002000 处开始的 1KB 中断向量表数据到 SRAM 中并且将 SCB->VTOR 指向对应 SRAM 的代码(这个过程可以参考 《Cortex-M中断向量表重定向方法》 一文)。

  上述改动完成之后,编译工程查看 map 文件,我们发现所有的相关代码都已经被链接在了 SRAM 里,但是 LPUART1_IRQHandler() 仍然在 Flash 里,很显然这种情况下中断发生时,仍然会有 Flash 访问行为(暂不考虑 L1-Cache 生效的情况),这就是客户遇到的问题。

  那么如何解决这个问题?其实 SDK 已经为你考虑到了,在 fsl_common_arm.c 文件中定义了 InstallIRQHandler() 函数(仅在 ENABLE_RAM_VECTOR_TABLE 宏存在的情况下生效),查看其源码,发现作用有两个:一、如果 SCB->VTOR 指向得不是 SRAM,那么将中断向量表从 Flash 拷贝到 SRAM 中,并且重置 VTOR;二、根据传入参数修改 SRAM 中的某个中断向量值。

  因此 lpuart_interrupt_transfer 例程中,如果需要彻底重定向中断处理函数,记得在 main 函数里的 LPUART_TransferCreateHandle() 函数调用之后加上如下一句代码,其作用除了重定向中断向量表之外,还将表里的 LPUART1 中断向量从 LPUART1_IRQHandler() 更换为了 LPUART1_DriverIRQHandler(),这样代码重定向就彻底了。

InstallIRQHandler(LPUART1_IRQn, (uint32_t)LPUART1_DriverIRQHandler);

  此时再编译工程下载运行,发现出现 hardfault,这是怎么回事?别急,因为 InstallIRQHandler() 函数里需要用到链接文件 MIMXRT1011xxxxx_flexspi_nor.icf 里定义的三个 Symbol,工程选项 Linker/Configuration file symbol 里必须添加 __ram_vector_table__=1 设置,那些 Symbol 才会真正产生重定向作用。

define symbol __ram_vector_table_size__        = isdefinedsymbol(__ram_vector_table__) ? 0x00000400 : 0;
define exported symbol __VECTOR_TABLE          = m_interrupts_start;
define exported symbol __VECTOR_RAM            = isdefinedsymbol(__ram_vector_table__) ? m_interrupts_ram_start : m_interrupts_start;
define exported symbol __RAM_VECTOR_TABLE_SIZE = __ram_vector_table_size__;

  至此,SDK2.0里事务型中断处理函数(DriverIRQHandler)的重定向注意事项痞子衡便介绍完毕了,掌声在哪里~~~

欢迎订阅

文章会同时发布到我的 博客园主页、CSDN主页、知乎主页、微信公众号 平台上。

微信搜索"痞子衡嵌入式"或者扫描下面二维码,就可以在手机上第一时间看了哦。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/832511.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于Java+SpringBoot+Mysql在线课程学习教育系统功能设计与实现六

部分功能:课程信息业务逻辑层Service、课程分类信息业务逻辑层Service、课程订单信息业务逻辑层Service、课程小节信息业务逻辑层Service、课程评论信息业务逻辑层Service 技术点:SpringBoot+SpringDataJPA+Mysql+Freemaker+Bootstrap+JS+CSS+HTML一、前言介绍: 免费获取:猿…

基于Java+SpringBoot+Mysql在线课程学习教育系统功能设计与实现五

技术点:SpringBoot+SpringDataJPA+Mysql+Freemaker+Bootstrap+JS+CSS+HTML 部分功能:前台用户信息数据层Dao、公告信息数据层Dao、评论信息数据层Dao、常见问题信息数据层Dao、优惠卷信息数据层Dao、论坛信息数据层Dao、论坛回复信息数据层Dao、学习资料信息数据层Dao、充值记…

全局平衡二叉树 (GBST) 小记

全局平衡二叉树 (GBST) 小记 以下全局平衡二叉树简称 \(\text{GBST(Globel Balanced Search Tree)}\)。 我认识的大多数人,对 \(\text{GBST}\) 的理解基本上都是 静态 \(\text{LCT}\),或者静态 \(\text{Top Tree}\),不过我对 \(\text{LCT}\) 的理解可能还差一点,所以我不打…

考研打卡(15)

开局(15) 开始时间 2024-11-12 20:37:51 结束时间 2024-11-12 22:41:32现在在敷泥膜数据结构设已知一稀疏矩阵的三元组表为:(1,2,3),(1,6,1),(3,1,5),(3,2,-1) (5,4,5),(5,1,-3),则其转置矩阵的三元组表中的第三个三元组为____(山东大学2013年) A (2,…

25. 使用MySQL之使用触发器

1. 触发器 MySQL语句在需要时被执行,存储过程也是如此。但是,如果你想要某条语句(或某些语句)在事件发生时自动执行,怎么办呢?例如:每当增加一个顾客到某个数据库表时,都检查其电话号码格式是 否正确,州的缩写是否为大写;每当订购一个产品时,都从库存数量中减去订购…

Mysql表分区实操

创建分区表 create table user(id int(11) not null,name varchar(32) not null) partition by range(id) ( partition p0 values less than(10), partition p1 values less than(20), partition p2 values less than(30), partition p3 values less than maxvalue )分区表数据…

【Azure Bot Service】部署NodeJS ChatBot代码到App Service中无法自动启动

问题描述 把NodeJS的ChatBot代码部署到App Service环境中,通过VS Code直接部署,显示部署成功。但是通过URL访问时候,却是 :( Application Error 。问题解答 App Service遇见Application Error,第一步,查看日志。 发现启动时候遇见: 2024-11-12T12:22:40.366223350Z Error…

Chrome如何卸载掉插件?如此简单!

前言 我之前由于好奇,一下子装了很多的chrome插件,这些插件虽然都有很多功能,但是,由于我的电脑性能不佳,浏览器一下子装这么多插件,有点带不动了。所以,我就想到了卸载谷歌浏览器,哦不,是卸载浏览器里面的插件。今天,我就来介绍下,如何在chrome里面卸载一些不常用的…

rocky9如何重置root密码

rocky9如何重置root密码 第一步第二步 在箭头指的地方把 "ro" 改成 "rw"另一个箭头指的地方写上如下内容:init=/bin/bash如图所示第三步 Ctrl + x 按当前的配置启动第四步 输入命令root 用户密码直接输入以下内容passwd如果更改其他用户第五步 最为关键的…

多校A层冲刺NOIP2024模拟赛21

多校A层冲刺NOIP2024模拟赛21\(T1\) A. 送信卒 \(90pts/100pts\)部分分\(90pts\)设最后的可能的最短路中左右共移动了 \(d\) 次,上下共移动了 \(x\) 次。则等价于求 \(\min \{ x_{i}k+d_{i} \}=s\) 的解,观察到 \(d \in [0,\min(\left\lceil \frac{nm}{2} \right\rceil,s)]\)…

[论文阅读] ZePo: Zero-Shot Portrait Stylization with Faster Sampling

写在前面 原文:ZePo GitHub:Github ZePo 关键词:肖像风格化、扩散模型、零样本快速生成 阅读理由:对扩散模型的改进,可以实现零样本快速生成图像,学习一下思路以及实验设计 前置知识:LCM以及GithubLCM(找时间写一下),可参考LCM&CM,一致性蒸馏、图像质量评价速览…

2024.11.12 鲜花

P11270 【MX-S5-T4】魔法少女们 题解这世界那么多人 这世界有那么多人 人群里 敞着一扇门 我迷朦的眼睛里长存 初见你蓝色清晨 这世界有那么多人 多幸运 我有个我们 这悠长命运中的晨昏 常让我 望远方出神 灰树叶飘转在池塘 看飞机轰的一声去远乡 光阴的长廊 脚步声叫嚷 灯一亮…