original_asm.S 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. #if __riscv_xlen == 64
  2. # define STORE sd
  3. # define LOAD ld
  4. # define LOG_REGBYTES 3
  5. #else
  6. # define STORE sw
  7. # define LOAD lw
  8. # define LOG_REGBYTES 2
  9. #endif
  10. #define REGBYTES (1 << LOG_REGBYTES)
  11. /*
  12. Entry point of all programs (_start).
  13. It initializes DWARF call frame information, the stack pointer, the
  14. frame pointer (needed for closures to work in start_rust) and the global
  15. pointer. Then it calls _start_rust.
  16. */
  17. .section .init, "ax"
  18. .global _start
  19. _start:
  20. /* Jump to the absolute address defined by the linker script. */
  21. // for 32bit
  22. .if __riscv_xlen == 32
  23. lui ra, %hi(_abs_start)
  24. jr %lo(_abs_start)(ra)
  25. .endif
  26. // for 64bit
  27. .if __riscv_xlen == 64
  28. .option push
  29. .option norelax // to prevent an unsupported R_RISCV_ALIGN relocation from being generated
  30. 1:
  31. auipc ra, %pcrel_hi(1f)
  32. ld ra, %pcrel_lo(1b)(ra)
  33. jr ra
  34. .align 3
  35. 1:
  36. .dword _abs_start
  37. .option pop
  38. .endif
  39. _abs_start:
  40. .cfi_startproc
  41. .cfi_undefined ra
  42. #ifdef SMODE
  43. csrw sie, 0 // interrupt disable
  44. csrw sip, 0 // no pending interrupts
  45. #else
  46. csrw mie, 0
  47. csrw mip, 0
  48. #endif
  49. li x1, 0
  50. li x2, 0
  51. li x3, 0
  52. li x4, 0
  53. li x5, 0
  54. li x6, 0
  55. li x7, 0
  56. li x8, 0
  57. li x9, 0
  58. // a0..a2 (x10..x12) skipped
  59. li x13,0
  60. li x14,0
  61. li x15,0
  62. li x16,0
  63. li x17,0
  64. li x18,0
  65. li x19,0
  66. li x20,0
  67. li x21,0
  68. li x22,0
  69. li x23,0
  70. li x24,0
  71. li x25,0
  72. li x26,0
  73. li x27,0
  74. li x28,0
  75. li x29,0
  76. li x30,0
  77. li x31,0
  78. .option push
  79. .option norelax
  80. la gp, __global_pointer$
  81. .option pop
  82. #ifdef SMODE
  83. // there is no equivalent of mhartid in supervisor mode.
  84. // instead, the hartid is passed as paramter by SMODE
  85. mv t2, a0
  86. #else
  87. csrr t2, mhartid
  88. #endif
  89. lui t0, %hi(_max_hart_id)
  90. add t0, t0, %lo(_max_hart_id)
  91. bgtu t2, t0, abort
  92. // Allocate stacks
  93. la sp, _stack_start
  94. lui t0, %hi(_hart_stack_size)
  95. add t0, t0, %lo(_hart_stack_size)
  96. #ifdef __riscv_mul
  97. mul t0, t2, t0
  98. #else
  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. #endif
  108. sub sp, sp, t0
  109. // Set frame pointer
  110. add s0, sp, zero
  111. jal zero, _start_rust
  112. .cfi_endproc
  113. /*
  114. Trap entry point (_start_trap)
  115. Saves caller saved registers ra, t0..6, a0..7, calls _start_trap_rust,
  116. restores caller saved registers and then returns.
  117. */
  118. .section .trap, "ax"
  119. .global default_start_trap
  120. default_start_trap:
  121. addi sp, sp, -16*REGBYTES
  122. STORE ra, 0*REGBYTES(sp)
  123. STORE t0, 1*REGBYTES(sp)
  124. STORE t1, 2*REGBYTES(sp)
  125. STORE t2, 3*REGBYTES(sp)
  126. STORE t3, 4*REGBYTES(sp)
  127. STORE t4, 5*REGBYTES(sp)
  128. STORE t5, 6*REGBYTES(sp)
  129. STORE t6, 7*REGBYTES(sp)
  130. STORE a0, 8*REGBYTES(sp)
  131. STORE a1, 9*REGBYTES(sp)
  132. STORE a2, 10*REGBYTES(sp)
  133. STORE a3, 11*REGBYTES(sp)
  134. STORE a4, 12*REGBYTES(sp)
  135. STORE a5, 13*REGBYTES(sp)
  136. STORE a6, 14*REGBYTES(sp)
  137. STORE a7, 15*REGBYTES(sp)
  138. add a0, sp, zero
  139. jal ra, _start_trap_rust
  140. LOAD ra, 0*REGBYTES(sp)
  141. LOAD t0, 1*REGBYTES(sp)
  142. LOAD t1, 2*REGBYTES(sp)
  143. LOAD t2, 3*REGBYTES(sp)
  144. LOAD t3, 4*REGBYTES(sp)
  145. LOAD t4, 5*REGBYTES(sp)
  146. LOAD t5, 6*REGBYTES(sp)
  147. LOAD t6, 7*REGBYTES(sp)
  148. LOAD a0, 8*REGBYTES(sp)
  149. LOAD a1, 9*REGBYTES(sp)
  150. LOAD a2, 10*REGBYTES(sp)
  151. LOAD a3, 11*REGBYTES(sp)
  152. LOAD a4, 12*REGBYTES(sp)
  153. LOAD a5, 13*REGBYTES(sp)
  154. LOAD a6, 14*REGBYTES(sp)
  155. LOAD a7, 15*REGBYTES(sp)
  156. addi sp, sp, 16*REGBYTES
  157. #ifdef SMODE
  158. sret
  159. #else
  160. mret
  161. #endif
  162. /* Make sure there is an abort when linking */
  163. .section .text.abort
  164. .globl abort
  165. abort:
  166. j abort