Skip to content

Commit

Permalink
Add bitfield helpers and use them in mstatus
Browse files Browse the repository at this point in the history
  • Loading branch information
jsgf committed May 8, 2024
1 parent 8de908c commit f968990
Show file tree
Hide file tree
Showing 3 changed files with 46 additions and 22 deletions.
17 changes: 17 additions & 0 deletions riscv/src/bits.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
/// Insert a new value into a bitfield
///
/// `value` is masked to `width` bits and inserted into `orig`.`
#[inline]
pub fn bf_insert(orig: usize, bit: usize, width: usize, value: usize) -> usize {
let mask = (1 << width) - 1;
orig & !(mask << bit) | ((value & mask) << bit)
}

/// Extract a value from a bitfield
///
/// Extracts `width` bits from bit offset `bit` and returns it shifted to bit 0.s
#[inline]
pub fn bf_extract(orig: usize, bit: usize, width: usize) -> usize {
let mask = (1 << width) - 1;
(orig >> bit) & mask
}
1 change: 1 addition & 0 deletions riscv/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@
#![allow(clippy::missing_safety_doc)]

pub mod asm;
pub(crate) mod bits;
pub mod delay;
pub mod interrupt;
pub mod register;
Expand Down
50 changes: 28 additions & 22 deletions riscv/src/register/mstatus.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
//! mstatus register
pub use super::misa::XLEN;
use crate::bits::{bf_extract, bf_insert};

/// mstatus register
#[derive(Clone, Copy, Debug)]
Expand Down Expand Up @@ -75,12 +76,17 @@ impl Mstatus {
/// Helper to insert a bitfield into Mstatus
#[inline]
fn bf_insert(&self, bit: usize, width: usize, val: usize) -> Self {
let mask = (1 << width) - 1;
Self {
bits: self.bits & !(mask << bit) | ((val & mask) << bit),
bits: bf_insert(self.bits, bit, width, val),
}
}

/// Helper to extract a bitfield from Mstatus
#[inline]
fn bf_extract(&self, bit: usize, width: usize) -> usize {
bf_extract(self.bits, bit, width)
}

/// Returns the contents of the register as raw bits
#[inline]
pub fn bits(&self) -> usize {
Expand All @@ -90,7 +96,7 @@ impl Mstatus {
/// Supervisor Interrupt Enable
#[inline]
pub fn sie(&self) -> bool {
self.bits & (1 << 1) != 0
self.bf_extract(1, 1) != 0
}

/// Update Supervisor Interrupt Enable
Expand All @@ -106,7 +112,7 @@ impl Mstatus {
/// Machine Interrupt Enable
#[inline]
pub fn mie(&self) -> bool {
self.bits & (1 << 3) != 0
self.bf_extract(3, 1) != 0
}

/// Update Machine Interrupt Enable
Expand All @@ -122,7 +128,7 @@ impl Mstatus {
/// Supervisor Previous Interrupt Enable
#[inline]
pub fn spie(&self) -> bool {
self.bits & (1 << 5) != 0
self.bf_extract(5, 1) != 0
}

/// Update Supervisor Previous Interrupt Enable
Expand All @@ -138,7 +144,7 @@ impl Mstatus {
/// U-mode non-instruction-fetch memory endianness
#[inline]
pub fn ube(&self) -> Endianness {
Endianness::from(self.bits & (1 << 6) != 0)
Endianness::from(self.bf_extract(6, 1) != 0)
}

/// Update U-mode non-instruction-fetch memory endianness
Expand All @@ -154,7 +160,7 @@ impl Mstatus {
/// Machine Previous Interrupt Enable
#[inline]
pub fn mpie(&self) -> bool {
self.bits & (1 << 7) != 0
self.bf_extract(7, 1) != 0
}

/// Update Machine Previous Interrupt Enable
Expand All @@ -170,7 +176,7 @@ impl Mstatus {
/// Supervisor Previous Privilege Mode
#[inline]
pub fn spp(&self) -> SPP {
match self.bits & (1 << 8) != 0 {
match self.bf_extract(7, 1) != 0 {
true => SPP::Supervisor,
false => SPP::User,
}
Expand All @@ -189,7 +195,7 @@ impl Mstatus {
/// Machine Previous Privilege Mode
#[inline]
pub fn mpp(&self) -> MPP {
let mpp = (self.bits >> 11) & 0x3; // bits 11-12
let mpp = self.bf_extract(11, 2); // bits 11-12
match mpp {
0b00 => MPP::User,
0b01 => MPP::Supervisor,
Expand All @@ -214,7 +220,7 @@ impl Mstatus {
/// and floating-point data registers `f0–f31`.
#[inline]
pub fn fs(&self) -> FS {
let fs = (self.bits >> 13) & 0x3; // bits 13-14
let fs = self.bf_extract(13, 2); // bits 13-14
match fs {
0b00 => FS::Off,
0b01 => FS::Initial,
Expand All @@ -240,7 +246,7 @@ impl Mstatus {
/// state.
#[inline]
pub fn xs(&self) -> XS {
let xs = (self.bits >> 15) & 0x3; // bits 15-16
let xs = self.bf_extract(15, 2); // bits 15-16
match xs {
0b00 => XS::AllOff,
0b01 => XS::NoneDirtyOrClean,
Expand All @@ -262,7 +268,7 @@ impl Mstatus {
/// Modify Memory PRiVilege
#[inline]
pub fn mprv(&self) -> bool {
self.bits & (1 << 17) != 0
self.bf_extract(17, 1) != 0
}

/// Update Modify Memory PRiVilege
Expand All @@ -278,7 +284,7 @@ impl Mstatus {
/// Permit Supervisor User Memory access
#[inline]
pub fn sum(&self) -> bool {
self.bits & (1 << 18) != 0
self.bf_extract(18, 1) != 0
}

/// Update Permit Supervisor User Memory access
Expand All @@ -294,7 +300,7 @@ impl Mstatus {
/// Make eXecutable Readable
#[inline]
pub fn mxr(&self) -> bool {
self.bits & (1 << 19) != 0
self.bf_extract(19, 1) != 0
}

/// Update Make eXecutable Readable
Expand All @@ -315,7 +321,7 @@ impl Mstatus {
/// TVM is hard-wired to 0 when S-mode is not supported.
#[inline]
pub fn tvm(&self) -> bool {
self.bits & (1 << 20) != 0
self.bf_extract(20, 1) != 0
}

/// Update Trap Virtual Memory
Expand All @@ -339,7 +345,7 @@ impl Mstatus {
/// TW is hard-wired to 0 when S-mode is not supported.
#[inline]
pub fn tw(&self) -> bool {
self.bits & (1 << 21) != 0
self.bf_extract(21, 1) != 0
}

/// Update Timeout Wait
Expand All @@ -360,7 +366,7 @@ impl Mstatus {
/// If S-mode is not supported, TSR bit is hard-wired to 0.
#[inline]
pub fn tsr(&self) -> bool {
self.bits & (1 << 22) != 0
self.bf_extract(22, 1) != 0
}

/// Update Trap SRET
Expand All @@ -382,7 +388,7 @@ impl Mstatus {
#[cfg(riscv32)]
() => XLEN::XLEN32,
#[cfg(not(riscv32))]
() => XLEN::from((self.bits >> 32) as u8 & 0x3),
() => XLEN::from(self.bf_extract(32, 2) as u8),
}
}

Expand All @@ -409,7 +415,7 @@ impl Mstatus {
#[cfg(riscv32)]
() => XLEN::XLEN32,
#[cfg(not(riscv32))]
() => XLEN::from((self.bits >> 34) as u8 & 0x3),
() => XLEN::from(self.bf_extract(34, 2) as u8),
}
}

Expand All @@ -435,7 +441,7 @@ impl Mstatus {
#[cfg(riscv32)]
() => super::mstatush::read().sbe(),
#[cfg(not(riscv32))]
() => Endianness::from(self.bits & (1 << 36) != 0),
() => Endianness::from(self.bf_extract(36, 1) != 0),
}
}

Expand All @@ -462,7 +468,7 @@ impl Mstatus {
#[cfg(riscv32)]
() => super::mstatush::read().mbe(),
#[cfg(not(riscv32))]
() => Endianness::from(self.bits & (1 << 37) != 0),
() => Endianness::from(self.bf_extract(37, 1) != 0),
}
}
/// Update M-mode non-instruction-fetch memory endianness
Expand All @@ -483,7 +489,7 @@ impl Mstatus {
/// Whether either the FS field or XS field signals the presence of some dirty state
#[inline]
pub fn sd(&self) -> bool {
self.bits & (1 << (usize::BITS as usize - 1)) != 0
self.bf_extract(usize::BITS as usize - 1, 1) != 0
}

/// Update whether either the FS field or XS field signals the presence of
Expand Down

0 comments on commit f968990

Please sign in to comment.