rfence.rs 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. use rustsbi::{HartMask, SbiRet};
  2. use spin::Mutex;
  3. use crate::board::SBI;
  4. use crate::fifo::{Fifo, FifoError};
  5. use crate::riscv_spec::current_hartid;
  6. use crate::trap_stack::NUM_HART_MAX;
  7. use crate::trap_stack::ROOT_STACK;
  8. use core::sync::atomic::{AtomicU32, Ordering};
  9. pub(crate) struct RFenceCell {
  10. queue: Mutex<Fifo<(RFenceCTX, usize)>>, // (ctx, source_hart_id)
  11. wait_sync_count: AtomicU32,
  12. }
  13. #[repr(C)]
  14. #[derive(Clone, Copy, Debug)]
  15. pub(crate) struct RFenceCTX {
  16. pub start_addr: usize,
  17. pub size: usize,
  18. pub asid: usize,
  19. pub vmid: usize,
  20. pub op: RFenceType,
  21. }
  22. #[allow(unused)]
  23. #[derive(Clone, Copy, Debug)]
  24. pub(crate) enum RFenceType {
  25. FenceI,
  26. SFenceVma,
  27. SFenceVmaAsid,
  28. HFenceGvmaVmid,
  29. HFenceGvma,
  30. HFenceVvmaAsid,
  31. HFenceVvma,
  32. }
  33. impl RFenceCell {
  34. pub fn new() -> Self {
  35. Self {
  36. queue: Mutex::new(Fifo::new()),
  37. wait_sync_count: AtomicU32::new(0),
  38. }
  39. }
  40. #[inline]
  41. pub unsafe fn local(&self) -> LocalRFenceCell<'_> {
  42. LocalRFenceCell(self)
  43. }
  44. /// 取出共享对象。
  45. #[inline]
  46. pub fn remote(&self) -> RemoteRFenceCell<'_> {
  47. RemoteRFenceCell(self)
  48. }
  49. }
  50. unsafe impl Sync for RFenceCell {}
  51. unsafe impl Send for RFenceCell {}
  52. /// 当前硬件线程的rfence上下文。
  53. pub struct LocalRFenceCell<'a>(&'a RFenceCell);
  54. /// 任意硬件线程的rfence上下文。
  55. pub struct RemoteRFenceCell<'a>(&'a RFenceCell);
  56. /// 获取当前hart的rfence上下文
  57. pub(crate) fn local_rfence() -> LocalRFenceCell<'static> {
  58. unsafe {
  59. ROOT_STACK
  60. .get_unchecked_mut(current_hartid())
  61. .hart_context()
  62. .rfence
  63. .local()
  64. }
  65. }
  66. /// 获取任意hart的rfence上下文
  67. pub(crate) fn remote_rfence(hart_id: usize) -> Option<RemoteRFenceCell<'static>> {
  68. unsafe {
  69. ROOT_STACK
  70. .get_mut(hart_id)
  71. .map(|x| x.hart_context().rfence.remote())
  72. }
  73. }
  74. #[allow(unused)]
  75. impl LocalRFenceCell<'_> {
  76. pub fn is_zero(&self) -> bool {
  77. self.0.wait_sync_count.load(Ordering::Relaxed) == 0
  78. }
  79. pub fn add(&self) {
  80. self.0.wait_sync_count.fetch_add(1, Ordering::Relaxed);
  81. }
  82. pub fn is_empty(&self) -> bool {
  83. self.0.queue.lock().is_empty()
  84. }
  85. pub fn get(&self) -> Option<(RFenceCTX, usize)> {
  86. match self.0.queue.lock().pop() {
  87. Ok(res) => Some(res),
  88. Err(_) => None,
  89. }
  90. }
  91. pub fn set(&self, ctx: RFenceCTX) {
  92. loop {
  93. match self.0.queue.lock().push((ctx, current_hartid())) {
  94. Ok(_) => break (),
  95. Err(FifoError::NoChange) => {
  96. crate::trap::rfence_signle_handler();
  97. continue;
  98. }
  99. _ => panic!("Unable to push fence ops to fifo"),
  100. }
  101. }
  102. }
  103. }
  104. #[allow(unused)]
  105. impl RemoteRFenceCell<'_> {
  106. pub fn set(&self, ctx: RFenceCTX) {
  107. // TODO: maybe deadlock
  108. loop {
  109. match self.0.queue.lock().push((ctx, current_hartid())) {
  110. Ok(_) => break (),
  111. Err(FifoError::NoChange) => {
  112. crate::trap::rfence_signle_handler();
  113. continue;
  114. }
  115. _ => panic!("Unable to push fence ops to fifo"),
  116. }
  117. }
  118. }
  119. pub fn sub(&self) {
  120. self.0.wait_sync_count.fetch_sub(1, Ordering::Relaxed);
  121. }
  122. }
  123. /// RFENCE
  124. pub(crate) struct RFence;
  125. fn remote_fence_process(rfence_ctx: RFenceCTX, hart_mask: HartMask) -> SbiRet {
  126. let sbi_ret = unsafe { SBI.assume_init_mut() }
  127. .clint
  128. .as_ref()
  129. .unwrap()
  130. .send_ipi_by_fence(hart_mask, rfence_ctx);
  131. if hart_mask.has_bit(current_hartid()) {
  132. crate::trap::rfence_handler();
  133. }
  134. sbi_ret
  135. }
  136. impl rustsbi::Fence for RFence {
  137. fn remote_fence_i(&self, hart_mask: HartMask) -> SbiRet {
  138. remote_fence_process(
  139. RFenceCTX {
  140. start_addr: 0,
  141. size: 0,
  142. asid: 0,
  143. vmid: 0,
  144. op: RFenceType::FenceI,
  145. },
  146. hart_mask,
  147. )
  148. }
  149. fn remote_sfence_vma(&self, hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet {
  150. // TODO: return SBI_ERR_INVALID_ADDRESS, when start_addr or size is not valid.
  151. let flush_size = match start_addr.checked_add(size) {
  152. None => usize::MAX,
  153. Some(_) => size,
  154. };
  155. remote_fence_process(
  156. RFenceCTX {
  157. start_addr,
  158. size: flush_size,
  159. asid: 0,
  160. vmid: 0,
  161. op: RFenceType::SFenceVma,
  162. },
  163. hart_mask,
  164. )
  165. }
  166. fn remote_sfence_vma_asid(
  167. &self,
  168. hart_mask: HartMask,
  169. start_addr: usize,
  170. size: usize,
  171. asid: usize,
  172. ) -> SbiRet {
  173. // TODO: return SBI_ERR_INVALID_ADDRESS, when start_addr or size is not valid.
  174. let flush_size = match start_addr.checked_add(size) {
  175. None => usize::MAX,
  176. Some(_) => size,
  177. };
  178. remote_fence_process(
  179. RFenceCTX {
  180. start_addr,
  181. size: flush_size,
  182. asid,
  183. vmid: 0,
  184. op: RFenceType::SFenceVmaAsid,
  185. },
  186. hart_mask,
  187. )
  188. }
  189. }
  190. pub fn hart_mask_clear(hart_mask: HartMask, hart_id: usize) -> HartMask {
  191. let (mask, mask_base) = hart_mask.into_inner();
  192. if mask_base == usize::MAX {
  193. return HartMask::from_mask_base(mask & (!(1 << hart_id)), 0);
  194. }
  195. let Some(idx) = hart_id.checked_sub(mask_base) else {
  196. return hart_mask;
  197. };
  198. if idx >= usize::BITS as usize {
  199. return hart_mask;
  200. }
  201. HartMask::from_mask_base(mask & (!(1 << hart_id)), mask_base)
  202. }