esp32-中断

中断(Interrupt)是一种非常重要和常用的机制,用于响应外部事件并执行特定的代码。当某个特定条件满足时(如 GPIO 引脚的信号状态发生变化),中断将会被触发,并执行与该中断相关联的 ISR(Interrupt Service Routine)函数。

要使用中断,必须先进行中断配置和初始化设置,然后注册 ISR 函数以便在中断触发时执行。以下是一些常用的 ESP32 IDF 中断 API 函数:

1. esp_intr_alloc()
用于分配一个可用的中断向量,并返回一个中断句柄(intr_handle_t)。函数原型如下:

esp_err_t esp_intr_alloc(int cpu_no, int intr_source, int flags, intr_handler_t intr_handler, void* arg, intr_handle_t* handle);

2. esp_intr_free()
用于释放中断句柄(intr_handle_t)所占用的资源。函数原型如下:

void esp_intr_free(intr_handle_t handle);

3. esp_intr_enable()
用于启用指定中断。函数原型如下:

void esp_intr_enable(intr_handle_t handle);

4. esp_intr_disable()
用于禁用指定中断。函数原型如下:

void esp_intr_disable(intr_handle_t handle);

通过使用这些 API 函数,我们可以很容易地配置和管理中断,并执行与中断相关联的处理函数。通常,我们可以将中断视为响应外部事件的一种重要机制,例如接收 GPS 定位数据、读取传感器数据或处理网络数据等。

下面是一个简单的示例,演示如何在 ESP32 上使用中断来处理 GPIO 引脚状态的变化。

#include <stdio.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "driver/gpio.h"

// 中断处理函数 ISR
void IRAM_ATTR gpio_isr_handler(void* arg)
{
    uint32_t gpio_num = (uint32_t)arg;
    printf("GPIO[%d] Interrupt!!!!!!\n", gpio_num);
}

void app_main()
{
    // 关闭 GPIO 中断
    gpio_intr_disable(GPIO_NUM_27);

    // 配置 GPIO 引脚为输入模式
    gpio_set_direction(GPIO_NUM_27, GPIO_MODE_INPUT);

    // 设置 GPIO 中断类型为上升沿触发
    gpio_set_intr_type(GPIO_NUM_27, GPIO_INTR_POSEDGE);

    // 通过 esp_intr_alloc() 函数分配中断向量
    intr_handle_t handle;
    esp_intr_alloc(ETS_GPIO_INTR_SOURCE, ESP_INTR_FLAG_EDGE, gpio_isr_handler, (void*)GPIO_NUM_27, &handle);

    // 通过 esp_intr_enable() 函数启用中断
    esp_intr_enable(handle);
    while(1) {
        printf("hello world!\n");
        vTaskDelay(pdMS_TO_TICKS(1000));
    }
}


 

在代码中,我们首先通过 gpio_intr_disable() 关闭 GPIO 中断,并使用 gpio_set_direction() 将 GPIO 设置为输入模式。然后,我们将 GPIO 中断类型设置为 GPIO_INTR_POSEDGE,指示中断在上升沿时触发。这样,在 GPIO 引脚状态发生变化时,中断将会被触发,并执行与中断相关联的处理函数(即 gpio_isr_handler)。

最后,我们使用 esp_intr_alloc() 分配中断向量并启动中断处理,通过 esp_intr_enable() 启用中断。在 while 循环中,我们通过 vTaskDelay() 函数暂停 1 秒钟,以便观察中断处理的效果。

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
配送一个自己写的串口驱动程序 用DMA接收数据 接收完会产生一个空闲中断 由此可判断接收完一个包的数据 再配送一个我自己写的动态内存管理 跟ESP8266的驱动 在项目中测试460800的波特率 30kb一秒的数据接收 一包1024个字节 每包需要应答15字节的情况下 AT指令处理是使用多个缓冲级来处理模块发送过来的数据 分别有模块应答缓冲级 跟等待应答的缓冲级、被忽略的AT指令集的缓冲级(例如注册一个SEND OK\r\n则模块应答此条指令立刻清除缓冲级释放内存无需等待超时直接忽略)还有需要回调的缓冲级(则出现此指令调用回调函数)都是通过注册的方式来实现 如果出现一包跟指令被分到一个包内 AT处理函数一样可以搜索到AT指令 使用strstr函数来实现 函数的缓冲级都是指针不占用内存 使用动态内存管理的方式 有数据则创建内存放入数据作为一个缓冲级 如果模块应答的数据在规定的时间内没有响应则删除此缓冲级 函数前都有注释介绍 下面介绍一些常用的函数: at_init初始化一些变量已经串口 at_time_task使用定时器回调 1毫秒回调一次 用来计数超时的指令缓冲&等待超时的计数 at_clear_all 在模块开机的时候可能会有很多乱数据 可以在初始化完毕后使用此函数清除所有缓冲级 释放所有内存 at_processing处理AT的应答超时的指令(做删除释放内存的动作),还有处理等待的AT指令 此函数一定要不断循环处理 可以加入到定时器 目前我实验是在UCOS上的 所以直接创建个任务来执行此函数 当程序在等待某个AT指令的时候此函数会寻找接收的缓冲级是否有等待的AT指令 at_cmd_cb_hand回调处理函数 如果接收缓冲级出现某个已经注册的指令则回调注册时所填写的函数地址 at_send_cmd 发送一个AT指令 可以用跟printf一样使用 %d等等 at_send_data 发送数据的时候所使用 需要填写长度 at_cmd_ignore_register 注册一个被忽略的AT指令 带入参数 *s (例如填写一个"SEND OK" 则模块应答的此条指令直接被忽略 释放内存 被忽略之前会检查此缓冲级会不会带有别的AT指令需要回调的) at_cmd_ignore_cancel 取消被忽略的指令 则取消已经注册的被忽略的AT指令 at_cmd_cb_register AT指令的回调注册 例如参数填写"+IPD",函数名a 则出现+IPD的时候回调a函数 a函数有类型 在at.h文件里面有 at_cmd_cb_cancel注销回调你懂得 at_wait_cmd 等待一个AT指令集或者超时则立刻返回 等待途中会不断调用OS的延迟程序 让系统能有时间去执行其他任务 使用方法例如{ at_send_cmd("AT+UART=%u,%u,%u,%u,%u\r\n",baudrate,databit,stopbit,parity,flow_control); return (esp_error)at_wait_cmd("\r\nOK\r\n",2000,NULL); } at_error at_wait2_cmd(char *s,char *s2,u16 timeout,u8 *index) 此函数是等待两个AT指令集 如果出现一个则立刻返回 返回值h文件有介绍 AT_DONE则出现此条指令 index参数则提取应答的缓冲首地址 使用at_buf_get函数获取首地址 使用完后要调用at_free_buf来清除并释放这个缓冲级 at_buf_len_get查询此应答的缓冲级长度 如果在index填写NULL则不需要缓冲级首地址 直接清除释放缓冲级

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值