Skip to content

Eplankton/mos-renode

Folders and files

NameName
Last commit message
Last commit date

Latest commit

Β 

History

12 Commits
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 

Repository files navigation

Introduction πŸš€

MOS is a real-time operating system (RTOS) project consists of a preemptive kernel and a command-line shell(both in C++) with optional user application components being ported(e.g., GuiLite and FatFS).

Repository 🌏

  • mos-core - The kernel and shell, check here.
  • mos-stm32 - Running on STM32 series, check here.
  • mos-renode - Testing on Renode emulation, check here.

Install πŸ“¦

  • Run git submodule init && git submodule update to pull the submodules.
  • Install arm-none-eabi-gcc toolchain.
  • Install EIDE plugin, open *.code-workspace with VSCode, and run Build script to compile.
  • Install Renode emulation platform, and add renode to /usr/bin path.
  • Run ./run.sh emulation/*.resc to start the emulation, type s to start, and q to quit.
  • Open a TCP connection to localhost:3333/3334 and observe the serial output.

Manual πŸ“š

Architecture πŸ”

.
β”œβ”€β”€ πŸ“ emulation             // Renode emulation script
β”œβ”€β”€ πŸ“ vendor                // Vendor HAL (SPL/HAL/LL/...)
β”œβ”€β”€ πŸ“ core
β”‚   β”œβ”€β”€ πŸ“ arch              // Architecture-Specific Code
β”‚   β”‚   └── cpu.hpp          // Initialization/Context Switch assembly code
β”‚   β”‚
β”‚   β”œβ”€β”€ πŸ“ kernel            // Kernel Layer (Architecture-Independent)
β”‚   β”‚   β”œβ”€β”€ macro.hpp        // Kernel Constants Macro
β”‚   β”‚   β”œβ”€β”€ type.hpp         // Basic Types
β”‚   β”‚   β”œβ”€β”€ concepts.hpp     // Type Constraints (Optional)
β”‚   β”‚   β”œβ”€β”€ data_type.hpp    // Basic Data Structures
β”‚   β”‚   β”œβ”€β”€ alloc.hpp        // Memory Management
β”‚   β”‚   β”œβ”€β”€ global.hpp       // Kernel Layer Global Variables
β”‚   β”‚   β”œβ”€β”€ printf.h/.c      // Thread-Safe printf (Reference Open Source Implementation)
β”‚   β”‚   β”œβ”€β”€ task.hpp         // Task Control
β”‚   β”‚   β”œβ”€β”€ sync.hpp         // Synchronization Primitives
β”‚   β”‚   β”œβ”€β”€ scheduler.hpp    // Scheduler
β”‚   β”‚   β”œβ”€β”€ ipc.hpp          // Inter-Process Communication
β”‚   β”‚   └── utils.hpp        // Other Utilities
β”‚   β”‚
β”‚   β”œβ”€β”€ config.h             // System Configuration
β”‚   β”œβ”€β”€ kernel.hpp           // Kernel Modules
β”‚   └── shell.hpp            // Shell Command Line
β”‚
└── πŸ“ app                   // User Code
    β”œβ”€β”€ main.cpp             // Entry Function
    └── test.hpp             // Test Code

Example 🍎

Shell Test shell_demo

Mutex Test(Priority Ceiling Protocol) mutex_test

LCD Driver & GUI Demo

Concurrent Task Period & Time Sequence

// 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 buffer
        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();
}

Boot Up ⚑

 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%
----------------------------------------

Version 🧾

πŸ“¦ v0.4

βœ… Done:

  • Adopt Renode emulation platform, stable support for Cortex-M series
  • [Experimental] Add scheduler lock Scheduler::suspend()

πŸ“¦ v0.3

βœ… Done:

  • Mapping Tids to BitMap_t
  • Message queue IPC::MsgQueue_t
  • Task::create allows generic function signatures as void fn(auto argv) with type checker
  • Added ESP32-C3 as a WiFi component
  • Added Driver::Device::SD_t, SD card driver, porting FatFs file system
  • Added Shell::usr_cmds for user-registered commands
  • [Experimental] Atomic types <stdatomic.h>
  • [Experimental] Utils::IrqGuard_t, nested interrupt critical sections
  • [Experimental] Simple formal verification of Scheduler + Mutex

πŸ“Œ Planned:

  • Inter-Process Communication: pipes/channels
  • FPU hardware float support
  • Performance benchmarking
  • Error handling with Result<T, E>, Option<T>
  • DMA_t DMA Driver
  • Software/hardware timers Timer
  • [Experimental] Adding POSIX support
  • [Experimental] Asynchronous stackless coroutines Async::{Future_t, async/await}
  • [Experimental] More real-time scheduling algorithms

πŸ“¦ v0.2

βœ… Done:

  • Synchronization primitives Sync::{Sema_t, Lock_t, Mutex_t<T>, CondVar_t, Barrier_t}
  • Scheduler::Policy::PreemptPri with RoundRobin scheduling for same priority levels
  • Task::terminate implicitly called upon task exit to reclaim resources
  • Simple command-line interaction Shell::{Command, CmdCall, launch}
  • HAL::STM32F4xx::SPI_t and Driver::Device::ST7735S_t, porting the GuiLite graphics library
  • Blocking delay with Kernel::Global::os_ticks and Task::delay
  • Refactored project organization into {kernel, arch, drivers}
  • Support for GCC compilation, compatible with STM32Cube HAL
  • Real-time calendar HAL::STM32F4xx::RTC_t, CmdCall::date_cmd, App::Calendar
  • idle uses Kernel::Global::zombie_list to reclaim inactive pages
  • Three basic page allocation policies Page_t::Policy::{POOL, DYNAMIC, STATIC}

πŸ“¦ v0.1

βœ… Done:

  • Basic data structures, scheduler, and task control, memory management

πŸ“Œ Planned:

  • Timers, round-robin scheduling
  • Inter-Process Communication (IPC), pipes, message queues
  • Process synchronization (Sync), semaphores, mutexes
  • Porting a simple Shell
  • Variable page sizes, memory allocator
  • SPI driver, porting GuiLite/LVGL graphics libraries
  • Porting to other boards/arch, e.g., ESP32-C3 (RISC-V)

References πŸ›Έ

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.

Releases

No releases published

Packages

No packages published