riscv64.inc 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  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. AVR
  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. {****************************************************************************
  13. fpu exception related stuff
  14. ****************************************************************************}
  15. const
  16. fpu_nx = 1 shl 0;
  17. fpu_uf = 1 shl 1;
  18. fpu_of = 1 shl 2;
  19. fpu_dz = 1 shl 3;
  20. fpu_nv = 1 shl 4;
  21. function getfflags: dword; nostackframe; assembler;
  22. asm
  23. frflags a0
  24. end;
  25. procedure setfflags(flags : dword); nostackframe; assembler;
  26. asm
  27. fsflags a0
  28. end;
  29. procedure RaisePendingExceptions;
  30. var
  31. fflags : dword;
  32. f: TFPUException;
  33. begin
  34. fflags:=getfflags;
  35. if (fflags and fpu_dz) <> 0 then
  36. float_raise(exZeroDivide);
  37. if (fflags and fpu_of) <> 0 then
  38. float_raise(exOverflow);
  39. if (fflags and fpu_uf) <> 0 then
  40. float_raise(exUnderflow);
  41. if (fflags and fpu_nv) <> 0 then
  42. float_raise(exInvalidOp);
  43. if (fflags and fpu_nx) <> 0 then
  44. float_raise(exPrecision);
  45. { now the soft float exceptions }
  46. for f in softfloat_exception_flags do
  47. float_raise(f);
  48. end;
  49. procedure fpc_throwfpuexception;[public,alias:'FPC_THROWFPUEXCEPTION'];
  50. var
  51. fflags : dword;
  52. begin
  53. fflags:=getfflags;
  54. { check, if the exception is masked }
  55. if ((fflags and fpu_dz) <> 0) and (exZeroDivide in softfloat_exception_mask) then
  56. fflags:=fflags and not(fpu_dz);
  57. if ((fflags and fpu_of) <> 0) and (exOverflow in softfloat_exception_mask) then
  58. fflags:=fflags and not(fpu_of);
  59. if ((fflags and fpu_uf) <> 0) and (exUnderflow in softfloat_exception_mask) then
  60. fflags:=fflags and not(fpu_uf);
  61. if ((fflags and fpu_nv) <> 0) and (exInvalidOp in softfloat_exception_mask) then
  62. fflags:=fflags and not(fpu_nv);
  63. if ((fflags and fpu_nx) <> 0) and (exPrecision in softfloat_exception_mask) then
  64. fflags:=fflags and not(fpu_nx);
  65. setfflags(fflags);
  66. if fflags<>0 then
  67. RaisePendingExceptions;
  68. end;
  69. procedure fpc_cpuinit;{$ifdef SYSTEMINLINE}inline;{$endif}
  70. begin
  71. softfloat_exception_mask:=[exPrecision,exUnderflow];
  72. end;
  73. {****************************************************************************
  74. stack frame related stuff
  75. ****************************************************************************}
  76. {$IFNDEF INTERNAL_BACKTRACE}
  77. {$define FPC_SYSTEM_HAS_GET_FRAME}
  78. function get_frame:pointer;assembler;nostackframe;
  79. asm
  80. addi a0, fp, 0
  81. end;
  82. {$ENDIF not INTERNAL_BACKTRACE}
  83. {$define FPC_SYSTEM_HAS_GET_CALLER_ADDR}
  84. function get_caller_addr(framebp:pointer;addr:pointer=nil):pointer;assembler;
  85. asm
  86. ld a0, -8*1(a0)
  87. end;
  88. {$define FPC_SYSTEM_HAS_GET_CALLER_FRAME}
  89. function get_caller_frame(framebp:pointer;addr:pointer=nil):pointer;assembler;
  90. asm
  91. ld a0, -8*2(a0)
  92. end;
  93. {$define FPC_SYSTEM_HAS_SPTR}
  94. Function Sptr : pointer;assembler;nostackframe;
  95. asm
  96. addi a0, sp, 0
  97. end;
  98. function InterLockedDecrement (var Target: longint) : longint; assembler; nostackframe;
  99. asm
  100. {$ifdef CPURV_HAS_ATOMIC}
  101. addi a1, x0, -1
  102. amoadd.w a0, a1, (a0)
  103. addw a0, a0, a1
  104. {$else CPURV_HAS_ATOMIC}
  105. lw a1, 0(a0)
  106. addiw a1, a1, -1
  107. sw a1, 0(a0)
  108. addi a0, a1, 0
  109. {$endif CPURV_HAS_ATOMIC}
  110. end;
  111. function InterLockedIncrement (var Target: longint) : longint; assembler; nostackframe;
  112. asm
  113. {$ifdef CPURV_HAS_ATOMIC}
  114. addi a1, x0, 1
  115. amoadd.w a0, a1, (a0)
  116. addw a0, a0, a1
  117. {$else CPURV_HAS_ATOMIC}
  118. lw a1, 0(a0)
  119. addiw a1, a1, 1
  120. sw a1, 0(a0)
  121. addi a0, a1, 0
  122. {$endif CPURV_HAS_ATOMIC}
  123. end;
  124. function InterLockedExchange (var Target: longint;Source : longint) : longint; assembler; nostackframe;
  125. asm
  126. {$ifdef CPURV_HAS_ATOMIC}
  127. amoswap.w a0, a1, (a0)
  128. {$else CPURV_HAS_ATOMIC}
  129. lw a2, 0(a0)
  130. sw a1, 0(a0)
  131. addi a0, a2
  132. {$endif CPURV_HAS_ATOMIC}
  133. end;
  134. function InterlockedCompareExchange(var Target: longint; NewValue: longint; Comperand: longint): longint; assembler; nostackframe;
  135. asm
  136. {$ifdef CPURV_HAS_ATOMIC}
  137. .LLoop:
  138. lr.w a3, 0(a0)
  139. bne a3, a2, .LFail
  140. sc.w a4, a1, 0(a0)
  141. bne a4, x0, .LLoop
  142. .LFail:
  143. addi a0, a3, 0
  144. {$else CPURV_HAS_ATOMIC}
  145. lw a3, 0(a0)
  146. bne a3, a2, .LFail
  147. sw a1, 0(a0)
  148. .LFail:
  149. addi a0, a3, 0
  150. {$endif CPURV_HAS_ATOMIC}
  151. end;
  152. function InterLockedExchangeAdd (var Target: longint;Source : longint) : longint; assembler; nostackframe;
  153. asm
  154. {$ifdef CPURV_HAS_ATOMIC}
  155. amoadd.w a0, a1, (a0)
  156. {$else CPURV_HAS_ATOMIC}
  157. lw a2, 0(a0)
  158. addiw a2, a2, a1
  159. sw a2, 0(a0)
  160. addi a0, a2, 0
  161. {$endif CPURV_HAS_ATOMIC}
  162. end;
  163. function InterLockedDecrement64 (var Target: int64) : int64; assembler; nostackframe;
  164. asm
  165. {$ifdef CPURV_HAS_ATOMIC}
  166. addi a1, x0, -1
  167. amoadd.d a0, a1, (a0)
  168. add a0, a0, a1
  169. {$else CPURV_HAS_ATOMIC}
  170. ld a1, 0(a0)
  171. addi a1, a1, -1
  172. sd a1, 0(a0)
  173. addi a0, a1, 0
  174. {$endif CPURV_HAS_ATOMIC}
  175. end;
  176. function InterLockedIncrement64 (var Target: int64) : int64; assembler; nostackframe;
  177. asm
  178. {$ifdef CPURV_HAS_ATOMIC}
  179. addi a1, x0, 1
  180. amoadd.d a0, a1, (a0)
  181. add a0, a0, a1
  182. {$else CPURV_HAS_ATOMIC}
  183. ld a1, 0(a0)
  184. addi a1, a1, 1
  185. sd a1, 0(a0)
  186. addi a0, a1, 0
  187. {$endif CPURV_HAS_ATOMIC}
  188. end;
  189. function InterLockedExchange64 (var Target: int64;Source : int64) : int64; assembler; nostackframe;
  190. asm
  191. {$ifdef CPURV_HAS_ATOMIC}
  192. amoswap.d a0, a1, (a0)
  193. {$else CPURV_HAS_ATOMIC}
  194. ld a2, 0(a0)
  195. sd a1, 0(a0)
  196. addi a0, a2
  197. {$endif CPURV_HAS_ATOMIC}
  198. end;
  199. function InterlockedCompareExchange64(var Target: int64; NewValue: int64; Comperand: int64): int64; assembler; nostackframe;
  200. asm
  201. {$ifdef CPURV_HAS_ATOMIC}
  202. .LLoop:
  203. lr.d a3, 0(a0)
  204. bne a3, a2, .LFail
  205. sc.d a4, a1, 0(a0)
  206. bne a4, x0, .LLoop
  207. .LFail:
  208. addi a0, a3, 0
  209. {$else CPURV_HAS_ATOMIC}
  210. ld a3, 0(a0)
  211. bne a3, a2, .LFail
  212. sd a1, 0(a0)
  213. .LFail:
  214. addi a0, a3, 0
  215. {$endif CPURV_HAS_ATOMIC}
  216. end;
  217. function InterLockedExchangeAdd64 (var Target: int64;Source : int64) : int64; assembler; nostackframe;
  218. asm
  219. {$ifdef CPURV_HAS_ATOMIC}
  220. amoadd.d a0, a1, (a0)
  221. {$else CPURV_HAS_ATOMIC}
  222. ld a2, 0(a0)
  223. addi a2, a2, a1
  224. sd a2, 0(a0)
  225. addi a0, a2, 0
  226. {$endif CPURV_HAS_ATOMIC}
  227. end;
  228. {$define FPC_SYSTEM_HAS_DECLOCKED_LONGINT}
  229. function declocked(var l: longint) : boolean; inline;
  230. begin
  231. Result:=InterLockedDecrement(l) = 0;
  232. end;
  233. {$define FPC_SYSTEM_HAS_INCLOCKED_LONGINT}
  234. procedure inclocked(var l: longint); inline;
  235. begin
  236. InterLockedIncrement(l);
  237. end;
  238. {$define FPC_SYSTEM_HAS_DECLOCKED_INT64}
  239. function declocked(var l:int64):boolean;
  240. begin
  241. Result:=InterLockedDecrement64(l) = 0;
  242. end;
  243. {$define FPC_SYSTEM_HAS_INCLOCKED_INT64}
  244. procedure inclocked(var l:int64);
  245. begin
  246. InterLockedIncrement64(l);
  247. end;
  248. {$define FPC_SYSTEM_HAS_MEM_BARRIER}
  249. procedure ReadBarrier; assembler; nostackframe;
  250. asm
  251. fence ir, ir
  252. end;
  253. procedure ReadDependencyBarrier;{$ifdef SYSTEMINLINE}inline;{$endif}
  254. begin
  255. end;
  256. procedure ReadWriteBarrier; assembler; nostackframe;
  257. asm
  258. fence iorw, iorw
  259. end;
  260. procedure WriteBarrier; assembler; nostackframe;
  261. asm
  262. fence ow, ow
  263. end;