STM32裸机开发转FreeRTOS教程

目录

  • 1. 简介
  • 2. RTOS设置
    • (1)分配内存
    • (2)查看任务剩余空间
    • (3)使用osDelay
  • 3. 队列的使用
    • (1)创建队列
    • (1)直接传值和指针传值
    • (2)发送/接收等待时间
    • (3)不要在硬件中断发送队列
  • 4. 数据传递和共享
    • (1)尽量用全局常量代替函数指针传参
    • (2)同一资源需要被多个线程访问的两种方法
  • 5. 开发调试
    • (1)修改任务名称前备份代码,否则都会被删除
    • (2)keil的字体和编码,vscode的使用
    • (3)DMA串口日志
    • (4)文档放在项目文件夹外面,以免被cube删除
  • 6. LCD乱码问题

1. 简介

之前都是用CubeMX+Keil裸机开发STM32,最近第一次启用了FreeRTOS,用它可以实现多线程,但是如果写代码不严谨,单片机容易卡死,非常头疼。

2. RTOS设置

(1)分配内存

config parameters选项卡里,有个totoal heap size,意思大概是freertos占用的总内存,这个数值的默认值是比较小的,后面线程和队列加多了可能会不够,可以手动增加。我设置成的8kB,STM32f103rct6有48kB的RAM,是很充足的:
在这里插入图片描述
可以在heap usage里面看到使用情况,"still available"和"used"加起来正好是上面设置的总大小:
在这里插入图片描述
还有个minimal stack size参数,这个相当于一个底线,分配给每个任务的空间大小不能小于这个值。注意这个是用Word(字)作单位,32位单片机的一个字占4字节。
在这里插入图片描述
下图是设置任务的界面,每个任务默认给了128个字(半个kB)这个大小是比较适中的, 足够大部分常规任务的应用,也不会太占用单片机内存。
在这里插入图片描述
如果想节省内存,可以把前面minimal stack size设为64 Words(不允许更小了),然后把那些变量比较少的线程空间大小设置为64 Word。调试期间可以用随后介绍的方法查看线程空间够不够。

(2)查看任务剩余空间

为了用uxTaskGetStackHighWaterMark()查看任务剩余空间,需要在cubemx中开启它对应的使能,如下图。
在FreeRTOSConfig.h里面改会和cubemx冲突。
在这里插入图片描述

(3)使用osDelay

所有线程(除了IDLE)的死循环里面都需要至少加个osDelay(1),否则容易卡死。
在cmsis_os.c里查看osDelay的函数体,可见它本质上就是vTaskDelay:
在这里插入图片描述

3. 队列的使用

(1)创建队列

在Cube的Tasks和Queues选项卡,添加队列:
在这里插入图片描述
Queue Size是队列长度,设置的别让队列溢出就行,可以用osMessageAvailableSpace()查询队列剩余长度。
Item Size是每个元素的长度,这个后面会讲。

生成代码之后,cube会在freertos.c里创建一个队列句柄:

osMessageQId ledQueHandle;

cube里面设置的item size,代表每个队列数据占用多少字节。但由于c语言属于初级语言,不能给函数传递不定长度的参数,添加队列元素的函数是:

osStatus osMessagePut (osMessageQId queue_id, uint32_t info, uint32_t millisec)

它的第二个参数info,始终是uint32_t类型的,占4个字节。那如何传递不同长度的数据呢?答案就是“指针传值”。

如果要传递的数据可以用4个字节表示,就用“直接传值”方法,item size设为4;如果单次数据量超过了4字节,可以把数据放在数组或结构体里面,用指针传值方法,item size为被传递的数组或结构体的大小。

(1)直接传值和指针传值

直接传值示例:

// 发送线程
void Task_Send(void const *arg)
{...int cmd;for(;;){...osMessagePut(ledQueHandle, (uint32_t)cmd, osWaitForever);// 参数是int或float等数值...}
}
// 接收线程:
void Task_Receive(void const * argument)
{/* USER CODE BEGIN Task_LED */osEvent evt;int cmd;for(;;)	{evt = osMessageGet(ledQueHandle,0); if(evt.status==osEventMessage){cmd=(int)evt.value.v; ...}osDelay(1);		}
}

指针传值示例:

// 发送线程
void Task_Send(void const *arg)
{...int cmd[4];//传递数组,队列的item size = 16// MyStructType cmd;//传递结构体,需要预先定义MyStructType类型,队列的item size = sizeof(MyStructType)for(;;){...osMessagePut(ledQueHandle, (uint32_t)cmd, osWaitForever);//传递数组指针
//		osMessagePut(ledQueHandle, (uint32_t)&cmd, osWaitForever);//传递结构体指针...}
}
// 接收线程:
void Task_Receive(void const * argument)
{/* USER CODE BEGIN Task_LED */osEvent evt;int* pcmd;//接收指针,需要和发送的指针类型一致
//	MyStructType * pcmd;for(;;)	{evt = osMessageGet(ledQueHandle,0); if(evt.status==osEventMessage){pcmd = (int*)evt.value.p;//需要强制转型
//			pcmd = (MyStructType*)evt.value.p;			...}osDelay(1);}
}

由以上可见,直接传值就是把要传送数据直接放到队列里,接收的时候用evt.value.v;指针传值是把被传递数据的指针放在队列里,接收的时候用evt.value.p。

(2)发送/接收等待时间

osMessagePut()和osMessageGet()的最后一个参数都是等待时间,发送函数的可以设置成osWaitForever,表示阻塞线程直到把数据放入队列;
接收函数的等待时间最好设置为0,同时在循环里加个osDelay()释放主控资源。设置成osWaitForever会卡死。

(3)不要在硬件中断发送队列

cmsis_os.h开头注释有:
在这里插入图片描述
意思是osMessagePut可以放中断,但是经过实测,在硬件中断中调用osMessagePut()函数会卡死。
所以,只能在操作系统函数(线程,定时器)操作队列,中断函数传值可以用全局变量。

4. 数据传递和共享

(1)尽量用全局常量代替函数指针传参

用指针传递维度高、数据量大的变量,容易导致各种错误。可以定义成全局变量,在函数里直接用。
如果全局变量需要被多个文件调用,可以先在.c文件定义,再在.h文件用 extern 声明一下,这样其他的C文件只要#include这个.h文件就能用全局变量了。

(2)同一资源需要被多个线程访问的两种方法

①互斥锁:在读写函数里面,先获取Mutex,操作之后再释放Mutex。
②队列:其他线程请求压入队列,再由资源访问线程接收处理。如果是读取操作,可以在队列元素里放个接收变量的指针(没验证过!)
经过测试,即便是4字节的变量,也要避免不同线程直接访问,不然会出错。

5. 开发调试

(1)修改任务名称前备份代码,否则都会被删除

在cube里面修改任务名称和入口函数前千万记得备份代码,否则重新生成代码之后,之前写的代码都会被擦除。
在这里插入图片描述

(2)keil的字体和编码,vscode的使用

在菜单栏Edit最下面打开configuration窗口,设置编码和字体:
在这里插入图片描述
在这里插入图片描述
Editor选项卡里面,编码设置有两个选择:

①Courier字体方案(字体易读):编码改成UTF-8,这是为了适配Courier字体。同时为了让cube适配UTF-8,需要添加一个系统环境变量,变量名称:JAVA_TOOL_OPTIONS,变量值:-Dfile.encoding=UTF-8。如果不加环境变量,cube会把中文注释搞成乱码。
②Keil默认字体方案(较难阅读):保持GB2312编码,也不用设置全局变量了。

同时勾选右边的“Automatic reload of externally modified files”,避免每次都提示要不要重新加载:
在这里插入图片描述
如果选Courier字体方案,还需要在Colors & Fonts选项卡设置:
在这里插入图片描述
开发过程中,可以用vscode打开项目文件夹,在里面写代码,再在keil里面编译下载。VSC的代码辅助比Keil好多了,而且深色主题更护眼。

(3)DMA串口日志

启用日志打印串口的发送DMA可以最小的干预主程序的运行。方法是在cube里面添加一个tx的dma通道,DMA参数默认
在这里插入图片描述
在NVIC页面里面,可以把DMA的中断关上,因为日志打印要求不高,不需要在DMA终端里面判断数据有没有发送完:
在这里插入图片描述
代码里面,可以先定义个全局数组作为发送缓冲区,在函数里用sprintf格式化字符串,先调用DMAStop,再发送,不然只能发送一次:

char uart_buf[50]; // 日志发送缓冲区
void Timer_Callback() // 要发送日志的函数,例如软件定时器
{sprintf(uart_buf,"%.2f %.2f %.2f %.2f\r\n",Mot.spd_sv, Mot.spd_pv, Mot.pos_sv, Mot.pos_pv);HAL_UART_DMAStop(&huart3);HAL_UART_Transmit_DMA(&huart3,uart_buf,strlen(uart_buf));
}

这个方法适用于周期循环发送日志的情况,发送周期基本上大于一次发送用时就行了,偶尔一次数据覆盖也没关系。如果日志量比较大,可以提高串口波特率。

(4)文档放在项目文件夹外面,以免被cube删除

如果要在项目里新建一个文件夹用来放文档,需要用全英文,避免特殊符号,以防被cube搞坏。或者把文档放项目文件夹外面。

6. LCD乱码问题

调试期间发现写入数据到芯片内部Flash之后,显示屏会出现字符错误。
解决方法是把把Flash写入地址往后移,从0x0800A000移到0x0800B000后,问题就消失了。
应该是代码地址和参数写入地址冲突了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/503827.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用高云小蜜蜂GW1N-2实现MIPI到LVDS(DVP)转换案例分享

作者:Hello,Panda 大家晚上好,熊猫君又来了。 今天要分享的是一个简单的MIPI到LVDS(DVP)接口转换的案例。目的就是要把低成本FPGA的应用潜力充分利用起来。 一、应用背景 这个案例的应用背景是:现在还在…

Express 加 sqlite3 写一个简单博客

例图: 搭建 命令: 前提已装好node.js 开始创建项目结构 npm init -y package.json:{"name": "ex01","version": "1.0.0","main": "index.js","scripts": {"test": &q…

git撤回提交、删除远端某版本、合并指定版本的更改

撤回提交 vscode的举例 一、只提交了还未推送的情况下 1.撤回最后一次提交,把最后一次提交的更改放到暂存区 git reset --soft HEAD~12.撤回最后一次提交,把最后一次提交的更改放到工作区 git reset --mixed HEAD~13.撤回最后一次提交,不…

[Linux]redis5.0.x升级至7.x完整操作流程

1. 从官网下载最新版redis: 官网地址:https://redis.io/download 注:下载需要的登录,如果选择使用github账号登录,那么需要提前在github账号中取消勾选“Keep my email addresses private”(隐藏我的邮箱…

xss-labs关卡记录15-20关

十五关 随便传一个参数,然后右击查看源码发现,这里有一个陌生的东西,就是ng-include。这里就是: ng-include指令就是文件包涵的意思,用来包涵外部的html文件,如果包涵的内容是地址,需要加引号。…

数据库回滚:大祸临头时

原文地址 什么是数据库回滚? 数据库技术中,回滚是通过撤销对数据库所做的一项或多项更改,将数据库返回到先前状态的操作。它是维护数据完整性和从错误中恢复的重要机制。 什么时候需要数据库回滚? 数据库回滚在以下几个场景中很…

年会抽奖Html

在这里插入图片描述 <!-- <video id"backgroundMusic" src"file:///D:/background.mp3" loop autoplay></video> --> <divstyle"width: 290px; height: 580px; margin-left: 20px; margin-top: 20px; background: url(D:/nianhu…

基于FPGA的出租车里程时间计费器

基于FPGA的出租车里程时间计费器 功能描述一、系统框图二、verilog代码里程增加模块时间增加模块计算价格模块上板视频演示 总结 功能描述 &#xff08;1&#xff09;&#xff1b;里程计费功能&#xff1a;3公里以内起步价8元&#xff0c;超过3公里后每公里2元&#xff0c;其中…

nginx-链路追踪(trace)实现

一. 需求场景&#xff1a; 在日常运维工作中&#xff0c;会经常遇到在有多重调用链的场景下&#xff0c;如请求遇到非致命error时&#xff0c;在各环节的定位会非常麻烦&#xff0c;举个例子&#xff1a;比如说&#xff0c;在一个有多重调用链的服务环境下&#xff0c;一个请求…

c#使用SevenZipSharp实现压缩文件和目录

封装了一个类&#xff0c;方便使用SevenZipSharp&#xff0c;支持加入进度显示事件。 双重加密压缩工具范例&#xff1a; using SevenZip; using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Text; using System.Threading.…

MySQL和Hive中的行转列、列转行

水善利万物而不争&#xff0c;处众人之所恶&#xff0c;故几于道&#x1f4a6; 文章目录 MySQL1.行转列2.列转行 Hive1.行转列2.列转行(1)侧窗(2)union MySQL 1.行转列 把多行转成列。直接group&#xff0c;sum(if()) 2.列转行 Hive 1.行转列 select name,sum(if(kmshuxu…

快速上手:采用Let‘sEncrypt免费SSL证书配置网站Https (示例环境:Centos7.9+Nginx+Let‘sEncrypt)

1 关于Let’s Encrypt与Cerbot DNS验证 Let’s Encrypt 是一个提供 免费证书 的 认证机构。 Cerbot 是 Let’s Encrypt 提供的一个工具&#xff0c;用于自动化生成、验证和续订证书。 DNS验证是 Cerbot 支持的验证方式之一。相比 HTTP 验证或 TLS-ALPN 验证&#xff0c;DNS …

【Unity3D】Text文本文字掉落效果

相关技术&#xff1a;Text、TextMesh、Rigidbody&#xff08;刚体&#xff09;、BoxCollider&#xff08;碰撞体&#xff09;、TextGenerator、文本网格、文字网格 原理&#xff1a;使用UGUI Text获取其文字的每个字符网格坐标&#xff0c;转世界坐标生成对应的3D文本(TextMesh…

flutter 专题二十四 Flutter性能优化在携程酒店的实践

Flutter性能优化在携程酒店的实践 一 、前言 携程酒店业务使用Flutter技术开发的时间快接近两年&#xff0c;这期间有列表页、详情页、相册页等页面使用了Flutter技术栈进行了跨平台整合&#xff0c;大大提高了研发效率。在开发过程中&#xff0c;也遇到了一些性能相关问题和…

设计模式 行为型 命令模式(Command Pattern)与 常见技术框架应用 解析

命令模式&#xff08;Command Pattern&#xff09;是一种行为型设计模式&#xff0c;它旨在将请求发送者和接收者解耦&#xff0c;通过将一个请求封装为一个对象&#xff0c;从而允许参数化客户端对象以进行不同的请求、排队请求或记录请求&#xff0c;并支持可撤销操作。 在软…

NodeLocal DNS 全攻略:从原理到应用实践

文章目录 一、NodeLocal DNS是什么&#xff1f;二、为什么使用NodeLocal DNS&#xff1f;三、工作原理架构图四、安装NodeLocal DNS五、在应用中使用NodeLocal DNSCache六、验证 一、NodeLocal DNS是什么&#xff1f; NodeLocal DNSCache 通过在集群节点上运行一个 DaemonSet …

jenkins入门12-- 权限管理

Jenkins的权限管理 由于jenkins默认的权限管理体系不支持用户组或角色的配置&#xff0c;因此需要安装第三发插件来支持角色的配置&#xff0c;我们使用Role-based Authorization Strategy 插件 只有项目读权限 只有某个项目执行权限

STM32-笔记34-4G遥控灯

4G接线 一、项目需求 服务器通过4G模块远程遥控开关灯。 二、项目实现 复制项目文件夹38-wifi控制风扇项目 重命名为39-4G遥控点灯 打开项目文件 加载文件 main.c #include "sys.h" #include "delay.h" #include "led.h" #include "ua…

STM32——系统滴答定时器(SysTick寄存器详解)

文章目录 1.SysTick简介2.工作原理3.SysTick寄存器4.代码延时逻辑5.附上整体代码6.一些重要解释 1.SysTick简介 Cortex-M处理器内集成了一个小型的名为SysTick(系统节拍)的定时器,它属于NVIC的一部分,且可以产生 SysTick异常(异常类型#15)。SysTick为简单的向下计数的24位计数…

数据库模型全解析:从文档存储到搜索引擎

目录 前言1. 文档存储&#xff08;Document Store&#xff09;1.1 概念与特点1.2 典型应用1.3 代表性数据库 2. 图数据库&#xff08;Graph DBMS&#xff09;2.1 概念与特点2.2 典型应用2.3 代表性数据库 3. 原生 XML 数据库&#xff08;Native XML DBMS&#xff09;3.1 概念与…