STM32CubeMX 生成的代码框架解析 - 最小化工程篇(含FreeRTOS)

文章目录

  • 前言
  • 1. 创建工程说明
  • 2. 目录结构介绍
    • 2.1 总目录
    • 2.2 Core文件夹
    • 2.3 Drive文件夹
    • 2.4 MKD-ARM
    • 2.5 Middlewares
    • 2.6 IDE配置文件
  • 3. 重点文件介绍
    • 3.1 Core文件夹
      • 3.1.1 src
        • freertos.c
        • main.c
        • stm32l4xx_hal_msp.c
        • stm32l4xx_it.c
        • system_stm32l4xx.c
      • 3.1. 2 include
        • FreeRTOSConfig.h
    • 3.2 Drive
    • 3.3 Middlewares
      • 3.3.1 CMSIS_RTOS_V2
        • cmsis_os.h
        • cmsis_os2.h/cmsis_os2.c
        • freertos_mpool.h
        • freertos_os2.h
      • 3.3.2 FreeRTOS
        • include
        • MemMang
        • src


前言

最近因为工作中要用到STM32+FreeRTOS进行开发,因此借助此次机会学习下STM32CubeMX生成的带有FreeRTOS的工程代码。熟悉下其生成代码的结构,以及一些细节。

这篇的话我想首先分析下,最简单的结构是什么样的,即所有的都按照默认来,不增加外设驱动,不对FreeRTOS进行过多的配置,我称之为最小化工程。

后面再通过写一些增加驱动、FreeRTOS的各项配置等来进行对比,进一步加强理解。

1. 创建工程说明

基于STM32L451的芯片进行生成的
在这里插入图片描述
这里选择只复制用到了的库文件,这样整体生成的会相对小一些,没有用到的就不会被copy出来

在这里插入图片描述
FreeRTOS采用CMSIS_V2的接口,这里我理解其实就是对FreeRTOS的封装,因为该软件除了FreeRTOS还支持其它的RTOS,为了生成代码风格统一,所以特制定了一套统一的对RTOS的封装吧。

在这里插入图片描述
如上图所示,硬件驱动相关的操作接口几乎也米有

2. 目录结构介绍

2.1 总目录

在这里插入图片描述
文件夹
总目录下分别有4个文件夹Core、Drivers、MDK-ARM、Middlewares

IDE配置文件
其中两个文件.mxproject和stm32L451_minimal.ioc都和STM32CubeMX这个工具有关

2.2 Core文件夹

在这里插入图片描述
STM32CubeMX 根据配置自动生成的核心代码,和应用相关的后续应该都在这里了。

在这里插入图片描述
我们打开.mxproject这个cubeMX的配置文件,可以看到Core目录下的内容放在
[PreviousGenFiles] 下,这说明该目录里面的内容应该就是根据我们在cubeMX上做的操作生成的。

后续我们可以关注下,是不是我们增加和减少了什么配置后是在这个目录下的文件中体现。

2.3 Drive文件夹

该文件夹存在的主要是对芯片的外设操作接口。

其总共分为两块,分别是CMSIS(处理器封装接口)和HAL(外设驱动的封装接口)

实际上两者的关系应该是HAL调用了CMSIS的接口,可以理解HAL是CMSIS上面的一层。
CMSIS实现

Drivers\CMSIS
Drivers\CMSIS\Device\ST\STM32L4xx\Include
Drivers\CMSIS\Include

CMSIS(Cortex Microcontroller Software Interface Standard)是一种为Cortex-M处理器系列定义一致接口的标准,旨在简化嵌入式软件开发。

这个文件夹包含了针对STMicroelectronics的STM32L4xx系列微控制器的CMSIS设备文件。这些文件包括了特定芯片的寄存器定义、中断向量表等信息,为开发者提供了在特定芯片上进行低层次编程的支持。
在这里插入图片描述

STM32 HAL库

Drivers\STM32L4xx_HAL_Driver
Drivers\STM32L4xx_HAL_Driver\Inc
Drivers\STM32L4xx_HAL_Driver\Src

这里就是放置HAL的驱动了
在这里插入图片描述

2.4 MKD-ARM

在这里插入图片描述
就是keil5的工程文件

2.5 Middlewares

该文件夹中主要放置通过STM32CubeMX生成的一些中间件,即和STM32本身没啥关系的,例如一些RTOS和通信协议等等,cubeMX上集成这些中间件主要也是想简化大家的开发流程。(例如这里FreeRTOS都不用去官网下载了)

我们这里主要用到了FreeRTOS。

在这里插入图片描述

上面我们有说到,middlewares中不仅支持FreeRTOS还有其它的RTOS,那么为了方便开发者能够无缝随时切换不同的RTOS进行开发,就需要根据STM32的一些常用方式形成一套通用的RTOS操作接口。

这样无论底层的RTOS用的是FreeRTOS还是Rt-Thread,我们只需要了解cmsis_os2.h这套接口怎么用就行了。

同时通过调用cmsis中的接口,去间接调用RTOS,还能够做到切换RTOS时,上层应用无需改动,从而大大降低开发的工作量。

2.6 IDE配置文件

.mxproject
STM32CubeMX 根据配置自动生成的核心代码,和应用相关的后续应该都在这里了。

在这里插入图片描述
我们打开.mxproject这个cubeMX的配置文件,可以看到Core目录下的内容放在
[PreviousGenFiles] 下,这说明该目录里面的内容应该就是根据我们在cubeMX上做的操作生成的。

后续我们可以关注下,是不是我们增加和减少了什么配置后是在这个目录下的文件中体现。

stm32L451_minimal.ioc
stm32L451_minimal.ioc文件包含了使用STM32CubeMX进行配置时的所有设置信息,包括引脚分配、时钟配置、外设初始化等内容。

3. 重点文件介绍

3.1 Core文件夹

作为核心文件来说,core下的每个文件可以说都很重要了。

3.1.1 src

我们先看一下src下面有哪些东西

freertos.c

在这里插入图片描述
该文件中应该主要存放应用利用FreeRTOS编写的核心业务代码

这个文件里面当前除了这些注释以外就没什么了,不知道的以为这是个头文件。

但实际上我们从往下看。

Private includes  - 添加自己所需要的头文件
Private typedef  - 定义自己所需要的typedef
....
Private function prototypes  -- 私有函数接口声明,也就是staic的这种要在开头声明下
Private application code -- 编写自己的应用代码

从上述信息中我们可以了解到后续应用代码可能会主要放在这里,又或者说操作FreeRTOS 的配置文件生成的task、timer、信号量等等应该都会在这个文件中产生出来

main.c

该文件中作为应用的起始文件,主要是实现了main()函数。
该函数中对驱动、时钟、os分别进行了初始化,并将os启动起来。

头文件
在这里插入图片描述这里我们看到当调用FreeRTOS时,他不是直接调用FreeRTOS相关的.h,而是直接调用了cmsis_os.h,而这个cmsis_os.h中又包含了cmsis_os2.h。

同时这里还定义了一个OS中task 属性配置的结构体,用于后续生成task。

不过令我感到奇怪的是,为什么task这些不是放到freertos.c中而是放到了main.c中呢?

初始化
在这里插入图片描述
该文件中作为应用的起始文件,主要是实现了main()函数。
该函数中对驱动、时钟、os分别进行了初始化,并将os启动起来。

stm32l4xx_hal_msp.c

时钟相关的操作
在这里插入图片描述

stm32l4xx_it.c

中断函数处理
在这里插入图片描述
中断相关的处理操作,这个里面已经透除了很多的中断处理接口,后续应该我们只需要根据不同的中断使用方式,填入我们自己的处理逻辑即可。

滴答定时器

这里其实有一个很重要的时钟中断,就是设备硬件滴答中断。
为什么说他重要呢?大家可以将其理解为是我们的钟表,钟表帮助我们计时,然后我们在此基础上我们才有了年月日、上下班时间等等。
因此它是一切的基础。

在OS端,在执行不同线程之间的切换时,一般都会采用时间片和优先级的方法,其中时间片的计算就依赖于该定时器。

因该定时器产生的速度很快很频繁,一般情况下1ms一次,所以我们不要在该定时器调用的函数中做什么大量业务的处理。否则会导致整个软件的功能不正常

system_stm32l4xx.c

实现外设的.c文件
在这里插入图片描述

3.1. 2 include

FreeRTOSConfig.h

在这里插入图片描述
一些参数配置,像是系统时钟,堆的大小,taks名称的长度
另外还有是否开启某些功能

3.2 Drive

这部分上面目录结构中已经说过了,这里就没有什么好说的了,主要就是
CMSIS-M4 -> HAL ->应用

3.3 Middlewares

上面目录结构介绍中,我们已经说到为了能够兼容不同的RTOS,ST官网定义了一套自己的CMSIS接口,然后不同的操作系统去适配这套接口,应用则是直接调用这套接口。

3.3.1 CMSIS_RTOS_V2

cmsis_os.h

这是FreeRTOS的头文件
应用层调用FreeRTOS的接口时,只需要将该.h包含进去即可。
它里面

#include "FreeRTOS.h"
#include "task.h"
#include "cmsis_os2.h"

在这里插入图片描述
它里面包含了很多CMSIS FreeRTOS封装的接口

在这里插入图片描述

但是我们看他的注释中有很多的这种接口说明之类的,因此我认为该文件其实更像是一个 CMSIS FreeRTOS的接口兼容层。
里面这里面提到的“replaced osPoolCreate with osMemoryPoolNew”,我们实际查看可以看到osPoolCreatecmsis_os.h
osMemoryPoolNewcmsis_os2.h中,且cmsis_os2.c中也只有osMemoryPoolNew的实现。

  • 那么为什么没有osPoolCreate的实现还保留了该接口呢?

    我认为就是为了兼容,比如这套接口我之前定义过了,他要兼容各个时期的实现,例如cmsis_os1.c中可能有该实现,且该接口也在被使用着,那么你不能让后续再生成代码的人发现重新生成后原先的接口不能用了,大面积报错要修改吧。 为了保证cmsis_os2.h 和cmsis_os1.h 中的接口都各自对应着自己的.c 实现,而不相互影响,又要兼容分别在使用不同接口的用户,所以cmsis_os.h就负责整合他两个的接口了。

cmsis_os2.h/cmsis_os2.c

ST CMSIS RTOS V2版本接口的定义和实现,其中.h中是接口,.c中是如何使用FreeRTOS实现这套接口的。

.h中的接口
在这里插入图片描述
.c中的实现
在这里插入图片描述

freertos_mpool.h

这个文件中主要声明了内存池相关的宏定义和结构体
在这里插入图片描述
然后我们可以看到这个结构体的使用并非是应用,而是cmsis_os2.c,在该.c中用于实现
内存池相关的接口。

freertos_os2.h

在这里插入图片描述
我们来翻译下上面框起来的这段信息。

“CMSIS-RTOS2函数osDelay使用FreeRTOS函数vTaskDelay。万一
osDelay没有在应用程序图像中使用,编译器会优化它。 设置#define INCLUDE_vTaskDelay 1来修复这个错误。”

在这里插入图片描述
我们再搜索一下INCLUDE_vTaskDelay 这个宏定义,通过上面的内容我们可以看到INCLUDE_vTaskDelay的赋值是在
FreeRTOS的配置文件中兼容的,那么再结合上面freertos_os2.h中的注释解释,我们大概就明白了。

ST借助FreeRTOS去实现cmsis_os2.h中的接口时,用了很多FreeRTOS上的功能,而这些功能需要FreeRTOS去开启,如果关闭了,那么可想而知cmsisi_os2.c中的实现肯定就有问题。

但是ST也控制不住一些人去修改FreeRTOS的配置,那么怎么做呢,它就通过该文件去检查自己所需要的FreeRTOS必要功能是否有开启,如果没开启那么就在编译时给出错误提示。或者直接让编译器进行修改。(这块我还不是很确定,我更加倾向于给出错误提示)

3.3.2 FreeRTOS

include

FreeRTOS.h
FreeRTOS的配置文件
在这里插入图片描述
这里好像还有一些扩展的结构体
在这里插入图片描述

MemMang

在 FreeRTOS 中,heap_1.c 到 heap_4.c 是 FreeRTOS 提供的不同内存分配算法的实现,用于动态内存分配。这些文件实现了 FreeRTOS 内存分配所需的函数,每个文件对应不同的内存分配策略,具体说明如下:

heap_1.c:
使用最简单的内存分配策略,所有任务共享一个固定大小的内存池,不支持内存块的释放(只能分配,不能释放)。

heap_2.c:
也是一个简单的内存分配策略,但允许内存块的释放,适合于仅需要少量的内存管理功能的系统。

heap_3.c:
实现了一个更复杂的内存分配算法,通过在运行时动态合并和分割内存块以提高内存利用率。

heap_4.c:
提供了一种高效的内存分配算法,使用了类似于二进制伙伴系统的方法来管理内存池,适用于需要高效内存管理的系统。

我们当前使用的是heap_4.c

src

这里面就是一些 list/task/timer…的实现了,这些我觉得等介绍FreeRTOS的时候再详细介绍,作为使用FreeRTOS来说的话我们最好还是先去熟悉FreeRTOS提供的这些接口,然后根据这些接口的使用方式去探索其实现。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

java 高级面试题(借鉴)(下)

雪花算法原理 第1位符号位固定为0,41位时间戳,10位workId,12位序列号,位数可以有不同实现。 优点:每个毫秒值包含的ID值很多,不够可以变动位数来增加,性能佳(依赖workId的实现…

挺后悔,我敷衍地回答了“程序员如何提升抽象思维“

分享是最有效的学习方式。 博客:https://blog.ktdaddy.com/ 大家好,我是老猫。 大概在月初的时候,我发了一篇文章【当程序员之后?(真心话)】,在这篇文章中,提及了抽象思维对一名程序员的重要性。可能说得也比较笼统&a…

UML 类关系表示

类之间关系:依赖、泛化(继承)、实现、关联、聚合与组合 类与类、类与接口之间的关系表示: 纵向关系:继承、实现 横向关系:依赖、关联、聚合与组合,从强到弱依次为:组合、聚合、关联…

文献学习-22-Surgical-VQLA:具有门控视觉语言嵌入的转换器,用于机器人手术中的视觉问题本地化回答

Authors: Long Bai1† , Mobarakol Islam2† , Lalithkumar Seenivasan3 and Hongliang Ren1,3,4∗ , Senior Member, IEEE Source: 2023 IEEE International Conference on Robotics and Automation (ICRA 2023) May 29 - June 2, 2023. London, UK Abstract: 尽管有计算机辅…

反沙箱思路总结

文章目录 反调试反沙箱时间对抗环境检测 反虚拟机黑DLL父进程检测傀儡进程后记 反调试 IsDebuggerPresent #include<windows.h> #include<stdio.h> BOOL check() {return IsDebuggerPresent(); } BOOL isPrime(long long number){if (number < 1)return FALSE…

水下蓝牙耳机哪个好?必看4款购买单,拒绝踩雷!

在当今的科技时代&#xff0c;无线蓝牙耳机已经成为了我们生活中不可或缺的一部分。无论是运动、工作还是休闲娱乐&#xff0c;一款好的蓝牙耳机都能为我们带来极大的便利和乐趣。然而&#xff0c;在水下使用蓝牙耳机却是一个相对特殊的应用场景&#xff0c;需要考虑到防水、防…

关系(一)利用python绘制散点图

关系&#xff08;一&#xff09;利用python绘制散点图 散点图 &#xff08;Scatterplot&#xff09;简介 在笛卡尔座标上放置一系列的数据点&#xff0c;检测两个变量之间的关系&#xff0c;这就是散点图。 散点图可以了解数据之间的各种相关性&#xff0c;如正比、反比、无相…

[linux]--关于进程概念(下)

目录 孤儿进程 环境变量 将程序放到PATH路径下 设置PATH环境变量 设置别名 环境变量相关的命令 环境变量的组织方式​编辑 通过系统调用获取环境变量 环境变量通常是具有全局属性的 进程优先级 查看系统进程 用top命令更改已存在进程的nice&#xff1a; 程序地址空…

【Python实战】——神经网络识别手写数字

&#x1f349;CSDN小墨&晓末:https://blog.csdn.net/jd1813346972 个人介绍: 研一&#xff5c;统计学&#xff5c;干货分享          擅长Python、Matlab、R等主流编程软件          累计十余项国家级比赛奖项&#xff0c;参与研究经费10w、40w级横向 文…

评测i5 1335U和r7 8840HS差距 酷睿i51335U和r78840HS对比

r7 8840HS采用 Zen 4架构 4 nm制作工艺8核 16线程主频 3.3GHz睿频5.1GHz 三 级缓存16MB TDP 功耗 28w 搭载AMD Radeon 780M核显 选i5 1335U还是i5 1235U这些点很重要http://www.adiannao.cn/dy i5 1335U处理器采用10nm工艺制程&#xff0c;拥有10核心12线程&#xff0c;最大睿…

接口关联和requests库

一、接口关联 postman的接口 postman的接口关联配置&#xff1a;js代码&#xff0c;重点在于思路。 // 定义jsonData这个变量 接受登录接口的返回结果 var jsonData JSON.parse(responseBody); // 从返回结果里提取token/id值&#xff0c;并赋值给token/id变量值作为环境变…

电视盒子哪个好?最具性价比网络电视盒子排行榜

挑选电视盒子的时候预算有限的消费者会把性价比放在第一位&#xff0c;如何才能在预算范围内选到配置最好的产品呢&#xff1f;小编这次盘点的是最具性价比的网络电视盒子排行榜&#xff0c;整理了五款最值得入手的电视盒子&#xff0c;不知道电视盒子哪个好可以看看以下这些。…

python -- 循环语句

你好, 我是木木, 目前正在做两件事   1. 沉淀自己的专业知识   2. 探索了解各种副业项目&#xff0c;同时将探索过程进行分享&#xff0c;帮助自己以及更多朋友找到副业, 做好副业 文末有惊喜 循环语句 1、什么是循环语句 一般编程语言都有循环语句&#xff0c;为什么呢&am…

蓝桥杯STM32 G431 hal库开发速成——输入捕获

蓝桥杯的输入捕获较为简单&#xff0c;基本不涉及溢出的问题。所以这里就不介绍溢出了。文末有源码。 核心思想&#xff1a;在第一次上升沿的时候计第一个数&#xff0c;第一次下降沿的时候计第二个数&#xff0c;第二次上升沿的时候计第三个数。 占空比 (第二个数-第一个数…

2024年必用的九大顶级Java分析器:性能优化的利器

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

基于Java的桃花峪滑雪场租赁系统(Vue.js+SpringBoot)

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 游客服务2.2 雪场管理 三、数据库设计3.1 教练表3.2 教练聘请表3.3 押金规则表3.4 器材表3.5 滑雪场表3.7 售票表3.8 器材损坏表 四、系统展示五、核心代码5.1 查询教练5.2 教练聘请5.3 查询滑雪场5.4 滑雪场预定5.5 新…

性能测试丨GreatSQL TPC-H 性能测试报告正式发布!

1、测试背景概述 本次测试针对GreatSQL开源数据库基于标准 TPC-H 场景的测试。 TPC-H&#xff08;商业智能计算测试&#xff09;是美国交易处理效能委员会&#xff08;TPC&#xff0c;TransactionProcessing Performance Council&#xff09;组织制定的用来模拟决策支持类应用…

数据容器-序列-集合-Python

师从黑马程序员 序列 序列的常用操作-切片 切片&#xff1a;从一个序列中&#xff0c;取出一个子序列 语法&#xff1a;序列[起始下标:结束下标&#xff0c;步长] 注&#xff1a;此操作不会影响序列本身&#xff0c;而是会得到一个新的序列 my_list[0.1,2,3,4,5,6] result1…

【大模型】VS Code(Visual Studio Code)上安装的扩展插件用不了,设置VS Code工作区信任

文章目录 一、找到【管理工作区信任】二、页面显示处于限制模式&#xff0c;改为【信任】三、测试四、总结 【运行环境】win 11 相关文章&#xff1a; 【大模型】直接在VS Code(Visual Studio Code)上安装CodeGeeX插件的过程 【问题】之前在 VS Code上安装 CodeGeeX 插件后&…

C语言中如何动态分配内存并进行操作

C语言文章更新目录 C语言学习资源汇总&#xff0c;史上最全面总结&#xff0c;没有之一 C/C学习资源&#xff08;百度云盘链接&#xff09; 计算机二级资料&#xff08;过级专用&#xff09; C语言学习路线&#xff08;从入门到实战&#xff09; 编写C语言程序的7个步骤和编程…