riscv64.inc 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. {
  2. This file is part of the Free Pascal run time library.
  3. Copyright (c) 2008 by the Free Pascal development team.
  4. Processor dependent implementation for the system unit for
  5. RiscV64
  6. See the file COPYING.FPC, included in this distribution,
  7. for details about the copyright.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  11. **********************************************************************}
  12. { Common RiscV stuff }
  13. {$I ../riscv/riscv.inc}
  14. {****************************************************************************
  15. stack frame related stuff
  16. ****************************************************************************}
  17. {$define FPC_SYSTEM_HAS_GET_CALLER_ADDR}
  18. function get_caller_addr(framebp:pointer;addr:pointer=nil):pointer;assembler;
  19. asm
  20. ld a0, -8*1(a0)
  21. end;
  22. {$define FPC_SYSTEM_HAS_GET_CALLER_FRAME}
  23. function get_caller_frame(framebp:pointer;addr:pointer=nil):pointer;assembler;
  24. asm
  25. ld a0, -8*2(a0)
  26. end;
  27. {****************************************************************************
  28. atomic operations
  29. ****************************************************************************}
  30. {$ifdef VER3_2}
  31. function InterLockedDecrement (var Target: longint) : longint; assembler; nostackframe;
  32. {$else VER3_2}
  33. {$define FPC_SYSTEM_HAS_ATOMIC_DEC_32}
  34. function fpc_atomic_dec_32 (var Target: longint) : longint; assembler; nostackframe;
  35. {$endif VER3_2}
  36. asm
  37. {$ifdef CPURV_HAS_ATOMIC}
  38. addi a1, x0, -1
  39. amoadd.w a0, a1, (a0)
  40. addw a0, a0, a1
  41. {$else CPURV_HAS_ATOMIC}
  42. lw a1, 0(a0)
  43. addiw a1, a1, -1
  44. sw a1, 0(a0)
  45. addi a0, a1, 0
  46. {$endif CPURV_HAS_ATOMIC}
  47. end;
  48. {$ifdef VER3_2}
  49. function InterLockedIncrement (var Target: longint) : longint; assembler; nostackframe;
  50. {$else VER3_2}
  51. {$define FPC_SYSTEM_HAS_ATOMIC_INC_32}
  52. function fpc_atomic_inc_32 (var Target: longint) : longint; assembler; nostackframe;
  53. {$endif VER3_2}
  54. asm
  55. {$ifdef CPURV_HAS_ATOMIC}
  56. addi a1, x0, 1
  57. amoadd.w a0, a1, (a0)
  58. addw a0, a0, a1
  59. {$else CPURV_HAS_ATOMIC}
  60. lw a1, 0(a0)
  61. addiw a1, a1, 1
  62. sw a1, 0(a0)
  63. addi a0, a1, 0
  64. {$endif CPURV_HAS_ATOMIC}
  65. end;
  66. {$ifdef VER3_2}
  67. function InterLockedExchange (var Target: longint;Source : longint) : longint; assembler; nostackframe;
  68. {$else VER3_2}
  69. {$define FPC_SYSTEM_HAS_ATOMIC_XCHG_32}
  70. function fpc_atomic_xchg_32 (var Target: longint;Source : longint) : longint; assembler; nostackframe;
  71. {$endif VER3_2}
  72. asm
  73. {$ifdef CPURV_HAS_ATOMIC}
  74. amoswap.w a0, a1, (a0)
  75. {$else CPURV_HAS_ATOMIC}
  76. lw a2, 0(a0)
  77. sw a1, 0(a0)
  78. addi a0, a2, 0
  79. {$endif CPURV_HAS_ATOMIC}
  80. end;
  81. {$ifdef VER3_2}
  82. function InterlockedCompareExchange(var Target: longint; NewValue: longint; Comperand: longint): longint; assembler; nostackframe;
  83. {$else VER3_2}
  84. {$define FPC_SYSTEM_HAS_ATOMIC_CMP_XCHG_32}
  85. function fpc_atomic_cmp_xchg_32 (var Target: longint; NewValue: longint; Comparand: longint) : longint; assembler; nostackframe;
  86. {$endif VER3_2}
  87. asm
  88. {$ifdef CPURV_HAS_ATOMIC}
  89. .LLoop:
  90. lr.w a3, 0(a0)
  91. bne a3, a2, .LFail
  92. sc.w a4, a1, 0(a0)
  93. bne a4, x0, .LLoop
  94. .LFail:
  95. addi a0, a3, 0
  96. {$else CPURV_HAS_ATOMIC}
  97. lw a3, 0(a0)
  98. bne a3, a2, .LFail
  99. sw a1, 0(a0)
  100. .LFail:
  101. addi a0, a3, 0
  102. {$endif CPURV_HAS_ATOMIC}
  103. end;
  104. {$ifdef VER3_2}
  105. function InterLockedExchangeAdd (var Target: longint;Source : longint) : longint; assembler; nostackframe;
  106. {$else VER3_2}
  107. {$define FPC_SYSTEM_HAS_ATOMIC_ADD_32}
  108. function fpc_atomic_add_32 (var Target: longint;Value: longint) : longint; assembler; nostackframe;
  109. {$endif VER3_2}
  110. asm
  111. {$ifdef CPURV_HAS_ATOMIC}
  112. amoadd.w a0, a1, (a0)
  113. {$else CPURV_HAS_ATOMIC}
  114. lw a2, 0(a0)
  115. addw a2, a2, a1
  116. sw a2, 0(a0)
  117. addi a0, a2, 0
  118. {$endif CPURV_HAS_ATOMIC}
  119. end;
  120. {$ifdef VER3_2}
  121. function InterLockedDecrement64 (var Target: int64) : int64; assembler; nostackframe;
  122. {$else VER3_2}
  123. {$define FPC_SYSTEM_HAS_ATOMIC_DEC_64}
  124. function fpc_atomic_dec_64 (var Target: int64) : int64; assembler; nostackframe;
  125. {$endif VER3_2}
  126. asm
  127. {$ifdef CPURV_HAS_ATOMIC}
  128. addi a1, x0, -1
  129. amoadd.d a0, a1, (a0)
  130. add a0, a0, a1
  131. {$else CPURV_HAS_ATOMIC}
  132. ld a1, 0(a0)
  133. addi a1, a1, -1
  134. sd a1, 0(a0)
  135. addi a0, a1, 0
  136. {$endif CPURV_HAS_ATOMIC}
  137. end;
  138. {$ifdef VER3_2}
  139. function InterLockedIncrement64 (var Target: int64) : int64; assembler; nostackframe;
  140. {$else VER3_2}
  141. {$define FPC_SYSTEM_HAS_ATOMIC_INC_64}
  142. function fpc_atomic_inc_64 (var Target: int64) : int64; assembler; nostackframe;
  143. {$endif VER3_2}
  144. asm
  145. {$ifdef CPURV_HAS_ATOMIC}
  146. addi a1, x0, 1
  147. amoadd.d a0, a1, (a0)
  148. add a0, a0, a1
  149. {$else CPURV_HAS_ATOMIC}
  150. ld a1, 0(a0)
  151. addi a1, a1, 1
  152. sd a1, 0(a0)
  153. addi a0, a1, 0
  154. {$endif CPURV_HAS_ATOMIC}
  155. end;
  156. {$ifdef VER3_2}
  157. function InterLockedExchange64 (var Target: int64;Source : int64) : int64; assembler; nostackframe;
  158. {$else VER3_2}
  159. {$define FPC_SYSTEM_HAS_ATOMIC_XCHG_64}
  160. function fpc_atomic_xchg_64 (var Target: int64;Source : int64) : int64; assembler; nostackframe;
  161. {$endif VER3_2}
  162. asm
  163. {$ifdef CPURV_HAS_ATOMIC}
  164. amoswap.d a0, a1, (a0)
  165. {$else CPURV_HAS_ATOMIC}
  166. ld a2, 0(a0)
  167. sd a1, 0(a0)
  168. addi a0, a2, 0
  169. {$endif CPURV_HAS_ATOMIC}
  170. end;
  171. {$ifdef VER3_2}
  172. function InterlockedCompareExchange64(var Target: int64; NewValue: int64; Comperand: int64): int64; assembler; nostackframe;
  173. {$else VER3_2}
  174. {$define FPC_SYSTEM_HAS_ATOMIC_CMP_XCHG_64}
  175. function fpc_atomic_cmp_xchg_64 (var Target: int64; NewValue: int64; Comparand: int64) : int64; [public,alias:'FPC_ATOMIC_CMP_XCHG_64']; assembler; nostackframe;
  176. {$endif VER3_2}
  177. asm
  178. {$ifdef CPURV_HAS_ATOMIC}
  179. .LLoop:
  180. lr.d a3, 0(a0)
  181. bne a3, a2, .LFail
  182. sc.d a4, a1, 0(a0)
  183. bne a4, x0, .LLoop
  184. .LFail:
  185. addi a0, a3, 0
  186. {$else CPURV_HAS_ATOMIC}
  187. ld a3, 0(a0)
  188. bne a3, a2, .LFail
  189. sd a1, 0(a0)
  190. .LFail:
  191. addi a0, a3, 0
  192. {$endif CPURV_HAS_ATOMIC}
  193. end;
  194. {$ifdef VER3_2}
  195. function InterLockedExchangeAdd64 (var Target: int64;Source : int64) : int64; assembler; nostackframe;
  196. {$else VER3_2}
  197. {$define FPC_SYSTEM_HAS_ATOMIC_ADD_64}
  198. function fpc_atomic_add_64 (var Target: int64;Value : int64) : int64; assembler; nostackframe;
  199. {$endif VER3_2}
  200. asm
  201. {$ifdef CPURV_HAS_ATOMIC}
  202. amoadd.d a0, a1, (a0)
  203. {$else CPURV_HAS_ATOMIC}
  204. ld a2, 0(a0)
  205. add a2, a2, a1
  206. sd a2, 0(a0)
  207. addi a0, a2, 0
  208. {$endif CPURV_HAS_ATOMIC}
  209. end;
  210. {$define FPC_SYSTEM_HAS_DECLOCKED_INT64}
  211. function declocked(var l:int64):boolean;
  212. begin
  213. Result:=InterLockedDecrement64(l) = 0;
  214. end;
  215. {$define FPC_SYSTEM_HAS_INCLOCKED_INT64}
  216. procedure inclocked(var l:int64);
  217. begin
  218. InterLockedIncrement64(l);
  219. end;
  220. {$ifdef CPURV_HAS_ZBB}
  221. {$define FPC_SYSTEM_HAS_SWAPENDIAN}
  222. function SwapEndian(const AValue: SmallInt): SmallInt; assembler; nostackframe;
  223. asm
  224. rev8 a0, a0
  225. srli a0, a0, 48
  226. end;
  227. function SwapEndian(const AValue: Word): Word; assembler; nostackframe;
  228. asm
  229. rev8 a0, a0
  230. srli a0, a0, 48
  231. end;
  232. function SwapEndian(const AValue: LongInt): LongInt; assembler; nostackframe;
  233. asm
  234. rev8 a0, a0
  235. srli a0, a0, 32
  236. end;
  237. function SwapEndian(const AValue: DWord): DWord; assembler; nostackframe;
  238. asm
  239. rev8 a0, a0
  240. srli a0, a0, 32
  241. end;
  242. function SwapEndian(const AValue: Int64): Int64; assembler; nostackframe;
  243. asm
  244. rev8 a0, a0
  245. end;
  246. function SwapEndian(const AValue: QWord): QWord; assembler; nostackframe;
  247. asm
  248. rev8 a0, a0
  249. end;
  250. {$endif CPURV_HAS_ZBB}
  251. {$define FPC_SYSTEM_HAS_UMUL64X64_128}
  252. function UMul64x64_128(a,b: uint64; out rHi: uint64): uint64; assembler; nostackframe;
  253. asm
  254. mulhu a3,a0,a1
  255. mul a0,a0,a1
  256. sd a3,(a2)
  257. end;