c - ESP 32 中断问题
问题描述
我正在使用 esp32 模块,并在其上实现中断功能。因此,基本上每当中断到来时,led 都应该发光,如果不是,led 应该处于关闭状态。令人惊讶的是,如果我尝试相反的情况,即“如果中断来自开关,则 LED 会关闭,并且通常它处于打开位置”,这是可行的。
这是我的代码:
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"
#include "driver/gpio.h"
#define op 18
#define ip 4
#define OP_PIN_SEL 1ULL<<op
#define IP_PIN_SEL 1ULL<<ip
#define ESP_INTR_FLAG_DEFAULT 0
static void gpio_isr_handler(void )
{
printf("Intrpt Act\n");
gpio_set_level(op,0);
vTaskDelay(1000/portTICK_PERIOD_MS);
fflush(stdout);
}
void app_main()
{
gpio_config_t io_conf;
// io_conf=malloc(0);
gpio_pad_select_gpio(op);
gpio_pad_select_gpio(ip);
gpio_set_direction (op,GPIO_MODE_OUTPUT);
gpio_set_direction(ip,GPIO_MODE_INPUT);
io_conf.intr_type=GPIO_PIN_INTR_ANYEDGE;
io_conf.mode=GPIO_MODE_OUTPUT;
io_conf.pin_bit_mask=OP_PIN_SEL;
io_conf.pull_down_en=1;
io_conf.pull_up_en=0;
gpio_config(&io_conf);
gpio_set_intr_type(ip,GPIO_INTR_POSEDGE);
gpio_install_isr_service(ESP_INTR_FLAG_DEFAULT);
gpio_isr_handler_add(ip,gpio_isr_handler,(void*)ip);
int cnt=0;
while(1)
{
printf("cnt: %d",cnt++);
gpio_set_level(op,1);
vTaskDelay(1000/portTICK_RATE_MS);
fflush(stdout);
}
}
当我通过开关触发中断时也会发生此错误:
线程 Thread-2 中的异常:
Traceback (most recent call last):
File "/usr/lib/python2.7/threading.py", line 801, in __bootstrap_inner
self.run()
File "/usr/lib/python2.7/threading.py", line 754, in run
self.__target(*self.__args, **self.__kwargs)
File "/home/dhananjay/esp/esp-idf/tools/idf_monitor.py", line 133, in _run_outer
self.run()
File "/home/dhananjay/esp/esp-idf/tools/idf_monitor.py", line 226, in run
data = self.serial.read(self.serial.in_waiting or 1)
File "/usr/lib/python2.7/dist-packages/serial/serialposix.py", line 495, in read
raise SerialException('device reports readiness to read but returned no data (device disconnected or multiple access on port?)')
SerialException: device reports readiness to read but returned no data (device disconnected or multiple access on port?)
解决方案
至少,您在中断处理程序 ( )中做的太多了。gpio_isr_handler()
中断处理程序会中断当前正在运行的任何代码流。它们可以在子程序中的指令之间的任何时间发生。除非代码采取预防措施以确保其数据结构保持一致状态,否则您将要求数据损坏和崩溃。为此,它必须锁定“关键部分”中的中断,这是一项非常昂贵的操作,并且可能导致错过中断。所以系统例程很少会这样做。
您还希望中断处理程序保持简短,因为它们会阻止系统上的所有其他活动 - 因此当您仍在中断处理程序中时,网络堆栈和其他内务处理功能不会发生。
您绝对不应该vTaskDelay()
从中断处理程序中调用。那有什么意思?充其量它会影响任何被中断的任务,如果有几个任务正在运行,那将是一个随机任务。
您也不应该从中断处理程序中调用任何 stdio 函数。同样,您可能会中断另一个 stdio 功能;它的数据结构可能处于不一致状态,如果它使用串行端口,则硬件也可能处于不一致状态。
正确的做法是让中断服务程序唤醒一个任务。将您当前的代码放入gpio_isr_handler()
带有 a 的无限循环中的任务中,启动任务app_main()
并gpio_isr_handler()
唤醒任务。在循环开始时使用vTaskSuspend()
让任务等到它被唤醒。
您可以使用以下之一安全地从中断处理程序中唤醒任务:
从 开始xTaskResumeFromISR()
,这是从中断处理程序中唤醒任务的最简单方法。
从结构上讲,您还应该将while()
循环移出app_main()
到它自己的任务中,这是您曾经存在的代码gpio_isr_handler()
应该恢复的任务。
推荐阅读
- vb.net - 对包含点和逗号的值求和的程序
- java - java math.mxparser 在参数名称为“e”时返回 NaN
- haskell - 在 Haskell 中,如何在不同的 Traversables 之间进行 fmap?
- android - 是否可以在颤动按钮开始时对齐文本
- r - 正则表达式从R中的字符串中删除所有内容,但表情符号?
- argparse - NameError:名称“argparse”未定义
- c++ - C++用户登录程序只读取文件的第一行
- linux - 使用 Bash 计算今天和本月下一个重复日期之间的距离?
- matlab - 从输入和输出数据估计传递函数系数
- c++ - 在 Unix 中将 json 作为命令行参数传递