|
@@ -8,92 +8,7 @@ pub mod pendings;
|
|
|
pub mod priorities;
|
|
|
pub mod threshold;
|
|
|
|
|
|
-/// Trait for enums of interrupt numbers.
|
|
|
-///
|
|
|
-/// This trait should be implemented by a peripheral access crate (PAC)
|
|
|
-/// on its enum of available external interrupts for a specific device.
|
|
|
-/// Each variant must convert to a `u16` of its interrupt number.
|
|
|
-///
|
|
|
-/// # Note
|
|
|
-///
|
|
|
-/// Recall that the interrupt number `0` is reserved as "no interrupt".
|
|
|
-///
|
|
|
-/// # Safety
|
|
|
-///
|
|
|
-/// * This trait must only be implemented on a PAC of a target with a PLIC peripheral.
|
|
|
-/// * This trait must only be implemented on enums of external interrupts.
|
|
|
-/// * Each enum variant must represent a distinct value (no duplicates are permitted),
|
|
|
-/// * Each enum variant must always return the same value (do not change at runtime).
|
|
|
-/// * All the interrupt numbers must be less than or equal to `MAX_INTERRUPT_NUMBER`.
|
|
|
-/// * `MAX_INTERRUPT_NUMBER` must coincide with the highest allowed interrupt number.
|
|
|
-pub unsafe trait InterruptNumber: Copy {
|
|
|
- /// Highest number assigned to an interrupt source.
|
|
|
- const MAX_INTERRUPT_NUMBER: u16;
|
|
|
-
|
|
|
- /// Converts an interrupt source to its corresponding number.
|
|
|
- fn number(self) -> u16;
|
|
|
-
|
|
|
- /// Tries to convert a number to a valid interrupt source.
|
|
|
- /// If the conversion fails, it returns an error with the number back.
|
|
|
- fn from_number(value: u16) -> Result<Self, u16>;
|
|
|
-}
|
|
|
-
|
|
|
-/// Trait for enums of priority levels.
|
|
|
-///
|
|
|
-/// This trait should be implemented by a peripheral access crate (PAC)
|
|
|
-/// on its enum of available priority numbers for a specific device.
|
|
|
-/// Each variant must convert to a `u8` of its priority level.
|
|
|
-///
|
|
|
-/// # Note
|
|
|
-///
|
|
|
-/// Recall that the priority number `0` is reserved as "never interrupt".
|
|
|
-///
|
|
|
-/// # Safety
|
|
|
-///
|
|
|
-/// * This trait must only be implemented on a PAC of a target with a PLIC peripheral.
|
|
|
-/// * This trait must only be implemented on enums of priority levels.
|
|
|
-/// * Each enum variant must represent a distinct value (no duplicates are permitted).
|
|
|
-/// * Each enum variant must always return the same value (do not change at runtime).
|
|
|
-/// * There must be a valid priority number set to 0 (i.e., never interrupt).
|
|
|
-/// * All the priority level numbers must be less than or equal to `MAX_PRIORITY_NUMBER`.
|
|
|
-/// * `MAX_PRIORITY_NUMBER` must coincide with the highest allowed priority number.
|
|
|
-pub unsafe trait PriorityNumber: Copy {
|
|
|
- /// Number assigned to the highest priority level.
|
|
|
- const MAX_PRIORITY_NUMBER: u8;
|
|
|
-
|
|
|
- /// Converts a priority level to its corresponding number.
|
|
|
- fn number(self) -> u8;
|
|
|
-
|
|
|
- /// Tries to convert a number to a valid priority level.
|
|
|
- /// If the conversion fails, it returns an error with the number back.
|
|
|
- fn from_number(value: u8) -> Result<Self, u8>;
|
|
|
-}
|
|
|
-
|
|
|
-/// Trait for enums of PLIC contexts.
|
|
|
-///
|
|
|
-/// This trait should be implemented by a peripheral access crate (PAC)
|
|
|
-/// on its enum of available contexts for a specific device.
|
|
|
-/// Each variant must convert to a `u16` of its context number.
|
|
|
-///
|
|
|
-/// # Safety
|
|
|
-///
|
|
|
-/// * This trait must only be implemented on a PAC of a target with a PLIC peripheral.
|
|
|
-/// * This trait must only be implemented on enums of contexts.
|
|
|
-/// * Each enum variant must represent a distinct value (no duplicates are permitted),
|
|
|
-/// * Each anum variant must always return the same value (do not change at runtime).
|
|
|
-/// * All the context numbers must be less than or equal to `MAX_CONTEXT_NUMBER`.
|
|
|
-/// * `MAX_CONTEXT_NUMBER` must coincide with the highest allowed context number.
|
|
|
-pub unsafe trait ContextNumber: Copy {
|
|
|
- /// Highest number assigned to a context.
|
|
|
- const MAX_CONTEXT_NUMBER: u16;
|
|
|
-
|
|
|
- /// Converts an context to its corresponding number.
|
|
|
- fn number(self) -> u16;
|
|
|
-
|
|
|
- /// Tries to convert a number to a valid context.
|
|
|
- /// If the conversion fails, it returns an error with the number back.
|
|
|
- fn from_number(value: u16) -> Result<Self, u16>;
|
|
|
-}
|
|
|
+pub use riscv_pac::{HartIdNumber, InterruptNumber, PriorityNumber}; // re-export useful riscv-pac traits
|
|
|
|
|
|
/// Trait for a PLIC peripheral.
|
|
|
///
|
|
@@ -144,11 +59,11 @@ impl<P: Plic> PLIC<P> {
|
|
|
unsafe { pendings::PENDINGS::new(P::BASE + Self::PENDINGS_OFFSET) }
|
|
|
}
|
|
|
|
|
|
- /// Returns a proxy to access to all the PLIC registers of a given context.
|
|
|
+ /// Returns a proxy to access to all the PLIC registers of a given HART context.
|
|
|
#[inline]
|
|
|
- pub fn ctx<C: ContextNumber>(context: C) -> CTX<P> {
|
|
|
+ pub fn ctx<H: HartIdNumber>(hart_id: H) -> CTX<P> {
|
|
|
// SAFETY: valid context number
|
|
|
- unsafe { CTX::new(context.number()) }
|
|
|
+ unsafe { CTX::new(hart_id.number()) }
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -216,7 +131,7 @@ impl<P: Plic> CTX<P> {
|
|
|
|
|
|
#[cfg(test)]
|
|
|
pub(crate) mod test {
|
|
|
- use super::{ContextNumber, InterruptNumber, PriorityNumber};
|
|
|
+ use super::{HartIdNumber, InterruptNumber, PriorityNumber};
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
|
|
#[repr(u16)]
|
|
@@ -282,8 +197,8 @@ pub(crate) mod test {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- unsafe impl ContextNumber for Context {
|
|
|
- const MAX_CONTEXT_NUMBER: u16 = 2;
|
|
|
+ unsafe impl HartIdNumber for Context {
|
|
|
+ const MAX_HART_ID_NUMBER: u16 = 2;
|
|
|
|
|
|
#[inline]
|
|
|
fn number(self) -> u16 {
|
|
@@ -292,7 +207,7 @@ pub(crate) mod test {
|
|
|
|
|
|
#[inline]
|
|
|
fn from_number(number: u16) -> Result<Self, u16> {
|
|
|
- if number > Self::MAX_CONTEXT_NUMBER {
|
|
|
+ if number > Self::MAX_HART_ID_NUMBER {
|
|
|
Err(number)
|
|
|
} else {
|
|
|
// SAFETY: valid context number
|
|
@@ -359,7 +274,7 @@ pub(crate) mod test {
|
|
|
assert_eq!(priorities.address(), 0x0C00_0000);
|
|
|
assert_eq!(pendings.address(), 0x0C00_1000);
|
|
|
|
|
|
- for i in 0..=Context::MAX_CONTEXT_NUMBER {
|
|
|
+ for i in 0..=Context::MAX_HART_ID_NUMBER {
|
|
|
let context = Context::from_number(i).unwrap();
|
|
|
let i = i as usize;
|
|
|
|