当前位置: 首页 > news >正文

ysyx学习:移植rt-thread

ysyx学习:移植rt-thread

OS中的上下文切换

上一期我们搞懂了yield-os.c的原理如何,我们理解 玩上下文切换的核心细节之后,我们可以把这些原理迁移到RT-Thread这个更大的操作系统中。
RT-Thread中有两个抽象层, 一个是BSP(Board Support Package), 另一个是libcpu。

负责把 RT-Thread 的线程结构(PCB)和抽象机器(AM)的上下文切换接口连接起来:调用处把目标/来源上下文地址临时放到线程的 user_data,触发 yield(),AM 回调读出这些信息并完成实际的上下文切换;另外还准备线程首次运行时的栈/参数(通过 kcontext)。

对于上下文的创建,需要先实现rt_uint8_t *rt_hw_stack_init(void *tentry, void *parameter, rt_uint8_t *stack_addr, void *texit)函数。

rt_uint8_t* rt_hw_stack_init(void* tentry, void* parameter, rt_uint8_t* stack_addr, void* texit) {stack_addr = (rt_uint8_t*)(((uintptr_t)stack_addr + sizeof(uintptr_t) - 1) & ~(sizeof(uintptr_t) - 1));stack_addr -= sizeof(temp_args); temp_args* args = (temp_args*)stack_addr; // 在栈上保存 入口 参数 出口 等信息// 设置参数args->tentry = (void (*)(void*))tentry;   // tentry是“接受一个 void* 参数、返回 void 的函数指针”args->parameter = parameter;args->texit = (void (*)(void))texit;      // texit是“不接受参数、返回 void 的函数指针”rt_uint8_t* c = (rt_uint8_t*)kcontext((Area){ stack_addr - sizeof(Area) , stack_addr }, (void*)wrap, args);return c;
}

一开始先对齐,然后拉出栈的位置,在栈上保存 入口 参数 出口 等信息,然后使用kcontext函数(详情见之前的随笔),将控制状态寄存器mepc设置为函数的地址,当你执行mret指令的时候就会跳转到该函数中去,因此会跳转到wrap中,附带参数args。

这个wrap()函数如下,这个是包裹函数,讲义中提到让包裹函数来调用tentry, 并在tentry返回后调用texit,T-Thread会保证代码不会从texit中返回,因此我这个代码这么写。
执行入口函数,并附带上参数,执行完之后执行退出函数。

void wrap(temp_args* args) {args->tentry(args->parameter);args->texit();// 不应该执行于此,RT-Thread会保证代码不会从texit中返回while (1);
}

接下来来看main函数,main函数代码如下:

int main() {ioe_init();
#ifdef __ISA_NATIVE__// trigger the real initialization of IOE to// perform SDL initialization int this main thread with large stackio_read(AM_TIMER_CONFIG);
#endifextern void __am_cte_init();__am_cte_init();extern int entry(void);entry();return 0;
}

先初始化一下ioe外设,在调用__am_cte_init()函数初始化cte。

void __am_cte_init() {cte_init(ev_handler); 
} 

意思就是把__am_asm_trap赋值给mtvec,方便后面ecall的时候直接跳进去__am_asm_trap函数。
然后把ev_handler函数注册进回调函数中。

ev_handler函数如下:定义一个当前的线程,可以通过rt_thread_self()这个函数获取当前线程,再使用这个线程中的user_data,可以用current->user_data来使用他,为什么用这个呢,因为这个user_data它用于存放线程的私有数据, 这意味着RT-Thread中调度相关的代码必定不会使用这个成员, 因此它很适合我们用来传递fromto的信息,这里的from是当前线程,这个to是你要跳往的线程,这里的ev_handler是回调函数,当系统陷入yield的时候,执行完一些列操作(看以前的随笔)后会执行这个回调函数了。这个函数主要作用就是把当前上下文保存到from中然后利用to作为下一个上下文进行跳转。

static Context* ev_handler(Event e, Context* c) {rt_thread_t current;rt_ubase_t* para;switch (e.event) {case EVENT_YIELD:current = rt_thread_self();para = (rt_ubase_t*)current->user_data;rt_ubase_t from = para[0];rt_ubase_t to = para[1];if (from) {*((Context**)from) = c; //保存上下文到from中}c = *(Context**)to;  // 返回to作为下一个上下文break;case EVENT_IRQ_TIMER:return c;default:printf("Unhandled event ID = %d\n", e.event);assert(0);}return c;
}

具体怎么上下文的切换呢,看这两个函数void rt_hw_context_switch_to(rt_ubase_t to)//切换到to的上下文
void rt_hw_context_switch(rt_ubase_t from, rt_ubase_t to)//切换到to的上下文,又要保存当前上下文到from上。

void rt_hw_context_switch_to(rt_ubase_t to) {rt_ubase_t data[2];rt_thread_t current = rt_thread_self();rt_ubase_t temp_ud = current->user_data;   //保存原来的 from 和 todata[1] = to;current->user_data = (rt_ubase_t)data;yield();current->user_data = temp_ud;
}
void rt_hw_context_switch(rt_ubase_t from, rt_ubase_t to) {rt_ubase_t data[2];rt_thread_t current = rt_thread_self();rt_ubase_t temp_ud = current->user_data;   //保存原来的 from 和 todata[0] = from;data[1] = to;current->user_data = (rt_ubase_t)data;yield();current->user_data = temp_ud;
}

这两个函数实现起来基本差不多,使用刚刚的user_data保存完fromto之后将,切换线程,然后切换完之后再将原先的user_data数据还回去。

然后启动rt-thread,成功!!!

但是很奇怪,我每次输入完一行命令,下次那一行命令会多一个/>,之前貌似不会,不知道改了什么之后就会了。
image

http://www.hskmm.com/?act=detail&tid=33328

相关文章:

  • 综合性题目
  • 实用指南:从入门到精通:Django的深度探索之旅
  • UML中9中数据流图总结
  • 两种树状数组
  • 斑马日记2025.10.17
  • CF Global Round 29(#2147) 总结
  • 详细介绍:C语言中#pragma的用法
  • JAVA 中断处理
  • 第十五天
  • 软件工程学习日志2025.10.17
  • 天黑了,睡觉
  • 升鲜宝生鲜配送供应链管理系统---- 门店收银 POS 离线工作设计文档(支持线上线下一体化)---02
  • 2025.10.16NOIP模拟
  • Python 基于Python开发的数据库同步检测工具
  • 当AI学会进化:荣耀与用户的“共生式成长”新范式
  • VSCode的下载安装以及配置
  • 2025年终极公众号排版神器排行榜 最新案例研究权威测评
  • NAS安装远程协作神器twake
  • 把三门问题做成了"游戏"
  • 下一代CPU驱动高性能计算革新
  • [KaibaMath]1010 关于关于收敛数列有界性的证明
  • 卫星地图匹配定位 - MKT
  • 10.17 —— (VP) 2021icpc沈阳
  • 10.17每日总结
  • 今天宝宝进面了
  • 《大象Thinking in Projects》读书笔记1
  • 20251017
  • MT签名去除签名校验分析
  • uml
  • P3643 [APIO2016] 划艇 分析