STM32完全学习——RT-thread在STM32F407上移植

ops/2025/2/2 2:11:12/

一、写在前面

关于源码的下载,以及在KEIL工程里面添加操作系统的源代码,这里就不再赘述了。需要注意的是RT-thread默认里面是会使用串口的,因此需要额外的进行串口的初始化,有些人可能会问,为什么不直接使用CubMAX直接进行系统的移植,经过我的尝试发现,这个玩意移植的RT-thread会有一些小的bug,比如直接生成的代码FinSH工作不正常,修改方法和第七节一样。虽然整个系统的调度啥的都是正常的。因此我决定另辟蹊径,其他的功能使用CubMAX进行设置,因为他简直太方便了,然后RT-thread的移植采用手动的方法进行。

二、移植前的准备工作

因为RT-thread的源码里面有这些中断处理函数的实现,因此需要将上面的这些都关闭掉,不然编译会出现重复定义的错误。

 我这里使用的是串口1,因此先完成他的相关设置。

三、将RT-thread的源码添加到工程里面

都添加进行以后,然后不要忘了设置头文件的包含路径。

四、进行第一次的编译

 会发现错误很多,不慌。透过查看错误发现是因为va_list没有定义引起的,我们知道va_list是定义在stdarg.h里面的,然后通过观察头文件的包含发现,这个头文件在下面这个文件里面定义着,显然是因为RT_USING_LIBC这个宏,没有打开导致的。在rtconfig.h里面添加这个宏定义即可。

 五、进行第二次编译

我们发现编译很成功,没有错误也没有警告,这时你可能已经觉得成功了,我只能说我当时也是这么想的,结果发现程序运行那是,相当的不正常。需要解决这个问题我们得知道RT-thread启动的流程,其实你只需要知道他在运行我们的main函数之前,会运行一段自己的代码,

 然后我们进入这个函数发现,这里是关于系统的初始化,有很多。

 接下来我们进入第一个初始化函数里面。将串口的初始化放在这里,注意需要包含相关的头文件。有了串口我们就可以使用printf函数来进行调试了。

 当我们进入到第二个初始化函数里面我们发现,这里有一些输出,但是要想使用这些输出我们需要实现void rt_hw_console_output(const char *str)这个函数,在board.c里面其实已经有关于这个函数的实现,他是用RT_USING_CONSOLE这个宏来进行控制的,因此需要在trconfig.h里面打开这个宏

 你会发现串口里面还是没有我们想要的输出,通过观察发现虽然我们自己将串口进行了初始化,但是系统这里使用的是串口2,因此需要将他改成你初始化的那个串口,我这里改成串口1。

 到这里我们的串口就能正常工作了。下面这个是串口的输出。

 到这里如果你创建一个任务,然后你会神奇的发现这个任务好像并没有执行,也就是说我们的系统并没有完全初始化成功。因为在RT-thread里面创建任务的时候有两种分配内存的方式,第一种就是静态分配,第二种就是动态分配。RT-thread默认使用的是动态分配,我想会不会是这里的问题,我就将关于动态分配的相关配置注释掉了。到这里整个系统就算是初始化完毕,也正常运行起来了。需要注意的是我们这里所有的内存分配都是静态的,因此创建变量的时候一定要分配它的大小,不然会有很奇怪的问题。

 六、使用静态方式创建任务

static struct rt_thread led1_thread;
/* 线程主体函数 */
static void led1_thread_entry(void* parameter);
/* 定义线程栈 */
static rt_uint8_t rt_led1_thread_stack[1024];static struct rt_thread led0_thread;
/* 线程主体函数 */
static void led0_thread_entry(void* parameter);
/* 定义线程栈 */
static rt_uint8_t rt_led0_thread_stack[1024];int main(void)
{MX_GPIO_Init();/* USER CODE BEGIN 2 */rt_thread_init(&led1_thread,                 /* 线程控制块 */"led1",                       /* 线程名字 */led1_thread_entry,            /* 线程入口函数 */RT_NULL,                      /* 线程入口函数参数 */&rt_led1_thread_stack[0],     /* 线程栈起始地址 */sizeof(rt_led1_thread_stack), /* 线程栈大小 */3,                            /* 线程的优先级 */20);                          /* 线程时间片 */rt_thread_init(&led0_thread,                 /* 线程控制块 */"led0",                       /* 线程名字 */led0_thread_entry,            /* 线程入口函数 */RT_NULL,                      /* 线程入口函数参数 */&rt_led0_thread_stack[0],     /* 线程栈起始地址 */sizeof(rt_led0_thread_stack), /* 线程栈大小 */4,                            /* 线程的优先级 */20);                          /* 线程时间片 */rt_thread_startup(&led1_thread);rt_thread_startup(&led0_thread);  while (1){}}//LED1线程
static void led1_thread_entry(void* parameter)
{	while(1){LED1 = 0;rt_thread_delay(200);   /* 延时200个tick */rt_kprintf("led1_thread running,LED1_ON\r\n");LED1 = 1;     rt_thread_delay(500);   /* 延时500个tick */rt_kprintf("led1_thread running,LED1_OFF\r\n");}
}static void led0_thread_entry(void* parameter)
{	while(1){LED0 = 0;rt_thread_delay(200);   /* 延时200个tick */rt_kprintf("led0_thread running,LED0_ON\r\n");LED0 = 1;     rt_thread_delay(500);   /* 延时500个tick */rt_kprintf("led0_thread running,LED0_OFF\r\n");}
}

七、实现FinSH

 

 这个头文件你可以通过CubMAX来生成,也可以在项目例程里面复制一个。然后运行有可能你的会运行成功,有可能并不会成功。失败的界面是这样的;

成功的界面是这样的

如果失败了,尝试调整下图的参数即可。


http://www.ppmy.cn/ops/154911.html

相关文章

【协议详解】卫星通信5G IoT NTN SIB32-NB 信令详解

一、SIB32信令概述 低轨卫星的移动性会导致地面用户设备覆盖不连续,为了解决这一问题,3GPP引入了SystemInformationBlockType32(SIB32)信令,为非连续覆盖预测提供卫星辅助信息。地面设备可以基于SIB32信令中的信息&am…

【matlab】绘图 离散数据--->连续函数

matlab绘图练习 离散数据及离散函数对离散区间进行细划分 达到连续效果画plot(y)图 与 复数的应用 离散数据及离散函数 例1 x1[1 2 4 6 7 8 10 11 12 14 16 17 18 20] y1[1 2 4 6 7 8 10 10 8 7 6 4 2 1] figure(1); plot(x1,y1,o,MarkerSize,15); x21:20; y2log(x2); figure…

【矩阵二分】力扣378. 有序矩阵中第 K 小的元素

给你一个 n x n 矩阵 matrix ,其中每行和每列元素均按升序排序,找到矩阵中第 k 小的元素。 请注意,它是 排序后 的第 k 小元素,而不是第 k 个 不同 的元素。 你必须找到一个内存复杂度优于 O(n2) 的解决方案。 示例 1&#xff1…

使用PyTorch实现逻辑回归:从训练到模型保存与加载

1. 引入必要的库 首先,需要引入必要的库。PyTorch用于构建和训练模型,pandas和numpy用于数据处理,matplotlib用于结果的可视化。 import torch import torch.nn as nn import torch.optim as optim from torch.utils.data import DataLoade…

CSS关系选择器详解

CSS关系选择器详解 学习前提什么是关系选择器?后代选择器(Descendant Combinator)语法示例注意事项 子代选择器(Child Combinator)语法示例注意事项 邻接兄弟选择器(Adjacent Sibling Combinator&#xff0…

双指针c++

双指针(Two Pointers)是一种常用的算法技巧,通常用于解决数组或链表中的问题,如滑动窗口、区间合并、有序数组的两数之和等。双指针的核心思想是通过两个指针的移动来优化时间复杂度,通常可以将 (O(n^2)) 的暴力解法优…

Titans 架构下MAC变体的探究

目前业界流行的 Transformer 模型架构虽然在大多数场景表现优秀,但其上下文窗口(Window)长度的限制,通常仅为几千到几万个 Token,这使得它们在处理长文本、多轮对话或需要大规模上下文记忆的任务中,往往无法…

flume和kafka整合 flume和kafka为什么一起用?

‌Flume和Kafka一起使用的主要原因是为了实现高效、可靠的数据采集和实时处理。‌‌12 实时流式日志处理的需求 Flume和Kafka结合使用的主要目的是为了完成实时流式的日志处理。Flume负责数据的采集和传输,而Kafka则作为消息缓存队列,能够有效地缓冲数据,防止数据堆积或丢…