STM32CubeMX在FreeRTOS下使用串口进行数据收发(不定长度)

2023-05-16

STM32CubeMX->FreeRTOS+USART接收不定长数据

由于本人做的一个项目功能相对复杂,要求使用操作系统,且项目工程中有很多需要串口操作的外设,所以需要对串口设计不定长的收发功能,裸机跑惯了的孩子就是比较野,一天瞎吉尔弄,现在是不是GG,这里先批评一下自己。为了解决这一问题并做个笔记,就有了这篇博客。

实现思路:
​ 使用串口、定时器记录接收到的数据,每5ms进入一次处理函数(一条数据处理完毕)
​ 使用一个计数位、一个标志位、一个暂存buff
​ 标志位置1则认为数据处理完成,将数据打印出来
使用硬件功能:串口3及中断,定时器7及中断
使用操作系统元素:一个互斥量
使能RTOS后会RTOS会强制调用系统时钟,所以在使能了操作系统之后需要将系统时钟换为定时器TIM1

操作步骤

  1. 启动STM32CubeMX并对芯片选型
    主要设置

    1. 串口3及中断使能,115200,8,None,1
    2. 定时器7及中断使能,预分频9000-1,自动重载50-1(时钟树APB1、2为90MHz)
    3. SYS时钟源设为定时器1
    4. 使能FreeRTOS,创建1个串口任务、创建1个互斥量
    5. 使能两个GPIO输出驱动LED
      以上都清楚的大佬转到代码功能实现继续
  2. 使能RCC时钟源为外部时钟


3. 设置SYS调试方式为串口,时钟源改为定时器1


4. 设置时钟树,是APB1、2为90MHz,或者自行设定,在后面定时器工作会用到

5. 使能串口3,并启用串口3中断,串口3设置为115200,8Bit,None,1(不知道什么意思的就没有必要看下去了)

在这里插入图片描述
6. 使能定时器7,并使能中断,定时器预分频系数与之前的90MHz相关,需要通过预分频系数和自动重载值将定时时间设为5ms【200Hz】,不会算的看我定时器博客(这里我用了一个不常用的定时器,用哪个都可以,只要注意好定时器隶属哪根时钟线就行)

在这里插入图片描述
7. 使能FreeRTOS并创建一个串口任务和一个二值量(二值信号量用于串口数据接收完成后进入串口任务)

在这里插入图片描述
在这里插入图片描述
8. 设置两个IO输出作为一个可视化依据

在这里插入图片描述
9. 项目命名、选择使用的程序编辑软件我的是MDK_ARM V5、分离.c.h文件,点击右上角GENERATE CODE生成工程
在这里插入图片描述

  1. 打开项目,先编译没有错误之后再进行修改

代码功能实现

  1. 首先在usart.c文件对串口3进行printf函数重定义,使串口3可用printf函数输出

    内容加入到/* USER CODE BEGIN 0 */框架内

    /* USER CODE BEGIN 0 */
    /********************************************************************************/
    #include "stdio.h"
    //加入以下代码,支持printf函数,而不需要选择use MicroLIB	  
    //#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)	
    #if 1
    #pragma import(__use_no_semihosting)             
    //标准库需要的支持函数                 
    struct __FILE 
    { 
    	int handle; 
    }; 
    
    FILE __stdout;       
    //定义_sys_exit()以避免使用半主机模式    
    void _sys_exit(int x) 
    { 
    	x = x; 
    } 
    //重定义fputc函数 
    int fputc(int ch, FILE *f)
    { 	
    	while((USART3->SR&0X40)==0);//循环发送,直到发送完毕   
    	USART3->DR = (uint8_t) ch;      
    	return ch;
    }
    #endif 
    /********************************************************************************/
    /* USER CODE END 0 */
    
  2. 定义串口使用的变量

    主要包括串口数据存储BUFF,串口数据长度计数BUFF,串口接收成功标志位(此处借鉴正点原子的串口处理方法)

    在main.c文件下的定义/* USER CODE BEGIN PV */框架内定义

    /* USER CODE BEGIN PV */
    /********************************************************************************/
    uint8_t RxBuffer[MAX_REC_LENGTH] = {0};		//串口数据存储BUFF		长度2048
    uint8_t RxFlag = 0;							//串口接收完成标志符
    uint16_t RxCounter = 0;						//串口长度计数
    uint8_t RxTemp[REC_LENGTH] = {0};			//串口数据接收暂存BUFF	长度1
    
    extern osSemaphoreId myBinarySem01Handle;	//操作系统定义的互斥量
    /********************************************************************************/
    /* USER CODE END PV */
    

    并在usart.h文件下的/* USER CODE BEGIN Private defines */框架下定义长度和链接串口用变量

    /* USER CODE BEGIN Private defines */
    /********************************************************************************/
    #define REC_LENGTH 1
    #define MAX_REC_LENGTH 2048
    
    extern uint8_t RxBuffer[MAX_REC_LENGTH];
    extern uint8_t RxFlag;
    extern uint16_t RxCounter;
    extern uint8_t RxTemp[REC_LENGTH];
    /********************************************************************************/	 
    /* USER CODE END Private defines */
    
  3. 串口中断回调函数,在main.c的/* USER CODE BEGIN 4 */框架下添加串口回调函数

    /********************************************************************************/
    void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)//串口3接收完成回调函数
    {
    	if(huart->Instance == USART3)
    	{
    		__HAL_TIM_SET_COUNTER(&htim7,0);							//清除定时器7计数值	
    		if(0 == RxCounter)											//如果是首字符(每帧数据开头)则开启定时器
    		{
    			__HAL_TIM_CLEAR_FLAG(&htim7, TIM_FLAG_UPDATE);			//清除中断标志位
    			HAL_TIM_Base_Start_IT(&htim7);							//开启基本定时器
    		}
    		RxBuffer[RxCounter] = RxTemp[0];							//缓存数据放入接收数组
    		RxCounter++;												//计数器加1
    		HAL_UART_Receive_IT(&huart3,(uint8_t *)RxTemp, REC_LENGTH);	//重新使能中断
    	}
    }
    /********************************************************************************/
    
  4. 定时器周期回调函数,由于在设置中系统时钟使用了定时器1,所以在main.c文件存在

    void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
    

    在函数 /* USER CODE BEGIN Callback 1 */框架下添加定时器7的回调函数

      /* USER CODE BEGIN Callback 1 */
    	/********************************************************************************/
    	if(htim->Instance == TIM7)
    	{
    		HAL_GPIO_TogglePin(DS1_GPIO_Port, DS1_Pin);	//LED反转一次
    		RxFlag = 1;																	//接收标志位置1
    		HAL_TIM_Base_Stop_IT(&htim7);								//关闭定时器
    		osSemaphoreRelease(myBinarySem01Handle);		//释放二值信号量,进入串口任务
    	}
    	/********************************************************************************/
      /* USER CODE END Callback 1 */
    
  5. 对freeRTOS添加任务

    首先在freertos.c文件的/* USER CODE BEGIN Includes */ 框架下添加串口使用的头文件

    /* USER CODE BEGIN Includes */     
    /********************************************************************************/
    #include "usart.h"
    #include "stdio.h"
    /********************************************************************************/
    /* USER CODE END Includes */
    
  6. 在freertos主任务中启动串口中断以及添加LED闪烁任务

    /* USER CODE END Header_StartDefaultTask */
    void StartDefaultTask(void const * argument)
    {
      /* USER CODE BEGIN StartDefaultTask */
    	HAL_UART_Receive_IT(&huart3,(uint8_t *)RxTemp, REC_LENGTH);
      /* Infinite loop */
      for(;;)
      {
    	HAL_GPIO_TogglePin(DS0_GPIO_Port,DS0_Pin);
        osDelay(200);
      }
      /* USER CODE END StartDefaultTask */
    }
    
  7. 在freertos串口子任务中添加串口数据打印功能

    /* USER CODE END Header_Start_U3_Task */
    void Start_U3_Task(void const * argument)
    {
      /* USER CODE BEGIN Start_U3_Task */
      /* Infinite loop */
      for(;;)
      {
    	/********************************************************************************/  
    	osSemaphoreWait(myBinarySem01Handle,100);						//等待二值信号量
    	if(RxFlag == 1)													//数据接收完成
    	{
    		for(int i = 0; i<RxCounter; i++)							//打印接收数组存储的内容
    			printf("%c",RxBuffer[i]);	
             printf("\r\n");											//打印完成换行
    		RxFlag = 0;													//接收标志清零
    		RxCounter = 0;												//接收计数清零
    		memset(RxBuffer ,0, MAX_REC_LENGTH);						//清空接收数组
    	}
    	/********************************************************************************/
        osDelay(1);
      }
      /* USER CODE END Start_U3_Task */
    }
    

实验结果

编译下载程序,打开串口调试助手,用串口转USB模块将串口3接入电脑,使用串口调试助手测试
在这里插入图片描述
上图为测试结果,发送ASWaterbenben后回复相同内容,每帧数据用换行符隔开,至此,实验成功!

DS0灯200ms反转一次,每次接收到一帧完整数据DS1灯就会反转一次。这个效果难以展示,你们自己试试就好!

如果本博客对您有帮助,希望点赞关注走一波,今后会看心情更新!哈哈哈哈哈哈哈哈哈

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

STM32CubeMX在FreeRTOS下使用串口进行数据收发(不定长度) 的相关文章

  • FreeRTOS打印任务对CPU的占有率

    1 配置RTOS 1 打开RTOS Config Parameter 找到Run Time And Task States gathering related definitions 使能GENERATE RUN TIME STATS US
  • freeRTOS使用uxTaskGetStackHighWaterMark函数查看任务堆栈空间的使用情况

    摘要 每个任务都有自己的堆栈 堆栈的总大小在创建任务的时候就确定了 此函数用于检查任务从创建好到现在的历史剩余最小值 这个值越小说明任务堆栈溢出的可能性就越大 FreeRTOS 把这个历史剩余最小值叫做 高水位线 此函数相对来说会多耗费一点
  • Error: L6218E: Undefined symbol vApplicationGetIdleTaskMemory (referred from tasks.o).

    我用的是F103ZET6的板子 移植成功后 编译出现两个错误是关于stm32f10x it c 里 void SVC Handler void void PendSV Handler void 两个函数的占用问题 随后编译出现以下两个问题
  • STM32CubeMX配置GPIO外部中断

    前言 用PA0来检测按键的输入信号 当按键按下时会由低电平变为高电平 1 配置RCC时钟 将RCC的High Speed Clock HSE 配置为Crystal Ceramic Resonator 将主频设置为72MHz 2 配置GPIO
  • FreeRTOS之软件定时器

    FreeRTOS之软件定时器 声明 本人按照正点原子的FreeRTOS例程进行学习的 欢迎各位大佬指责和批评 谢谢 include sys h include delay h include usart h include led h in
  • FreeRTOS之事件

    FreeRTOS之事件 声明 本人按照正点原子的FreeRTOS例程进行学习的 欢迎各位大佬指责和批评 谢谢 一 事件定义 事件 事件集 与高数上的集合意义差不多 事件啊 其实是实现任务间通信的机制 主要用于实现多任务间的同步 但是事件类型
  • 啊哈C的简单使用

    打开啊哈C 新建一个程序输出hello world include
  • Arduino IDE将FreeRTOS用于STM32

    介绍 适用于STM32F103C8的FreeRTOS STM32F103C是一种能够使用FreeRTOS的ARM Cortex M3处理器 我们直接在Arduino IDE中开始使用STM32F103C8的FreeRTOS 我们也可以使用K
  • FreeRTOS学习笔记(8)---- 软件定时器

    使用FreeRTOS软件定时器需要在文件FreeRTOSConfig h先做如下配置 1 configUSE TIMERS 使能软件定时器 2 configTIMER TASK PRIORITY 定时器任务优先级 3 configTIMER
  • [FreeRTOS入门学习笔记]定时器

    定时器的使用步骤 1 定义一个handle xTimerCreate创建 2 启动定时器 在Task1中调用 通过队列通知守护任务来执行定时器任务 要再config头文件中定义守护任务相关配置 虽然定时器是在task1中启动 但是定时器的任
  • FreeRTOS学习---“定时器”篇

    总目录 FreeRTOS学习 任务 篇 FreeRTOS学习 消息队列 篇 FreeRTOS学习 信号量 篇 FreeRTOS学习 事件组 篇 FreeRTOS学习 定时器 篇 FreeRTOS提供了一种软件定时器 用来快速实现一些周期性的
  • RT-Thread记录(五、RT-Thread 临界区保护与FreeRTOS的比较)

    本文聊聊临界区 以及RT Thread对临界区的处理 通过源码分析一下 RT Thread 对临界区保护的实现以及与 FreeRTOS 处理的不同 目录 前言 一 临界区 1 1 什么是临界区 1 2 RTOS中的临界区 二 RT Thre
  • warning: #940-D: missing return statement at end of non-void function “fgetc“解决方案

    问题描述 warning 940 D missing return statement at end of non void function fgetc 解决方案 解决措施 引入头文件stdio h
  • 【FreeRTOS 事件】任务通知事件

    普通任务通知事件创建创建及运行 参阅安富莱电子demo define BIT 0 1 lt lt 0 define BIT 1 1 lt lt 1 static TaskHandle t xHandleTaskUserIF NULL sta
  • freeRTOS出现任务卡死的情况。

    最近在做一个产品二代升级的项目 代码是上一任工程师留下的 很多BUG 而且融合了HAL库和LL库 以及github上下载的GSM源码 很不好用 我这边是将2G模块换成了4G 且添加了单独的BLE模块 因此只在源码的基础上 去除2G和BLE代
  • 【STM32CubeMX】位置式PID调节控制输出电压(超详解)

    本文将借助STM32CubeMX来配置ADC DMA DAC USART 并利用PID位置式算法实现对输出电压进行AD采集通过PID算法调节DAC 获取到我们想要的电压值 讲解的主要知识 何为PID以及为何需要PID STM32CubeMX
  • STM32CubeMX学习六 之ADC配置

    文章目录 前言 一 本地环境 二 开始 1 定时器配置 2 引脚配置 在这里插入图片描述 https img blog csdnimg cn e5b6f155a1b8468cb15046a0a9d031cd png 3 内部时钟配置 4 A
  • 当一个任务写入变量而其他任务读取该变量时,我们是否需要信号量?

    我正在研究 freeRtos 并且我有一个名为 x 的变量 现在 每秒只有一个任务正在写入该变量 而其他任务正在读取该变量值 我需要用互斥锁来保护变量吗 如果变量为 32 位或更小 并且其值是独立的并且不与任何其他变量一起解释 则不需要互斥
  • VS Code 有没有办法导入 Makefile 项目?

    正如标题所说 我可以从现有的 Makefile 自动填充 c cpp properties json 吗 Edit 对于其他尝试导入 makefile 的人 我找到了一组脚本 它们完全可以实现我想要实现的目标 即通过 VS Code 管理
  • 小型 ARM 微控制器的 RTOS 内核之间的可量化差异 [关闭]

    Closed 这个问题是基于意见的 help closed questions 目前不接受答案 有许多不同的 RTOS 可用于微控制器 我专门寻找支持 ARM Cortex M 处理器的 RTOS 另外 我对闭源解决方案不感兴趣 试图从网站

随机推荐