avr.inc 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  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. {$asmmode gas}
  13. const
  14. {$i cpuinnr.inc}
  15. { Reads SREG and then disables interrupts, returns contents of SREG }
  16. function avr_save: byte;[INTERNPROC: in_avr_save];
  17. { Restores SREG }
  18. procedure avr_restore(old_sreg: byte); [INTERNPROC: in_avr_restore];
  19. procedure fpc_cpuinit;{$ifdef SYSTEMINLINE}inline;{$endif}
  20. begin
  21. end;
  22. {$define FPC_SYSTEM_HAS_MOVE}
  23. procedure Move(const source;var dest;count:SizeInt);[public, alias: 'FPC_MOVE']; assembler; nostackframe;
  24. asm
  25. push r28
  26. push r29
  27. movw r26, r24 // Src=X
  28. movw r28, r22 // Dest=Y
  29. movw r30, r20 // Count=Z
  30. cp r1, r30
  31. cpc r1, r31
  32. brge .Lexit // if 0 >= Count
  33. cp r28, r26
  34. cpc r29, r27
  35. breq .Lexit // if dest = source
  36. brlo .LForwardMove // if dest < source
  37. // Add count to both pointers
  38. add r26, r30
  39. adc r27, r31
  40. add r28, r30
  41. adc r29, r31
  42. .LBackwardMove:
  43. ld r18, -X
  44. st -Y, r18
  45. sbiw r30, 1
  46. brne .LBackwardMove
  47. rjmp .Lexit
  48. .LForwardMove:
  49. ld r18, X+
  50. st Y+, r18
  51. sbiw r30, 1
  52. brne .LForwardMove
  53. .Lexit:
  54. pop r29
  55. pop r28
  56. end;
  57. {$define FPC_SYSTEM_HAS_FILLCHAR}
  58. Procedure FillChar(var x;count:SizeInt;value:byte);
  59. var
  60. pdest,pend : pbyte;
  61. v : ptruint;
  62. begin
  63. if count <= 0 then
  64. exit;
  65. pdest:=@x;
  66. pend:=pdest+count;
  67. while pdest<pend do
  68. begin
  69. pdest^:=value;
  70. inc(pdest);
  71. end;
  72. end;
  73. {$IFNDEF INTERNAL_BACKTRACE}
  74. {$define FPC_SYSTEM_HAS_GET_FRAME}
  75. function get_frame:pointer;assembler;nostackframe;
  76. asm
  77. end;
  78. {$ENDIF not INTERNAL_BACKTRACE}
  79. {$define FPC_SYSTEM_HAS_GET_CALLER_ADDR}
  80. function get_caller_addr(framebp:pointer;addr:pointer=nil):pointer;assembler;nostackframe;
  81. asm
  82. end;
  83. {$define FPC_SYSTEM_HAS_GET_CALLER_FRAME}
  84. function get_caller_frame(framebp:pointer;addr:pointer=nil):pointer;assembler;nostackframe;
  85. asm
  86. end;
  87. {$define FPC_SYSTEM_HAS_SPTR}
  88. Function Sptr : pointer;assembler;nostackframe;
  89. asm
  90. in r24, 0x3d
  91. in r25, 0x3e
  92. end;
  93. function InterLockedDecrement (var Target: longint) : longint;
  94. var
  95. temp_sreg : byte;
  96. begin
  97. { block interrupts }
  98. temp_sreg:=avr_save();
  99. Result:=Target-1;
  100. Target:=Result;
  101. { release interrupts }
  102. avr_restore(temp_sreg);
  103. end;
  104. function InterLockedIncrement (var Target: longint) : longint;
  105. var
  106. temp_sreg : byte;
  107. begin
  108. { block interrupts }
  109. temp_sreg:=avr_save();
  110. Result:=Target+1;
  111. Target:=Result;
  112. { release interrupts }
  113. avr_restore(temp_sreg);
  114. end;
  115. function InterLockedExchange (var Target: longint;Source : longint) : longint;
  116. var
  117. temp_sreg : byte;
  118. begin
  119. { block interrupts }
  120. temp_sreg:=avr_save();
  121. Result:=Target;
  122. Target:=Source;
  123. { release interrupts }
  124. avr_restore(temp_sreg);
  125. end;
  126. function InterlockedCompareExchange(var Target: longint; NewValue: longint; Comperand: longint): longint;
  127. var
  128. temp_sreg : byte;
  129. begin
  130. { block interrupts }
  131. temp_sreg:=avr_save();
  132. Result:=Target;
  133. if Result=Comperand then
  134. Target:=NewValue;
  135. { release interrupts }
  136. avr_restore(temp_sreg);
  137. end;
  138. function InterLockedExchangeAdd (var Target: longint;Source : longint) : longint;
  139. var
  140. temp_sreg : byte;
  141. begin
  142. { block interrupts }
  143. temp_sreg:=avr_save();
  144. Result:=Target;
  145. Target:=Result+Source;
  146. { release interrupts }
  147. avr_restore(temp_sreg);
  148. end;
  149. function InterLockedDecrement (var Target: smallint) : smallint;
  150. var
  151. temp_sreg : byte;
  152. begin
  153. { block interrupts }
  154. temp_sreg:=avr_save();
  155. Result:=Target-1;
  156. Target:=Result;
  157. { release interrupts }
  158. avr_restore(temp_sreg);
  159. end;
  160. function InterLockedIncrement (var Target: smallint) : smallint;
  161. var
  162. temp_sreg : byte;
  163. begin
  164. { block interrupts }
  165. temp_sreg:=avr_save();
  166. Result:=Target+1;
  167. Target:=Result;
  168. { release interrupts }
  169. avr_restore(temp_sreg);
  170. end;
  171. function InterLockedExchange (var Target: smallint;Source : smallint) : smallint;
  172. var
  173. temp_sreg : byte;
  174. begin
  175. { block interrupts }
  176. temp_sreg:=avr_save();
  177. Result:=Target;
  178. Target:=Source;
  179. { release interrupts }
  180. avr_restore(temp_sreg);
  181. end;
  182. function InterlockedCompareExchange(var Target: smallint; NewValue: smallint; Comperand: smallint): smallint;
  183. var
  184. temp_sreg : byte;
  185. begin
  186. { block interrupts }
  187. temp_sreg:=avr_save();
  188. Result:=Target;
  189. if Result=Comperand then
  190. Target:=NewValue;
  191. { release interrupts }
  192. avr_restore(temp_sreg);
  193. end;
  194. function InterLockedExchangeAdd (var Target: smallint;Source : smallint) : smallint;
  195. var
  196. temp_sreg : byte;
  197. begin
  198. { block interrupts }
  199. temp_sreg:=avr_save();
  200. Result:=Target;
  201. Target:=Result+Source;
  202. { release interrupts }
  203. avr_restore(temp_sreg);
  204. end;
  205. {include hand-optimized assembler code}
  206. {$i math.inc}