asm.rs 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. use core::arch::global_asm;
  2. /// Parse cfg attributes inside a global_asm call.
  3. macro_rules! cfg_global_asm {
  4. {@inner, [$($x:tt)*], } => {
  5. global_asm!{$($x)*}
  6. };
  7. (@inner, [$($x:tt)*], #[cfg($meta:meta)] $asm:literal, $($rest:tt)*) => {
  8. #[cfg($meta)]
  9. cfg_global_asm!{@inner, [$($x)* $asm,], $($rest)*}
  10. #[cfg(not($meta))]
  11. cfg_global_asm!{@inner, [$($x)*], $($rest)*}
  12. };
  13. {@inner, [$($x:tt)*], $asm:literal, $($rest:tt)*} => {
  14. cfg_global_asm!{@inner, [$($x)* $asm,], $($rest)*}
  15. };
  16. {$($asms:tt)*} => {
  17. cfg_global_asm!{@inner, [], $($asms)*}
  18. };
  19. }
  20. // Entry point of all programs (_start). It initializes DWARF call frame information,
  21. // the stack pointer, the frame pointer (needed for closures to work in start_rust)
  22. // and the global pointer. Then it calls _start_rust.
  23. cfg_global_asm!(
  24. ".section .init, \"ax\"
  25. .global _start
  26. _start:",
  27. #[cfg(riscv32)]
  28. "lui ra, %hi(_abs_start)
  29. jr %lo(_abs_start)(ra)",
  30. #[cfg(riscv64)]
  31. ".option push
  32. .option norelax // to prevent an unsupported R_RISCV_ALIGN relocation from being generated
  33. 1:
  34. auipc ra, %pcrel_hi(1f)
  35. ld ra, %pcrel_lo(1b)(ra)
  36. jr ra
  37. .align 3
  38. 1:
  39. .dword _abs_start
  40. .option pop",
  41. "
  42. _abs_start:
  43. .option norelax
  44. .cfi_startproc
  45. .cfi_undefined ra",
  46. #[cfg(feature = "s-mode")]
  47. "csrw sie, 0
  48. csrw sip, 0",
  49. #[cfg(not(feature = "s-mode"))]
  50. "csrw mie, 0
  51. csrw mip, 0",
  52. "li x1, 0
  53. li x2, 0
  54. li x3, 0
  55. li x4, 0
  56. li x5, 0
  57. li x6, 0
  58. li x7, 0
  59. li x8, 0
  60. li x9, 0
  61. // a0..a2 (x10..x12) skipped
  62. li x13, 0
  63. li x14, 0
  64. li x15, 0
  65. li x16, 0
  66. li x17, 0
  67. li x18, 0
  68. li x19, 0
  69. li x20, 0
  70. li x21, 0
  71. li x22, 0
  72. li x23, 0
  73. li x24, 0
  74. li x25, 0
  75. li x26, 0
  76. li x27, 0
  77. li x28, 0
  78. li x29, 0
  79. li x30, 0
  80. li x31, 0
  81. .option push
  82. .option norelax
  83. la gp, __global_pointer$
  84. .option pop
  85. // Allocate stacks",
  86. #[cfg(all(not(feature = "single-hart"), feature = "s-mode"))]
  87. "mv t2, a0 // the hartid is passed as parameter by SMODE",
  88. #[cfg(all(not(feature = "single-hart"), not(feature = "s-mode")))]
  89. "csrr t2, mhartid",
  90. #[cfg(not(feature = "single-hart"))]
  91. "lui t0, %hi(_max_hart_id)
  92. add t0, t0, %lo(_max_hart_id)
  93. bgtu t2, t0, abort
  94. lui t0, %hi(_hart_stack_size)
  95. add t0, t0, %lo(_hart_stack_size)",
  96. #[cfg(all(not(feature = "single-hart"), riscvm))]
  97. "mul t0, t2, t0",
  98. #[cfg(all(not(feature = "single-hart"), not(riscvm)))]
  99. "beqz t2, 2f // Jump if single-hart
  100. mv t1, t2
  101. mv t3, t0
  102. 1:
  103. add t0, t0, t3
  104. addi t1, t1, -1
  105. bnez t1, 1b
  106. 2: ",
  107. "la t1, _stack_start",
  108. #[cfg(not(feature = "single-hart"))]
  109. "sub t1, t1, t0",
  110. "andi sp, t1, -16 // Force 16-byte alignment
  111. // Set frame pointer
  112. add s0, sp, zero
  113. jal zero, _start_rust
  114. .cfi_endproc",
  115. );
  116. /// Trap entry point (_start_trap). It saves caller saved registers, calls
  117. /// _start_trap_rust, restores caller saved registers and then returns.
  118. ///
  119. /// # Usage
  120. ///
  121. /// The macro takes 5 arguments:
  122. /// - `$STORE`: the instruction used to store a register in the stack (e.g. `sd` for riscv64)
  123. /// - `$LOAD`: the instruction used to load a register from the stack (e.g. `ld` for riscv64)
  124. /// - `$BYTES`: the number of bytes used to store a register (e.g. 8 for riscv64)
  125. /// - `$TRAP_SIZE`: the number of registers to store in the stack (e.g. 32 for all the user registers)
  126. /// - list of tuples of the form `($REG, $LOCATION)`, where:
  127. /// - `$REG`: the register to store/load
  128. /// - `$LOCATION`: the location in the stack where to store/load the register
  129. #[rustfmt::skip]
  130. macro_rules! trap_handler {
  131. ($STORE:ident, $LOAD:ident, $BYTES:literal, $TRAP_SIZE:literal, [$(($REG:ident, $LOCATION:literal)),*]) => {
  132. // ensure we do not break that sp is 16-byte aligned
  133. const _: () = assert!(($TRAP_SIZE * $BYTES) % 16 == 0);
  134. global_asm!(
  135. "
  136. .section .trap, \"ax\"
  137. .global default_start_trap
  138. default_start_trap:",
  139. // save space for trap handler in stack
  140. concat!("addi sp, sp, -", stringify!($TRAP_SIZE * $BYTES)),
  141. // save registers in the desired order
  142. $(concat!(stringify!($STORE), " ", stringify!($REG), ", ", stringify!($LOCATION * $BYTES), "(sp)"),)*
  143. // call rust trap handler
  144. "add a0, sp, zero
  145. jal ra, _start_trap_rust",
  146. // restore registers in the desired order
  147. $(concat!(stringify!($LOAD), " ", stringify!($REG), ", ", stringify!($LOCATION * $BYTES), "(sp)"),)*
  148. // free stack
  149. concat!("addi sp, sp, ", stringify!($TRAP_SIZE * $BYTES)),
  150. );
  151. cfg_global_asm!(
  152. // return from trap
  153. #[cfg(feature = "s-mode")]
  154. "sret",
  155. #[cfg(not(feature = "s-mode"))]
  156. "mret",
  157. );
  158. };
  159. }
  160. #[rustfmt::skip]
  161. #[cfg(riscv32)]
  162. trap_handler!(
  163. sw, lw, 4, 16,
  164. [(ra, 0), (t0, 1), (t1, 2), (t2, 3), (t3, 4), (t4, 5), (t5, 6), (t6, 7),
  165. (a0, 8), (a1, 9), (a2, 10), (a3, 11), (a4, 12), (a5, 13), (a6, 14), (a7, 15)]
  166. );
  167. #[rustfmt::skip]
  168. #[cfg(riscv64)]
  169. trap_handler!(
  170. sd, ld, 8, 16,
  171. [(ra, 0), (t0, 1), (t1, 2), (t2, 3), (t3, 4), (t4, 5), (t5, 6), (t6, 7),
  172. (a0, 8), (a1, 9), (a2, 10), (a3, 11), (a4, 12), (a5, 13), (a6, 14), (a7, 15)]
  173. );
  174. // Make sure there is an abort when linking
  175. global_asm!(
  176. ".section .text.abort
  177. .globl abort
  178. abort:
  179. j abort"
  180. );