MOS 是一个实时操作系统(RTOS)项目,包含一个抢占式内核和一个简易命令行(均使用C++编写), 并移植了用户应用程序组件(例如,GuiLite 和 FatFS)。
- 运行
git submodule init && git submodule update
拉取子模块core
- 安装
arm-none-eabi-gcc
工具链 - 安装 EIDE 插件, 使用
VSCode
打开*.code-workspace
, 执行Build
编译脚本 - 安装 Renode 仿真平台, 将
renode
添加到/usr/bin
路径 - 运行
bash ./run.sh emulation/*.resc
开始仿真, 输入s
启动,q
退出 - 打开
TCP
连接localhost:3333/3334
, 观察串口的输出
.
├── 📁 emulation // Renode 仿真脚本
├── 📁 vendor // 硬件抽象层(SPL/HAL/LL/...)
├── 📁 core
│ ├── 📁 arch // 架构相关
│ │ └── cpu.hpp // 初始化/上下文切换
│ │
│ ├── 📁 kernel // 内核层(架构无关)
│ │ ├── macro.hpp // 内核常量宏
│ │ ├── type.hpp // 基础类型
│ │ ├── concepts.hpp // 类型约束(可选)
│ │ ├── data_type.hpp // 基本数据结构
│ │ ├── alloc.hpp // 内存管理
│ │ ├── global.hpp // 内核层全局变量
│ │ ├── printf.h/.c // 线程安全的 printf(参考开源实现)
│ │ ├── task.hpp // 任务控制
│ │ ├── sync.hpp // 同步原语
│ │ ├── scheduler.hpp // 调度器
│ │ ├── ipc.hpp // 进程间通信
│ │ └── utils.hpp // 其他工具
│ │
│ ├── config.h // 系统配置
│ ├── kernel.hpp // 内核模块
│ └── shell.hpp // Shell 命令行
│
└── 📁 app // 用户层
├── main.cpp // 入口函数
└── test.hpp // 测试代码
并发任务周期与抢占
// MOS Kernel & Shell
#include "mos/kernel.hpp"
#include "mos/shell.hpp"
// HAL and Device
#include "drivers/stm32f4xx/hal.hpp"
#include "drivers/device/led.hpp"
namespace MOS::User::Global
{
using namespace HAL::STM32F4xx;
using namespace Driver::Device;
using namespace DataType;
// Shell I/O UART and Buffer
auto& stdio = STM32F4xx::convert(USARTx);
DataType::SyncRxBuf_t<16> io_buf;
// LED red, green, blue
Device::LED_t leds[] = {...};
}
namespace MOS::User::BSP
{
using namespace Driver;
using namespace Global;
void LED_Config()
{
for (auto& led: leds) {
led.init();
}
}
void USART_Config()
{
stdio.init(9600-8-1-N)
.rx_config(PXa) // RX -> PXa
.tx_config(PYb) // TX -> PYb
.it_enable(RXNE) // Enable RXNE interrupt
.enable(); // Enable UART
}
...
}
namespace MOS::User::App
{
Sync::Barrier_t bar {2}; // Set Barrier to sync tasks
void led1(Device::LED_t leds[])
{
bar.wait();
for (auto _: Range(0, 20)) {
leds[1].toggle(); // green
Task::delay(250_ms);
}
kprintf(
"%s exits...\n",
Task::current()->get_name()
);
}
void led0(Device::LED_t leds[])
{
Task::create(
led1,
leds,
Task::current()->get_pri(),
"led1"
);
bar.wait();
while (true) {
leds[0].toggle(); // red
Task::delay(500_ms);
}
}
...
}
int main()
{
using namespace MOS;
using namespace Kernel;
using namespace User;
using namespace User::Global;
BSP::config(); // Init hardware and clocks
Task::create( // Create Calendar with RTC
App::time_init, nullptr, 0, "time/init"
);
Task::create( // Create Shell with stdio.buf
Shell::launch, &stdio.buf, 1, "shell"
);
/* User Tasks */
Task::create(App::led_init, &leds, 2, "led/init");
...
/* Test examples */
Test::MutexTest();
Test::MsgQueueTest();
...
// Start scheduling, never return
Scheduler::launch();
}
A_A _ Version @ x.x.x(...)
o'' )_____// Build @ TIME, DATE
`_/ MOS ) Chip @ MCU, ARCH
(_(_/--(_/ 2023-2024 Copyright by Eplankton
Tid Name Priority Status Mem%
----------------------------------------
#0 idle 15 READY 10%
#1 shell 1 BLOCKED 21%
#2 led0 2 RUNNING 9%
----------------------------------------
📦 v0.4
✅ 完成:
- 平台迁移,使用
Renode
仿真平台, 稳定支持Cortex-M
系列- [实验性] 添加调度器锁
Scheduler::suspend()
📦 v0.3
✅ 完成:
Tids
映射到BitMap_t
IPC::MsgQueue_t
,消息队列Task::create
允许泛型函数签名为void fn(auto argv)
,提供类型检查- 添加
ESP32-C3
作为WiFi
元件- 添加
Driver::Device::SD_t
,SD
卡驱动,移植FatFs
文件系统- 添加
Shell::usr_cmds
,用户注册命令- [实验性] 原子类型
<stdatomic.h>
- [实验性]
Utils::IrqGuard_t
,嵌套中断临界区- [实验性]
Scheduler + Mutex
简单的形式化验证📌 计划:
- 进程间通信:管道/通道
FPU
硬件浮点支持- 性能基准测试
Result<T, E>, Option<T>
,错误处理DMA_t
驱动- 软/硬件定时器
Timer
- [实验性] 添加
POSIX
支持- [实验性] 异步无栈协程
Async::{Future_t, async/await}
- [实验性] 更多实时调度算法
📦 v0.2
✅ 完成:
Sync::{Sema_t, Lock_t, Mutex_t<T>, CondVar_t, Barrier_t}
同步原语Scheduler::Policy::PreemptPri
,在相同优先级下则以时间片轮转RoundRobin
调度Task::terminate
在任务退出时隐式调用,回收资源Shell::{Command, CmdCall, launch}
,简单的命令行交互HAL::STM32F4xx::SPI_t
和Driver::Device::ST7735S_t
, 移植GuiLite
图形库Kernel::Global::os_ticks
和Task::delay
,阻塞延时- 重构项目组织为
{kernel, arch, drivers}
- 支持
GCC
编译,兼容STM32Cube HAL
HAL::STM32F4xx::RTC_t
,CmdCall::date_cmd
,App::Calendar
实时日历idle
使用Kernel::Global::zombie_list
回收非活动页面- 三种基本的页面分配策略
Page_t::Policy::{POOL(池), DYNAMIC(动态), STATIC(静态)}
📦 v0.1
✅ 完成:
- 基本的数据结构、调度器与任务控制、内存管理
📌 计划:
- 定时器,时间片轮转调度
- 进程间通信
IPC
,管道、消息队列- 进程同步
Sync
,信号量、互斥锁- 移植简单的
Shell
- 可变页面大小,内存分配器
SPI
驱动,移植GuiLite/LVGL
图形库- 移植到其他开发板/架构,例如
ESP32-C3(RISC-V)
- How to build a Real-Time Operating System(RTOS)
- PeriodicScheduler_Semaphore
- STM32F4-LCD_ST7735s
- A printf/sprintf Implementation for Embedded Systems
- GuiLite
- STMViewer
- FatFs
- The Zephyr Project
- Eclipse ThreadX
- Embassy
- Renode
I've seen things you people wouldn't believe.
Attack ships on fire off the shoulder of Orion.
I watched C-beams glitter in the dark near the Tannhäuser Gate.
All those moments will be lost in time, like tears in rain.
Time to die.