当前位置: 首页 > article >正文

外设的中断控制

如ADC、SPI、I2C、TIM等使用STM32 HAL库时的中断函数调用方式和UART非常类似,都有底层直接使能中断和上层库函数管理两种方式。下面详细说明几种典型外设:


一、ADC外设

(1)直接使能中断(底层控制):

只需调用一次,一直有效。

// 开启ADC转换完成中断
__HAL_ADC_ENABLE_IT(&hadc1, ADC_IT_EOC);

// ADC IRQ中断函数(stm32f1xx_it.c)
void ADC1_IRQHandler(void)
{
    if(__HAL_ADC_GET_FLAG(&hadc1, ADC_FLAG_EOC))
    {
        uint16_t adc_value = HAL_ADC_GetValue(&hadc1);  // 手动读取ADC数据
        __HAL_ADC_CLEAR_FLAG(&hadc1, ADC_FLAG_EOC);     // 清除标志位
    }
}

(2)使用HAL库中断(库函数控制):

一般调用一次即启动一次ADC转换,中断每次触发后自动关闭,需再次调用。

// 启动一次ADC中断转换
HAL_ADC_Start_IT(&hadc1);

// 中断回调函数(用户实现)
void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc)
{
    uint16_t adc_value = HAL_ADC_GetValue(hadc);
    // 处理数据...

    HAL_ADC_Start_IT(hadc); // 再次启动中断以实现连续采样
}

// ADC中断服务函数(自动调用回调)
void ADC1_IRQHandler(void)
{
    HAL_ADC_IRQHandler(&hadc1);
}

二、SPI外设

(1)直接使能中断(底层控制):

通常只调用一次保持长期有效。

// 启用SPI接收缓冲区非空中断
__HAL_SPI_ENABLE_IT(&hspi1, SPI_IT_RXNE);

// SPI中断函数
void SPI1_IRQHandler(void)
{
    if(__HAL_SPI_GET_FLAG(&hspi1, SPI_FLAG_RXNE))
    {
        uint8_t data = *(__IO uint8_t *)&hspi1.Instance->DR; // 手动读取数据寄存器
        // 数据处理
    }
}

(2)使用HAL库中断(库函数控制):

需要在每次传输完成后再次调用。

uint8_t rx_data[10];
HAL_SPI_Receive_IT(&hspi1, rx_data, 10); // 开启一次接收

// 回调函数
void HAL_SPI_RxCpltCallback(SPI_HandleTypeDef *hspi)
{
    // 接收完成处理
    HAL_SPI_Receive_IT(hspi, rx_data, 10); // 再次调用以连续接收
}

// SPI中断函数
void SPI1_IRQHandler(void)
{
    HAL_SPI_IRQHandler(&hspi1);
}

三、I2C外设

(1)直接使能中断(底层控制):

使能一次即可长期有效,但需手动处理通信时序(不推荐新手使用)。

// 启用I2C接收中断
__HAL_I2C_ENABLE_IT(&hi2c1, I2C_IT_RXNE);

// I2C中断函数
void I2C1_EV_IRQHandler(void)
{
    if(__HAL_I2C_GET_FLAG(&hi2c1, I2C_FLAG_RXNE))
    {
        uint8_t data = hi2c1.Instance->DR; // 手动读取数据寄存器
        // 数据处理
    }
}

(2)使用HAL库中断(库函数控制):

每次传输完成后需要再次调用。

uint8_t rx_data[10];
HAL_I2C_Slave_Receive_IT(&hi2c1, rx_data, 10); // 开启一次接收

// 回调函数
void HAL_I2C_SlaveRxCpltCallback(I2C_HandleTypeDef *hi2c)
{
    // 数据处理...
    HAL_I2C_Slave_Receive_IT(hi2c, rx_data, 10); // 再次调用
}

// I2C中断函数
void I2C1_EV_IRQHandler(void)
{
    HAL_I2C_EV_IRQHandler(&hi2c1);
}

四、TIM定时器外设

(1)直接使能中断(底层控制):

调用一次后自动循环触发中断(只要计数器开启)。

// 使能更新中断(溢出中断)
__HAL_TIM_ENABLE_IT(&htim1, TIM_IT_UPDATE);
HAL_TIM_Base_Start(&htim1); // 启动定时器计数

// 定时器中断函数
void TIM1_UP_IRQHandler(void)
{
    if(__HAL_TIM_GET_FLAG(&htim1, TIM_FLAG_UPDATE))
    {
        __HAL_TIM_CLEAR_FLAG(&htim1, TIM_FLAG_UPDATE);
        // 定时器溢出处理
    }
}

(2)使用HAL库中断(库函数控制):

调用一次后自动循环触发(推荐使用,使用简单)。

// 启动TIM定时器中断模式
HAL_TIM_Base_Start_IT(&htim1);

// 回调函数(用户实现)
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
    if(htim->Instance == TIM1)
    {
        // 定时处理
    }
}

// TIM中断函数
void TIM1_UP_IRQHandler(void)
{
    HAL_TIM_IRQHandler(&htim1);
}

不同外设两种方式的共同规律总结:

方式中断生效时长数据处理方式再次调用需求
底层控制 (__HAL_xxx_ENABLE_IT)持续有效(一次调用长期有效)用户手动读取寄存器一般无需
库函数控制 (HAL_xxx_IT)一次有效(传输结束自动关闭)库自动管理缓冲区,使用回调函数必须重复调用

推荐方式:

  • 一般应用场景下,推荐使用HAL_xxx_IT(易用、稳定)。
  • 对于底层、实时要求严格、或者对数据有精细控制需求的场景,才考虑使用__HAL_xxx_ENABLE_IT手动管理。

http://www.kler.cn/a/611396.html

相关文章:

  • Go 语言 sync 包使用教程
  • 内存型数据库深入解析:Memcache、Redis 与 Squid 的对比与应用
  • 数据结构二叉树进阶
  • SylixOS 中 select 原理及使用分析
  • 计算机三级信息安全技术核心知识点详细定义解析,按章节分类并重点阐述关键概念定义
  • 【加密社】如何创建自己的币圈工具站
  • 解决用户同时登录轮询获取用户信息错乱,使用WebSocket和Server-Sent Events (SSE)
  • 数据可视化TensorboardX和tensorBoard安装及使用
  • MySQL - 数据库基础操作
  • 【每日算法】Day 8-1:广度优先搜索(BFS)算法精讲——层序遍历与最短路径实战(C++实现)
  • 二十五、实战开发 uni-app x 项目(仿京东)- 前后端轮播图
  • 2025最新Chatbox全攻略:一键配置Claude/GPT/DeepSeek等主流模型(亲测可用)
  • # WebSocket 与 Socket.IO 对比与优化
  • RustDesk部署到linux(自建服务器)
  • How to use pgbench to test performance for PostgreSQL?
  • 完全背包模板
  • 突破反爬困境:SDK架构设计,为什么选择独立服务模式(四)
  • 本地部署 LangManus
  • K8S学习之基础五十一:k8s部署jenkins
  • 面试常问系列(二)-神经网络参数初始化之自注意力机制