asm.rs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. //! Assembly instructions
  2. macro_rules! instruction {
  3. ($(#[$attr:meta])*, $fnname:ident, $asm:expr) => (
  4. $(#[$attr])*
  5. #[inline]
  6. pub unsafe fn $fnname() {
  7. match () {
  8. #[cfg(riscv)]
  9. () => core::arch::asm!($asm),
  10. #[cfg(not(riscv))]
  11. () => unimplemented!(),
  12. }
  13. }
  14. )
  15. }
  16. instruction!(
  17. /// `nop` instruction wrapper
  18. ///
  19. /// Generates a no-operation. Useful to prevent delay loops from being optimized away.
  20. , nop, "nop");
  21. instruction!(
  22. /// `EBREAK` instruction wrapper
  23. ///
  24. /// Generates a breakpoint exception.
  25. , ebreak, "ebreak");
  26. instruction!(
  27. /// `WFI` instruction wrapper
  28. ///
  29. /// Provides a hint to the implementation that the current hart can be stalled until an interrupt might need servicing.
  30. /// The WFI instruction is just a hint, and a legal implementation is to implement WFI as a NOP.
  31. , wfi, "wfi");
  32. instruction!(
  33. /// `SFENCE.VMA` instruction wrapper (all address spaces and page table levels)
  34. ///
  35. /// Synchronizes updates to in-memory memory-management data structures with current execution.
  36. /// Instruction execution causes implicit reads and writes to these data structures; however, these implicit references
  37. /// are ordinarily not ordered with respect to loads and stores in the instruction stream.
  38. /// Executing an `SFENCE.VMA` instruction guarantees that any stores in the instruction stream prior to the
  39. /// `SFENCE.VMA` are ordered before all implicit references subsequent to the `SFENCE.VMA`.
  40. , sfence_vma_all, "sfence.vma");
  41. instruction!(
  42. /// `FENCE` instruction wrapper
  43. ///
  44. /// The FENCE instruction is used to order device I/O and memory accesses as viewed by other RISC-V
  45. /// harts and external devices or coprocessors. Any combination of device input (I), device output
  46. /// (O), memory reads (R), and memory writes (W) may be ordered with respect to any combination
  47. /// of the same. Informally, no other RISC-V hart or external device can observe any operation in the
  48. /// successor set following a FENCE before any operation in the predecessor set preceding the FENCE.
  49. /// Chapter 17 provides a precise description of the RISC-V memory consistency model.
  50. ///
  51. /// The FENCE instruction also orders memory reads and writes made by the hart as observed by
  52. /// memory reads and writes made by an external device. However, FENCE does not order observations
  53. /// of events made by an external device using any other signaling mechanism.
  54. , fence, "fence");
  55. /// `SFENCE.VMA` instruction wrapper
  56. ///
  57. /// Synchronizes updates to in-memory memory-management data structures with current execution.
  58. /// Instruction execution causes implicit reads and writes to these data structures; however, these implicit references
  59. /// are ordinarily not ordered with respect to loads and stores in the instruction stream.
  60. /// Executing an `SFENCE.VMA` instruction guarantees that any stores in the instruction stream prior to the
  61. /// `SFENCE.VMA` are ordered before all implicit references subsequent to the `SFENCE.VMA`.
  62. #[inline]
  63. #[allow(unused_variables)]
  64. pub unsafe fn sfence_vma(asid: usize, addr: usize) {
  65. match () {
  66. #[cfg(riscv)]
  67. () => core::arch::asm!("sfence.vma {0}, {1}", in(reg) addr, in(reg) asid),
  68. #[cfg(not(riscv))]
  69. () => unimplemented!(),
  70. }
  71. }
  72. /// Blocks the program for *at least* `cycles` CPU cycles.
  73. ///
  74. /// This is implemented in assembly so its execution time is independent of the optimization
  75. /// level, however it is dependent on the specific architecture and core configuration.
  76. ///
  77. /// NOTE that the delay can take much longer if interrupts are serviced during its execution
  78. /// and the execution time may vary with other factors. This delay is mainly useful for simple
  79. /// timer-less initialization of peripherals if and only if accurate timing is not essential. In
  80. /// any other case please use a more accurate method to produce a delay.
  81. #[inline]
  82. #[allow(unused_variables)]
  83. pub unsafe fn delay(cycles: u32) {
  84. match () {
  85. #[cfg(riscv)]
  86. () => {
  87. let real_cyc = 1 + cycles / 2;
  88. core::arch::asm!(
  89. "1:",
  90. "addi {0}, {0}, -1",
  91. "bne {0}, zero, 1b",
  92. inout(reg) real_cyc => _,
  93. options(nomem, nostack),
  94. )
  95. }
  96. #[cfg(not(riscv))]
  97. () => unimplemented!(),
  98. }
  99. }