rfence.rs 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. use crate::ecall::SbiRet;
  2. use crate::hart_mask::HartMask;
  3. use crate::util::AmoOncePtr;
  4. /// Remote fence support
  5. ///
  6. /// The remote fence function acts as a full TLB flush if
  7. /// - `start_addr` and `size` are both 0, or
  8. /// - `size` is equal to `usize::MAX`.
  9. pub trait Rfence: Send + Sync {
  10. /// Instructs remote harts to execute `FENCE.I` instruction.
  11. ///
  12. /// # Return value
  13. ///
  14. /// Returns `SBI_SUCCESS` when remote fence was sent to all the targeted harts successfully.
  15. fn remote_fence_i(&self, hart_mask: HartMask) -> SbiRet;
  16. /// Instructs the remote harts to execute one or more `SFENCE.VMA` instructions,
  17. /// covering the range of virtual addresses between `start_addr` and `size`.
  18. ///
  19. /// # Return value
  20. ///
  21. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  22. ///
  23. /// | Return code | Description
  24. /// |:--------------------------|:----------------------------------------------
  25. /// | SBI_SUCCESS | Remote fence was sent to all the targeted harts successfully.
  26. /// | SBI_ERR_INVALID_ADDRESS | `start_addr` or `size` is not valid.
  27. fn remote_sfence_vma(&self, hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet;
  28. /// Instruct the remote harts to execute one or more `SFENCE.VMA` instructions,
  29. /// covering the range of virtual addresses between `start_addr` and `size`.
  30. /// This covers only the given address space by `asid`.
  31. ///
  32. /// # Return value
  33. ///
  34. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  35. ///
  36. /// | Return code | Description
  37. /// |:--------------------------|:----------------------------------------------
  38. /// | SBI_SUCCESS | Remote fence was sent to all the targeted harts successfully.
  39. /// | SBI_ERR_INVALID_ADDRESS | `start_addr` or `size` is not valid.
  40. fn remote_sfence_vma_asid(
  41. &self,
  42. hart_mask: HartMask,
  43. start_addr: usize,
  44. size: usize,
  45. asid: usize,
  46. ) -> SbiRet;
  47. /// Instruct the remote harts to execute one or more `HFENCE.GVMA` instructions,
  48. /// covering the range of guest physical addresses between `start_addr` and `size`
  49. /// only for the given virtual machine by `vmid`.
  50. ///
  51. /// This function call is only valid for harts implementing hypervisor extension.
  52. ///
  53. /// # Return value
  54. ///
  55. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  56. ///
  57. /// | Return code | Description
  58. /// |:--------------------------|:----------------------------------------------
  59. /// | SBI_SUCCESS | Remote fence was sent to all the targeted harts successfully.
  60. /// | SBI_ERR_NOT_SUPPORTED | This function is not supported as it is not implemented or one of the target hart doesn’t support hypervisor extension.
  61. /// | SBI_ERR_INVALID_ADDRESS | `start_addr` or `size` is not valid.
  62. fn remote_hfence_gvma_vmid(
  63. &self,
  64. hart_mask: HartMask,
  65. start_addr: usize,
  66. size: usize,
  67. vmid: usize,
  68. ) -> SbiRet {
  69. let _ = (hart_mask, start_addr, size, vmid);
  70. SbiRet::not_supported()
  71. }
  72. /// Instruct the remote harts to execute one or more `HFENCE.GVMA` instructions,
  73. /// covering the range of guest physical addresses between `start_addr` and `size`
  74. /// for all the guests.
  75. ///
  76. /// This function call is only valid for harts implementing hypervisor extension.
  77. ///
  78. /// # Return value
  79. ///
  80. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  81. ///
  82. /// | Return code | Description
  83. /// |:--------------------------|:----------------------------------------------
  84. /// | SBI_SUCCESS | Remote fence was sent to all the targeted harts successfully.
  85. /// | SBI_ERR_NOT_SUPPORTED | This function is not supported as it is not implemented or one of the target hart does not support hypervisor extension.
  86. /// | SBI_ERR_INVALID_ADDRESS | `start_addr` or `size` is not valid.
  87. fn remote_hfence_gvma(&self, hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet {
  88. let _ = (hart_mask, start_addr, size);
  89. SbiRet::not_supported()
  90. }
  91. /// Instruct the remote harts to execute one or more `HFENCE.VVMA` instructions,
  92. /// covering the range of guest virtual addresses between `start_addr` and `size` for the given
  93. /// address space by `asid` and current virtual machine (by `vmid` in `hgatp` CSR)
  94. /// of calling hart.
  95. ///
  96. /// This function call is only valid for harts implementing hypervisor extension.
  97. ///
  98. /// # Return value
  99. ///
  100. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  101. ///
  102. /// | Return code | Description
  103. /// |:--------------------------|:----------------------------------------------
  104. /// | SBI_SUCCESS | Remote fence was sent to all the targeted harts successfully.
  105. /// | SBI_ERR_NOT_SUPPORTED | This function is not supported as it is not implemented or one of the target hart does not support hypervisor extension.
  106. /// | SBI_ERR_INVALID_ADDRESS | `start_addr` or `size` is not valid.
  107. fn remote_hfence_vvma_asid(
  108. &self,
  109. hart_mask: HartMask,
  110. start_addr: usize,
  111. size: usize,
  112. asid: usize,
  113. ) -> SbiRet {
  114. let _ = (hart_mask, start_addr, size, asid);
  115. SbiRet::not_supported()
  116. }
  117. /// Instruct the remote harts to execute one or more `HFENCE.VVMA` instructions,
  118. /// covering the range of guest virtual addresses between `start_addr` and `size`
  119. /// for current virtual machine (by `vmid` in `hgatp` CSR) of calling hart.
  120. ///
  121. /// This function call is only valid for harts implementing hypervisor extension.
  122. ///
  123. /// # Return value
  124. ///
  125. /// The possible return error codes returned in `SbiRet.error` are shown in the table below:
  126. ///
  127. /// | Return code | Description
  128. /// |:--------------------------|:----------------------------------------------
  129. /// | SBI_SUCCESS | Remote fence was sent to all the targeted harts successfully.
  130. /// | SBI_ERR_NOT_SUPPORTED | This function is not supported as it is not implemented or one of the target hart doesn’t support hypervisor extension.
  131. /// | SBI_ERR_INVALID_ADDRESS | `start_addr` or `size` is not valid.
  132. fn remote_hfence_vvma(&self, hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet {
  133. let _ = (hart_mask, start_addr, size);
  134. SbiRet::not_supported()
  135. }
  136. }
  137. static RFENCE: AmoOncePtr<dyn Rfence> = AmoOncePtr::new();
  138. pub fn init_rfence(rfence: &'static dyn Rfence) {
  139. if !RFENCE.try_call_once(rfence) {
  140. panic!("load sbi module when already loaded")
  141. }
  142. }
  143. #[inline]
  144. pub(crate) fn probe_rfence() -> bool {
  145. RFENCE.get().is_some()
  146. }
  147. pub(crate) fn remote_fence_i(hart_mask: HartMask) -> SbiRet {
  148. if let Some(rfence) = RFENCE.get() {
  149. rfence.remote_fence_i(hart_mask)
  150. } else {
  151. SbiRet::not_supported()
  152. }
  153. }
  154. pub(crate) fn remote_sfence_vma(hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet {
  155. if let Some(rfence) = RFENCE.get() {
  156. rfence.remote_sfence_vma(hart_mask, start_addr, size)
  157. } else {
  158. SbiRet::not_supported()
  159. }
  160. }
  161. pub(crate) fn remote_sfence_vma_asid(
  162. hart_mask: HartMask,
  163. start_addr: usize,
  164. size: usize,
  165. asid: usize,
  166. ) -> SbiRet {
  167. if let Some(rfence) = RFENCE.get() {
  168. rfence.remote_sfence_vma_asid(hart_mask, start_addr, size, asid)
  169. } else {
  170. SbiRet::not_supported()
  171. }
  172. }
  173. pub(crate) fn remote_hfence_gvma_vmid(
  174. hart_mask: HartMask,
  175. start_addr: usize,
  176. size: usize,
  177. vmid: usize,
  178. ) -> SbiRet {
  179. if let Some(rfence) = RFENCE.get() {
  180. rfence.remote_hfence_gvma_vmid(hart_mask, start_addr, size, vmid)
  181. } else {
  182. SbiRet::not_supported()
  183. }
  184. }
  185. pub(crate) fn remote_hfence_gvma(hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet {
  186. if let Some(rfence) = RFENCE.get() {
  187. rfence.remote_hfence_gvma(hart_mask, start_addr, size)
  188. } else {
  189. SbiRet::not_supported()
  190. }
  191. }
  192. pub(crate) fn remote_hfence_vvma_asid(
  193. hart_mask: HartMask,
  194. start_addr: usize,
  195. size: usize,
  196. asid: usize,
  197. ) -> SbiRet {
  198. if let Some(rfence) = RFENCE.get() {
  199. rfence.remote_hfence_vvma_asid(hart_mask, start_addr, size, asid)
  200. } else {
  201. SbiRet::not_supported()
  202. }
  203. }
  204. pub(crate) fn remote_hfence_vvma(hart_mask: HartMask, start_addr: usize, size: usize) -> SbiRet {
  205. if let Some(rfence) = RFENCE.get() {
  206. rfence.remote_hfence_vvma(hart_mask, start_addr, size)
  207. } else {
  208. SbiRet::not_supported()
  209. }
  210. }