1/// An MMIO register which can only be read from.
2#[derive(Default)]
3#[repr(transparent)]
4pub struct ReadOnly<T: Copy>(pub(crate) T);
56impl<T: Copy> ReadOnly<T> {
7/// Construct a new instance for testing.
8pub const fn new(value: T) -> Self {
9Self(value)
10 }
11}
1213/// An MMIO register which can only be written to.
14#[derive(Default)]
15#[repr(transparent)]
16pub struct WriteOnly<T: Copy>(pub(crate) T);
1718/// An MMIO register which may be both read and written.
19#[derive(Default)]
20#[repr(transparent)]
21pub struct Volatile<T: Copy>(T);
2223impl<T: Copy> Volatile<T> {
24/// Construct a new instance for testing.
25pub const fn new(value: T) -> Self {
26Self(value)
27 }
28}
2930/// A trait implemented by MMIO registers which may be read from.
31pub trait VolatileReadable<T> {
32/// Performs a volatile read from the MMIO register.
33unsafe fn vread(self) -> T;
34}
3536impl<T: Copy> VolatileReadable<T> for *const ReadOnly<T> {
37unsafe fn vread(self) -> T {
38self.read_volatile().0
39}
40}
4142impl<T: Copy> VolatileReadable<T> for *const Volatile<T> {
43unsafe fn vread(self) -> T {
44self.read_volatile().0
45}
46}
4748/// A trait implemented by MMIO registers which may be written to.
49pub trait VolatileWritable<T> {
50/// Performs a volatile write to the MMIO register.
51unsafe fn vwrite(self, value: T);
52}
5354impl<T: Copy> VolatileWritable<T> for *mut WriteOnly<T> {
55unsafe fn vwrite(self, value: T) {
56 (self as *mut T).write_volatile(value)
57 }
58}
5960impl<T: Copy> VolatileWritable<T> for *mut Volatile<T> {
61unsafe fn vwrite(self, value: T) {
62 (self as *mut T).write_volatile(value)
63 }
64}
6566/// Performs a volatile read from the given field of pointer to a struct representing an MMIO region.
67///
68/// # Usage
69/// ```compile_fail
70/// # use core::ptr::NonNull;
71/// # use virtio_drivers::volatile::{ReadOnly, volread};
72/// struct MmioDevice {
73/// field: ReadOnly<u32>,
74/// }
75///
76/// let device: NonNull<MmioDevice> = NonNull::new(0x1234 as *mut MmioDevice).unwrap();
77/// let value = unsafe { volread!(device, field) };
78/// ```
79macro_rules! volread {
80 ($nonnull:expr, $field:ident) => {
81$crate::volatile::VolatileReadable::vread(core::ptr::addr_of!((*$nonnull.as_ptr()).$field))
82 };
83}
8485/// Performs a volatile write to the given field of pointer to a struct representing an MMIO region.
86///
87/// # Usage
88/// ```compile_fail
89/// # use core::ptr::NonNull;
90/// # use virtio_drivers::volatile::{WriteOnly, volread};
91/// struct MmioDevice {
92/// field: WriteOnly<u32>,
93/// }
94///
95/// let device: NonNull<MmioDevice> = NonNull::new(0x1234 as *mut MmioDevice).unwrap();
96/// unsafe { volwrite!(device, field, 42); }
97/// ```
98macro_rules! volwrite {
99 ($nonnull:expr, $field:ident, $value:expr) => {
100$crate::volatile::VolatileWritable::vwrite(
101core::ptr::addr_of_mut!((*$nonnull.as_ptr()).$field),
102$value,
103 )
104 };
105}
106107pub(crate) use volread;
108pub(crate) use volwrite;