FreeRtos——24、STM32中断处理体系及软件定时器按键消抖
第一节:STM32中断处理体系结构1.中断处理路径:2.NVIC中断控制器的中断优先级2.1 中断号在NVIC中对于硬件产生的任何一个中断都分配了一个中断号中断号是一个唯一的标识符用于识别每个外设设备的中断。NVIC使用中断号来配置中断的优先级和使用的状态。如在F103数据手册上的介绍在向量表中列出的 位置Position实际上就是中断号IRQ number。这些中断号用于在嵌套向量中断控制器NVIC中配置和管理中断2.2 中断优先级组成NVIC中对中断优先级进行了分组优先级分别有两部分组成抢占优先级 子优先级抢占优先级决定了是否可以抢占。子优先级决定了优先级的处理顺序。中断的分组优先级的抢占优先级与子优先级大部分可由用户自行配置。如果未配置则使用默认值数据越小优先级越高。例如当两个相同的分组优先相同子优先不同的中断不会发生抢占而是按子优先级顺序依次执行。3.中断处理例程ISR对中断进行响应的函数又称中断服务例程是CPU中断当前任务保存现场后去执行的函数。在中断处理例程中要执行一些实时性高的操作不可以出现阻塞。也不允许在ISR使用可能出现阻塞的操作。对于每一个中断号,MCU的启动文件startup_stmf103c8t6.s的汇编启动文中已经定义好了相应的ISR中断全程函数的句柄。在启动文件中定义两部分中断的ISR,一部分是系统内置的中断服务例程的句柄另一部则是外部中断源的中例ISR大多也都是一些外设中断源的ISR句柄。4.__weak弱函数可实现的函数是一种函数回调的方式。__weak函数修饰符是ARM编译器中的一种语法形式。此函数可以实新实现类似于C虚函数。本质是个函数回调。在代码中重新实现即可。编译时会直接编译重新实现的函数。重点一句话以上讲解了很多的概念但是在开发中只需要搞清楚与中断事件对应的ISR的回调的__weak函数重新实现相关的ISR逻辑即可注册ISR的回调的函数中不可以有阻塞。5.HAL库快速配置GPIO引脚为中断复用引脚1. 看电路图使用cubeMX快速配置引脚复用使用EXTI14的NVIC中的中断功能。6.重写__weak函数实现ISR逻辑:实现按键响应的功能在小彩灯的驱动文件中重写相关中断句柄的回调的_weak函数即可实现中断控制的逻辑#include color_led_driver.h #include main.h //全采led灯的驱动 void color_led_set(uint8_t red, uint8_t green, uint8_t blue) { //红灯 HAL_GPIO_WritePin(GPIOA, GPIO_PIN_2, (GPIO_PinState)!red); //绿灯 HAL_GPIO_WritePin(GPIOA, GPIO_PIN_15, (GPIO_PinState)!green); //蓝灯 HAL_GPIO_WritePin(GPIOB, GPIO_PIN_3, (GPIO_PinState)!blue); } //外部按键通过中断控制小彩灯驱动 void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) { //实现功能就是通过按键key是否按下来控制小彩灯 if(GPIO_Pin GPIO_PIN_14) { if(HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_14))//抬起 { color_led_set(0,0,0); }else{//按下 color_led_set(1,0,1); } } }第二节、FreeRTOS中对中断上下文的处理由于材料的物理特性及电学特性当按键按下的瞬间会短时间出现电平的高低变化如图所示按键抖动的问题如何消抖下面介绍几种方式1. FreeRTOS软件定时器进行延时消抖1. 简述FreeRTOS定时器在RTOS调度启动时会自动创建一个空闲idle任务与一个定时器守护任务。那么定时器的功能就是由这个定时器守护任务执行相应的回调函数实现的。软件定时器有一个定时周期还有一个回调函数。根据回调函数执行的频度有两种类型的软件定时器单次定时器one-shot timer,即回调函数被执行一次后定时器停止工作。周期定时器periodic timer,回调函数会被循环执行定时器一直工作。2.定时器有休眠与运行两种状态休眠状态定时器使用xTimerCreate创建后就处于休眠状态单次定时器执行一次回调函数后进入休眠状态。定时器使用函数xTimerStop停止后进行休眠状态。运行状态处于运行状态的定时器在超时时间到达后就会执行其回调函数不定单次定时器还是周期定时器。定时器在以下几种状态下处于运行状态。使用函数xTimerStart启动后定时器进行运行状态。定时器在运行状态时被执行函数xTimerReset复位起始时间后继续处于运行状态。软件定时器的各种操作实际上是在FreeRTOS系统的定时器守护任务中完成的。3.软件定时器的机制与使用配置在用户任务里执行的各种定时器操作函数都是通过一个队列发送给这个定时器守护任务的所以这个执行同步的中间的队列也被称为定时器指令队列定时器守护任务读取定时器指令队列中的指令然后执行相庆的操作如图所示为了保证守护定时器任务可以得到时间片最好配置守护任务的调度优先级高于用户任务的优先级。同时如果要使用软件定时器需要在进行如下配置官方文档介绍即需要在FreeRTOSConfig.h中进行以上宏定义并指定合适的值。4. 软件定时器常用API接口函数1. 动态创建定时器timers.h TimerHandle_t xTimerCreate ( const char * const pcTimerName, const TickType_t xTimerPeriod, const UBaseType_t uxAutoReload, void * const pvTimerID, TimerCallbackFunction_t pxCallbackFunction ); 功能创建一个新的软件定时器实例 并返回一个可以引用定时器的句柄。 参数 pcTimerName 分配给定时器的可读文本名称。 这样做纯粹是为了协助 调试。 xTimerPeriod 定时器的周期。 以 tick 为单位指定此周期 宏 pdMS_TO_TICKS() 可用于将以毫秒为单位指定的时间转换为以 tick 为单位指定的时间。 uxAutoReload pdTRUE重新加载pdFALSE执行一次结束。 pvTimerID 分配给正在创建的定时器的标识符。传递给回调函数的参数。 pxCallbackFunction 定时器到期时调用的函数 返回值 如果定时器创建成功 则返回新创建的定时器的句柄。失败返回NULL; 当于此动态创建定时器也有静态创建定时器详阅官方文档。2.启动定时器阻塞版xTimerStart与非阻塞版xTimerStartFromISR1. BaseType_t xTimerStart( TimerHandle_t xTimer, TickType_t xBlockTime ); 功能启动定时器在队列满后会阻塞当前调用任务。 参数 xTimer 正在启动/重新启动的定时器的句柄。 xBlockTime 在调用 xTimerStart() 时队列已满的情况下 指定调用任务应保持阻塞状态 以等待启动命令成功发送到定时器命令队列的时间 以 tick 为单位。 如果在 启动 RTOS 调度器之前调用 xTimerStop()则会忽略 xBlockTime 成功返回pdPASS,失败返回pdFAIL. 2. BaseType_t xTimerStartFromISR(TimerHandle_t xTimer, BaseType_t *pxHigherPriorityTaskWoken); 功能可从中断服务例程调用的 xTimerStart() 的非阻塞版本 参数 xTimer 正在启动/重新启动的定时器的句柄 pxHigherPriorityTaskWoken : 如果守护进程的优先级高于当前任务会返回pdTRUE如果为pdTRUE 那么应在中断退出前执行上下文切换这样可以让高优先级任度马上得到执行。 可以调用内核函数portYIELD_FROM_ISR(xHigherPriorityTaskWoken);出让时间片。 成功返回pdPASS,失败返回pdFAIL3.定时器运行中重置定时器函数xTimerReset 与xTimerResetFromISRtimers.h 1.BaseType_t xTimerReset( TimerHandle_t xTimer, TickType_t xBlockTime ); 功能介绍xTimerReset() 重新启动先前使用 xTimerCreate() API 函数创建的定时器。 如果定时器已经启动且已处于活动状态 那么 xTimerReset() 会使定时器 重新评估其到期时间 因此到期时间与 xTimerReset() 的调用时间有关。 如果定时器处于休眠状态则 xTimerReset() 具有与 xTimerStart() API 函数等效的功能。 参数 xTimer 正在重置/启动/重新启动的定时器的句柄。 xBlockTime 队列满中的阻塞时间 返回值 成功返回pdPASS, 失败返回pdFAIL 2.BaseType_t xTimerResetFromISR(TimerHandle_t xTimer, BaseType_t *pxHigherPriorityTaskWoken); 功能与之reset阻塞版相同。 参数与start一样的参数功能 返回值与start一样的返回值。4.删除定时器xTimerDelete:BaseType_t xTimerDelete( TimerHandle_t xTimer, TickType_t xBlockTime ); 功能删除正在使用的定时器并回收内存。 成功返回pdPASS,失败返回pdTRUE5.定时器消抖实验实例演示配置FreeRTOSConfig.h:#define configUSE_QUEUE_SETS 1 #define configUSE_TIMERS 1 #define configTIMER_TASK_PRIORITY configMAX_PRIORITIES-1 #define configTIMER_QUEUE_LENGTH 10 #define configTIMER_TASK_STACK_DEPTH 256freeRTOS.c://添加定时器对象并动态创建定时器 TimerHandle_t mykeyTimer; mykeyTimer xTimerCreate(mykeytimer,10, pdFALSE, NULL, mykeyTimer_function); configASSERT(mykeyTimer);key_app.c:#include key_app.h #include timers.h extern TimerHandle_t mykeyTimer; //外部按键通过中断控制小彩灯驱动 void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) { //实现功能就是通过按键key是否按下来控制小彩灯 if(GPIO_Pin GPIO_PIN_14) { BaseType_t status pdFALSE; //复位定时器 xTimerResetFromISR(mykeyTimer,status); if(status pdTRUE) { portYIELD_FROM_ISR(status); } } } //超时处理函数 void mykeyTimer_function( TimerHandle_t xTimer ) { if(isKeyPress()) { color_led_set(1,0,1); } else{ color_led_set(0,0,0); } }2.FreeRTOS同步机制处理需要延时处理的中断下文以FreeRTOS中最简洁的同步方式任务通知进行消抖freeRTOS.c:xTaskCreate(mykey_dithering, mykey_dithering, 256, NULL, osPriorityNormal,key_dithering_taskhandle ); configASSERT(key_dithering_taskhandle);key_app.c:#include key_app.h #include timers.h #include stdbool.h extern TaskHandle_t key_dithering_taskhandle; // 外部按键通过中断控制小彩灯驱动 void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) { // 实现功能就是通过按键key是否按下来控制小彩灯 if (GPIO_Pin GPIO_PIN_14) { BaseType_t status pdFALSE; // 发出任务的通知 xTaskNotifyFromISR(key_dithering_taskhandle, 0, eNoAction, status); if (status pdTRUE) { portYIELD_FROM_ISR(status); } } } void mykey_dithering(void *arg) { uint32_t data 0; while (true) { // 等待通知 xTaskNotifyWait(pdFALSE, pdFALSE, data, portMAX_DELAY); //阻塞10ms,过滤杂波 vTaskDelay(10); if (isKeyPress()) { color_led_set(1, 0, 1); } else { color_led_set(0, 0, 0); } } }
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2468943.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!