#![no_std] #![no_main] #![feature(alloc_error_handler)] #![feature(global_asm)] #![feature(llvm_asm)] use core::alloc::Layout; use core::panic::PanicInfo; use k210_hal::{clock::Clocks, fpioa, pac, prelude::*}; use linked_list_allocator::LockedHeap; use rustsbi::{enter_privileged, print, println}; use riscv::register::{ mcause::{self, Exception, Interrupt, Trap}, medeleg, mepc, mhartid, mideleg, mie, mip, misa::{self, MXL}, mstatus::{self, MPP}, mtval, mtvec::{self, TrapMode}, }; #[global_allocator] static ALLOCATOR: LockedHeap = LockedHeap::empty(); #[panic_handler] fn panic(_info: &PanicInfo) -> ! { loop {} } #[alloc_error_handler] fn oom(_layout: Layout) -> ! { loop {} } fn mp_hook() -> bool { use riscv::asm::wfi; use k210_hal::clint::msip; let hartid = mhartid::read(); if hartid == 0 { true } else { unsafe { // Clear IPI msip::clear_ipi(hartid); // Start listening for software interrupts mie::set_msoft(); loop { wfi(); if mip::read().msoft() { break; } } // Stop listening for software interrupts mie::clear_msoft(); // Clear IPI msip::clear_ipi(hartid); } false } } #[export_name = "_start"] #[link_section = ".text.entry"] // this is stable fn main() -> ! { unsafe { llvm_asm!( " csrr a2, mhartid lui t0, %hi(_max_hart_id) add t0, t0, %lo(_max_hart_id) bgtu a2, t0, _start_abort la sp, _stack_start lui t0, %hi(_hart_stack_size) add t0, t0, %lo(_hart_stack_size) .ifdef __riscv_mul mul t0, a2, t0 .else beqz a2, 2f // Jump if single-hart mv t1, a2 mv t2, t0 1: add t0, t0, t2 addi t1, t1, -1 bnez t1, 1b 2: .endif sub sp, sp, t0 csrw mscratch, zero j _start_success _start_abort: wfi j _start_abort _start_success: " ) }; if mp_hook() { extern "C" { static mut _ebss: u32; static mut _sbss: u32; static mut _edata: u32; static mut _sdata: u32; static _sidata: u32; } unsafe { r0::zero_bss(&mut _sbss, &mut _ebss); r0::init_data(&mut _sdata, &mut _edata, &_sidata); } } extern "C" { fn _start_trap(); } unsafe { mtvec::write(_start_trap as usize, TrapMode::Direct); } if mhartid::read() == 0 { extern "C" { fn _sheap(); fn _heap_size(); } let sheap = &mut _sheap as *mut _ as usize; let heap_size = &_heap_size as *const _ as usize; unsafe { ALLOCATOR.lock().init(sheap, heap_size); } let p = pac::Peripherals::take().unwrap(); let mut sysctl = p.SYSCTL.constrain(); let fpioa = p.FPIOA.split(&mut sysctl.apb0); let clocks = Clocks::new(); let _uarths_tx = fpioa.io5.into_function(fpioa::UARTHS_TX); let _uarths_rx = fpioa.io4.into_function(fpioa::UARTHS_RX); // Configure UART let serial = p.UARTHS.configure(115_200.bps(), &clocks); let (tx, rx) = serial.split(); use rustsbi::legacy_stdio::init_legacy_stdio_embedded_hal_fuse; init_legacy_stdio_embedded_hal_fuse(tx, rx); struct Ipi; impl rustsbi::Ipi for Ipi { fn max_hart_id(&self) -> usize { 1 } fn send_ipi_many(&mut self, hart_mask: rustsbi::HartMask) { use k210_hal::clint::msip; for i in 0..=1 { if hart_mask.has_bit(i) { msip::set_ipi(i); msip::clear_ipi(i); } } } } use rustsbi::init_ipi; init_ipi(Ipi); struct Timer; impl rustsbi::Timer for Timer { fn set_timer(&mut self, stime_value: u64) { // This function must clear the pending timer interrupt bit as well. use k210_hal::clint::mtimecmp; mtimecmp::write(mhartid::read(), stime_value); unsafe { mip::clear_mtimer() }; } } use rustsbi::init_timer; init_timer(Timer); } unsafe { mideleg::set_sext(); mideleg::set_stimer(); mideleg::set_ssoft(); medeleg::set_instruction_misaligned(); medeleg::set_breakpoint(); medeleg::set_user_env_call(); medeleg::set_instruction_page_fault(); medeleg::set_load_page_fault(); medeleg::set_store_page_fault(); mie::set_mext(); // 不打开mie::set_mtimer mie::set_msoft(); } if mhartid::read() == 0 { println!("[rustsbi] Version 0.1.0"); println!("{}", rustsbi::LOGO); println!("[rustsbi] Platform: K210"); let isa = misa::read(); if let Some(isa) = isa { let mxl_str = match isa.mxl() { MXL::XLEN32 => "RV32", MXL::XLEN64 => "RV64", MXL::XLEN128 => "RV128", }; print!("[rustsbi] misa: {}", mxl_str); for ext in 'A'..='Z' { if isa.has_extension(ext) { print!("{}", ext); } } println!(""); } println!("[rustsbi] mideleg: {:#x}", mideleg::read().bits()); println!("[rustsbi] medeleg: {:#x}", medeleg::read().bits()); println!("[rustsbi] Kernel entry: 0x80020000"); } extern "C" { fn _s_mode_start(); } unsafe { mepc::write(_s_mode_start as usize); mstatus::set_mpp(MPP::Supervisor); enter_privileged(mhartid::read(), 0x2333333366666666); } } global_asm!( " .section .text .globl _s_mode_start _s_mode_start: 1: auipc ra, %pcrel_hi(1f) ld ra, %pcrel_lo(1b)(ra) jr ra .align 3 1: .dword 0x80020000 " ); // todo: configurable target address global_asm!( " .equ REGBYTES, 8 .macro STORE reg, offset sd \\reg, \\offset*REGBYTES(sp) .endm .macro LOAD reg, offset ld \\reg, \\offset*REGBYTES(sp) .endm .section .text .global _start_trap .p2align 2 _start_trap: csrrw sp, mscratch, sp bnez sp, 1f /* from M level, load sp */ csrrw sp, mscratch, zero 1: addi sp, sp, -16 * REGBYTES STORE ra, 0 STORE t0, 1 STORE t1, 2 STORE t2, 3 STORE t3, 4 STORE t4, 5 STORE t5, 6 STORE t6, 7 STORE a0, 8 STORE a1, 9 STORE a2, 10 STORE a3, 11 STORE a4, 12 STORE a5, 13 STORE a6, 14 STORE a7, 15 mv a0, sp call _start_trap_rust LOAD ra, 0 LOAD t0, 1 LOAD t1, 2 LOAD t2, 3 LOAD t3, 4 LOAD t4, 5 LOAD t5, 6 LOAD t6, 7 LOAD a0, 8 LOAD a1, 9 LOAD a2, 10 LOAD a3, 11 LOAD a4, 12 LOAD a5, 13 LOAD a6, 14 LOAD a7, 15 addi sp, sp, 16 * REGBYTES csrrw sp, mscratch, sp mret " ); #[allow(unused)] struct TrapFrame { ra: usize, t0: usize, t1: usize, t2: usize, t3: usize, t4: usize, t5: usize, t6: usize, a0: usize, a1: usize, a2: usize, a3: usize, a4: usize, a5: usize, a6: usize, a7: usize, } #[export_name = "_start_trap_rust"] extern "C" fn start_trap_rust(trap_frame: &mut TrapFrame) { let cause = mcause::read().cause(); match cause { Trap::Exception(Exception::SupervisorEnvCall) => { let params = [trap_frame.a0, trap_frame.a1, trap_frame.a2, trap_frame.a3]; let ans = rustsbi::ecall(trap_frame.a7, trap_frame.a6, params); trap_frame.a0 = ans.error; trap_frame.a1 = ans.value; mepc::write(mepc::read().wrapping_add(4)); } Trap::Interrupt(Interrupt::MachineSoft) => { unsafe { mip::set_ssoft(); mie::clear_msoft(); } } Trap::Interrupt(Interrupt::MachineTimer) => { unsafe { mip::set_stimer(); mie::clear_mtimer(); } } Trap::Exception(Exception::IllegalInstruction) => { #[inline] unsafe fn get_vaddr_u32(vaddr: usize) -> u32 { let mut ans: u32; llvm_asm!(" li t0, (1 << 17) mv t1, $1 csrrs t0, mstatus, t0 lwu t1, 0(t1) csrw mstatus, t0 mv $0, t1 " :"=r"(ans) :"r"(vaddr) :"t0", "t1"); ans } let vaddr = mepc::read(); let ins = unsafe { get_vaddr_u32(vaddr) }; if ins & 0xFFFFF07F == 0xC0102073 { // rdtime let rd = ((ins >> 7) & 0b1_1111) as u8; let mtime = k210_hal::clint::mtime::read(); let time_usize = mtime as usize; match rd { 10 => trap_frame.a0 = time_usize, 11 => trap_frame.a1 = time_usize, 12 => trap_frame.a2 = time_usize, 13 => trap_frame.a3 = time_usize, 14 => trap_frame.a4 = time_usize, 15 => trap_frame.a5 = time_usize, 16 => trap_frame.a6 = time_usize, 17 => trap_frame.a7 = time_usize, 5 => trap_frame.t0 = time_usize, 6 => trap_frame.t1 = time_usize, 7 => trap_frame.t2 = time_usize, 28 => trap_frame.t3 = time_usize, 29 => trap_frame.t4 = time_usize, 30 => trap_frame.t5 = time_usize, 31 => trap_frame.t6 = time_usize, _ => panic!("invalid target"), } mepc::write(mepc::read().wrapping_add(4)); } else { panic!("invalid instruction, mepc: {:016x?}, instruction: {:016x?}", mepc::read(), ins); } } cause => panic!( "Unhandled exception! mcause: {:?}, mepc: {:016x?}, mtval: {:016x?}", cause, mepc::read(), mtval::read() ), } }