|
@@ -3,11 +3,12 @@
|
|
|
// NOTE: Adapted from cortex-m/src/interrupt.rs
|
|
|
|
|
|
pub mod machine {
|
|
|
- use crate::register::mstatus;
|
|
|
+ use crate::register::{mepc, mstatus};
|
|
|
|
|
|
/// Disables all interrupts in the current hart (machine mode).
|
|
|
#[inline]
|
|
|
pub fn disable() {
|
|
|
+ // SAFETY: It is safe to disable interrupts
|
|
|
unsafe { mstatus::clear_mie() }
|
|
|
}
|
|
|
|
|
@@ -49,13 +50,55 @@ pub mod machine {
|
|
|
|
|
|
r
|
|
|
}
|
|
|
+
|
|
|
+ /// Execute closure `f` with interrupts enabled in the current hart (machine mode).
|
|
|
+ ///
|
|
|
+ /// This method is assumed to be called within an interrupt handler, and allows
|
|
|
+ /// nested interrupts to occur. After the closure `f` is executed, the [`mstatus`]
|
|
|
+ /// and [`mepc`] registers are properly restored to their previous values.
|
|
|
+ ///
|
|
|
+ /// # Safety
|
|
|
+ ///
|
|
|
+ /// - Do not call this function inside a critical section.
|
|
|
+ /// - This method is assumed to be called within an interrupt handler.
|
|
|
+ /// - Make sure to clear the interrupt flag that caused the interrupt before calling
|
|
|
+ /// this method. Otherwise, the interrupt will be re-triggered before executing `f`.
|
|
|
+ #[inline]
|
|
|
+ pub unsafe fn nested<F, R>(f: F) -> R
|
|
|
+ where
|
|
|
+ F: FnOnce() -> R,
|
|
|
+ {
|
|
|
+ let mstatus = mstatus::read();
|
|
|
+ let mepc = mepc::read();
|
|
|
+
|
|
|
+ // enable interrupts to allow nested interrupts
|
|
|
+ enable();
|
|
|
+
|
|
|
+ let r = f();
|
|
|
+
|
|
|
+ // If the interrupts were inactive before our `enable` call, then re-disable
|
|
|
+ // them. Otherwise, keep them enabled
|
|
|
+ if !mstatus.mie() {
|
|
|
+ disable();
|
|
|
+ }
|
|
|
+
|
|
|
+ // Restore MSTATUS.PIE, MSTATUS.MPP, and SEPC
|
|
|
+ if mstatus.mpie() {
|
|
|
+ mstatus::set_mpie();
|
|
|
+ }
|
|
|
+ mstatus::set_mpp(mstatus.mpp());
|
|
|
+ mepc::write(mepc);
|
|
|
+
|
|
|
+ r
|
|
|
+ }
|
|
|
}
|
|
|
pub mod supervisor {
|
|
|
- use crate::register::sstatus;
|
|
|
+ use crate::register::{sepc, sstatus};
|
|
|
|
|
|
/// Disables all interrupts in the current hart (supervisor mode).
|
|
|
#[inline]
|
|
|
pub fn disable() {
|
|
|
+ // SAFETY: It is safe to disable interrupts
|
|
|
unsafe { sstatus::clear_sie() }
|
|
|
}
|
|
|
|
|
@@ -97,6 +140,46 @@ pub mod supervisor {
|
|
|
|
|
|
r
|
|
|
}
|
|
|
+
|
|
|
+ /// Execute closure `f` with interrupts enabled in the current hart (supervisor mode).
|
|
|
+ /// This method is assumed to be called within an interrupt handler, and allows
|
|
|
+ /// nested interrupts to occur. After the closure `f` is executed, the [`sstatus`]
|
|
|
+ /// and [`sepc`] registers are properly restored to their previous values.
|
|
|
+ ///
|
|
|
+ /// # Safety
|
|
|
+ ///
|
|
|
+ /// - Do not call this function inside a critical section.
|
|
|
+ /// - This method is assumed to be called within an interrupt handler.
|
|
|
+ /// - Make sure to clear the interrupt flag that caused the interrupt before calling
|
|
|
+ /// this method. Otherwise, the interrupt will be re-triggered before executing `f`.
|
|
|
+ #[inline]
|
|
|
+ pub unsafe fn nested<F, R>(f: F) -> R
|
|
|
+ where
|
|
|
+ F: FnOnce() -> R,
|
|
|
+ {
|
|
|
+ let sstatus = sstatus::read();
|
|
|
+ let sepc = sepc::read();
|
|
|
+
|
|
|
+ // enable interrupts to allow nested interrupts
|
|
|
+ enable();
|
|
|
+
|
|
|
+ let r = f();
|
|
|
+
|
|
|
+ // If the interrupts were inactive before our `enable` call, then re-disable
|
|
|
+ // them. Otherwise, keep them enabled
|
|
|
+ if !sstatus.sie() {
|
|
|
+ disable();
|
|
|
+ }
|
|
|
+
|
|
|
+ // Restore SSTATUS.SPIE, SSTATUS.SPP, and SEPC
|
|
|
+ if sstatus.spie() {
|
|
|
+ sstatus::set_spie();
|
|
|
+ }
|
|
|
+ sstatus::set_spp(sstatus.spp());
|
|
|
+ sepc::write(sepc);
|
|
|
+
|
|
|
+ r
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
#[cfg(not(feature = "s-mode"))]
|