|
1 | 1 | //! stvec register
|
2 | 2 |
|
3 | 3 | pub use crate::register::mtvec::TrapMode;
|
| 4 | +use crate::result::{Error, Result}; |
4 | 5 |
|
5 |
| -/// stvec register |
6 |
| -#[derive(Clone, Copy, Debug)] |
7 |
| -pub struct Stvec { |
8 |
| - bits: usize, |
| 6 | +const TRAP_MASK: usize = 0b11; |
| 7 | + |
| 8 | +read_write_csr! { |
| 9 | + /// stvec register |
| 10 | + Stvec: 0x105, |
| 11 | + mask: usize::MAX, |
| 12 | +} |
| 13 | + |
| 14 | +read_write_csr_field! { |
| 15 | + Stvec, |
| 16 | + /// Returns the trap-vector mode |
| 17 | + trap_mode, |
| 18 | + TrapMode: [0:1], |
9 | 19 | }
|
10 | 20 |
|
11 | 21 | impl Stvec {
|
12 |
| - /// Returns the contents of the register as raw bits |
| 22 | + /// Returns the trap-vector base-address |
13 | 23 | #[inline]
|
14 |
| - pub fn bits(&self) -> usize { |
15 |
| - self.bits |
| 24 | + pub const fn address(&self) -> usize { |
| 25 | + self.bits & !TRAP_MASK |
16 | 26 | }
|
17 | 27 |
|
18 |
| - /// Returns the trap-vector base-address |
| 28 | + /// Sets the trap-vector base-address. |
| 29 | + /// |
| 30 | + /// # Note |
| 31 | + /// |
| 32 | + /// Panics if the address is not aligned to 4-bytes. |
19 | 33 | #[inline]
|
20 |
| - pub fn address(&self) -> usize { |
21 |
| - self.bits - (self.bits & 0b11) |
| 34 | + pub fn set_address(&mut self, address: usize) { |
| 35 | + self.try_set_address(address).unwrap(); |
22 | 36 | }
|
23 | 37 |
|
24 |
| - /// Returns the trap-vector mode |
| 38 | + /// Attempts to set the trap-vector base-address. |
| 39 | + /// |
| 40 | + /// # Note |
| 41 | + /// |
| 42 | + /// Returns an error if the address is not aligned to 4-bytes. |
25 | 43 | #[inline]
|
26 |
| - pub fn trap_mode(&self) -> Option<TrapMode> { |
27 |
| - let mode = self.bits & 0b11; |
28 |
| - match mode { |
29 |
| - 0 => Some(TrapMode::Direct), |
30 |
| - 1 => Some(TrapMode::Vectored), |
31 |
| - _ => None, |
| 44 | + pub fn try_set_address(&mut self, address: usize) -> Result<()> { |
| 45 | + // check for four-byte alignment |
| 46 | + if (address & TRAP_MASK) != 0 { |
| 47 | + Err(Error::InvalidFieldVariant { |
| 48 | + field: "stvec::address", |
| 49 | + value: address, |
| 50 | + }) |
| 51 | + } else { |
| 52 | + self.bits = address | (self.bits & TRAP_MASK); |
| 53 | + Ok(()) |
32 | 54 | }
|
33 | 55 | }
|
34 | 56 | }
|
35 |
| - |
36 |
| -read_csr_as!(Stvec, 0x105); |
37 |
| -write_csr!(0x105); |
38 |
| - |
39 |
| -/// Writes the CSR |
40 |
| -#[inline] |
41 |
| -pub unsafe fn write(addr: usize, mode: TrapMode) { |
42 |
| - _write(addr + mode as usize); |
43 |
| -} |
|
0 commit comments