0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

STM32Cube HAL库中断处理机制 以及回调函数实现原理

黄工的嵌入式技术圈 来源:黄工的嵌入式技术圈 2020-03-03 14:01 次阅读

最近有较多关于STM32Cube HAL的问题,侧面反应了使用STM32CubeMX的人不少。所以,最近可能会重点写这方面内容。

1写在前面

很多人都知道STM32CubeMX这套工具的一个目的:减少开发者对STM32底层驱动的开发时间,把重心放在应用代码上。

但是,STM32CubeMX只是生成了底层驱动的初始化代码。所以,我们还需要掌握:应用层代码如何调用HAL库函数接口,以及HAL库中断处理机制等相关知识。

HAL库牵涉的内容较多,本文拿HAL库中断处理来讲解,以及相关的回调函数。

2

HAL库中断处理机制

之前使用标准外设库开发时,中断程序(函数)由我们自己实现。

而HAL库的中断处理函数是按照HAL处理机制来实现,如USART1,统一由HAL_UART_IRQHandler来进行处理,如下图:

其它大部分外设(TIM、SPI、CAN...)中断都类似,HAL进行统一处理。

也就是说,HAL已经帮我们把中断处理函数写好了,我们只需要调用相应函数来编写应用程序就行了。

HAL_xxx_IRQHandler里面做了哪些处理?我们以STM32F1的HAL_UART_IRQHandler为例:

void HAL_UART_IRQHandler(UART_HandleTypeDef *huart){ uint32_t isrflags = READ_REG(huart->Instance->SR); uint32_t cr1its = READ_REG(huart->Instance->CR1); uint32_t cr3its = READ_REG(huart->Instance->CR3); uint32_t errorflags = 0x00U; uint32_t dmarequest = 0x00U; /* If no error occurs */ errorflags = (isrflags & (uint32_t)(USART_SR_PE | USART_SR_FE | USART_SR_ORE | USART_SR_NE)); if(errorflags == RESET) { /* UART in mode Receiver -------------------------------------------------*/ if(((isrflags & USART_SR_RXNE) != RESET) && ((cr1its & USART_CR1_RXNEIE) != RESET)) { UART_Receive_IT(huart); return; } } /* If some errors occur */ if((errorflags != RESET) && (((cr3its & USART_CR3_EIE) != RESET) || ((cr1its & (USART_CR1_RXNEIE | USART_CR1_PEIE)) != RESET))) {/*··删减了部分代码·*/ } /* End if some error occurs */ /* UART in mode Transmitter ------------------------------------------------*/ if(((isrflags & USART_SR_TXE) != RESET) && ((cr1its & USART_CR1_TXEIE) != RESET)) { UART_Transmit_IT(huart); return;} /* UART in mode Transmitter end --------------------------------------------*/ if(((isrflags & USART_SR_TC) != RESET) && ((cr1its & USART_CR1_TCIE) != RESET)) { UART_EndTransmit_IT(huart); return; }}

其实,大家认真看一下代码应该能明白,这些和我们编写的中断处理函数是不是有类似之处?

这是无非就是接收中断、发送中断、错误中断等一系列处理。只是这里又进行了再次封装,比如接收中断UART_Receive_IT。

当然,这个UART_Receive_IT接收中断实现方式又可能存在不同。像F0、F1...就是直接调用这个接收中断函数来进一步处理。

像L0、G0...是通过执行指针函数RxISR来进一步处理。G0的接收中断处理为:huart->RxISR(huart);

void HAL_UART_IRQHandler(UART_HandleTypeDef *huart){//删除了前面代码 /* If no error occurs */ errorflags = (isrflags & (uint32_t)(USART_ISR_PE | USART_ISR_FE | USART_ISR_ORE | USART_ISR_NE)); if (errorflags == 0U) { /* UART in mode Receiver ---------------------------------------------------*/ if (((isrflags & USART_ISR_RXNE_RXFNE) != 0U) && (((cr1its & USART_CR1_RXNEIE_RXFNEIE) != 0U) || ((cr3its & USART_CR3_RXFTIE) != 0U))) { if (huart->RxISR != NULL) { huart->RxISR(huart); } return; } }//删除了后面代码}

看了上面USART中断处理的函数,大家有没有得到什么启发?

其实,HAL库里面处理机制基本一致,只是实现方式上有所不同。

如果你摸清楚了HAL库基本原理,相信阅读HAL库,或者使用HAL库编写应用代码不是问题。

3

回调函数实现原理

在HAL库中存在大量类似HAL_XXX_XXXCallback这样的函数,这些都是回调函数。

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。

回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

---来自百度百科

HAL库中断处理使用了较多的回调函数,还是拿UART接收中断来举例说明。

初始化配置好UART中断接收,如果有中断请求,就会执行回调函数HAL_UART_RxCpltCallback。

看上面回调函数的定义,通过特定条件调用『回调函数』,这里触发的条件就是中断。

4

扩展说明

这里也简单说几点:

1.初学者想直接使用HAL不是不行,需要有一定C语言功底

针对大部分初学者来说,是不建议直接上手HAL。但是,有部分C功底较好的,还是建议直接上手。

2.学HAL,建议参看官网例程

很多人不知道如何找资源,我不止一次强调,官方的才是最好。在HAL库中Projects目录下就有很多例程Examples。

3.我们追求效率,可以HAL库源码

如果你想修改HAL库源码,允许修改少部分。如果要大量修改,还是别折腾了。

4.实际项目需做一定修改

STM32CubeMX仅仅是生成初始化代码和工程,你实际项目中一般都有自己的软件架构。

特别是项目越大,软件架构就需要更加规范。

比如:生成的gpio.c文件名,你需要修改成bsp_gpio.c.

再比如:函数MX_USART2_UART_Init改成MX_DEBUG_UART_Init.

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 底层驱动
    +关注

    关注

    0

    文章

    4

    浏览量

    8432
  • stm32cubemx
    +关注

    关注

    5

    文章

    280

    浏览量

    14746
  • HAL库
    +关注

    关注

    1

    文章

    114

    浏览量

    6168
收藏 人收藏

    评论

    相关推荐

    STM32Cube和standard peripheral library有什么区别?

    STM32Cube和standard peripheral library有什么区别
    发表于 05-16 06:52

    HALIAP失败的原因?

    第一个项目HALSTM32Cube FW_F1 V1.2.0,生成A_boot.bin,A_app.bin,从A_boot跳转A_app成功 第二个项目HAL
    发表于 05-10 08:29

    请问hal的串口接收完成函数中如何区分是不是DMA接收完成?

    HAL_UART_RxCpltCallback 这个函数在串口中断接收完成后会被调用,在DMA接收完成后也会被调用 那么能否在这个
    发表于 05-07 07:39

    STM32 hal无法接收串口数据是怎么回事?

    一个字节产生中断。在接收中断函数处理保存数据并重新开启接收一个字节的
    发表于 04-29 06:11

    stm32中断函数改变的变量在while中不变怎么解决?

    关于stm32 中断函数改变的变量在while中不变问题
    发表于 04-26 07:44

    使用HAL的USB模块时,当设置要接收的数据长度为大于64字节时,无法从接收结束函数的原因?

    两个方面:1、HAL包调用接收结束的函数的条件为:接收完预设的数据长度 或者 接收到小于max_packet长度的包。 当接收到第一个64字节后,因为还不满要求接收的长度,因此
    发表于 04-12 06:07

    请问Hall多串口接收时串口中断函数重入导致HardFault 0xFFFFFFF1怎么解决?

    _BUSY){ UART3_RestartRecIT=1; } } } 按照正常集成的设计,各串口都有独立的串口中断响应函数,所以不会存在该问题。但是HAL
    发表于 04-03 06:44

    STM32CUBUMX定时器1中断函数就是进不去怎么解决?

    );HAL_TIM_PWM_Start( htim1, TIM_CHANNEL_3); HAL_TIM_PWM_Start_IT( htim1, TIM_CHANNEL_3); 就死活进不去,
    发表于 03-28 07:58

    HAL_UART_RxCpltCallback HAL函数,通过485发送命令与串口三通信后,整个函数就不执行了的原因?

    HAL_UART_RxCpltCallbackHAL函数两个串口同时进行中断接受数据,一
    发表于 03-22 06:53

    Stm32G474 DMA传输全部完毕后,需要产生中断并调运回函数完成相关操作,如何处理

    Stm32G474,DMA传输全部完毕后,需要产生中断并调运回函数完成相关操作。请问我如何处理?没有看到常规的
    发表于 03-15 08:26

    函数(callback)是什么?函数实现方法

    函数是一种特殊的函数,它作为参数传递给另一个函数,并在被调用函数执行完毕后被调用。
    发表于 03-12 11:46 2697次阅读

    ​​嵌入式中函数实现方法

    函数的命名规范没有固定的标准,但是根据通用惯例和编码规范,函数的命名应该能够反映
    发表于 03-04 14:49 647次阅读

    stm32f1如何将外部中断关掉hal

    Layer)来简化芯片和外设的驱动开发。在STM32F1中,外部中断的关闭是通过HAL中的相应函数
    的头像 发表于 12-22 13:52 3187次阅读

    函数指针的五大作用

    1,函数机制函数指针常用于实现
    的头像 发表于 12-06 08:00 1834次阅读
    <b class='flag-5'>函数</b>指针的五大作用

    一文解析BLE观察者模式机制

    nRF5 SDK从版本14开始,对事件机制做了更新,引入了观察者模式,以解耦不同BLE Layer对BLE事件的
    的头像 发表于 11-27 10:07 922次阅读
    一文解析BLE观察者模式<b class='flag-5'>回</b><b class='flag-5'>调</b><b class='flag-5'>机制</b>