1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
/// The public interface of the Platform-Level Interrupt Controller driver.
use r3_core::kernel::{InterruptNum, InterruptPriority};

use super::plic_regs;

/// Implement [`InterruptController`] and [`Plic`] on the given kernel trait
/// type using the Platform-Level Interrupt Controller (PLIC) on the target.
/// **Requires [`PlicOptions`] and [`InterruptControllerToPort`].**
///
/// [`InterruptController`]: crate::InterruptController
/// [`InterruptControllerToPort`]: crate::InterruptControllerToPort
///
/// This macro adds a method `const fn configure_plic(b: &mut Cfg<C>)` to the
/// kernel trait type. **It should be called by your application's configuration
/// function.** See the following example:
///
/// ```rust,ignore
/// r3_port_riscv::use_plic!(unsafe impl InterruptController for SystemTraits);
///
/// impl r3_port_riscv::PlicOptions for SystemTraits {
///     // SiFive E
///     const MAX_PRIORITY: InterruptPriority = 7;
///     const MAX_NUM: InterruptNum = 127;
///     const PLIC_BASE: usize = 0x0c00_0000;
/// }
///
/// const fn configure_app(b: &mut r3_kernel::Cfg<SystemTraits>) -> Objects {
///     SystemTraits::configure_plic(b);
///     /* ... */
/// }
/// ```
///
/// # Safety
///
///  - The target must really include a PLIC.
///  - `PlicOptions` should be configured correctly and the memory-mapped
///    registers should be accessible.
///
#[macro_export]
macro_rules! use_plic {
    (unsafe impl InterruptController for $Traits:ty) => {
        const _: () = {
            use $crate::{
                core::ops::Range,
                plic::{imp, plic_regs},
                r3_core::kernel::{
                    traits, Cfg, ClearInterruptLineError, EnableInterruptLineError, InterruptNum,
                    InterruptPriority, PendInterruptLineError, QueryInterruptLineError,
                    SetInterruptLinePriorityError,
                },
                r3_kernel::{PortInterrupts, System},
                InterruptController, Plic, PlicOptions,
            };

            unsafe impl Plic for $Traits {
                fn plic_regs() -> &'static plic_regs::Plic {
                    unsafe { &*(<$Traits as PlicOptions>::PLIC_BASE as *const plic_regs::Plic) }
                }
            }

            impl $Traits {
                pub const fn configure_plic<C>(b: &mut Cfg<C>)
                where
                    C: ~const traits::CfgInterruptLine<System = System<Self>>,
                {
                    imp::configure::<_, Self>(b)
                }
            }

            impl InterruptController for $Traits {
                #[inline]
                unsafe fn init() {
                    imp::init::<Self>()
                }

                const MANAGED_INTERRUPT_PRIORITY_RANGE: Range<InterruptPriority> =
                    0..(<$Traits as PlicOptions>::MAX_PRIORITY + 1);

                #[inline]
                unsafe fn set_interrupt_line_priority(
                    line: InterruptNum,
                    priority: InterruptPriority,
                ) -> Result<(), SetInterruptLinePriorityError> {
                    imp::set_interrupt_line_priority::<Self>(line, priority)
                }

                #[inline]
                unsafe fn enable_interrupt_line(
                    line: InterruptNum,
                ) -> Result<(), EnableInterruptLineError> {
                    imp::enable_interrupt_line::<Self>(line)
                }

                #[inline]
                unsafe fn disable_interrupt_line(
                    line: InterruptNum,
                ) -> Result<(), EnableInterruptLineError> {
                    imp::disable_interrupt_line::<Self>(line)
                }

                #[inline]
                unsafe fn pend_interrupt_line(
                    _line: InterruptNum,
                ) -> Result<(), PendInterruptLineError> {
                    Err(PendInterruptLineError::NotSupported)
                }

                #[inline]
                unsafe fn clear_interrupt_line(
                    _line: InterruptNum,
                ) -> Result<(), ClearInterruptLineError> {
                    Err(ClearInterruptLineError::NotSupported)
                }

                #[inline]
                unsafe fn is_interrupt_line_pending(
                    line: InterruptNum,
                ) -> Result<bool, QueryInterruptLineError> {
                    imp::is_interrupt_line_pending::<Self>(line)
                }
            }
        };
    };
}

/// The options for [`use_plic!`].
pub trait PlicOptions {
    /// The base address of PLIC's memory-mapped registers.
    const PLIC_BASE: usize;

    /// The maximum (highest) interrupt priority supported by the PLIC
    /// implementation.
    const MAX_PRIORITY: InterruptPriority;

    /// The last interrupt source supported by the PLIC implementation. Must be
    /// in range `0..=1023`.
    const MAX_NUM: InterruptNum;

    /// The PLIC context for the hart on which the kernel runs.
    const CONTEXT: usize = 0;

    /// Enables the trick for nested interrupt processing.
    ///
    /// PLIC is not designed to allow nested interrupt processing. When this
    /// flag is enabled, the driver will signal completion earlier to start
    /// accepting higher-priority interrupts.
    ///
    /// The following advices should be taken into consideration when enabling
    /// this option:
    ///
    ///  - This should be disabled when there is at least one interrupt source
    ///    configured to target multiple contexts.
    ///
    ///  - Some PLIC gateway implementations don't clear the pending flag when
    ///    an incoming interrupt request signal is deasserted. The pending flag
    ///    gets set again as soon as completion is signaled, meaning the
    ///    interrupt will be claimed twice every time it's taken.
    ///    The PLIC in FE310 has this issue.
    ///
    /// Defaults to `false` when unspecified.
    const USE_NESTING: bool = false;
}

/// Provides access to a system-global PLIC instance. Implemented by
/// [`use_plic!`].
///
/// # Safety
///
/// This trait is not intended to be implemented in any other means.
pub unsafe trait Plic: PlicOptions {
    #[doc(hidden)]
    /// Get [`plic_regs::Plic`] representing the memory-mapped interface for the
    /// PLIC instance.
    fn plic_regs() -> &'static plic_regs::Plic;
}