2024-05-14 11:03:08 -04:00
|
|
|
// src/main.rs
|
2024-05-13 18:31:37 -04:00
|
|
|
#![no_std]
|
|
|
|
#![no_main]
|
|
|
|
#![feature(naked_functions)]
|
|
|
|
|
2024-05-16 00:17:55 -04:00
|
|
|
use core::panic::PanicInfo;
|
2024-05-13 18:31:37 -04:00
|
|
|
|
2024-05-18 10:26:17 -04:00
|
|
|
mod entry;
|
2024-05-14 11:03:08 -04:00
|
|
|
mod heap;
|
2024-05-16 12:21:27 -04:00
|
|
|
mod trap;
|
2024-05-13 23:12:29 -04:00
|
|
|
mod uart;
|
|
|
|
|
2024-05-13 18:31:37 -04:00
|
|
|
#[naked]
|
|
|
|
#[no_mangle]
|
|
|
|
#[link_section = ".text.init"]
|
2024-05-15 21:44:54 -04:00
|
|
|
unsafe extern "C" fn _enter() -> ! {
|
2024-05-16 11:39:53 -04:00
|
|
|
// TODO see if possible to replace this somehow...
|
2024-05-13 18:31:37 -04:00
|
|
|
use core::arch::asm;
|
|
|
|
asm!(
|
2024-05-18 09:00:48 -04:00
|
|
|
// load hartid into `tp``
|
|
|
|
// if hartid =/= 0, then jump to busy loop
|
|
|
|
"csrr tp, mhartid",
|
|
|
|
"bnez tp, 3f",
|
2024-05-16 09:23:08 -04:00
|
|
|
|
2024-05-14 11:06:57 -04:00
|
|
|
// before we use the `la` pseudo-instruction for the first time,
|
2024-05-16 09:19:23 -04:00
|
|
|
// we need to set `gp` (look up linker relaxation)
|
2024-05-16 00:33:36 -04:00
|
|
|
".option push", // pushes the current option stack
|
|
|
|
".option norelax", // disable relaxation so we can properly set `gp`
|
|
|
|
// this instruction compiles to:
|
2024-05-18 01:07:42 -04:00
|
|
|
// 1:
|
2024-05-16 00:33:36 -04:00
|
|
|
// auipc gp, %pcrel_hi(_global_pointer)
|
|
|
|
// addi gp, gp, %pcrel_lo(1b)
|
|
|
|
// if relaxation were on, this would simply be `mv gp, gp` or `addi gp, gp, 0`
|
2024-05-14 11:06:57 -04:00
|
|
|
"la gp, _global_pointer",
|
2024-05-16 00:33:36 -04:00
|
|
|
".option pop", // pops the current option stack
|
2024-05-13 18:31:37 -04:00
|
|
|
|
2024-05-17 11:18:37 -04:00
|
|
|
// delegate all traps to S-mode trap handler
|
|
|
|
"li t5, 0xffff",
|
|
|
|
"csrw medeleg, t5", // delegate all machine exceptions
|
|
|
|
"csrw mideleg, t5", // delegate all machine interrupts
|
2024-05-16 12:21:27 -04:00
|
|
|
|
2024-05-14 11:06:57 -04:00
|
|
|
// set the stack pointer
|
2024-05-18 21:02:40 -04:00
|
|
|
"la sp, _stack_end",
|
2024-05-13 18:31:37 -04:00
|
|
|
|
2024-05-17 11:18:37 -04:00
|
|
|
// Make sure machine mode is set, and enable coarse interrupts
|
|
|
|
"li t0, (0b11 << 11) | (1 << 7) | (1 << 3)",
|
|
|
|
"csrw mstatus, t0",
|
2024-05-16 12:47:44 -04:00
|
|
|
|
2024-05-17 11:18:37 -04:00
|
|
|
// Set mtvec to the location of our trap handler function
|
|
|
|
"la t1, {trap_vector}",
|
|
|
|
"csrw mtvec, t1",
|
2024-05-16 12:47:44 -04:00
|
|
|
|
2024-05-17 11:18:37 -04:00
|
|
|
// Set MSIE, MTIE, and MEIE on machine interrupt enable CSR:
|
2024-05-17 11:55:43 -04:00
|
|
|
// (1 << 3) = MSIE to enable machine-/M-mode software interrupts
|
2024-05-17 11:40:55 -04:00
|
|
|
// | (1 << 7) = MTIE to enable M-mode timer interrupts (disabled for now)
|
|
|
|
// | (1 << 11) = MEIE to enable M-mode external interrupts
|
|
|
|
"li t2, (1 << 3) | (1 << 11)",
|
|
|
|
"csrw mie, t2",
|
2024-05-16 12:21:27 -04:00
|
|
|
|
2024-05-14 11:06:57 -04:00
|
|
|
// clear the BSS
|
|
|
|
"la t0, _bss_start",
|
|
|
|
"la t1, _bss_end",
|
|
|
|
"bgeu t0, t1, 2f",
|
2024-05-13 23:59:41 -04:00
|
|
|
"1:",
|
2024-05-14 11:06:57 -04:00
|
|
|
"sb zero, 0(t0)",
|
|
|
|
"addi t0, t0, 1",
|
|
|
|
"bne t0, t1, 1b",
|
2024-05-13 23:59:41 -04:00
|
|
|
"2:",
|
2024-05-16 09:19:15 -04:00
|
|
|
"j 4f",
|
2024-05-16 12:21:27 -04:00
|
|
|
// BSS is clear!
|
2024-05-16 09:21:50 -04:00
|
|
|
|
|
|
|
// busy loop if hartid =/= 0
|
2024-05-16 09:19:15 -04:00
|
|
|
"3:",
|
2024-05-16 09:21:50 -04:00
|
|
|
"wfi", // wait for interrupt (or nop)
|
2024-05-16 09:19:15 -04:00
|
|
|
"j 3b",
|
2024-05-16 09:21:50 -04:00
|
|
|
|
2024-05-16 09:19:15 -04:00
|
|
|
"4:",
|
2024-05-15 21:44:54 -04:00
|
|
|
// "tail-call" to {start} (call without saving a return address)
|
|
|
|
"tail {start}",
|
|
|
|
start = sym start, // {start} refers to the function [start] below
|
2024-05-17 11:18:37 -04:00
|
|
|
trap_vector = sym trap::trap_handler, // {trap_vector} refers to function [trap::trap_handler]
|
2024-05-14 11:06:57 -04:00
|
|
|
options(noreturn) // we must handle "returning" from assembly
|
2024-05-13 18:31:37 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-05-15 21:44:54 -04:00
|
|
|
extern "C" fn start() -> ! {
|
2024-05-14 11:03:08 -04:00
|
|
|
// UNSAFE: Called exactly once, right here.
|
|
|
|
unsafe { heap::init() };
|
|
|
|
|
|
|
|
// Now we're free to use dynamic allocation!
|
2024-05-16 01:37:21 -04:00
|
|
|
// {
|
|
|
|
// extern crate alloc;
|
|
|
|
// use alloc::boxed::Box;
|
|
|
|
// let _my_heap_pointer = Box::new(10);
|
|
|
|
// // _my_heap_pointer lives on the heap!
|
|
|
|
// }
|
2024-05-14 11:03:08 -04:00
|
|
|
|
2024-05-16 01:37:21 -04:00
|
|
|
// println!("This should not print because the console is not initialised.");
|
2024-05-13 23:59:41 -04:00
|
|
|
unsafe { uart::init_console(0x1000_0000) };
|
|
|
|
println!("Hello, world!");
|
|
|
|
|
2024-05-16 09:23:08 -04:00
|
|
|
// print current hartid
|
2024-05-16 01:36:19 -04:00
|
|
|
println!("hartid: {}", riscv::register::mhartid::read());
|
|
|
|
|
2024-05-16 09:23:08 -04:00
|
|
|
// poll console for input and print characters back
|
2024-05-13 23:59:41 -04:00
|
|
|
loop {
|
|
|
|
let c = uart::CONSOLE.lock().as_mut().and_then(uart::Device::get);
|
2024-05-17 10:54:07 -04:00
|
|
|
if let Some(c) = c { print!("{}", c as char); }
|
2024-05-13 18:31:37 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-16 12:38:01 -04:00
|
|
|
#[no_mangle]
|
|
|
|
extern "C" fn eh_personality() {}
|
|
|
|
|
2024-05-13 18:31:37 -04:00
|
|
|
#[panic_handler]
|
2024-05-16 12:38:01 -04:00
|
|
|
fn panic(info: &PanicInfo) -> ! {
|
|
|
|
// print panic info and abort
|
2024-05-16 00:13:10 -04:00
|
|
|
println!("{}", info);
|
2024-05-16 12:38:01 -04:00
|
|
|
abort();
|
|
|
|
}
|
2024-05-16 00:13:10 -04:00
|
|
|
|
2024-05-16 12:38:01 -04:00
|
|
|
#[no_mangle]
|
|
|
|
extern "C" fn abort() -> ! {
|
2024-05-18 10:26:17 -04:00
|
|
|
// loop wait for interrupt
|
|
|
|
loop { riscv::asm::wfi(); }
|
2024-05-13 18:31:37 -04:00
|
|
|
}
|
2024-05-14 11:03:08 -04:00
|
|
|
|
2024-05-14 11:06:57 -04:00
|
|
|
// TODO unit testing
|