痞子衡嵌入式:探析i.MXRT1050在GPIO上增加RC延时电路后导致边沿中断误触发问题(上篇)
大家好,我是痞子衡,是正经搞技术的痞子。今天痞子衡给大家分享的是i.MXRT1050在GPIO上增加RC延时电路后导致边沿中断误触发问题探析。
前段时间有一个 RT1052 客户反馈了一个有趣的问题,他们设计得是一个带 LCD 屏交互的应用,应用以官方 SDK 里的 lvgl_demo_widgets_bm 例程为基础。当客户在这个例程基础上增加了 GPIO 输入边沿中断检测,并且硬件上给 GPIO 增加了 RC 延时电路后,发现边沿中断触发得不太准确,这是怎么回事?今天痞子衡带大家还原现场:
一、问题描述
客户做得硬件改动很简单,在 GPIO_AD_B1_04 引脚和 GPIO_AD_B1_10 引脚之间加了如下的 RC 延时电路。GPIO_AD_B1_04 上产生得是 500Hz 的方波(既可以是 GPIO 模块输出,也可以去掉 R290 后直接接信号发生器),这个方波经过 RC 电路之后输出给 GPIO_AD_B1_10,然后通过其输入边沿中断来检测电平变化,并且在每个边沿中断里都翻转一次 GPIO_AD_B1_11 电平。
代码改动也足够简单,只需要在 \SDK_2_15_000_EVKB-IMXRT1050\boards\evkbimxrt1050\lvgl_examples\lvgl_demo_widgets_bm 工程里添加 test_gpio_irq() 函数调用即可(这里假定 GPIO_AD_B1_04 上的方波是由外部信号发生器提供的):
void GPIO1_Combined_16_31_IRQHandler(void)
{
// 检测到 GPIO_AD_B1_10 边沿
if ((GPIO1->ISR & (1U << 26)) && (GPIO1->IMR & (1U << 26)))
{
GPIO_PortClearInterruptFlags(GPIO1, 1U << 26);
// 翻转 GPIO_AD_B1_11 电平
GPIO_PortToggle(GPIO1, 1 << 27);
__DSB();
}
}
void config_rc_in_gpio(void)
{
// 配置 GPIO_AD_B1_10 为边沿中断输入检测模式
gpio_pin_config_t in_config = { kGPIO_DigitalInput, 1, kGPIO_NoIntmode };
IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_10_GPIO1_IO26, 1);
IOMUXC_SetPinConfig(IOMUXC_GPIO_AD_B1_10_GPIO1_IO26, 0x011030U);
GPIO_PinInit(GPIO1, 26, &in_config);
GPIO_SetPinInterruptConfig(GPIO1, 26, kGPIO_IntRisingOrFallingEdge);
EnableIRQ(GPIO1_Combined_16_31_IRQn);
GPIO_PortEnableInterrupts(GPIO1, 1U << 26);
}
void config_user_out_gpio(void)
{
// 配置 GPIO_AD_B1_11 为普通输出模式
gpio_pin_config_t out_config = { kGPIO_DigitalOutput, 1, kGPIO_NoIntmode };
IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_11_GPIO1_IO27, 0);
GPIO_PinInit(GPIO1, 27, &out_config);
GPIO_PinWrite(GPIO1, 27, 0U);
}
void test_gpio_irq(void)
{
config_rc_in_gpio();
config_user_out_gpio();
}
如果 GPIO_AD_B1_10 边沿中断检测无误,那么输出的 GPIO_AD_B1_11 信号应该是和原始输入 GPIO_AD_B1_04 完全同频的方波,而事实上客户用示波器抓到的 GPIO_AD_B1_11 信号偶尔会出现如下情况,很显然有边沿中断误触发的情况发生:
并且更有趣的是,这样的测试仅在 lvgl_demo_widgets_bm 工程里能复现,而在普通 input_interrupt 工程下没有任何问题。
- Note1:在 lvgl_demo_widgets_bm 工程下出现的 GPIO 边沿中断误触发问题仅在增加 RC 电路时存在。
- Note2:在普通 input_interrupt 工程下即使增加 RC 电路,GPIO 边沿中断误触发问题也不存在。
二、问题复现
理论上分析 GPIO_AD_B1_10 引脚输入的信号频率是 500Hz,那么其边沿中断应该是每 1ms 产生一次,而从上一节客户抓取的 GPIO_AD_B1_11 实际信号反推,似乎有时候边沿中断在 10us 内连续产生了两次。
为了从软件角度抓到这个中断误触发现象,痞子衡稍微改了一下代码,将 GPIO_AD_B1_04 上信号改为软件输出(在 SysTick 1ms 一次的中断响应里翻转电平),并且用了两个计数器 s_outputPinEdgeCount、s_inputRcPinIrqCount 来分别记录 GPIO_AD_B1_04、GPIO_AD_B1_10 边沿次数。如果边沿中断触发无误的话,这两个计数器的值应该是永远相等的,但是实际跑了一段时间后发现 s_inputRcPinIrqCount 会超过 s_outputPinEdgeCount,并且随着时间累积,差距会越来越大。这说明边沿中断误触发现象是一直存在的。
volatile uint32_t s_inputRcPinIrqCount = 0;
volatile uint32_t s_outputPinEdgeCount = 0;
void GPIO1_Combined_16_31_IRQHandler(void)
{
// 检测到 GPIO_AD_B1_10 边沿
if ((GPIO1->ISR & (1U << 26)) && (GPIO1->IMR & (1U << 26)))
{
GPIO_PortClearInterruptFlags(GPIO1, 1U << 26);
// 计数 GPIO_AD_B1_10 边沿
s_inputRcPinIrqCount++;
__DSB();
}
}
void config_rc_out_gpio(void)
{
// 配置 GPIO_AD_B1_04 为普通输出模式
IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_04_GPIO1_IO20, 0);
GPIO_PinInit(GPIO1, 20, &out_config);
GPIO_PinWrite(GPIO1, 20, 0U);
}
void test_gpio_irq(void)
{
config_rc_in_gpio();
config_rc_out_gpio();
}
void SysTick_Handler(void)
{
// 计数 GPIO_AD_B1_04 边沿
s_outputPinEdgeCount++;
GPIO_PortToggle(GPIO1, 1 << 20);
__DSB();
// 原应用代码省略
}
三、问题定位
描述至此,你的第一反应到底是哪里出了问题?痞子衡想你可能会觉得罪魁祸首是 RC 延时电路,它将标准的方波上升、下降过程变得平缓,导致信号电压处于临界区的时间变长(极端情况下,对于高频信号,可能会导致其一直处于临界区),这个可能会影响 GPIO 电平跳变判定。既如此,我们先翻看一下 RT1050 的 datasheet,找到如下 GPIO DC 参数表,其高、低电平判定值分别是 70%、30% NVCC_XXXX,此外备注里说明了只要电平变化是单调的(随着时间单向增大或减小),且转换时间范围在 0.1ns - 1s 内均会被认定为有效跳变。
这时候我们再根据 RC 延时电路标准时间常数公式 t = RC * $\ln (\frac{(V1-V0)}{V1-Vt})$ 来推算(V1 电源电压、V0 电容初始时刻电压、$V_t$ 为 t 时刻电容电压)。如果 NVCC 为 3.3V,那么上升沿时从 0V 充电到 2.31V 的时间是 12us,显然这个 12us 充电时间对于 500Hz 的方波来说不足以影响其跳变判定。
有没有方法能抓住这个异常边沿中断发生时,GPIO_AD_B1_10 信号当时的波形状态呢?当然是可以的,我们可以再修改一下边沿中断处理函数代码,在里面计算两次中断之间的 Tick 间隔,如果间隔 Tick 低于一定值,说明是误触发,此时翻转一次 GPIO_AD_B1_11 电平用作标记。
volatile uint32_t s_systickCurVal = 0;
volatile uint32_t s_systickLastVal = 0;
volatile uint32_t s_systickCurCount = 0;
volatile uint32_t s_systickLastCount = 0;
volatile uint32_t s_systickDeltaVal;
uint32_t s_systickReloadVal = 0;
void GPIO1_Combined_16_31_IRQHandler(void)
{
/* clear the interrupt status */
if ((GPIO1->ISR & (1U << 26)) && (GPIO1->IMR & (1U << 26)))
{
s_systickCurVal = SysTick->VAL;
s_systickCurCount = s_outputPinEdgeCount;
GPIO_PortClearInterruptFlags(GPIO1, 1U << 26);
// 计算两次中断之间的 Tick 间隔
s_systickDeltaVal = (s_outputPinEdgeCount - s_systickLastCount) * s_systickReloadVal + s_systickLastVal - s_systickCurVal;
s_systickLastVal = s_systickCurVal;
s_systickLastCount = s_systickCurCount;
// 当间隔 Tick 低于一定值时,说明是误触发,此时翻转一次 GPIO_AD_B1_11 电平
if (s_systickDeltaVal <= s_systickReloadVal / 2)
{
GPIO_PortToggle(GPIO1, 1 << 27);
}
__DSB();
}
}
int main(void)
{
// 应用代码省略...
test_gpio_irq();
s_systickReloadVal = SystemCoreClock / (LVGL_TICK_MS * 1000U);
s_inputRcPinIrqCount = 0;
s_systickLastVal = s_systickReloadVal;
DEMO_SetupTick();
// 应用代码省略...
}
如果用示波器以 GPIO_AD_B1_11 跳变为触发信号(ch2),即能看到案发现场 GPIO_AD_B1_10 状态(ch1),确实我们看到充放电时间内出现了短时脉冲波干扰(glitch),这个脉冲导致了电平变化不是单调的,因而产生了 GPIO 中断误触发。本篇仅是定位问题,下一篇我们会具体分析这个 glitch 是如何产生的!
至此,i.MXRT1050在GPIO上增加RC延时电路后导致边沿中断误触发问题探析(上篇)痞子衡便介绍完毕了,掌声在哪里~~~
欢迎订阅
文章会同时发布到我的 博客园主页、CSDN主页、知乎主页、微信公众号 平台上。
微信搜索"痞子衡嵌入式"或者扫描下面二维码,就可以在手机上第一时间看了哦。

痞子衡嵌入式:探析i.MXRT1050在GPIO上增加RC延时电路后导致边沿中断误触发问题(上篇)的更多相关文章
- 痞子衡嵌入式:其实i.MXRT1050,1020,1015系列ROM也提供了FlexSPI driver API
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT1050/1020/1015系列ROM中的FlexSPI驱动API使用. 今天痞子衡去4S店给爱车做保养了,保养一次要等两小 ...
- 痞子衡嵌入式:深入i.MXRT1050系列ROM中串行NOR Flash启动初始化流程
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是深入i.MXRT1050系列ROM中串行NOR Flash启动初始化流程. 从外部串行NOR Flash启动问题是i.MXRT系列开发最 ...
- 痞子衡嵌入式:16MB以上NOR Flash使用不当可能会造成软复位后i.MXRT无法正常启动
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是i.MXRT上使用16MB以上NOR Flash软复位无法正常启动问题的分析解决经验. 痞子衡这几天在支持一个i.MXRT1050客户项 ...
- 痞子衡嵌入式:揭秘i.MXRT600的ISP模式下用J-Link连接后PC总是停在0x1c04a的原因(Debug Mailbox)
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT600中的Debug Mailbox实现对JLink调试的影响. 事情缘起痞子衡的同事 - 喜欢打破砂锅问到底的Kerry小 ...
- 痞子衡嵌入式:以i.MXRT1xxx的GPIO模块为例谈谈中断处理函数(IRQHandler)的标准流程
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是以i.MXRT的GPIO模块为例谈谈中断处理函数(IRQHandler)的标准流程. 在痞子衡旧文 <串口(UART)自动波特率识 ...
- 痞子衡嵌入式:再测i.MXRT1060,1170上的普通GPIO与高速GPIO极限翻转频率
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT1060/1170上的普通GPIO与高速GPIO极限翻转频率. 按照上一篇文章 <实测i.MXRT1010上的普通GP ...
- 痞子衡嵌入式:探析开启CRC完整性校验的IAR工程生成.out和.bin文件先后顺序
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是开启CRC完整性校验的IAR工程生成.out和.bin文件先后顺序问题. 痞子衡之前写了一篇 <在IAR开发环境下为工程开启CRC ...
- 痞子衡嵌入式:简析i.MXRT1170 Cortex-M4 L-MEM ECC功能特点、开启步骤、性能影响
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是恩智浦i.MXRT1170上Cortex-M4内核的L-MEM ECC功能. 本篇是 <简析i.MXRT1170 Cortex-M ...
- 痞子衡嵌入式:简析i.MXRT1170 XECC功能特点及其保护串行NOR Flash和SDRAM之道
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是i.MXRT1170 XECC功能特点及其保护串行NOR Flash和SDRAM之道. ECC 是 "Error Correc ...
- 痞子衡嵌入式:简析i.MXRT1170 Cortex-M7 FlexRAM ECC功能特点、开启步骤、性能影响
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是恩智浦i.MXRT1170上Cortex-M7内核的FlexRAM ECC功能. ECC是"Error Correcting ...
随机推荐
- 用ESP8266-NodeMCU开发板显示一下我的QQ头像
诶,说好的自己写esp8266的开发板固件的我回来了. 20年说好的,今天回来还愿了 ESP8266串口WiFi模块 - WiFi杀手 今天我们把OLED显示屏也接上,我此次买的是4脚的OLED(12 ...
- 记录一次MySQL多表查询,order by不走索引的情况.
首先是表结构,部分字段脱敏已删除 CREATE TABLE `log_device_heart` ( `id` int unsigned NOT NULL AUTO_INCREMENT, `devic ...
- poj1338 ugly number 题解 打表
类似的题目有HDU1058 humble number(翻译下来都是丑陋的数字). Description Ugly numbers are numbers whose only prime fact ...
- 异构智联Wi-Fi6+蓝牙模组,重新定义多屏互联体验!
下班回家打开门,电灯.电视.空调.音响.电动窗帘.扫地机器人--一呼百应,有序开工,原本冰冷的房子立刻变成了温暖港湾.可以说,舒适便捷的智能设备已经完全融入了我们的生活中. 从单一场景.单一设备,到现 ...
- 核对不同文件夹所含内容的差异并提取缺失内容:Python代码
本文介绍基于Python语言,以一个大文件夹作为标准,对另一个大文件夹所包含的子文件夹或文件加以查漏补缺,并将查漏补缺的结果输出的方法. 首先,来明确一下本文所需实现的具体需求.现有一个大文件 ...
- 数据源dataSource以及事务tx的xml文件配置方式及代码配置方式
所需要使用的依赖 <dependencies> <!--spring jdbc Spring 持久化层支持jar包--> <dependency> <grou ...
- 屏幕分辨率基础概念PX,PT,DP,DPR,DPI说明
屏幕分辨率基础概念说明 缩写 全称 说明 PX Device Pixels 设备像素,指设备的物理像素 PX CSS Pixels CSS像素,指CSS样式代码中使用的逻辑像素 DOT Dot 点,屏 ...
- 统计里面PV 和 UV代表什么意思
1.网站流量bai统计中"PV"它所代表的意思是访问量了,具体指的du就是网站zhi的页面点击量或是浏览量,亦或是页面的刷新量dao了,网站的页面每刷新一次,就统计一个" ...
- 吐血整理如何在Google Earth Engine上写循环 五个代码实例详细拆解
在这里同步一篇本人的原创文章.原文发布于2023年发布在知乎专栏,转移过来时略有修改.全文共计3万余字,希望帮助到GEE小白快速进阶. 引言 这篇文章主要解答GEE中.map()和.iterate() ...
- 网络基础 Modbus协议学习总结
协议简介 Modbus协议,首先从字面理解它包括Mod和Bus两部分,首先它是一种bus,即总线协议,总线就意味着有主机,有从机,这些设备在同一条总线上. Modbus支持单主机,多个从机,最多支持2 ...