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

【星云 Orbit • STM32F4】10. 在串口接收中断里即时解析数据头的程序框架

【星云 Orbit • STM32F4】10. 串口中断中即时解析数据头的程序开发:实现高效实时数据处理


摘要

在嵌入式开发中,串口中断处理是实现高效实时数据传输的关键技术之一。本文将详细介绍如何在STM32F407微控制器上开发一个在串口接收中断中即时解析数据头的程序框架。通过本教程,你将掌握如何在中断服务函数中快速解析数据头,实现高效的数据接收与处理。


关键词

STM32F407、串口中断、数据头解析、实时数据处理、嵌入式开发


1. 引言

本教程旨在帮助嵌入式开发小白从零开始,学习如何在STM32F407微控制器上实现一个在串口接收中断里即时解析数据头的特殊程序框架。该程序能够通过中断方式接收串口数据,并在接收到数据头后立即解析后续数据内容,支持基本的错误检测和数据处理功能。教程内容涵盖基础知识、配置步骤、HAL库函数详解,并提供配套例程和代码注释。

2. 硬件准备

  • STM32F407开发板
  • 串口调试工具(如串口助手)

3. 软件准备

  • Keil MDK-ARM开发环境
  • STM32F407标准库

4. 知识储备

在开始编程之前,需要了解以下基础知识:

  • STM32F407的串口(USART)外设
  • 基本的串口通信知识(波特率、数据位、停止位、校验位)
  • C语言编程基础
  • 中断服务函数的基本概念

5. 程序设计:从需求到实现

5.1 程序功能概述

程序的主要功能包括:

  • 初始化STM32F407的串口外设
  • 通过中断方式接收串口数据
  • 在中断服务函数中即时解析数据头
  • 存储接收到的数据到缓冲区
  • 提供数据处理接口
5.2 程序模块划分

程序分为以下几个模块:

  • main.c:主程序文件,负责初始化和程序运行逻辑
  • usart.husart.c:串口驱动文件,负责串口的初始化和数据收发
  • protocol.hprotocol.c:协议处理模块,负责数据帧的解析和校验
5.3 程序流程图

在这里插入图片描述

程序流程图

  • 系统初始化:程序开始时进行系统初始化,包括时钟配置、GPIO配置等。
  • 串口初始化:配置串口参数(如波特率、数据位、停止位等),使串口准备好接收数据。
  • 等待中断触发:程序进入等待状态,等待串口接收到数据后触发中断。
  • 接收数据:中断触发后,程序从串口接收数据。
  • 检测数据头:检查接收到的数据是否包含有效的数据头,以确保数据的完整性。
  • 解析数据帧:根据数据帧结构,解析接收到的数据,提取数据类型、数据长度、数据内容和校验码等信息。
  • 存储数据:将解析后的数据存储到内存或外部存储器中。
  • 数据处理:对存储的数据进行进一步的处理,如计算、分析等,处理完成后返回等待中断触发状态,继续接收新的数据

6. 代码实现:从初始化到数据处理

6.1 串口初始化 (usart.c)
#include "usart.h"

void USART_Init(uint32_t baudrate) {
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;

    // 使能GPIO和USART时钟
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);

    // 配置GPIO引脚
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    // 配置USART
    USART_InitStructure.USART_BaudRate = baudrate;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;
    USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
    USART_Init(USART1, &USART_InitStructure);

    // 使能USART
    USART_Cmd(USART1, ENABLE);

    // 配置 NVIC 优先级
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
    NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure);
}
6.2 协议处理模块 (protocol.c)
#include "protocol.h"

#define DATA_BUFFER_SIZE 100
#define HEADER_SIZE 3
#define HEADER_1 0xEB
#define HEADER_2 0x00
#define HEADER_3 0x55

static uint8_t data_buffer[DATA_BUFFER_SIZE];
static uint16_t buffer_index = 0;
static uint8_t header_received = 0;

void Protocol_Init(void) {
    buffer_index = 0;
    header_received = 0;
}

void USART1_IRQHandler(void) {
    if (USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) {
        uint8_t received_data = USART_ReceiveData(USART1);

        if (header_received < HEADER_SIZE) {
            if (received_data == header[header_received]) {
                header_received++;
                if (header_received == HEADER_SIZE) {
                    // 数据头检测完成,开始接收数据
                    buffer_index = 0;
                }
            } else {
                header_received = 0;
            }
        } else {
            if (buffer_index < DATA_BUFFER_SIZE) {
                data_buffer[buffer_index++] = received_data;
            }
        }
    }
}

uint8_t* GetDataBuffer(void) {
    return data_buffer;
}

uint16_t GetBufferDataLength(void) {
    return buffer_index;
}
6.3 主程序 (main.c)
#include "usart.h"
#include "protocol.h"

int main(void) {
    // 系统时钟配置
    SystemClock_Config();

    // 初始化串口
    USART_Init(9600);

    // 初始化协议处理模块
    Protocol_Init();

    while (1) {
        // 主循环可以添加其他任务
    }
}

7. 使用示例:快速上手的实践指南

7.1 初始化串口

main.c 中调用 USART_Init 函数,配置串口参数:

USART_Init(9600); // 配置波特率为9600
7.2 初始化协议处理模块

调用 Protocol_Init 函数,初始化协议处理模块:

Protocol_Init();
7.3 数据接收与处理

在中断服务函数 USART1_IRQHandler 中,接收到的数据会被存储到 data_buffer 中。主程序可以通过以下函数获取接收到的数据:

uint8_t* data = GetDataBuffer();
uint16_t data_length = GetBufferDataLength();

8. 状态图:直观理解程序运行机制

在这里插入图片描述

串口收发状态图

  • 系统初始化:程序开始时进行系统初始化,包括时钟配置、GPIO配置等。
  • 串口初始化:初始化完成后,进入串口初始化状态,配置串口参数。
  • 等待中断触发:串口配置完成后,进入等待中断触发状态,等待串口接收到数据后触发中断。
  • 接收数据:中断触发后,进入接收数据状态,从串口接收数据。
  • 检测数据头:接收到数据后,进入检测数据头状态,检查数据头的有效性。
  • 解析数据帧:如果数据头有效,进入解析数据帧状态,解析数据内容。
  • 存储数据:解析完成后,进入存储数据状态,将数据存储到缓冲区。
  • 数据处理:存储完成后,进入数据处理状态,对数据进行进一步处理。
  • 返回等待中断触发:数据处理完成后,返回等待中断触发状态,继续接收新的数据。
  • 错误处理:如果在数据处理过程中发生错误,返回系统初始化状态,重新初始化系统。
  • 数据头无效:如果检测到数据头无效,返回等待中断触发状态,重新开始接收数据。

9. 总结

通过本教程,读者可以掌握如何在STM32F407上实现一个在串口接收中断里即时解析数据头的特殊程序框架。程序通过中断方式接收数据,并在接收到数据头后立即解析后续数据内容,支持基本的错误检测和数据处理功能。教程内容从零开始,详细讲解了串口配置、数据接收、中断处理等关键步骤,并提供了完整的代码示例和注释,帮助读者快速上手。


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

相关文章:

  • 测试是如何跟进和管理 bug
  • Prompt Engineering for Large Language Models
  • 【C++学习篇】智能指针
  • 【C#】Clipboard中SetImage(BitmapSource image)的用法
  • Elasticsearch 限制索引大小与索引模板匹配冲突解决方案
  • 安装gcc8编译工具和centos7中的yum冲突,恢复原本yum
  • 集合遍历的多种方式
  • vulnhub靶场之【digitalworld.local系列】的JOY靶机
  • LeetCode hot 100 每日一题(3)--128. 最长连续序列
  • 鸿蒙中打开相机相册
  • Electron、Tauri及其它跨平台方案终极对比
  • 腾讯云 | 微搭低代码快速开发数据表单应用
  • C#里定义对象序列化保存的例子
  • 证明:曲线的可导点不能同时为极值点和拐点
  • Nest系列:从环境变量到工程化实践-2
  • 你了解 Java 线程池的原理吗?
  • Beyond Compare for mac v5.0.6.30713 文件对比利器 支持M、Intel芯片
  • 动态规划背包问题
  • Celia智能助手系统架构设计与技术实现全解析
  • 深入探索Python机器学习算法:模型评估