本帖最后由 st780206 于 2026-1-22 09:51 编辑
参考 (八)零基础开发小安派-Eyes-S1【外设篇】——GPIO中断编程
上期的GPIO输入输出编程是基于在main函数中设置while循环不断的改变IO0的电平,同时打印IO1的电平状态。本期了解关于中断编程的概念方法。 一、中断概念了解我们在常规的程序中,都是在main函数中设置While函数不断运行代码。 而中断的概念则是打断当前的程序,并保护现场,执行另外的程序,而后回到最初打断的位置,举个恰当的例子,比如我们在敲代码,忽然外卖小哥敲门说外卖到了,于是我们先ctrl+s保存,然后去拿完外卖顺便吃了回来继续敲代码。敲代码就是我们的常规程序(发量危),中断源就是外卖小哥的敲门声,而拿外卖吃外卖的过程就是中断服务函数。这就是中断的基本概念。 为什么要使用中断?中断是为了应对各种突发事件,如敲代码的时候忽然有人敲门送外卖。正如使用按键扫描检测和外部中断的区别来说,按键扫描会一直在循环中持续扫描按键在状态,正如我们在敲代码时,因为不知道外卖什么时候送来,还要时不时看看有没有人敲门,注意力都用在盯着门上了,怎么能认真敲代码。而中断正如我们已经设置好了耳朵,我们可以认真敲代码,当有人敲门时我们再自然而然的去开门即可,不需要一直盯着门看。 1.中断源在上面的例子中,中断源就是外卖小哥的敲门声。我们在开门之后他就会停止敲门,但是实际上我们还是需要手动的告诉MCU不用敲门了,这也就是要清除中断源
的操作。 2.中断服务函数中断服务函数,也就是我们触发中断后需要执行的操作,这里的吃外卖过程就是中断服务函数的执行过程。一般会在中断服务函数内清除中断源,也就是让外卖小哥别敲门了。 二、中断的配置函数1.bflb_gpio_int_init说明: gpio 外部中断初始化 void bflb_gpio_int_init(struct bflb_device_s *dev, uint8_t pin, uint8_t trig_mode);[td]
| parameter | description | dev | 设备句柄 | pin | gpio pin | trig_mode | 中断触发模式 | 2.bflb_gpio_int_mask说明:gpio 外部中断屏蔽开关 void bflb_gpio_int_mask(struct bflb_device_s *dev, uint8_t pin, bool mask);[td]
| parameter | description | dev | 设备句柄 | pin | gpio pin | mask | 是否屏蔽中断 | 3.bflb_gpio_get_intstatus说明: 获取 gpio 外部中断是否触发的标志 bool bflb_gpio_get_intstatus(struct bflb_device_s *dev, uint8_t pin);[td]
| parameter | description | dev | 设备句柄 | pin | gpio pin | return | true为触发,false未触发 | 4.bflb_gpio_int_clear说明: 清除 gpio 中断标志 void bflb_gpio_int_clear(struct bflb_device_s *dev, uint8_t pin);[td]
| parameter | description | dev | 设备句柄 | pin | gpio pin | 5.bflb_irq_attach说明:设置触发中断后进入的回调函数 int bflb_irq_attach(int irq, irq_callback isr, void *arg)[td]
| parameter | description | irq | 中断号 | isr | 中断服务函数 | arg | 设备句柄 | 6.bflb_irq_enable说明:中断使能 void bflb_irq_enable(int irq)[td]
| parameter | description | irq | 中断号 | 其他:bflb_gpio_uart_init说明: gpio 配置成 uart 的某一个功能,可配置的可能参考上一期的gpio uart function void bflb_gpio_uart_init(struct bflb_device_s *dev, uint8_t pin, uint8_t uart_func);[td]
| parameter | description | dev | 设备句柄 | pin | gpio pin | uart_func | uart具体某一功能 | 三、示例讲解这里还是对SDK的中断示例稍作修改 main
#include "bflb_gpio.h"
#include "bflb_mtimer.h"
#include "board.h"
struct bflb_device_s *gpio; //设置控制的外设句柄,取名
gpiovoid gpio_isr(int irq, void *arg) //中断服务函数
{
bool intstatus = bflb_gpio_get_intstatus(gpio, GPIO_PIN_0);//检测中断是否发生
if (intstatus) { //前面的instatus是bool类型,为true,也就是中断发生
bflb_gpio_int_clear(gpio, GPIO_PIN_0);//清除中断源,也就是例子里的敲门声
printf("Finished eating\r\n");//输出信息,“吃完了”
}
}
int main(void){
board_init();
gpio = bflb_device_get_by_name("gpio"); //给外设句柄复位gpio句柄
printf("gpio interrupt\r\n");
bflb_gpio_int_init(gpio, GPIO_PIN_0, GPIO_INT_TRIG_MODE_SYNC_LOW_LEVEL);//第三个参数设置为低电平触发
bflb_gpio_int_mask(gpio, GPIO_PIN_0, false);//是否屏蔽中断,设置为false
bflb_irq_attach(gpio->irq_num, gpio_isr, gpio);//第二个参数为中断服务函数的函数名
bflb_irq_enable(gpio->irq_num);//中断使能
while (1) {
printf("I am typing the code\r\n");//在常规的while程序中输出“我在敲代码”
bflb_mtimer_delay_ms(2000);
}
}
效果
当正常识别串口信息时,只会每两秒打印一次“I am typing the code”
当我用IO0去碰一下TTL的GND脚,也就是拉低,触发了中断,每进入一次中断会打印一次“Finished eating”
这里打印了许多次,说明我触发了多次中断,毕竟只是使用杜邦线碰一下地,程序的运行和检测是十分快的,所以及时碰一下还是触发了许多次中断,实际上使用按键会更加灵敏,需要注意的是,不要在中断中使用delay延迟函数!中断是为了应对突发状况,处理的越快越好,而延迟会让整个系统暂停在中断里等待,相当于堵塞在这里,直到延迟结束。
|