Toggle navigation
首页
问答
文章
积分商城
专家
专区
更多专区...
文档中心
返回主站
搜索
提问
会员
中心
登录
注册
多线程
2024-RSOC
【2024-RSOC】day02基于RTT的多线程建立及应用+示例介绍
发布于 2024-07-24 02:28:42 浏览:582
订阅该版
[tocm] # 实时系统的引入 实习系统是指能在指定时间内完成规定功能并且对外部异步事件进行响应的计算机系统 ## 实时系统分为 硬实时系统:结果迟到产生灾难性的后果 FIRM实习系统:结果迟到产生难以接受的质量上的降低 软实时系统:结果迟到产生质量下降,但是系统可以自主恢复 ## 衡量实时性的指标 本地响应时间 生存时间 吞吐量 ## 嵌入式实时系统的功能介绍 ![](C:\Users\LAO\Desktop\14.jpg) # RT-Thread的引入 ## 传统操作系统(Linux&&RTOS) ![](C:\Users\LAO\Desktop\21.png) > 传统的操作系统Linux重点在于通用泛用,但是存在着移植难度大,开销大,实时性并不保证的缺点 > > RTOS轻量化,保证实时性,但是其功能较少需要额外的成本或者集成工作,仅能满足部分单片机的全部使用 > 而RT-Thread拥有更好的性能,相比前两家兼具其优,并且配有开发社区预计丰富的外设资源 # 裸机开发模式 ## 后台 后台是一个无限的循环,其中调用相应的处理函数,完成相应的操作 ADC SPI USB LCD Audio_Decode File_Write ## 前台 ```c void USB_ISR(void) { Clear interrupt; Read packet; } ``` > 逻辑模式的缺陷在于后台循环调用的函数中如果存在延时函数那么对于整个系统的实时性会造成影响 # RT-Thread启动流程 ## 多任务系统 ![](C:\Users\LAO\Desktop\41.png) ## 启动流程分析 ![](C:\Users\LAO\Desktop\42.png) ## 线程控制块 ## 构成 线程控制块是一种结构体双向链表,其主要结构体为struct rt_thread定义并形成线程内核对象,再链接到内核对象容器中进行管理 ## 线程控制块继承关系 ![](C:\Users\LAO\Desktop\51.png) ## 线程控制块的定义 一个线程包括有一个线程栈一个入口函数以及一个线程控制块 下面展示struct rt_thread的定义 ``` struct rt_thread { char name[RT_NAME_MAX]; rt_unit8_t type; rt_unit8_t flags; rt_list_t list; rt_list_t tlist; void *sp; void *entry void *parameter void *stack_addr; rt_unit32_t stack_size; rt_err_t error; rt_unit8_t stat; rt_unit8_t current_priority; rt_unit8_t init_priority; rt_unit32_t number_mask; ... rt_ubase_t init_tick; rt_ubase_t remaining_tick; struct rt_timer thread_timer; void(*cleanip)(struct rt_thread*tid); rt_unit32_t user_data; } ``` > 从上往下有线程名称,对象类型,标志位,对象列表,栈指针等待,包含很多方面的结构体描述 ## 线程栈 RT-Thread的线程具有独立的空间栈,当进行线程转换时,当前线程的上下文会存在栈中。 线程栈还可以用来存放函数中的局部变量 ## 线程状态以及转换 ![](C:\Users\LAO\Desktop\52.png) ![](C:\Users\LAO\Desktop\53.png) ## 线程属性 ### 优先级 有专门的优先级位,数值越小的优先级越高 ### 时间片 每个线程都需要有时间片,当时间片仅对优先级相同的就绪态线程有效。系统对优先级相同的就绪态线程进行调度时,时间片起到的作用为单次运行时间,其单位时一个系统节拍(OSTick) ### 入口函数 ![](C:\Users\LAO\Desktop\54.png) ### 错误码 ![](C:\Users\LAO\Desktop\55.png) # 系统线程 ## 空闲线程 最低优先级的线程,其状态永远保持为就绪态,当系统无其他就绪态存在时,系统将进入空闲线程,且一般都是死循环,且永远不能被挂起 有着几个重要的作用 1.回收被删除线程的资源 2.提供狗子函数的接口,或者看门狗喂狗 # 线程管理API ## 线程初始化 ![](C:\Users\LAO\Desktop\56.png) ![](C:\Users\LAO\Desktop\57.png) ![](C:\Users\LAO\Desktop\58.png) ## 线程启动 rt_err_t rt_thread_startup(rt_thread_t thread) 调用这个函数时,将把线程的状态更改为就绪状态,并根据优先级队列中等待调度。 ## 线程延时 抢占时系统线程在不使用CPU时需要让出CPU的使用权,让其他线程得以运行。 通常有如下三个函数 ```c rtt_err_t rt_thread_sleep(rt_tick_t tick);//OS Tick为单位 rtt_err_t rt_thread_delay(rt_tick_t tick);//OS Tick为单位 rtt_err_t rt_thread_mdelay(rt_int32_t ms);//以毫秒为单位,这个最小 ``` # 学习示例 ## 程序流程 分别定义一个静态线程和一个动态线程 在两者成功建立时向电脑发送ready确认 而后在进程中有进行GPIO引脚亮灭的任务,两个进程控制的GPIO引脚各不相同 ## 初始化部分(头文件和GPIO) ```c #include
#include
#include
#ifndef RT_USING_NANO #include
#endif /* RT_USING_NANO */ #define GPIO_LED_B GET_PIN(F, 11) #define GPIO_LED_R GET_PIN(F, 12) ``` ## 初始化部分(静态线程) ```c //静态线程任务 static rt_uint8_t thread1_stack[512]; //线程栈 static struct rt_thread static_thread; //线程控制块 static void static_entry(void *param) { static int cnt = 0; while (cnt++<2) { rt_kprintf("static_thread is run:%d\n",cnt); rt_pin_write(GPIO_LED_B, PIN_LOW); rt_thread_mdelay(1000); rt_pin_write(GPIO_LED_B, PIN_HIGH); rt_thread_mdelay(500); } } ``` ## 初始化部分(动态线程) ```c //动态线程任务 static void dynamic_entry(void *param) { static int cnt = 0; while (cnt++<2) { rt_kprintf("dynamic_thread is run:%d\n",cnt); rt_pin_write(GPIO_LED_R, PIN_LOW); rt_thread_mdelay(1000); rt_pin_write(GPIO_LED_R, PIN_HIGH); rt_thread_mdelay(500); } } ``` ## 建立线程 ```c int thread_sample(void) { static rt_thread_t thread_id = RT_NULL; thread_id = rt_thread_create("dynamic_th", //名称 dynamic_entry, //线程代码 RT_NULL, //参数 1024, //栈大小 15, //优先级 20); //时间片 if (thread_id != RT_NULL) { rt_thread_startup(thread_id); //线程进入就绪态 rt_kprintf("dynamic_th ready!"); } thread_id=rt_thread_init(&static_thread, //线程handle "static_thread", //线程名称 static_entry, //线程入口函数 RT_NULL, //线程入口参数 &thread1_stack[0], //线程栈地址 sizeof(thread1_stack), //线程栈大小 15, //线程优先级 5); //线程时间片 /. if (thread_id != RT_NULL) { rt_thread_startup(thread_id); //线程进入就绪态 rt_kprintf("static_th ready!"); } return RT_EOK; } /* 导出到 msh 命令列表中 */ MSH_CMD_EXPORT(thread_sample, thread sample); ``` ## 主函数部分 ```c int main(void) { rt_err_t ERR; rt_pin_mode(GPIO_LED_R, PIN_MODE_OUTPUT); rt_pin_mode(GPIO_LED_B, PIN_MODE_OUTPUT); rt_pin_write(GPIO_LED_R, PIN_HIGH); rt_pin_write(GPIO_LED_B, PIN_HIGH); ERR=thread_sample(); while(ERR==RT_EOK); while(1); return RT_EOK; } ``` ## 结果展示 开发板上的灯正常亮灭,但是串口没有发送数据不知为何仍在努力中
0
条评论
默认排序
按发布时间排序
登录
注册新账号
关于作者
鲁迪乌斯
这家伙很懒,什么也没写!
文章
5
回答
0
被采纳
0
关注TA
发私信
相关文章
1
大家有没有遇到过多线程使用同一个串口发送数据,数据是交叉的
2
workqueue中调用rt_i2c_transfer互斥锁线程bug
3
micropython和本地线程的通信
4
大神们,我发现一个问题,RTT这个系统是不是没有查看其他线程状态的函数?
5
不知道该如何使用线程?
6
线程错误怎么判断哪里的问题(其他线程都ok)
7
rt_kprintf 多线程使用问题
8
线程优先级被中断影响
9
挂载在操作系统上的文件系统操作,需要用互斥锁防止多线程操作问题吗?
10
socket多线程操作转为单线程处理,断开连接检测代码展示
推荐文章
1
RT-Thread应用项目汇总
2
玩转RT-Thread系列教程
3
国产MCU移植系列教程汇总,欢迎查看!
4
机器人操作系统 (ROS2) 和 RT-Thread 通信
5
五分钟玩转RT-Thread新社区
6
【技术三千问】之《玩转ART-Pi》,看这篇就够了!干货汇总
7
关于STM32H7开发板上使用SDIO接口驱动SD卡挂载文件系统的问题总结
8
STM32的“GPU”——DMA2D实例详解
9
RT-Thread隐藏的宝藏之completion
10
【ART-PI】RT-Thread 开启RTC 与 Alarm组件
热门标签
RT-Thread Studio
串口
Env
LWIP
SPI
AT
Bootloader
Hardfault
CAN总线
FinSH
ART-Pi
USB
DMA
文件系统
RT-Thread
SCons
RT-Thread Nano
线程
MQTT
STM32
RTC
FAL
rt-smart
I2C_IIC
ESP8266
UART
WIZnet_W5500
ota在线升级
cubemx
PWM
flash
freemodbus
BSP
packages_软件包
潘多拉开发板_Pandora
定时器
ADC
flashDB
GD32
socket
编译报错
中断
Debug
rt_mq_消息队列_msg_queue
SFUD
msh
keil_MDK
ulog
C++_cpp
MicroPython
本月问答贡献
xusiwei1236
8
个答案
2
次被采纳
踩姑娘的小蘑菇
1
个答案
2
次被采纳
用户名由3_15位
9
个答案
1
次被采纳
bernard
4
个答案
1
次被采纳
RTT_逍遥
3
个答案
1
次被采纳
本月文章贡献
聚散无由
2
篇文章
15
次点赞
catcatbing
2
篇文章
5
次点赞
Wade
2
篇文章
4
次点赞
Ghost_Girls
1
篇文章
7
次点赞
YZRD
1
篇文章
2
次点赞
回到
顶部
发布
问题
投诉
建议
回到
底部