1// Licensed under the Apache License, Version 2.0 or the MIT License.
2// SPDX-License-Identifier: Apache-2.0 OR MIT
3// Copyright Tock Contributors 2022.
45use core::fmt::Write;
6use kernel::debug;
7use kernel::hil::time::Freq32KHz;
8use kernel::platform::chip::InterruptService;
9use kernel::utilities::registers::interfaces::Readable;
1011use crate::clint;
12use crate::interrupts;
13use rv32i::pmp::{simple::SimplePMP, PMPUserMPU};
1415extern "C" {
16fn _start_trap();
17}
1819pub type ArtyExxClint<'a> = sifive::clint::Clint<'a, Freq32KHz>;
2021pub struct ArtyExx<'a, I: InterruptService + 'a> {
22 pmp: PMPUserMPU<2, SimplePMP<4>>,
23 userspace_kernel_boundary: rv32i::syscall::SysCall,
24 clic: rv32i::clic::Clic,
25 machinetimer: &'a ArtyExxClint<'a>,
26 interrupt_service: &'a I,
27}
2829pub struct ArtyExxDefaultPeripherals<'a> {
30pub machinetimer: ArtyExxClint<'a>,
31pub gpio_port: crate::gpio::Port<'a>,
32pub uart0: sifive::uart::Uart<'a>,
33}
3435impl ArtyExxDefaultPeripherals<'_> {
36pub fn new() -> Self {
37Self {
38 machinetimer: ArtyExxClint::new(&clint::CLINT_BASE),
39 gpio_port: crate::gpio::Port::new(),
40 uart0: sifive::uart::Uart::new(crate::uart::UART0_BASE, 32_000_000),
41 }
42 }
4344// Resolves any circular dependencies and sets up deferred calls
45pub fn init(&'static self) {
46 kernel::deferred_call::DeferredCallClient::register(&self.uart0);
47 }
48}
4950impl InterruptService for ArtyExxDefaultPeripherals<'_> {
51unsafe fn service_interrupt(&self, interrupt: u32) -> bool {
52match interrupt {
53 interrupts::MTIP => self.machinetimer.handle_interrupt(),
5455 interrupts::GPIO0 => self.gpio_port[0].handle_interrupt(),
56 interrupts::GPIO1 => self.gpio_port[1].handle_interrupt(),
57 interrupts::GPIO2 => self.gpio_port[2].handle_interrupt(),
58 interrupts::GPIO3 => self.gpio_port[3].handle_interrupt(),
59 interrupts::GPIO4 => self.gpio_port[4].handle_interrupt(),
60 interrupts::GPIO5 => self.gpio_port[5].handle_interrupt(),
61 interrupts::GPIO6 => self.gpio_port[6].handle_interrupt(),
62 interrupts::GPIO7 => self.gpio_port[7].handle_interrupt(),
63 interrupts::GPIO8 => self.gpio_port[8].handle_interrupt(),
64 interrupts::GPIO9 => self.gpio_port[9].handle_interrupt(),
65 interrupts::GPIO10 => self.gpio_port[10].handle_interrupt(),
66 interrupts::GPIO11 => self.gpio_port[11].handle_interrupt(),
67 interrupts::GPIO12 => self.gpio_port[12].handle_interrupt(),
68 interrupts::GPIO13 => self.gpio_port[13].handle_interrupt(),
69 interrupts::GPIO14 => self.gpio_port[14].handle_interrupt(),
70 interrupts::GPIO15 => self.gpio_port[15].handle_interrupt(),
7172 interrupts::UART0 => self.uart0.handle_interrupt(),
7374_ => return false,
75 }
76true
77}
78}
7980impl<'a, I: InterruptService + 'a> ArtyExx<'a, I> {
81pub unsafe fn new(machinetimer: &'a ArtyExxClint<'a>, interrupt_service: &'a I) -> Self {
82// Make a bit-vector of all interrupt locations that we actually intend
83 // to use on this chip.
84 // 0001 1111 1111 1111 1111 0000 0000 1000 0000
85let in_use_interrupts: u64 = 0x1FFFF0080;
8687Self {
88 pmp: PMPUserMPU::new(SimplePMP::new().unwrap()),
89 userspace_kernel_boundary: rv32i::syscall::SysCall::new(),
90 clic: rv32i::clic::Clic::new(in_use_interrupts),
91 machinetimer,
92 interrupt_service,
93 }
94 }
9596pub fn enable_all_interrupts(&self) {
97self.clic.enable_all();
98 }
99100/// By default the machine timer is enabled and will trigger interrupts. To
101 /// prevent that we can make the compare register very large to effectively
102 /// stop the interrupt from triggering, and then the machine timer can be
103 /// used later as needed.
104pub unsafe fn disable_machine_timer(&self) {
105self.machinetimer.disable_machine_timer();
106 }
107108/// Setup the function that should run when a trap happens.
109 ///
110 /// This needs to be chip specific because how the CLIC works is configured
111 /// when the trap handler address is specified in mtvec, and that is only
112 /// valid for platforms with a CLIC.
113#[cfg(any(doc, all(target_arch = "riscv32", target_os = "none")))]
114pub unsafe fn configure_trap_handler(&self) {
115use core::arch::asm;
116asm!(
117"
118 // The csrw instruction writes a Control and Status Register (CSR)
119 // with a new value.
120 //
121 // CSR 0x305 (mtvec, 'Machine trap-handler base address.') sets the
122 // address of the trap handler. We do not care about its old value,
123 // so we don't bother reading it. We want to enable direct CLIC mode
124 // so we set the second lowest bit.
125 lui t0, %hi(_start_trap)
126 addi t0, t0, %lo(_start_trap)
127 ori t0, t0, 0x02 // Set CLIC direct mode
128 csrw 0x305, t0 // Write the mtvec CSR.
129 ",
130 out("t0") _
131);
132 }
133134// Mock implementation for tests on Travis-CI.
135#[cfg(not(any(doc, all(target_arch = "riscv32", target_os = "none"))))]
136pub unsafe fn configure_trap_handler(&self) {
137unimplemented!()
138 }
139140/// Generic helper initialize function to setup all of the chip specific
141 /// operations. Different boards can call the functions that `initialize()`
142 /// calls directly if it needs to use a custom setup operation.
143pub unsafe fn initialize(&self) {
144self.disable_machine_timer();
145self.configure_trap_handler();
146 }
147}
148149impl<'a, I: InterruptService + 'a> kernel::platform::chip::Chip for ArtyExx<'a, I> {
150type MPU = PMPUserMPU<2, SimplePMP<4>>;
151type UserspaceKernelBoundary = rv32i::syscall::SysCall;
152153fn mpu(&self) -> &Self::MPU {
154&self.pmp
155 }
156157fn userspace_kernel_boundary(&self) -> &rv32i::syscall::SysCall {
158&self.userspace_kernel_boundary
159 }
160161fn service_pending_interrupts(&self) {
162unsafe {
163while let Some(interrupt) = self.clic.next_pending() {
164if !self.interrupt_service.service_interrupt(interrupt) {
165debug!("unhandled interrupt: {:?}", interrupt);
166 }
167168// Mark that we are done with this interrupt and the hardware
169 // can clear it.
170self.clic.complete(interrupt);
171 }
172 }
173 }
174175fn has_pending_interrupts(&self) -> bool {
176self.clic.has_pending()
177 }
178179fn sleep(&self) {
180unsafe {
181 rv32i::support::wfi();
182 }
183 }
184185unsafe fn atomic<F, R>(&self, f: F) -> R
186where
187F: FnOnce() -> R,
188 {
189 rv32i::support::atomic(f)
190 }
191192unsafe fn print_state(&self, write: &mut dyn Write) {
193 rv32i::print_riscv_state(write);
194 }
195}
196197/// Trap handler for board/chip specific code.
198///
199/// For the arty-e21 this gets called when an interrupt occurs while the chip is
200/// in kernel mode. All we need to do is check which interrupt occurred and
201/// disable it.
202#[export_name = "_start_trap_rust_from_kernel"]
203pub extern "C" fn start_trap_rust() {
204let mcause = rv32i::csr::CSR.mcause.extract();
205206match rv32i::csr::mcause::Trap::from(mcause) {
207 rv32i::csr::mcause::Trap::Interrupt(_interrupt) => {
208// Since we are using the CLIC, the hardware includes the interrupt
209 // index in the mcause register. The interrupt number is the lowest
210 // 8 bits.
211let interrupt_index = mcause.read(rv32i::csr::mcause::mcause::reason) & 0xFF;
212unsafe {
213 rv32i::clic::disable_interrupt(interrupt_index as u32);
214 }
215 }
216217 rv32i::csr::mcause::Trap::Exception(_exception) => {
218// Otherwise, the kernel encountered a fault...so panic!()?
219panic!("kernel exception");
220 }
221 }
222}
223224/// Function that gets called if an interrupt occurs while an app was running.
225///
226/// mcause is passed in, and this function should correctly handle disabling the
227/// interrupt that fired so that it does not trigger again.
228#[export_name = "_disable_interrupt_trap_rust_from_app"]
229pub extern "C" fn disable_interrupt_trap_handler(mcause: u32) {
230// The interrupt number is then the lowest 8
231 // bits.
232let interrupt_index = mcause & 0xFF;
233unsafe {
234 rv32i::clic::disable_interrupt(interrupt_index);
235 }
236}