单片机STM32有什么推荐的裸机编程架构_stm32裸机多任务程序架构

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Golang全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Go语言开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024b (备注go)
img

正文

本项目为2021年自己从零到一写出来的简易RTOS系统,目的是为了能更加深入的学习RTOS系统及底层运行的逻辑,项目命名为"LinRTOS",为什么会取这个名字呢?

LinRTOS开发基于MDK平台,目前工程工程自动适配Cortex M0/M3/M4,通过软件验证过所有的功能模块均正常工作

参考说明

写该系统主要参考的有两个知名的RTOS框架:为FreeRTOSRT-Thread,FreeRTOS从远源码角度进行学习的,RT-Thread从Api的命名及系统框架结构上进行了一定借鉴。

这里想对国内的RT-Thread做定义说明,都知道老牌FreeRTOS是免费商业使用,其实RT-Thread也是可免费商业使用,只是RT-Thread有扩展的增值服务才需要进行收费,RT-Thread相对FreeRTOS来说有一个很好的有点,就是有大量的官方中文文档可以进行学习和参考。

  • github master代码仓上你能看到的都是开源、免费的,可以免费商用

LinRTOS命名

函数命名规则主要参照FreeRTOS:

变量名

定义变量的时候往往会把变量的类型当作前缀加在变量上,这样的好处是让用户一看到这个变量就知道该变量的类型:

  • c : char 型变量.
  • s : short 型变量.
  • l : long型变量.
  • x : portBASE_TYPE类型变量,数据结构,任务句柄,队列句柄等定义的变量名.
  • u : 无符号型的前面.
  • p : 指针变量的前面.

例:当我们定义一个无符号的 char 型变量的时候会加一个 uc 前缀,当定义一个char 型的指针变量的时候会有一个 pc 前缀。

函数名

函数名包含了函数返回值的类型、函数所在的文件名和函数的功能,如果是私有的函数则会加一个 prv(private)的前缀。

在函数名中加入了函数所在的文件名,这大大的帮助了用户提高寻找函数定义的效率和了解函数作用的目的,具体的举例如下:

  • vTaskPrioritySet()函数的返回值为 void 型,在 task.c这个文件中定义。
  • xQueueReceive()函数的返回值为 portBASE_TYPE 型,在 queue.c 这个文件中定义。
  • vSemaphoreCreateBinary()函数的返回值为 void 型,在 semphr.h 这个文件中定义。

LinRTOS系统框架

这里称系统架构其实也不是很合理的,因为LinRTOS只提供了在ARM的架构之上的内核层面的任务调度和消息处理和对硬件进行一定的抽象,内核层上面的组件框架和在上层的业务逻辑层是没有去实现的。

虽然目前只实现了内核层部分,想要这个系统更加的完善,还需要添加部分必要的组件和安全的框架进去,这里先给自己埋一个坑吧,后面有机会在来进行进一步的完善。

img

LinRTOS Kernel

工程文件结构

img

Kernel框架

img

对RTOS来说,任务调度是最核心的东西,这里面涉及到优先级调度算法,调度逻辑,及系统运行在硬件平台的一些基础知识点,上面说的所有的东西都是在Task.c和Task.h中来实现的。所以,Task也是RTOS中最重要的文件之一。

LinRTOS Config: 整个系统的配置文件,可以配置需要使用那些RTOS资源,优先级的个数等。

Task: RTOS最核心的模块,负责具体的任务调度的实现。

Event:事假核心文件,提供给各种类型的消息,信号公用的模块。

  • Sem: 信号量: 可用于任务的同步和计数,提供增删改查功能。
  • Mutex: 互斥信号量,用于任务的同步,具有互斥属性(用了来防止优先级翻转问题),提供增删改查功能。
  • EventGroup: 事件标志组,用了多任务之间的同步,和信号量不同的是,可以实现一对多,多对多的同步。 即一个任务可以等待多个事件的发生:可以是任意一个事件发生时唤醒任务进行事件处理;也可以是几个事件都发生后才唤醒任务进行事件处理。同样,也可以是多个任务同步多个事件。
  • Mailbox: 邮箱通讯,同于任务间数据信息传输,提供增删改查功能。

Scheduler: 操作硬件相关的任务中断,为任务的切换提供入口。

Mempool:为任务分配空间资源的管理文件。

List: 提供最基础的双向通用列表,可以进行增删改查。

Bitmap: 优先级调度算法的数组遍历映射文件

Timer: 给真个RTOS计数的模块。


裸机还真没考虑过什么架构,就像楼上说的main里面while到底。

说到底应该还是对rtos不太了解导致的死机吧。

先查下是整个板子死机还是某几个线程死了。

额外开个优先级最高的线程,间隔1秒print一些调试信息。在死机时查查print信息是否还有在输出,可以确认这个线程挂了没。

如果线程全挂,重点查下每个线程的stack size是不是太小,默认值在函数嵌套比较深时是不太够用的,还有子函数是否有声明较大的局部变量,另外查查野指针之类的问题。

如果最高优先级线程活着,查查是否信号量互锁的问题,以及多线程之间的协作设计有误。

遇到问题需要先查出原因而不是逃避,就算逃到裸机也会遇到其它问题的。

F4这么好的性能不上rtos可惜了,裸机编程折腾死你


作者:追风筝的人
链接:https://www.zhihu.com/question/438340661/answer/2443068345
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

可以参考我的开源项目:基于循环时间的跨平台多任务管理系统(可用于MCS51,STM32等单片机)

1. 简介

1.1 系统简介

基于循环时间的跨平台任务管理系统(可用于MCS51,STM32等单片机)

1.2 文件目录说明

  • TaskManager_c:c语言实现的任务管理系统
  • TaskManager_cpp:c++实现的任务管理系统
  • Demo:提供多种示例,涵盖电脑模拟器(MSVC),STM32,8081等多种运行环境

2. 用法(C语言版)

2.1 API用法

系统配置(TaskManager_config.h):

  • MAX_TASK_NUM:最大任务数量,默认为10个,根据自己的需要修改
  • SYS_CYCLE_TIME:系统的循环时间(单位为ms),默认为 1,表示1ms 【注】在MSVC下,SYS_CYCLE_TIME 只能为1,系统循环时间为1s
  • 编译器:目前支持的编译器为 ARM_KILL (AC5和AV6均可) 和 WIN_MSVC

API接口(TaskManager_c):

  • TaskMsg:【结构体】任务信息,结构体成员为 任务函数,开始时间,周期时间,运行次数,PID任务函数:只能是无参数,无返回值的函数,不能是阻塞函数开始时间:任务开始运行的时间,START_NOW 表示立即开始运行 周期时间:任务每周期时间运行一次 运行次数:任务需要运行的次数,RUN_FOREVER 表示无穷次 PID:任务编号号,初始化的时候传入 PID_INIT
  • void TM_init():系统初始化
  • uint32_t TM_add_task(TaskMsg* new_task_msg):添加任务
  • void TM_kill_by_PID(uint32_t PID):通过任务序号删除任务
  • void TM_kill_by_taskmsg(TaskMsg* task_msg):通过任务信息删除任务
  • void TM_run(void):运行系统

示例代码1

#include “taskmanager.h”
#include “TaskManager_config.h”
#include “stdio.h”

// 任务1
void task1(void)
{
printf(“task1!\n”);
}

// 任务2
void task2(void)
{
printf(“task2!\n”);
}

void main(void)
{
TM_init();
// 任务1立即启动,每2ms中运行一次,无休无止的运行
TaskMsg tasks_msg1 = { task1,START_NOW,2,RUN_FOREVER,0 };
TM_add_task(&tasks_msg1);
// 任务2在1ms之后启动,每4ms运行一次,运行5次自动结束
TaskMsg tasks_msg2 = { task2,1,4,5,0 };
TM_add_task(&tasks_msg2);
while (1)
{
// 任务管理器启动
TM_run();
}
}

示例代码2 [Demo/MSVC/c_language]

#include “…/TaskManager/taskmanager.h”
#include “…/TaskManager/TaskManager_config.h”

#include “stdio.h”

void task0(void)
{
static i = 1;
printf_s(“\n%d:”, i);
i++;
}

void task1(void)
{
printf_s(“task1!”);
}

void task2(void)
{
printf_s(“task2!”);
}

void main(void)
{
TM_init(1);
TaskMsg tasks_msg0 = { task0,START_NOW,1,RUN_FOREVER,0 };
int PID0 = TM_add_task(&tasks_msg0);
TaskMsg tasks_msg1 = { task1,START_NOW,2,RUN_FOREVER,0 };
int PID1 = TM_add_task(&tasks_msg1);
TaskMsg tasks_msg2 = { task2,START_NOW,4,RUN_FOREVER,0 };
int PID2 = TM_add_task(&tasks_msg2);
while (1)
{
TM_run();
if (get_systime() > 10)
tasks_msg1.period = 1;
if (get_systime() > 20)
TM_kill_by_PID(PID1);
if (get_systime() > 30)
TM_kill_by_taskmsg(&tasks_msg2);
if (get_systime() > 40)
return;
}
}

运行结果:

1:task1!task2!
2:
3:task1!
4:
5:task1!task2!
6:
7:task1!
8:
9:task1!task2!
10:
11:task1!
12:
13:task1!task2!
14:task1!
15:task1!
16:task1!
17:task1!task2!
18:task1!
19:task1!
20:task1!
21:task2!
22:
23:
24:
25:task2!
26:
27:
28:
29:task2!
30:
31:
32:
33:
34:
35:
36:
37:
38:
39:
40:
C:\Users\Harry\Desktop\TaskManager\Demo\MSVC\c_language\Debug\c_language.exe (进程 41888)已退出,返回代码为: 0。
若要在调试停止时自动关闭控制台,请启用“工具”->“选项”->“调试”->“调试停止时自动关闭控制台”。
按任意键关闭此窗口…

2.2 移植方法

将 TaskManager_c 添加到程序目录下,引用 taskmanager.h , TaskManager_config.h 和 systime.h 并配置时钟即可。

以移植STM32F107为例,说明如下:

Step1:添加项目文件

  • 添加c文件到项目目录

img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Go)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
加c文件到项目目录

img

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Go)
[外链图片转存中…(img-UpPAypKP-1713471732165)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 20
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值