rfence.rs 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. use rustsbi::{HartMask, SbiRet};
  2. use spin::Mutex;
  3. use crate::board::SBI_IMPL;
  4. use crate::sbi::fifo::{Fifo, FifoError};
  5. use crate::riscv_spec::current_hartid;
  6. use crate::sbi::trap_stack::ROOT_STACK;
  7. use crate::sbi::trap;
  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 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 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 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() -> Option<LocalRFenceCell<'static>> {
  58. unsafe {
  59. ROOT_STACK
  60. .get_mut(current_hartid())
  61. .map(|x| x.hart_context().rfence.local())
  62. }
  63. }
  64. /// 获取任意hart的rfence上下文
  65. pub(crate) fn remote_rfence(hart_id: usize) -> Option<RemoteRFenceCell<'static>> {
  66. unsafe {
  67. ROOT_STACK
  68. .get_mut(hart_id)
  69. .map(|x| x.hart_context().rfence.remote())
  70. }
  71. }
  72. #[allow(unused)]
  73. impl LocalRFenceCell<'_> {
  74. pub fn is_sync(&self) -> bool {
  75. self.0.wait_sync_count.load(Ordering::Relaxed) == 0
  76. }
  77. pub fn add(&self) {
  78. self.0.wait_sync_count.fetch_add(1, Ordering::Relaxed);
  79. }
  80. pub fn is_empty(&self) -> bool {
  81. self.0.queue.lock().is_empty()
  82. }
  83. pub fn get(&self) -> Option<(RFenceCTX, usize)> {
  84. match self.0.queue.lock().pop() {
  85. Ok(res) => Some(res),
  86. Err(_) => None,
  87. }
  88. }
  89. pub fn set(&self, ctx: RFenceCTX) {
  90. loop {
  91. match self.0.queue.lock().push((ctx, current_hartid())) {
  92. Ok(_) => break,
  93. Err(FifoError::Full) => {
  94. trap::rfence_signle_handler();
  95. continue;
  96. }
  97. _ => panic!("Unable to push fence ops to fifo"),
  98. }
  99. }
  100. }
  101. }
  102. #[allow(unused)]
  103. impl RemoteRFenceCell<'_> {
  104. pub fn set(&self, ctx: RFenceCTX) {
  105. // TODO: maybe deadlock
  106. loop {
  107. match self.0.queue.lock().push((ctx, current_hartid())) {
  108. Ok(_) => break,
  109. Err(FifoError::Full) => {
  110. trap::rfence_signle_handler();
  111. continue;
  112. }
  113. _ => panic!("Unable to push fence ops to fifo"),
  114. }
  115. }
  116. }
  117. pub fn sub(&self) {
  118. self.0.wait_sync_count.fetch_sub(1, Ordering::Relaxed);
  119. }
  120. }
  121. /// RFENCE
  122. pub(crate) struct SbiRFence;
  123. fn remote_fence_process(rfence_ctx: RFenceCTX, hart_mask: HartMask) -> SbiRet {
  124. let sbi_ret = unsafe { SBI_IMPL.assume_init_mut() }
  125. .ipi
  126. .as_ref()
  127. .unwrap()
  128. .send_ipi_by_fence(hart_mask, rfence_ctx);
  129. sbi_ret
  130. }
  131. impl rustsbi::Fence for SbiRFence {
  132. fn remote_fence_i(&self, hart_mask: HartMask) -> SbiRet {
  133. remote_fence_process(
  134. RFenceCTX {
  135. start_addr: 0,
  136. size: 0,
  137. asid: 0,
  138. vmid: 0,
  139. op: RFenceType::FenceI,
  140. },
  141. hart_mask,
  142. )
  143. }
  144. fn remote_sfence_vma(&self, hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet {
  145. // TODO: return SBI_ERR_INVALID_ADDRESS, when start_addr or size is not valid.
  146. let flush_size = match start_addr.checked_add(size) {
  147. None => usize::MAX,
  148. Some(_) => size,
  149. };
  150. remote_fence_process(
  151. RFenceCTX {
  152. start_addr,
  153. size: flush_size,
  154. asid: 0,
  155. vmid: 0,
  156. op: RFenceType::SFenceVma,
  157. },
  158. hart_mask,
  159. )
  160. }
  161. fn remote_sfence_vma_asid(
  162. &self,
  163. hart_mask: HartMask,
  164. start_addr: usize,
  165. size: usize,
  166. asid: usize,
  167. ) -> SbiRet {
  168. // TODO: return SBI_ERR_INVALID_ADDRESS, when start_addr or size is not valid.
  169. let flush_size = match start_addr.checked_add(size) {
  170. None => usize::MAX,
  171. Some(_) => size,
  172. };
  173. remote_fence_process(
  174. RFenceCTX {
  175. start_addr,
  176. size: flush_size,
  177. asid,
  178. vmid: 0,
  179. op: RFenceType::SFenceVmaAsid,
  180. },
  181. hart_mask,
  182. )
  183. }
  184. }