|
@@ -2,16 +2,21 @@ use aclint::SifiveClint as Clint;
|
|
|
use core::arch::asm;
|
|
|
use fast_trap::{trap_entry, FastContext, FastResult};
|
|
|
use riscv::register::{
|
|
|
- mepc, mie, mstatus};
|
|
|
+ mcause::{self, Exception as E, Trap as T},
|
|
|
+ mepc, mie, mstatus, mtval, satp, sstatus,
|
|
|
+};
|
|
|
use rustsbi::RustSBI;
|
|
|
|
|
|
use crate::board::SBI;
|
|
|
use crate::clint::{self, SIFIVECLINT};
|
|
|
use crate::console::{MachineConsole, CONSOLE};
|
|
|
-use crate::hart_id;
|
|
|
+use crate::current_hartid;
|
|
|
use crate::hsm::local_hsm;
|
|
|
+use crate::rfence::{local_rfence, RFenceType};
|
|
|
use crate::riscv_spec::{CSR_TIME, CSR_TIMEH};
|
|
|
|
|
|
+const PAGE_SIZE: usize = 4096;
|
|
|
+
|
|
|
#[naked]
|
|
|
pub(crate) unsafe extern "C" fn trap_vec() {
|
|
|
asm!(
|
|
@@ -32,8 +37,8 @@ pub(crate) unsafe extern "C" fn trap_vec() {
|
|
|
"j {default}", // machine external
|
|
|
".option pop",
|
|
|
default = sym trap_entry,
|
|
|
- mtimer = sym mtimer,
|
|
|
msoft = sym msoft,
|
|
|
+ mtimer = sym mtimer,
|
|
|
options(noreturn)
|
|
|
)
|
|
|
}
|
|
@@ -91,49 +96,189 @@ unsafe extern "C" fn mtimer() {
|
|
|
|
|
|
/// machine soft 中断代理
|
|
|
///
|
|
|
-/// # Safety
|
|
|
-///
|
|
|
-/// 裸函数。
|
|
|
#[naked]
|
|
|
-unsafe extern "C" fn msoft() {
|
|
|
+unsafe extern "C" fn msoft() -> ! {
|
|
|
asm!(
|
|
|
- // 换栈:
|
|
|
- // sp : M sp
|
|
|
- // mscratch: S sp
|
|
|
- " csrrw sp, mscratch, sp",
|
|
|
- // 保护
|
|
|
- " addi sp, sp, -3*8
|
|
|
- sd ra, 0*8(sp)
|
|
|
- sd a0, 1*8(sp)
|
|
|
- sd a1, 2*8(sp)
|
|
|
- ",
|
|
|
- // 清除 msip 设置 ssip
|
|
|
- " la a0, {clint_ptr}
|
|
|
- ld a0, (a0)
|
|
|
- csrr a1, mhartid
|
|
|
- call {clear_msip}
|
|
|
- csrrsi zero, mip, 1 << 1
|
|
|
- ",
|
|
|
- // 恢复
|
|
|
- " ld ra, 0*8(sp)
|
|
|
- ld a0, 1*8(sp)
|
|
|
- ld a1, 2*8(sp)
|
|
|
- addi sp, sp, 3*8
|
|
|
- ",
|
|
|
- // 换栈:
|
|
|
- // sp : S sp
|
|
|
- // mscratch: M sp
|
|
|
- " csrrw sp, mscratch, sp",
|
|
|
- // 返回
|
|
|
- " mret",
|
|
|
- clint_ptr = sym SIFIVECLINT,
|
|
|
- // Clint::clear_msip_naked(&self, hart_idx)
|
|
|
- clear_msip = sym Clint::clear_msip_naked,
|
|
|
+ ".align 2",
|
|
|
+ "csrrw sp, mscratch, sp",
|
|
|
+ "addi sp, sp, -32*8",
|
|
|
+ "sd ra, 0*8(sp)
|
|
|
+ sd gp, 2*8(sp)
|
|
|
+ sd tp, 3*8(sp)
|
|
|
+ sd t0, 4*8(sp)
|
|
|
+ sd t1, 5*8(sp)
|
|
|
+ sd t2, 6*8(sp)
|
|
|
+ sd s0, 7*8(sp)
|
|
|
+ sd s1, 8*8(sp)
|
|
|
+ sd a0, 9*8(sp)
|
|
|
+ sd a1, 10*8(sp)
|
|
|
+ sd a2, 11*8(sp)
|
|
|
+ sd a3, 12*8(sp)
|
|
|
+ sd a4, 13*8(sp)
|
|
|
+ sd a5, 14*8(sp)
|
|
|
+ sd a6, 15*8(sp)
|
|
|
+ sd a7, 16*8(sp)
|
|
|
+ sd s2, 17*8(sp)
|
|
|
+ sd s3, 18*8(sp)
|
|
|
+ sd s4, 19*8(sp)
|
|
|
+ sd s5, 20*8(sp)
|
|
|
+ sd s6, 21*8(sp)
|
|
|
+ sd s7, 22*8(sp)
|
|
|
+ sd s8, 23*8(sp)
|
|
|
+ sd s9, 24*8(sp)
|
|
|
+ sd s10, 25*8(sp)
|
|
|
+ sd s11, 26*8(sp)
|
|
|
+ sd t3, 27*8(sp)
|
|
|
+ sd t4, 28*8(sp)
|
|
|
+ sd t5, 29*8(sp)
|
|
|
+ sd t6, 30*8(sp)",
|
|
|
+ "csrr t0, mepc
|
|
|
+ sd t0, 31*8(sp)",
|
|
|
+ "csrr t2, mscratch",
|
|
|
+ "sd t2, 1*8(sp)",
|
|
|
+ "mv a0, sp",
|
|
|
+ "call {msoft_hanlder}",
|
|
|
+ "ld t0, 31*8(sp)
|
|
|
+ csrw mepc, t0",
|
|
|
+ "ld ra, 0*8(sp)
|
|
|
+ ld gp, 2*8(sp)
|
|
|
+ ld tp, 3*8(sp)
|
|
|
+ ld t0, 4*8(sp)
|
|
|
+ ld t1, 5*8(sp)
|
|
|
+ ld t2, 6*8(sp)
|
|
|
+ ld s0, 7*8(sp)
|
|
|
+ ld s1, 8*8(sp)
|
|
|
+ ld a0, 9*8(sp)
|
|
|
+ ld a1, 10*8(sp)
|
|
|
+ ld a2, 11*8(sp)
|
|
|
+ ld a3, 12*8(sp)
|
|
|
+ ld a4, 13*8(sp)
|
|
|
+ ld a5, 14*8(sp)
|
|
|
+ ld a6, 15*8(sp)
|
|
|
+ ld a7, 16*8(sp)
|
|
|
+ ld s2, 17*8(sp)
|
|
|
+ ld s3, 18*8(sp)
|
|
|
+ ld s4, 19*8(sp)
|
|
|
+ ld s5, 20*8(sp)
|
|
|
+ ld s6, 21*8(sp)
|
|
|
+ ld s7, 22*8(sp)
|
|
|
+ ld s8, 23*8(sp)
|
|
|
+ ld s9, 24*8(sp)
|
|
|
+ ld s10, 25*8(sp)
|
|
|
+ ld s11, 26*8(sp)
|
|
|
+ ld t3, 27*8(sp)
|
|
|
+ ld t4, 28*8(sp)
|
|
|
+ ld t5, 29*8(sp)
|
|
|
+ ld t6, 30*8(sp)",
|
|
|
+ "addi sp, sp, 32*8",
|
|
|
+ "csrrw sp, mscratch, sp",
|
|
|
+ "mret",
|
|
|
+ msoft_hanlder = sym msoft_hanlder,
|
|
|
options(noreturn)
|
|
|
- )
|
|
|
+ );
|
|
|
+}
|
|
|
+
|
|
|
+pub extern "C" fn msoft_hanlder(ctx: &mut SupervisorContext) {
|
|
|
+ #[inline]
|
|
|
+ fn boot(ctx: &mut SupervisorContext, start_addr: usize, opaque: usize) {
|
|
|
+ unsafe {
|
|
|
+ sstatus::clear_sie();
|
|
|
+ satp::write(0);
|
|
|
+ }
|
|
|
+ ctx.a0 = current_hartid();
|
|
|
+ ctx.a1 = opaque;
|
|
|
+ ctx.mepc = start_addr;
|
|
|
+ }
|
|
|
+
|
|
|
+ match local_hsm().start() {
|
|
|
+ // HSM Start
|
|
|
+ Ok(next_stage) => {
|
|
|
+ clint::clear_msip();
|
|
|
+ unsafe {
|
|
|
+ mstatus::set_mpie();
|
|
|
+ mstatus::set_mpp(next_stage.next_mode);
|
|
|
+ mie::set_msoft();
|
|
|
+ mie::set_mtimer();
|
|
|
+ }
|
|
|
+ info!(
|
|
|
+ "hart {} => start_addr : 0x{:16x}",
|
|
|
+ current_hartid(),
|
|
|
+ next_stage.start_addr
|
|
|
+ );
|
|
|
+ boot(ctx, next_stage.start_addr, next_stage.opaque);
|
|
|
+ }
|
|
|
+ Err(rustsbi::spec::hsm::HART_STOP) => {
|
|
|
+ clint::clear_msip();
|
|
|
+ unsafe {
|
|
|
+ mie::set_msoft();
|
|
|
+ }
|
|
|
+ riscv::asm::wfi();
|
|
|
+ }
|
|
|
+ // RFence
|
|
|
+ _ => {
|
|
|
+ msoft_rfence_handler();
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
-/// Fast Trap
|
|
|
+pub fn msoft_rfence_handler() {
|
|
|
+ let rfence_context = local_rfence().get();
|
|
|
+ match rfence_context {
|
|
|
+ Some(ctx) => match ctx.op {
|
|
|
+ RFenceType::FenceI => unsafe {
|
|
|
+ asm!("fence.i");
|
|
|
+ local_rfence().sync();
|
|
|
+ clint::clear_msip();
|
|
|
+ },
|
|
|
+ RFenceType::SFenceVma => {
|
|
|
+ if (ctx.start_addr == 0 && ctx.size == 0) || (ctx.size == usize::MAX) {
|
|
|
+ unsafe {
|
|
|
+ asm!("sfence.vma");
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ for offset in (0..ctx.size).step_by(PAGE_SIZE) {
|
|
|
+ let addr = ctx.start_addr + offset;
|
|
|
+ unsafe {
|
|
|
+ asm!("sfence.vma {}", in(reg) addr);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ local_rfence().sync();
|
|
|
+ clint::clear_msip();
|
|
|
+ }
|
|
|
+ RFenceType::SFenceVmaAsid => {
|
|
|
+ let asid = ctx.asid;
|
|
|
+ if (ctx.start_addr == 0 && ctx.size == 0) || (ctx.size == usize::MAX) {
|
|
|
+ unsafe {
|
|
|
+ asm!("sfence.vma {}, {}", in(reg) 0, in(reg) asid);
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ for offset in (0..ctx.size).step_by(PAGE_SIZE) {
|
|
|
+ let addr = ctx.start_addr + offset;
|
|
|
+ unsafe {
|
|
|
+ asm!("sfence.vma {}, {}", in(reg) addr, in(reg) asid);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ local_rfence().sync();
|
|
|
+ clint::clear_msip();
|
|
|
+ }
|
|
|
+ _ => {
|
|
|
+ error!("Unsupported RFence Type!");
|
|
|
+ local_rfence().sync();
|
|
|
+ clint::clear_msip();
|
|
|
+ }
|
|
|
+ },
|
|
|
+ None => {
|
|
|
+ clint::clear_msip();
|
|
|
+ unsafe {
|
|
|
+ riscv::register::mip::set_ssoft();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/// Fast trap
|
|
|
pub extern "C" fn fast_handler(
|
|
|
mut ctx: FastContext,
|
|
|
a1: usize,
|
|
@@ -144,18 +289,13 @@ pub extern "C" fn fast_handler(
|
|
|
a6: usize,
|
|
|
a7: usize,
|
|
|
) -> FastResult {
|
|
|
- use riscv::register::{
|
|
|
- mcause::{self, Exception as E, Trap as T},
|
|
|
- mtval, satp, sstatus,
|
|
|
- };
|
|
|
-
|
|
|
#[inline]
|
|
|
fn boot(mut ctx: FastContext, start_addr: usize, opaque: usize) -> FastResult {
|
|
|
unsafe {
|
|
|
sstatus::clear_sie();
|
|
|
satp::write(0);
|
|
|
}
|
|
|
- ctx.regs().a[0] = hart_id();
|
|
|
+ ctx.regs().a[0] = current_hartid();
|
|
|
ctx.regs().a[1] = opaque;
|
|
|
ctx.regs().pc = start_addr;
|
|
|
ctx.call(2)
|
|
@@ -163,20 +303,26 @@ pub extern "C" fn fast_handler(
|
|
|
loop {
|
|
|
match local_hsm().start() {
|
|
|
Ok(next_stage) => {
|
|
|
+ clint::clear_msip();
|
|
|
unsafe {
|
|
|
mstatus::set_mpie();
|
|
|
mstatus::set_mpp(next_stage.next_mode);
|
|
|
mie::set_msoft();
|
|
|
mie::set_mtimer();
|
|
|
}
|
|
|
+ info!(
|
|
|
+ "hart {} => start_addr : 0x{:16x}",
|
|
|
+ current_hartid(),
|
|
|
+ next_stage.start_addr
|
|
|
+ );
|
|
|
break boot(ctx, next_stage.start_addr, next_stage.opaque);
|
|
|
}
|
|
|
Err(rustsbi::spec::hsm::HART_STOP) => {
|
|
|
+ clint::clear_msip();
|
|
|
unsafe {
|
|
|
mie::set_msoft();
|
|
|
}
|
|
|
riscv::asm::wfi();
|
|
|
- clint::clear_msip();
|
|
|
}
|
|
|
_ => match mcause::read().cause() {
|
|
|
// SBI call
|
|
@@ -263,7 +409,7 @@ pub extern "C" fn fast_handler(
|
|
|
mepc::read(),
|
|
|
mtval::read()
|
|
|
);
|
|
|
- panic!("stopped with unsupported trap")
|
|
|
+ panic!("Stopped with unsupported trap")
|
|
|
}
|
|
|
},
|
|
|
}
|
|
@@ -272,15 +418,14 @@ pub extern "C" fn fast_handler(
|
|
|
|
|
|
#[inline]
|
|
|
fn delegate() {
|
|
|
- use riscv::register::{sepc, scause, stval, stvec, sstatus, mepc, mcause, mtval};
|
|
|
+ use riscv::register::{mcause, mepc, mtval, scause, sepc, sstatus, stval, stvec};
|
|
|
unsafe {
|
|
|
-
|
|
|
// TODO: 当支持中断嵌套时,需要从ctx里获取mpec。当前ctx.reg().pc与mepc不一致
|
|
|
sepc::write(mepc::read());
|
|
|
scause::write(mcause::read().bits());
|
|
|
stval::write(mtval::read());
|
|
|
sstatus::clear_sie();
|
|
|
- if mstatus::read().mpp() == mstatus::MPP::Supervisor {
|
|
|
+ if mstatus::read().mpp() == mstatus::MPP::Supervisor {
|
|
|
sstatus::set_spp(sstatus::SPP::Supervisor);
|
|
|
} else {
|
|
|
sstatus::set_spp(sstatus::SPP::User);
|
|
@@ -291,8 +436,8 @@ fn delegate() {
|
|
|
|
|
|
#[inline]
|
|
|
fn illegal_instruction_handler(ctx: &mut FastContext) -> bool {
|
|
|
- use riscv_decode::{decode, Instruction};
|
|
|
use riscv::register::{mepc, mtval};
|
|
|
+ use riscv_decode::{decode, Instruction};
|
|
|
|
|
|
let inst = decode(mtval::read() as u32);
|
|
|
match inst {
|
|
@@ -303,7 +448,11 @@ fn illegal_instruction_handler(ctx: &mut FastContext) -> bool {
|
|
|
"Unsupported CSR rd: {}",
|
|
|
csr.rd()
|
|
|
);
|
|
|
- ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_mut() }.clint.as_ref().unwrap().get_time();
|
|
|
+ ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_mut() }
|
|
|
+ .clint
|
|
|
+ .as_ref()
|
|
|
+ .unwrap()
|
|
|
+ .get_time();
|
|
|
}
|
|
|
CSR_TIMEH => {
|
|
|
assert!(
|
|
@@ -311,7 +460,11 @@ fn illegal_instruction_handler(ctx: &mut FastContext) -> bool {
|
|
|
"Unsupported CSR rd: {}",
|
|
|
csr.rd()
|
|
|
);
|
|
|
- ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_mut() }.clint.as_ref().unwrap().get_timeh();
|
|
|
+ ctx.regs().a[(csr.rd() - 10) as usize] = unsafe { SBI.assume_init_mut() }
|
|
|
+ .clint
|
|
|
+ .as_ref()
|
|
|
+ .unwrap()
|
|
|
+ .get_timeh();
|
|
|
}
|
|
|
_ => return false,
|
|
|
},
|
|
@@ -320,3 +473,40 @@ fn illegal_instruction_handler(ctx: &mut FastContext) -> bool {
|
|
|
mepc::write(mepc::read() + 4);
|
|
|
true
|
|
|
}
|
|
|
+
|
|
|
+#[derive(Debug)]
|
|
|
+#[repr(C)]
|
|
|
+pub struct SupervisorContext {
|
|
|
+ pub ra: usize, // 0
|
|
|
+ pub sp: usize,
|
|
|
+ pub gp: usize,
|
|
|
+ pub tp: usize,
|
|
|
+ pub t0: usize,
|
|
|
+ pub t1: usize,
|
|
|
+ pub t2: usize,
|
|
|
+ pub s0: usize,
|
|
|
+ pub s1: usize,
|
|
|
+ pub a0: usize,
|
|
|
+ pub a1: usize,
|
|
|
+ pub a2: usize,
|
|
|
+ pub a3: usize,
|
|
|
+ pub a4: usize,
|
|
|
+ pub a5: usize,
|
|
|
+ pub a6: usize,
|
|
|
+ pub a7: usize,
|
|
|
+ pub s2: usize,
|
|
|
+ pub s3: usize,
|
|
|
+ pub s4: usize,
|
|
|
+ pub s5: usize,
|
|
|
+ pub s6: usize,
|
|
|
+ pub s7: usize,
|
|
|
+ pub s8: usize,
|
|
|
+ pub s9: usize,
|
|
|
+ pub s10: usize,
|
|
|
+ pub s11: usize,
|
|
|
+ pub t3: usize,
|
|
|
+ pub t4: usize,
|
|
|
+ pub t5: usize,
|
|
|
+ pub t6: usize, // 30
|
|
|
+ pub mepc: usize, // 31
|
|
|
+}
|