volatile.rs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. macro_rules! volatile_read {
  2. ($data: expr) => {
  3. unsafe { core::ptr::read_volatile(core::ptr::addr_of!($data)) }
  4. };
  5. }
  6. macro_rules! volatile_write {
  7. ($data: expr, $value: expr) => {
  8. unsafe { core::ptr::write_volatile(core::ptr::addr_of_mut!($data), $value) }
  9. };
  10. }
  11. /// @brief: 用于volatile设置某些bits
  12. /// @param val: 设置这些位
  13. /// @param flag: true表示设置这些位为1; false表示设置这些位为0;
  14. macro_rules! volatile_set_bit {
  15. ($data: expr, $val: expr, $flag: expr) => {
  16. volatile_write!(
  17. $data,
  18. match $flag {
  19. true => core::ptr::read_volatile(core::ptr::addr_of!($data)) | $val,
  20. false => core::ptr::read_volatile(core::ptr::addr_of!($data)) & (!$val),
  21. }
  22. )
  23. };
  24. }
  25. /// @param data: volatile变量
  26. /// @param bits: 置1的位才有效,表示写这些位
  27. /// @param val: 要写的值
  28. /// 比如: 写 x 的 2至8bit, 为 10, 可以这么写 volatile_write_bit(x, (1<<8)-(1<<2), 10<<2);
  29. macro_rules! volatile_write_bit {
  30. ($data: expr, $bits: expr, $val: expr) => {
  31. volatile_set_bit!($data, $bits, false);
  32. volatile_set_bit!($data, ($val) & ($bits), true);
  33. };
  34. }
  35. /// 以下代码来自于virtio-drivers 0.2.0
  36. /// 在对已经MMIO映射对虚拟地址的寄存器的操作中,我们经常遇到有的寄存器是只读或可读写的
  37. /// 那么我们就可以使用结构体ReadOnly WriteOnly Volatile对其进行区分
  38. /// 例:
  39. /// #[repr(C)]
  40. /// struct CommonCfg {
  41. /// device_feature_select: Volatile<u32>,
  42. /// device_feature: ReadOnly<u32>,
  43. /// driver_feature_select: Volatile<u32>,
  44. /// driver_feature: Volatile<u32>,
  45. /// msix_config: Volatile<u16>,
  46. /// num_queues: ReadOnly<u16>,
  47. /// device_status: Volatile<u8>,
  48. /// config_generation: ReadOnly<u8>,
  49. /// queue_select: Volatile<u16>,
  50. /// queue_size: Volatile<u16>,
  51. /// queue_msix_vector: Volatile<u16>,
  52. /// queue_enable: Volatile<u16>,
  53. /// queue_notify_off: Volatile<u16>,
  54. /// queue_desc: Volatile<u64>,
  55. /// queue_driver: Volatile<u64>,
  56. /// queue_device: Volatile<u64>,
  57. /// }
  58. ///
  59. /// 对CommonCfg里面的某个寄存器进行读写:
  60. /// volwrite!(self.common_cfg, queue_enable, 0);
  61. ///
  62. /// 这样做不仅使代码的可读性提高了,也避免了对只读寄存器进行写入的误操作
  63. /// 只读寄存器
  64. #[derive(Default)]
  65. #[repr(transparent)]
  66. pub struct ReadOnly<T: Copy>(T);
  67. #[allow(dead_code)]
  68. impl<T: Copy> ReadOnly<T> {
  69. /// Construct a new instance for testing.
  70. pub fn new(value: T) -> Self {
  71. Self(value)
  72. }
  73. }
  74. /// 只写寄存器
  75. #[derive(Default)]
  76. #[repr(transparent)]
  77. pub struct WriteOnly<T: Copy>(T);
  78. /// 写读寄存器
  79. #[derive(Default)]
  80. #[repr(transparent)]
  81. pub struct Volatile<T: Copy>(T);
  82. #[allow(dead_code)]
  83. impl<T: Copy> Volatile<T> {
  84. /// Construct a new instance for testing.
  85. pub fn new(value: T) -> Self {
  86. Self(value)
  87. }
  88. }
  89. /// A trait implemented by MMIO registers which may be read from.
  90. pub trait VolatileReadable<T> {
  91. /// Performs a volatile read from the MMIO register.
  92. unsafe fn vread(self) -> T;
  93. }
  94. impl<T: Copy> VolatileReadable<T> for *const ReadOnly<T> {
  95. unsafe fn vread(self) -> T {
  96. self.read_volatile().0
  97. }
  98. }
  99. impl<T: Copy> VolatileReadable<T> for *const Volatile<T> {
  100. unsafe fn vread(self) -> T {
  101. self.read_volatile().0
  102. }
  103. }
  104. /// A trait implemented by MMIO registers which may be written to.
  105. pub trait VolatileWritable<T> {
  106. /// Performs a volatile write to the MMIO register.
  107. unsafe fn vwrite(self, value: T);
  108. }
  109. impl<T: Copy> VolatileWritable<T> for *mut WriteOnly<T> {
  110. unsafe fn vwrite(self, value: T) {
  111. (self as *mut T).write_volatile(value)
  112. }
  113. }
  114. impl<T: Copy> VolatileWritable<T> for *mut Volatile<T> {
  115. unsafe fn vwrite(self, value: T) {
  116. (self as *mut T).write_volatile(value)
  117. }
  118. }
  119. /// Performs a volatile read from the given field of pointer to a struct representing an MMIO region.
  120. ///
  121. /// # Usage
  122. /// ```compile_fail
  123. /// # use core::ptr::NonNull;
  124. /// # use virtio_drivers::volatile::{ReadOnly, volread};
  125. /// struct MmioDevice {
  126. /// field: ReadOnly<u32>,
  127. /// }
  128. ///
  129. /// let device: NonNull<MmioDevice> = NonNull::new(0x1234 as *mut MmioDevice).unwrap();
  130. /// let value = unsafe { volread!(device, field) };
  131. /// ```
  132. macro_rules! volread {
  133. ($nonnull:expr, $field:ident) => {
  134. crate::libs::volatile::VolatileReadable::vread(core::ptr::addr_of!(
  135. (*$nonnull.as_ptr()).$field
  136. ))
  137. };
  138. }
  139. /// Performs a volatile write to the given field of pointer to a struct representing an MMIO region.
  140. ///
  141. /// # Usage
  142. /// ```compile_fail
  143. /// # use core::ptr::NonNull;
  144. /// # use virtio_drivers::volatile::{WriteOnly, volread};
  145. /// struct MmioDevice {
  146. /// field: WriteOnly<u32>,
  147. /// }
  148. ///
  149. /// let device: NonNull<MmioDevice> = NonNull::new(0x1234 as *mut MmioDevice).unwrap();
  150. /// unsafe { volwrite!(device, field, 42); }
  151. /// ```
  152. macro_rules! volwrite {
  153. ($nonnull:expr, $field:ident, $value:expr) => {
  154. crate::libs::volatile::VolatileWritable::vwrite(
  155. core::ptr::addr_of_mut!((*$nonnull.as_ptr()).$field),
  156. $value,
  157. )
  158. };
  159. }
  160. pub(crate) use volread;
  161. pub(crate) use volwrite;