cgcpu.pas 81 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130
  1. {
  2. Copyright (c) 2003 by Florian Klaempfl
  3. Member of the Free Pascal development team
  4. This unit implements the code generator for the ARM
  5. This program is free software; you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation; either version 2 of the License, or
  8. (at your option) any later version.
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with this program; if not, write to the Free Software
  15. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  16. ****************************************************************************
  17. }
  18. unit cgcpu;
  19. {$i fpcdefs.inc}
  20. interface
  21. uses
  22. globtype,symtype,symdef,
  23. cgbase,cgutils,cgobj,
  24. aasmbase,aasmcpu,aasmtai,aasmdata,
  25. parabase,
  26. cpubase,cpuinfo,node,cg64f32,rgcpu;
  27. type
  28. tcgarm = class(tcg)
  29. { true, if the next arithmetic operation should modify the flags }
  30. cgsetflags : boolean;
  31. procedure init_register_allocators;override;
  32. procedure done_register_allocators;override;
  33. procedure a_param_const(list : TAsmList;size : tcgsize;a : aint;const paraloc : TCGPara);override;
  34. procedure a_param_ref(list : TAsmList;size : tcgsize;const r : treference;const paraloc : TCGPara);override;
  35. procedure a_paramaddr_ref(list : TAsmList;const r : treference;const paraloc : TCGPara);override;
  36. procedure a_call_name(list : TAsmList;const s : string);override;
  37. procedure a_call_reg(list : TAsmList;reg: tregister);override;
  38. procedure a_call_ref(list : TAsmList;ref: treference);override;
  39. procedure a_op_const_reg(list : TAsmList; Op: TOpCG; size: TCGSize; a: aint; reg: TRegister); override;
  40. procedure a_op_reg_reg(list : TAsmList; Op: TOpCG; size: TCGSize; src, dst: TRegister); override;
  41. procedure a_op_const_reg_reg(list: TAsmList; op: TOpCg;
  42. size: tcgsize; a: aint; src, dst: tregister); override;
  43. procedure a_op_reg_reg_reg(list: TAsmList; op: TOpCg;
  44. size: tcgsize; src1, src2, dst: tregister); override;
  45. procedure a_op_const_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; a: aint; src, dst: tregister;setflags : boolean;var ovloc : tlocation);override;
  46. procedure a_op_reg_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; src1, src2, dst: tregister;setflags : boolean;var ovloc : tlocation);override;
  47. { move instructions }
  48. procedure a_load_const_reg(list : TAsmList; size: tcgsize; a : aint;reg : tregister);override;
  49. procedure a_load_reg_ref(list : TAsmList; fromsize, tosize: tcgsize; reg : tregister;const ref : treference);override;
  50. procedure a_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);override;
  51. procedure a_load_reg_reg(list : TAsmList; fromsize, tosize : tcgsize;reg1,reg2 : tregister);override;
  52. function a_internal_load_reg_ref(list : TAsmList; fromsize, tosize: tcgsize; reg : tregister;const ref : treference):treference;
  53. function a_internal_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister):treference;
  54. { fpu move instructions }
  55. procedure a_loadfpu_reg_reg(list: TAsmList; size: tcgsize; reg1, reg2: tregister); override;
  56. procedure a_loadfpu_ref_reg(list: TAsmList; size: tcgsize; const ref: treference; reg: tregister); override;
  57. procedure a_loadfpu_reg_ref(list: TAsmList; size: tcgsize; reg: tregister; const ref: treference); override;
  58. procedure a_paramfpu_ref(list : TAsmList;size : tcgsize;const ref : treference;const paraloc : TCGPara);override;
  59. { comparison operations }
  60. procedure a_cmp_const_reg_label(list : TAsmList;size : tcgsize;cmp_op : topcmp;a : aint;reg : tregister;
  61. l : tasmlabel);override;
  62. procedure a_cmp_reg_reg_label(list : TAsmList;size : tcgsize;cmp_op : topcmp;reg1,reg2 : tregister;l : tasmlabel); override;
  63. procedure a_jmp_name(list : TAsmList;const s : string); override;
  64. procedure a_jmp_always(list : TAsmList;l: tasmlabel); override;
  65. procedure a_jmp_flags(list : TAsmList;const f : TResFlags;l: tasmlabel); override;
  66. procedure g_flags2reg(list: TAsmList; size: TCgSize; const f: TResFlags; reg: TRegister); override;
  67. procedure g_proc_entry(list : TAsmList;localsize : longint;nostackframe:boolean);override;
  68. procedure g_proc_exit(list : TAsmList;parasize : longint;nostackframe:boolean); override;
  69. procedure a_loadaddr_ref_reg(list : TAsmList;const ref : treference;r : tregister);override;
  70. procedure g_concatcopy(list : TAsmList;const source,dest : treference;len : aint);override;
  71. procedure g_concatcopy_unaligned(list : TAsmList;const source,dest : treference;len : aint);override;
  72. procedure g_concatcopy_move(list : TAsmList;const source,dest : treference;len : aint);
  73. procedure g_concatcopy_internal(list : TAsmList;const source,dest : treference;len : aint;aligned : boolean);
  74. procedure g_overflowcheck(list: TAsmList; const l: tlocation; def: tdef); override;
  75. procedure g_overflowCheck_loc(List:TAsmList;const Loc:TLocation;def:TDef;ovloc : tlocation);override;
  76. procedure g_save_standard_registers(list : TAsmList);override;
  77. procedure g_restore_standard_registers(list : TAsmList);override;
  78. procedure a_jmp_cond(list : TAsmList;cond : TOpCmp;l: tasmlabel);
  79. procedure fixref(list : TAsmList;var ref : treference);
  80. function handle_load_store(list:TAsmList;op: tasmop;oppostfix : toppostfix;reg:tregister;ref: treference):treference;
  81. procedure g_intf_wrapper(list: TAsmList; procdef: tprocdef; const labelname: string; ioffset: longint);override;
  82. end;
  83. tcg64farm = class(tcg64f32)
  84. procedure a_op64_reg_reg(list : TAsmList;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);override;
  85. procedure a_op64_const_reg(list : TAsmList;op:TOpCG;size : tcgsize;value : int64;reg : tregister64);override;
  86. procedure a_op64_const_reg_reg(list: TAsmList;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64);override;
  87. procedure a_op64_reg_reg_reg(list: TAsmList;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64);override;
  88. procedure a_op64_const_reg_reg_checkoverflow(list: TAsmList;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64;setflags : boolean;var ovloc : tlocation);override;
  89. procedure a_op64_reg_reg_reg_checkoverflow(list: TAsmList;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64;setflags : boolean;var ovloc : tlocation);override;
  90. end;
  91. const
  92. OpCmp2AsmCond : Array[topcmp] of TAsmCond = (C_NONE,C_EQ,C_GT,
  93. C_LT,C_GE,C_LE,C_NE,C_LS,C_CC,C_CS,C_HI);
  94. winstackpagesize = 4096;
  95. function get_fpu_postfix(def : tdef) : toppostfix;
  96. implementation
  97. uses
  98. globals,verbose,systems,cutils,
  99. fmodule,
  100. symconst,symsym,
  101. tgobj,
  102. procinfo,cpupi,
  103. paramgr;
  104. function get_fpu_postfix(def : tdef) : toppostfix;
  105. begin
  106. if def.typ=floatdef then
  107. begin
  108. case tfloatdef(def).floattype of
  109. s32real:
  110. result:=PF_S;
  111. s64real:
  112. result:=PF_D;
  113. s80real:
  114. result:=PF_E;
  115. else
  116. internalerror(200401272);
  117. end;
  118. end
  119. else
  120. internalerror(200401271);
  121. end;
  122. procedure tcgarm.init_register_allocators;
  123. begin
  124. inherited init_register_allocators;
  125. { currently, we save R14 always, so we can use it }
  126. rg[R_INTREGISTER]:=trgintcpu.create(R_INTREGISTER,R_SUBWHOLE,
  127. [RS_R0,RS_R1,RS_R2,RS_R3,RS_R4,RS_R5,RS_R6,RS_R7,RS_R8,
  128. RS_R9,RS_R10,RS_R12,RS_R14],first_int_imreg,[]);
  129. rg[R_FPUREGISTER]:=trgcpu.create(R_FPUREGISTER,R_SUBNONE,
  130. [RS_F0,RS_F1,RS_F2,RS_F3,RS_F4,RS_F5,RS_F6,RS_F7],first_fpu_imreg,[]);
  131. rg[R_MMREGISTER]:=trgcpu.create(R_MMREGISTER,R_SUBNONE,
  132. [RS_S0,RS_S1,RS_R2,RS_R3,RS_R4,RS_S31],first_mm_imreg,[]);
  133. end;
  134. procedure tcgarm.done_register_allocators;
  135. begin
  136. rg[R_INTREGISTER].free;
  137. rg[R_FPUREGISTER].free;
  138. rg[R_MMREGISTER].free;
  139. inherited done_register_allocators;
  140. end;
  141. procedure tcgarm.a_param_const(list : TAsmList;size : tcgsize;a : aint;const paraloc : TCGPara);
  142. var
  143. ref: treference;
  144. begin
  145. paraloc.check_simple_location;
  146. case paraloc.location^.loc of
  147. LOC_REGISTER,LOC_CREGISTER:
  148. a_load_const_reg(list,size,a,paraloc.location^.register);
  149. LOC_REFERENCE:
  150. begin
  151. reference_reset(ref);
  152. ref.base:=paraloc.location^.reference.index;
  153. ref.offset:=paraloc.location^.reference.offset;
  154. a_load_const_ref(list,size,a,ref);
  155. end;
  156. else
  157. internalerror(2002081101);
  158. end;
  159. end;
  160. procedure tcgarm.a_param_ref(list : TAsmList;size : tcgsize;const r : treference;const paraloc : TCGPara);
  161. var
  162. tmpref, ref: treference;
  163. location: pcgparalocation;
  164. sizeleft: aint;
  165. begin
  166. location := paraloc.location;
  167. tmpref := r;
  168. sizeleft := paraloc.intsize;
  169. while assigned(location) do
  170. begin
  171. case location^.loc of
  172. LOC_REGISTER,LOC_CREGISTER:
  173. a_load_ref_reg(list,location^.size,location^.size,tmpref,location^.register);
  174. LOC_REFERENCE:
  175. begin
  176. reference_reset_base(ref,location^.reference.index,location^.reference.offset);
  177. { doubles in softemu mode have a strange order of registers and references }
  178. if location^.size=OS_32 then
  179. g_concatcopy(list,tmpref,ref,4)
  180. else
  181. begin
  182. g_concatcopy(list,tmpref,ref,sizeleft);
  183. if assigned(location^.next) then
  184. internalerror(2005010710);
  185. end;
  186. end;
  187. LOC_FPUREGISTER,LOC_CFPUREGISTER:
  188. case location^.size of
  189. OS_F32, OS_F64:
  190. a_loadfpu_ref_reg(list,location^.size,tmpref,location^.register);
  191. else
  192. internalerror(2002072801);
  193. end;
  194. LOC_VOID:
  195. begin
  196. // nothing to do
  197. end;
  198. else
  199. internalerror(2002081103);
  200. end;
  201. inc(tmpref.offset,tcgsize2size[location^.size]);
  202. dec(sizeleft,tcgsize2size[location^.size]);
  203. location := location^.next;
  204. end;
  205. end;
  206. procedure tcgarm.a_paramaddr_ref(list : TAsmList;const r : treference;const paraloc : TCGPara);
  207. var
  208. ref: treference;
  209. tmpreg: tregister;
  210. begin
  211. paraloc.check_simple_location;
  212. case paraloc.location^.loc of
  213. LOC_REGISTER,LOC_CREGISTER:
  214. a_loadaddr_ref_reg(list,r,paraloc.location^.register);
  215. LOC_REFERENCE:
  216. begin
  217. reference_reset(ref);
  218. ref.base := paraloc.location^.reference.index;
  219. ref.offset := paraloc.location^.reference.offset;
  220. tmpreg := getintregister(list,OS_ADDR);
  221. a_loadaddr_ref_reg(list,r,tmpreg);
  222. a_load_reg_ref(list,OS_ADDR,OS_ADDR,tmpreg,ref);
  223. end;
  224. else
  225. internalerror(2002080701);
  226. end;
  227. end;
  228. procedure tcgarm.a_call_name(list : TAsmList;const s : string);
  229. begin
  230. list.concat(taicpu.op_sym(A_BL,current_asmdata.RefAsmSymbol(s)));
  231. {
  232. the compiler does not properly set this flag anymore in pass 1, and
  233. for now we only need it after pass 2 (I hope) (JM)
  234. if not(pi_do_call in current_procinfo.flags) then
  235. internalerror(2003060703);
  236. }
  237. include(current_procinfo.flags,pi_do_call);
  238. end;
  239. procedure tcgarm.a_call_reg(list : TAsmList;reg: tregister);
  240. var
  241. r : tregister;
  242. begin
  243. list.concat(taicpu.op_reg_reg(A_MOV,NR_R14,NR_PC));
  244. list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,reg));
  245. {
  246. the compiler does not properly set this flag anymore in pass 1, and
  247. for now we only need it after pass 2 (I hope) (JM)
  248. if not(pi_do_call in current_procinfo.flags) then
  249. internalerror(2003060703);
  250. }
  251. include(current_procinfo.flags,pi_do_call);
  252. end;
  253. procedure tcgarm.a_call_ref(list : TAsmList;ref: treference);
  254. begin
  255. a_reg_alloc(list,NR_R12);
  256. a_load_ref_reg(list,OS_ADDR,OS_ADDR,ref,NR_R12);
  257. list.concat(taicpu.op_reg_reg(A_MOV,NR_R14,NR_PC));
  258. list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R12));
  259. a_reg_dealloc(list,NR_R12);
  260. include(current_procinfo.flags,pi_do_call);
  261. end;
  262. procedure tcgarm.a_op_const_reg(list : TAsmList; Op: TOpCG; size: TCGSize; a: aint; reg: TRegister);
  263. begin
  264. a_op_const_reg_reg(list,op,size,a,reg,reg);
  265. end;
  266. procedure tcgarm.a_op_reg_reg(list : TAsmList; Op: TOpCG; size: TCGSize; src, dst: TRegister);
  267. begin
  268. case op of
  269. OP_NEG:
  270. list.concat(taicpu.op_reg_reg_const(A_RSB,dst,src,0));
  271. OP_NOT:
  272. begin
  273. list.concat(taicpu.op_reg_reg(A_MVN,dst,src));
  274. case size of
  275. OS_8 :
  276. a_op_const_reg_reg(list,OP_AND,OS_INT,$ff,dst,dst);
  277. OS_16 :
  278. a_op_const_reg_reg(list,OP_AND,OS_INT,$ffff,dst,dst);
  279. end;
  280. end
  281. else
  282. a_op_reg_reg_reg(list,op,OS_32,src,dst,dst);
  283. end;
  284. end;
  285. const
  286. op_reg_reg_opcg2asmop: array[TOpCG] of tasmop =
  287. (A_NONE,A_MOV,A_ADD,A_AND,A_NONE,A_NONE,A_MUL,A_MUL,A_NONE,A_NONE,A_ORR,
  288. A_NONE,A_NONE,A_NONE,A_SUB,A_EOR);
  289. procedure tcgarm.a_op_const_reg_reg(list: TAsmList; op: TOpCg;
  290. size: tcgsize; a: aint; src, dst: tregister);
  291. var
  292. ovloc : tlocation;
  293. begin
  294. a_op_const_reg_reg_checkoverflow(list,op,size,a,src,dst,false,ovloc);
  295. end;
  296. procedure tcgarm.a_op_reg_reg_reg(list: TAsmList; op: TOpCg;
  297. size: tcgsize; src1, src2, dst: tregister);
  298. var
  299. ovloc : tlocation;
  300. begin
  301. a_op_reg_reg_reg_checkoverflow(list,op,size,src1,src2,dst,false,ovloc);
  302. end;
  303. procedure tcgarm.a_op_const_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; a: aint; src, dst: tregister;setflags : boolean;var ovloc : tlocation);
  304. var
  305. shift : byte;
  306. tmpreg : tregister;
  307. so : tshifterop;
  308. l1 : longint;
  309. begin
  310. ovloc.loc:=LOC_VOID;
  311. if is_shifter_const(-a,shift) then
  312. case op of
  313. OP_ADD:
  314. begin
  315. op:=OP_SUB;
  316. a:=dword(-a);
  317. end;
  318. OP_SUB:
  319. begin
  320. op:=OP_ADD;
  321. a:=dword(-a);
  322. end
  323. end;
  324. if is_shifter_const(a,shift) and not(op in [OP_IMUL,OP_MUL]) then
  325. case op of
  326. OP_NEG,OP_NOT,
  327. OP_DIV,OP_IDIV:
  328. internalerror(200308281);
  329. OP_SHL:
  330. begin
  331. if a>32 then
  332. internalerror(200308294);
  333. if a<>0 then
  334. begin
  335. shifterop_reset(so);
  336. so.shiftmode:=SM_LSL;
  337. so.shiftimm:=a;
  338. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));
  339. end
  340. else
  341. list.concat(taicpu.op_reg_reg(A_MOV,dst,src));
  342. end;
  343. OP_SHR:
  344. begin
  345. if a>32 then
  346. internalerror(200308292);
  347. shifterop_reset(so);
  348. if a<>0 then
  349. begin
  350. so.shiftmode:=SM_LSR;
  351. so.shiftimm:=a;
  352. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));
  353. end
  354. else
  355. list.concat(taicpu.op_reg_reg(A_MOV,dst,src));
  356. end;
  357. OP_SAR:
  358. begin
  359. if a>32 then
  360. internalerror(200308295);
  361. if a<>0 then
  362. begin
  363. shifterop_reset(so);
  364. so.shiftmode:=SM_ASR;
  365. so.shiftimm:=a;
  366. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src,so));
  367. end
  368. else
  369. list.concat(taicpu.op_reg_reg(A_MOV,dst,src));
  370. end;
  371. else
  372. list.concat(setoppostfix(
  373. taicpu.op_reg_reg_const(op_reg_reg_opcg2asmop[op],dst,src,a),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))
  374. ));
  375. if (cgsetflags or setflags) and (size in [OS_8,OS_16,OS_32]) then
  376. begin
  377. ovloc.loc:=LOC_FLAGS;
  378. case op of
  379. OP_ADD:
  380. ovloc.resflags:=F_CS;
  381. OP_SUB:
  382. ovloc.resflags:=F_CC;
  383. end;
  384. end;
  385. end
  386. else
  387. begin
  388. { there could be added some more sophisticated optimizations }
  389. if (op in [OP_MUL,OP_IMUL]) and (a=1) then
  390. a_load_reg_reg(list,size,size,src,dst)
  391. else if (op in [OP_MUL,OP_IMUL]) and (a=0) then
  392. a_load_const_reg(list,size,0,dst)
  393. else if (op in [OP_IMUL]) and (a=-1) then
  394. a_op_reg_reg(list,OP_NEG,size,src,dst)
  395. { we do this here instead in the peephole optimizer because
  396. it saves us a register }
  397. else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a,l1) and not(cgsetflags or setflags) then
  398. a_op_const_reg_reg(list,OP_SHL,size,l1,src,dst)
  399. { for example : b=a*5 -> b=a*4+a with add instruction and shl }
  400. else if (op in [OP_MUL,OP_IMUL]) and ispowerof2(a-1,l1) and not(cgsetflags or setflags) then
  401. begin
  402. if l1>32 then{roozbeh does this ever happen?}
  403. internalerror(200308296);
  404. shifterop_reset(so);
  405. so.shiftmode:=SM_LSL;
  406. so.shiftimm:=l1;
  407. list.concat(taicpu.op_reg_reg_reg_shifterop(A_ADD,dst,src,src,so));
  408. end
  409. else
  410. begin
  411. tmpreg:=getintregister(list,size);
  412. a_load_const_reg(list,size,a,tmpreg);
  413. a_op_reg_reg_reg_checkoverflow(list,op,size,tmpreg,src,dst,setflags,ovloc);
  414. end;
  415. end;
  416. end;
  417. procedure tcgarm.a_op_reg_reg_reg_checkoverflow(list: TAsmList; op: TOpCg; size: tcgsize; src1, src2, dst: tregister;setflags : boolean;var ovloc : tlocation);
  418. var
  419. so : tshifterop;
  420. tmpreg,overflowreg : tregister;
  421. asmop : tasmop;
  422. begin
  423. ovloc.loc:=LOC_VOID;
  424. case op of
  425. OP_NEG,OP_NOT,
  426. OP_DIV,OP_IDIV:
  427. internalerror(200308281);
  428. OP_SHL:
  429. begin
  430. shifterop_reset(so);
  431. so.rs:=src1;
  432. so.shiftmode:=SM_LSL;
  433. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src2,so));
  434. end;
  435. OP_SHR:
  436. begin
  437. shifterop_reset(so);
  438. so.rs:=src1;
  439. so.shiftmode:=SM_LSR;
  440. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src2,so));
  441. end;
  442. OP_SAR:
  443. begin
  444. shifterop_reset(so);
  445. so.rs:=src1;
  446. so.shiftmode:=SM_ASR;
  447. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,dst,src2,so));
  448. end;
  449. OP_IMUL,
  450. OP_MUL:
  451. begin
  452. if cgsetflags or setflags then
  453. begin
  454. overflowreg:=getintregister(list,size);
  455. if op=OP_IMUL then
  456. asmop:=A_SMULL
  457. else
  458. asmop:=A_UMULL;
  459. { the arm doesn't allow that rd and rm are the same }
  460. if dst=src2 then
  461. begin
  462. if dst<>src1 then
  463. list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,src1,src2))
  464. else
  465. begin
  466. tmpreg:=getintregister(list,size);
  467. a_load_reg_reg(list,size,size,src2,dst);
  468. list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,tmpreg,src1));
  469. end;
  470. end
  471. else
  472. list.concat(taicpu.op_reg_reg_reg_reg(asmop,dst,overflowreg,src2,src1));
  473. if op=OP_IMUL then
  474. begin
  475. shifterop_reset(so);
  476. so.shiftmode:=SM_ASR;
  477. so.shiftimm:=31;
  478. list.concat(taicpu.op_reg_reg_shifterop(A_CMP,overflowreg,dst,so));
  479. end
  480. else
  481. list.concat(taicpu.op_reg_const(A_CMP,overflowreg,0));
  482. ovloc.loc:=LOC_FLAGS;
  483. ovloc.resflags:=F_NE;
  484. end
  485. else
  486. begin
  487. { the arm doesn't allow that rd and rm are the same }
  488. if dst=src2 then
  489. begin
  490. if dst<>src1 then
  491. list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,src1,src2))
  492. else
  493. begin
  494. tmpreg:=getintregister(list,size);
  495. a_load_reg_reg(list,size,size,src2,dst);
  496. list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,tmpreg,src1));
  497. end;
  498. end
  499. else
  500. list.concat(taicpu.op_reg_reg_reg(A_MUL,dst,src2,src1));
  501. end;
  502. end;
  503. else
  504. list.concat(setoppostfix(
  505. taicpu.op_reg_reg_reg(op_reg_reg_opcg2asmop[op],dst,src2,src1),toppostfix(ord(cgsetflags or setflags)*ord(PF_S))
  506. ));
  507. end;
  508. end;
  509. procedure tcgarm.a_load_const_reg(list : TAsmList; size: tcgsize; a : aint;reg : tregister);
  510. var
  511. imm_shift : byte;
  512. l : tasmlabel;
  513. hr : treference;
  514. tmpreg : tregister;
  515. begin
  516. if not(size in [OS_8,OS_S8,OS_16,OS_S16,OS_32,OS_S32]) then
  517. internalerror(2002090902);
  518. if is_shifter_const(a,imm_shift) then
  519. list.concat(taicpu.op_reg_const(A_MOV,reg,a))
  520. else if is_shifter_const(not(a),imm_shift) then
  521. list.concat(taicpu.op_reg_const(A_MVN,reg,not(a)))
  522. { loading of constants with mov and orr }
  523. {else [if (is_shifter_const(a-byte(a),imm_shift)) then
  524. begin
  525. }{ roozbeh:why using tmpreg later causes error in compiling of system.pp,and also those other similars}
  526. {list.concat(taicpu.op_reg_const(A_MOV,reg,a-byte(a)));
  527. list.concat(taicpu.op_reg_reg_const(A_ORR,reg,reg,byte(a)));
  528. end
  529. else if (is_shifter_const(a-word(a),imm_shift)) and (is_shifter_const(word(a),imm_shift)) then
  530. begin
  531. list.concat(taicpu.op_reg_const(A_MOV,reg,a-word(a)));
  532. list.concat(taicpu.op_reg_reg_const(A_ORR,reg,reg,word(a)));
  533. end
  534. else if (is_shifter_const(a-(longint(a) shl 8) shr 8,imm_shift)) and (is_shifter_const((longint(a) shl 8) shr 8,imm_shift)) then
  535. begin
  536. list.concat(taicpu.op_reg_const(A_MOV,reg,a-(longint(a) shl 8)shr 8));
  537. list.concat(taicpu.op_reg_reg_const(A_ORR,reg,reg,(longint(a) shl 8)shr 8));
  538. end}
  539. else
  540. begin
  541. reference_reset(hr);
  542. current_asmdata.getjumplabel(l);
  543. cg.a_label(current_procinfo.aktlocaldata,l);
  544. hr.symboldata:=current_procinfo.aktlocaldata.last;
  545. current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(longint(a)));
  546. hr.symbol:=l;
  547. list.concat(taicpu.op_reg_ref(A_LDR,reg,hr));
  548. end;
  549. end;
  550. function tcgarm.handle_load_store(list:TAsmList;op: tasmop;oppostfix : toppostfix;reg:tregister;ref: treference):treference;
  551. var
  552. tmpreg,tmpreg2 : tregister;
  553. tmpref : treference;
  554. l : tasmlabel;
  555. so : tshifterop;
  556. begin
  557. tmpreg:=NR_NO;
  558. { Be sure to have a base register }
  559. if (ref.base=NR_NO) then
  560. begin
  561. if ref.shiftmode<>SM_None then
  562. internalerror(200308294);
  563. ref.base:=ref.index;
  564. ref.index:=NR_NO;
  565. end;
  566. { absolute symbols can't be handled directly, we've to store the symbol reference
  567. in the text segment and access it pc relative
  568. For now, we assume that references where base or index equals to PC are already
  569. relative, all other references are assumed to be absolute and thus they need
  570. to be handled extra.
  571. A proper solution would be to change refoptions to a set and store the information
  572. if the symbol is absolute or relative there.
  573. }
  574. if (assigned(ref.symbol) and
  575. not(is_pc(ref.base)) and
  576. not(is_pc(ref.index))
  577. ) or
  578. { [#xxx] isn't a valid address operand }
  579. ((ref.base=NR_NO) and (ref.index=NR_NO)) or
  580. (ref.offset<-4095) or
  581. (ref.offset>4095) or
  582. ((oppostfix in [PF_SB,PF_H,PF_SH]) and
  583. ((ref.offset<-255) or
  584. (ref.offset>255)
  585. )
  586. ) or
  587. ((op in [A_LDF,A_STF]) and
  588. ((ref.offset<-1020) or
  589. (ref.offset>1020) or
  590. { the usual pc relative symbol handling assumes possible offsets of +/- 4095 }
  591. assigned(ref.symbol)
  592. )
  593. ) then
  594. begin
  595. reference_reset(tmpref);
  596. { load symbol }
  597. tmpreg:=getintregister(list,OS_INT);
  598. if assigned(ref.symbol) then
  599. begin
  600. current_asmdata.getjumplabel(l);
  601. cg.a_label(current_procinfo.aktlocaldata,l);
  602. tmpref.symboldata:=current_procinfo.aktlocaldata.last;
  603. current_procinfo.aktlocaldata.concat(tai_const.create_sym_offset(ref.symbol,ref.offset));
  604. { load consts entry }
  605. tmpref.symbol:=l;
  606. tmpref.base:=NR_R15;
  607. list.concat(taicpu.op_reg_ref(A_LDR,tmpreg,tmpref));
  608. end
  609. else
  610. a_load_const_reg(list,OS_ADDR,ref.offset,tmpreg);
  611. if (ref.base<>NR_NO) then
  612. begin
  613. if ref.index<>NR_NO then
  614. begin
  615. list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));
  616. ref.base:=tmpreg;
  617. end
  618. else
  619. begin
  620. ref.index:=tmpreg;
  621. ref.shiftimm:=0;
  622. ref.signindex:=1;
  623. ref.shiftmode:=SM_None;
  624. end;
  625. end
  626. else
  627. ref.base:=tmpreg;
  628. ref.offset:=0;
  629. ref.symbol:=nil;
  630. end;
  631. if (ref.base<>NR_NO) and (ref.index<>NR_NO) and (ref.offset<>0) then
  632. begin
  633. if tmpreg<>NR_NO then
  634. a_op_const_reg_reg(list,OP_ADD,OS_ADDR,ref.offset,tmpreg,tmpreg)
  635. else
  636. begin
  637. tmpreg:=getintregister(list,OS_ADDR);
  638. a_op_const_reg_reg(list,OP_ADD,OS_ADDR,ref.offset,ref.base,tmpreg);
  639. ref.base:=tmpreg;
  640. end;
  641. ref.offset:=0;
  642. end;
  643. { floating point operations have only limited references
  644. we expect here, that a base is already set }
  645. if (op in [A_LDF,A_STF]) and (ref.index<>NR_NO) then
  646. begin
  647. if ref.shiftmode<>SM_none then
  648. internalerror(200309121);
  649. if tmpreg<>NR_NO then
  650. begin
  651. if ref.base=tmpreg then
  652. begin
  653. if ref.signindex<0 then
  654. list.concat(taicpu.op_reg_reg_reg(A_SUB,tmpreg,tmpreg,ref.index))
  655. else
  656. list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,tmpreg,ref.index));
  657. ref.index:=NR_NO;
  658. end
  659. else
  660. begin
  661. if ref.index<>tmpreg then
  662. internalerror(200403161);
  663. if ref.signindex<0 then
  664. list.concat(taicpu.op_reg_reg_reg(A_SUB,tmpreg,ref.base,tmpreg))
  665. else
  666. list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));
  667. ref.base:=tmpreg;
  668. ref.index:=NR_NO;
  669. end;
  670. end
  671. else
  672. begin
  673. tmpreg:=getintregister(list,OS_ADDR);
  674. list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,ref.index));
  675. ref.base:=tmpreg;
  676. ref.index:=NR_NO;
  677. end;
  678. end;
  679. list.concat(setoppostfix(taicpu.op_reg_ref(op,reg,ref),oppostfix));
  680. Result := ref;
  681. end;
  682. procedure tcgarm.a_load_reg_ref(list : TAsmList; fromsize, tosize: tcgsize; reg : tregister;const ref : treference);
  683. var
  684. oppostfix:toppostfix;
  685. usedtmpref,usedtmpref2: treference;
  686. tmpreg,tmpreg2 : tregister;
  687. so : tshifterop;
  688. begin
  689. case ToSize of
  690. { signed integer registers }
  691. OS_8,
  692. OS_S8:
  693. oppostfix:=PF_B;
  694. OS_16,
  695. OS_S16:
  696. oppostfix:=PF_H;
  697. OS_32,
  698. OS_S32:
  699. oppostfix:=PF_None;
  700. else
  701. InternalError(200308295);
  702. end;
  703. if ref.alignment<>0 then
  704. begin
  705. case FromSize of
  706. OS_16,OS_S16:
  707. begin
  708. shifterop_reset(so);so.shiftmode:=SM_LSR;so.shiftimm:=8;
  709. tmpreg:=getintregister(list,OS_INT);
  710. usedtmpref:=a_internal_load_reg_ref(list,OS_8,OS_8,reg,Ref);
  711. inc(usedtmpref.offset);
  712. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,tmpreg,reg,so));
  713. a_internal_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref);
  714. end;
  715. OS_32,OS_S32:
  716. begin
  717. shifterop_reset(so);so.shiftmode:=SM_LSR;so.shiftimm:=8;
  718. tmpreg:=getintregister(list,OS_INT);
  719. usedtmpref:=a_internal_load_reg_ref(list,OS_8,OS_8,reg,Ref);
  720. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,tmpreg,reg,so));
  721. inc(usedtmpref.offset);
  722. a_internal_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref);
  723. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,tmpreg,tmpreg,so));
  724. inc(usedtmpref.offset);
  725. a_internal_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref);
  726. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,tmpreg,tmpreg,so));
  727. inc(usedtmpref.offset);
  728. a_internal_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref);
  729. end
  730. else
  731. handle_load_store(list,A_STR,oppostfix,reg,ref);
  732. end;
  733. end
  734. else
  735. handle_load_store(list,A_STR,oppostfix,reg,ref);
  736. end;
  737. procedure tcgarm.a_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister);
  738. var
  739. oppostfix:toppostfix;
  740. usedtmpref,usedtmpref2: treference;
  741. tmpreg,tmpreg2,tmpreg3 : tregister;
  742. so : tshifterop;
  743. begin
  744. case FromSize of
  745. { signed integer registers }
  746. OS_8:
  747. oppostfix:=PF_B;
  748. OS_S8:
  749. oppostfix:=PF_SB;
  750. OS_16:
  751. oppostfix:=PF_H;
  752. OS_S16:
  753. oppostfix:=PF_SH;
  754. OS_32,
  755. OS_S32:
  756. oppostfix:=PF_None;
  757. else
  758. InternalError(200308297);
  759. end;
  760. if Ref.alignment<>0 then
  761. begin
  762. case FromSize of
  763. OS_16,OS_S16:
  764. begin
  765. a_loadaddr_ref_reg(list,ref,reg);
  766. reference_reset_base(usedtmpref,reg,0);
  767. shifterop_reset(so);so.shiftmode:=SM_LSL;so.shiftimm:=8;
  768. tmpreg:=getintregister(list,OS_INT);
  769. a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);
  770. inc(usedtmpref.offset);
  771. tmpreg2:=getintregister(list,OS_INT);
  772. if FromSize=OS_16 then
  773. a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg2)
  774. else
  775. a_internal_load_ref_reg(list,OS_S8,OS_S8,usedtmpref,tmpreg2);
  776. list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,tmpreg,tmpreg2,so));
  777. end;
  778. OS_32,OS_S32:
  779. begin
  780. tmpreg:=getintregister(list,OS_INT);
  781. tmpreg2:=getintregister(list,OS_INT);
  782. tmpreg3:=getintregister(list,OS_INT);
  783. shifterop_reset(so);so.shiftmode:=SM_LSL;so.shiftimm:=8;
  784. a_loadaddr_ref_reg(list,ref,tmpreg3);
  785. reference_reset_base(usedtmpref,tmpreg3,0);
  786. a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,reg);
  787. inc(usedtmpref.offset);
  788. a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);
  789. list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,tmpreg2,reg,tmpreg,so));
  790. inc(usedtmpref.offset);
  791. a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,reg);
  792. so.shiftimm:=16;
  793. list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,tmpreg,tmpreg2,reg,so));
  794. inc(usedtmpref.offset);
  795. a_internal_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg2);
  796. so.shiftimm:=24;
  797. list.concat(taicpu.op_reg_reg_reg_shifterop(A_ORR,reg,tmpreg,tmpreg2,so));
  798. end
  799. else
  800. handle_load_store(list,A_LDR,oppostfix,reg,ref);
  801. end;
  802. end
  803. else
  804. handle_load_store(list,A_LDR,oppostfix,reg,ref);
  805. end;
  806. function tcgarm.a_internal_load_reg_ref(list : TAsmList; fromsize, tosize: tcgsize; reg : tregister;const ref : treference):treference;
  807. var
  808. oppostfix:toppostfix;
  809. begin
  810. case ToSize of
  811. { signed integer registers }
  812. OS_8,
  813. OS_S8:
  814. oppostfix:=PF_B;
  815. OS_16,
  816. OS_S16:
  817. oppostfix:=PF_H;
  818. OS_32,
  819. OS_S32:
  820. oppostfix:=PF_None;
  821. else
  822. InternalError(2003082910);
  823. end;
  824. result:=handle_load_store(list,A_STR,oppostfix,reg,ref);
  825. end;
  826. function tcgarm.a_internal_load_ref_reg(list : TAsmList; fromsize, tosize : tcgsize;const Ref : treference;reg : tregister):treference;
  827. var
  828. oppostfix:toppostfix;
  829. begin
  830. case FromSize of
  831. { signed integer registers }
  832. OS_8:
  833. oppostfix:=PF_B;
  834. OS_S8:
  835. oppostfix:=PF_SB;
  836. OS_16:
  837. oppostfix:=PF_H;
  838. OS_S16:
  839. oppostfix:=PF_SH;
  840. OS_32,
  841. OS_S32:
  842. oppostfix:=PF_None;
  843. else
  844. InternalError(200308291);
  845. end;
  846. result:=handle_load_store(list,A_LDR,oppostfix,reg,ref);
  847. end;
  848. procedure tcgarm.a_load_reg_reg(list : TAsmList; fromsize, tosize : tcgsize;reg1,reg2 : tregister);
  849. var
  850. instr: taicpu;
  851. so : tshifterop;
  852. begin
  853. shifterop_reset(so);
  854. if (tcgsize2size[tosize] < tcgsize2size[fromsize]) or
  855. (
  856. (tcgsize2size[tosize] = tcgsize2size[fromsize]) and
  857. (tosize <> fromsize) and
  858. not(fromsize in [OS_32,OS_S32])
  859. ) then
  860. begin
  861. case tosize of
  862. OS_8:
  863. list.concat(taicpu.op_reg_reg_const(A_AND,
  864. reg2,reg1,$ff));
  865. OS_S8:
  866. begin
  867. so.shiftmode:=SM_LSL;
  868. so.shiftimm:=24;
  869. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg1,so));
  870. so.shiftmode:=SM_ASR;
  871. so.shiftimm:=24;
  872. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg2,so));
  873. end;
  874. OS_16:
  875. begin
  876. so.shiftmode:=SM_LSL;
  877. so.shiftimm:=16;
  878. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg1,so));
  879. so.shiftmode:=SM_LSR;
  880. so.shiftimm:=16;
  881. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg2,so));
  882. end;
  883. OS_S16:
  884. begin
  885. so.shiftmode:=SM_LSL;
  886. so.shiftimm:=16;
  887. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg1,so));
  888. so.shiftmode:=SM_ASR;
  889. so.shiftimm:=16;
  890. list.concat(taicpu.op_reg_reg_shifterop(A_MOV,reg2,reg2,so));
  891. end;
  892. OS_32,OS_S32:
  893. begin
  894. instr:=taicpu.op_reg_reg(A_MOV,reg2,reg1);
  895. list.concat(instr);
  896. add_move_instruction(instr);
  897. end;
  898. else internalerror(2002090901);
  899. end;
  900. end
  901. else
  902. begin
  903. if reg1<>reg2 then
  904. begin
  905. { same size, only a register mov required }
  906. instr:=taicpu.op_reg_reg(A_MOV,reg2,reg1);
  907. list.Concat(instr);
  908. { Notify the register allocator that we have written a move instruction so
  909. it can try to eliminate it. }
  910. add_move_instruction(instr);
  911. end;
  912. end;
  913. end;
  914. procedure tcgarm.a_paramfpu_ref(list : TAsmList;size : tcgsize;const ref : treference;const paraloc : TCGPara);
  915. var
  916. href,href2 : treference;
  917. hloc : pcgparalocation;
  918. begin
  919. href:=ref;
  920. hloc:=paraloc.location;
  921. while assigned(hloc) do
  922. begin
  923. case hloc^.loc of
  924. LOC_FPUREGISTER,LOC_CFPUREGISTER:
  925. a_loadfpu_ref_reg(list,size,ref,hloc^.register);
  926. LOC_REGISTER :
  927. case hloc^.size of
  928. OS_F32:
  929. a_load_ref_reg(list,OS_32,OS_32,href,hloc^.register);
  930. OS_64,
  931. OS_F64:
  932. cg64.a_param64_ref(list,href,paraloc);
  933. else
  934. a_load_ref_reg(list,hloc^.size,hloc^.size,href,hloc^.register);
  935. end;
  936. LOC_REFERENCE :
  937. begin
  938. reference_reset_base(href2,hloc^.reference.index,hloc^.reference.offset);
  939. { concatcopy should choose the best way to copy the data }
  940. g_concatcopy(list,href,href2,tcgsize2size[size]);
  941. end;
  942. else
  943. internalerror(200408241);
  944. end;
  945. inc(href.offset,tcgsize2size[hloc^.size]);
  946. hloc:=hloc^.next;
  947. end;
  948. end;
  949. procedure tcgarm.a_loadfpu_reg_reg(list: TAsmList; size: tcgsize; reg1, reg2: tregister);
  950. begin
  951. list.concat(setoppostfix(taicpu.op_reg_reg(A_MVF,reg2,reg1),cgsize2fpuoppostfix[size]));
  952. end;
  953. procedure tcgarm.a_loadfpu_ref_reg(list: TAsmList; size: tcgsize; const ref: treference; reg: tregister);
  954. var
  955. oppostfix:toppostfix;
  956. begin
  957. case size of
  958. OS_32,
  959. OS_F32:
  960. oppostfix:=PF_S;
  961. OS_64,
  962. OS_F64:
  963. oppostfix:=PF_D;
  964. OS_F80:
  965. oppostfix:=PF_E;
  966. else
  967. InternalError(200309021);
  968. end;
  969. handle_load_store(list,A_LDF,oppostfix,reg,ref);
  970. end;
  971. procedure tcgarm.a_loadfpu_reg_ref(list: TAsmList; size: tcgsize; reg: tregister; const ref: treference);
  972. var
  973. oppostfix:toppostfix;
  974. begin
  975. case size of
  976. OS_F32:
  977. oppostfix:=PF_S;
  978. OS_F64:
  979. oppostfix:=PF_D;
  980. OS_F80:
  981. oppostfix:=PF_E;
  982. else
  983. InternalError(200309022);
  984. end;
  985. handle_load_store(list,A_STF,oppostfix,reg,ref);
  986. end;
  987. { comparison operations }
  988. procedure tcgarm.a_cmp_const_reg_label(list : TAsmList;size : tcgsize;cmp_op : topcmp;a : aint;reg : tregister;
  989. l : tasmlabel);
  990. var
  991. tmpreg : tregister;
  992. b : byte;
  993. begin
  994. if is_shifter_const(a,b) then
  995. list.concat(taicpu.op_reg_const(A_CMP,reg,a))
  996. { CMN reg,0 and CMN reg,$80000000 are different from CMP reg,$ffffffff
  997. and CMP reg,$7fffffff regarding the flags according to the ARM manual }
  998. else if (a<>$7fffffff) and (a<>-1) and is_shifter_const(-a,b) then
  999. list.concat(taicpu.op_reg_const(A_CMN,reg,-a))
  1000. else
  1001. begin
  1002. tmpreg:=getintregister(list,size);
  1003. a_load_const_reg(list,size,a,tmpreg);
  1004. list.concat(taicpu.op_reg_reg(A_CMP,reg,tmpreg));
  1005. end;
  1006. a_jmp_cond(list,cmp_op,l);
  1007. end;
  1008. procedure tcgarm.a_cmp_reg_reg_label(list : TAsmList;size : tcgsize;cmp_op : topcmp;reg1,reg2 : tregister;l : tasmlabel);
  1009. begin
  1010. list.concat(taicpu.op_reg_reg(A_CMP,reg2,reg1));
  1011. a_jmp_cond(list,cmp_op,l);
  1012. end;
  1013. procedure tcgarm.a_jmp_name(list : TAsmList;const s : string);
  1014. var
  1015. ai : taicpu;
  1016. begin
  1017. ai:=taicpu.op_sym(A_B,current_asmdata.RefAsmSymbol(s));
  1018. ai.is_jmp:=true;
  1019. list.concat(ai);
  1020. end;
  1021. procedure tcgarm.a_jmp_always(list : TAsmList;l: tasmlabel);
  1022. var
  1023. ai : taicpu;
  1024. begin
  1025. ai:=taicpu.op_sym(A_B,l);
  1026. ai.is_jmp:=true;
  1027. list.concat(ai);
  1028. end;
  1029. procedure tcgarm.a_jmp_flags(list : TAsmList;const f : TResFlags;l: tasmlabel);
  1030. var
  1031. ai : taicpu;
  1032. begin
  1033. ai:=setcondition(taicpu.op_sym(A_B,l),flags_to_cond(f));
  1034. ai.is_jmp:=true;
  1035. list.concat(ai);
  1036. end;
  1037. procedure tcgarm.g_flags2reg(list: TAsmList; size: TCgSize; const f: TResFlags; reg: TRegister);
  1038. var
  1039. ai : taicpu;
  1040. begin
  1041. list.concat(setcondition(taicpu.op_reg_const(A_MOV,reg,1),flags_to_cond(f)));
  1042. list.concat(setcondition(taicpu.op_reg_const(A_MOV,reg,0),inverse_cond(flags_to_cond(f))));
  1043. end;
  1044. procedure tcgarm.g_proc_entry(list : TAsmList;localsize : longint;nostackframe:boolean);
  1045. var
  1046. ref,href : treference;
  1047. shift : byte;
  1048. firstfloatreg,lastfloatreg,
  1049. r : byte;
  1050. i : aint;
  1051. again : tasmlabel;
  1052. regs : tcpuregisterset;
  1053. begin
  1054. LocalSize:=align(LocalSize,4);
  1055. if not(nostackframe) then
  1056. begin
  1057. firstfloatreg:=RS_NO;
  1058. { save floating point registers? }
  1059. for r:=RS_F0 to RS_F7 do
  1060. if r in rg[R_FPUREGISTER].used_in_proc-paramanager.get_volatile_registers_fpu(pocall_stdcall) then
  1061. begin
  1062. if firstfloatreg=RS_NO then
  1063. firstfloatreg:=r;
  1064. lastfloatreg:=r;
  1065. end;
  1066. a_reg_alloc(list,NR_STACK_POINTER_REG);
  1067. if current_procinfo.framepointer<>NR_STACK_POINTER_REG then
  1068. begin
  1069. a_reg_alloc(list,NR_FRAME_POINTER_REG);
  1070. a_reg_alloc(list,NR_R12);
  1071. list.concat(taicpu.op_reg_reg(A_MOV,NR_R12,NR_STACK_POINTER_REG));
  1072. end;
  1073. { save int registers }
  1074. reference_reset(ref);
  1075. ref.index:=NR_STACK_POINTER_REG;
  1076. ref.addressmode:=AM_PREINDEXED;
  1077. regs:=rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall);
  1078. if current_procinfo.framepointer<>NR_STACK_POINTER_REG then
  1079. regs:=regs+[RS_R11,RS_R12,RS_R14,RS_R15]
  1080. else
  1081. if (regs<>[]) or (pi_do_call in current_procinfo.flags) then
  1082. include(regs,RS_R14);
  1083. if regs<>[] then
  1084. list.concat(setoppostfix(taicpu.op_ref_regset(A_STM,ref,regs),PF_FD));
  1085. if current_procinfo.framepointer<>NR_STACK_POINTER_REG then
  1086. list.concat(taicpu.op_reg_reg_const(A_SUB,NR_FRAME_POINTER_REG,NR_R12,4));
  1087. { allocate necessary stack size
  1088. not necessary according to Yury Sidorov
  1089. { don't use a_op_const_reg_reg here because we don't allow register allocations
  1090. in the entry/exit code }
  1091. if (target_info.system in [system_arm_wince]) and
  1092. (localsize>=winstackpagesize) then
  1093. begin
  1094. if localsize div winstackpagesize<=5 then
  1095. begin
  1096. if is_shifter_const(localsize,shift) then
  1097. list.concat(Taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,localsize))
  1098. else
  1099. begin
  1100. a_load_const_reg(list,OS_ADDR,localsize,NR_R12);
  1101. list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));
  1102. end;
  1103. for i:=1 to localsize div winstackpagesize do
  1104. begin
  1105. if localsize-i*winstackpagesize<4096 then
  1106. reference_reset_base(href,NR_STACK_POINTER_REG,-(localsize-i*winstackpagesize))
  1107. else
  1108. begin
  1109. a_load_const_reg(list,OS_ADDR,-(localsize-i*winstackpagesize),NR_R12);
  1110. reference_reset_base(href,NR_STACK_POINTER_REG,0);
  1111. href.index:=NR_R12;
  1112. end;
  1113. { the data stored doesn't matter }
  1114. list.concat(Taicpu.op_reg_ref(A_STR,NR_R0,href));
  1115. end;
  1116. a_reg_dealloc(list,NR_R12);
  1117. reference_reset_base(href,NR_STACK_POINTER_REG,0);
  1118. { the data stored doesn't matter }
  1119. list.concat(Taicpu.op_reg_ref(A_STR,NR_R0,href));
  1120. end
  1121. else
  1122. begin
  1123. current_asmdata.getjumplabel(again);
  1124. list.concat(Taicpu.op_reg_const(A_MOV,NR_R12,localsize div winstackpagesize));
  1125. a_label(list,again);
  1126. { always shifterop }
  1127. list.concat(Taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,winstackpagesize));
  1128. reference_reset_base(href,NR_STACK_POINTER_REG,0);
  1129. { the data stored doesn't matter }
  1130. list.concat(Taicpu.op_reg_ref(A_STR,NR_R0,href));
  1131. list.concat(Taicpu.op_reg_reg_const(A_SUB,NR_R12,NR_R12,1));
  1132. a_jmp_cond(list,OC_NE,again);
  1133. if is_shifter_const(localsize mod winstackpagesize,shift) then
  1134. list.concat(Taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,localsize mod winstackpagesize))
  1135. else
  1136. begin
  1137. a_load_const_reg(list,OS_ADDR,localsize mod winstackpagesize,NR_R12);
  1138. list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));
  1139. end;
  1140. a_reg_dealloc(list,NR_R12);
  1141. reference_reset_base(href,NR_STACK_POINTER_REG,0);
  1142. { the data stored doesn't matter }
  1143. list.concat(Taicpu.op_reg_ref(A_STR,NR_R0,href));
  1144. end
  1145. end
  1146. else
  1147. }
  1148. if LocalSize<>0 then
  1149. if not(is_shifter_const(localsize,shift)) then
  1150. begin
  1151. if current_procinfo.framepointer=NR_STACK_POINTER_REG then
  1152. a_reg_alloc(list,NR_R12);
  1153. a_load_const_reg(list,OS_ADDR,LocalSize,NR_R12);
  1154. list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));
  1155. a_reg_dealloc(list,NR_R12);
  1156. end
  1157. else
  1158. begin
  1159. a_reg_dealloc(list,NR_R12);
  1160. list.concat(taicpu.op_reg_reg_const(A_SUB,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize));
  1161. end;
  1162. if firstfloatreg<>RS_NO then
  1163. begin
  1164. reference_reset(ref);
  1165. if tarmprocinfo(current_procinfo).floatregstart<=-1023 then
  1166. begin
  1167. a_load_const_reg(list,OS_ADDR,-tarmprocinfo(current_procinfo).floatregstart,NR_R12);
  1168. list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_R12,NR_FRAME_POINTER_REG,NR_R12));
  1169. ref.base:=NR_R12;
  1170. end
  1171. else
  1172. begin
  1173. ref.base:=NR_FRAME_POINTER_REG;
  1174. ref.offset:=tarmprocinfo(current_procinfo).floatregstart;
  1175. end;
  1176. list.concat(taicpu.op_reg_const_ref(A_SFM,newreg(R_FPUREGISTER,firstfloatreg,R_SUBWHOLE),
  1177. lastfloatreg-firstfloatreg+1,ref));
  1178. end;
  1179. end;
  1180. end;
  1181. procedure tcgarm.g_proc_exit(list : TAsmList;parasize : longint;nostackframe:boolean);
  1182. var
  1183. ref : treference;
  1184. firstfloatreg,lastfloatreg,
  1185. r : byte;
  1186. shift : byte;
  1187. regs : tcpuregisterset;
  1188. LocalSize : longint;
  1189. begin
  1190. if not(nostackframe) then
  1191. begin
  1192. { restore floating point register }
  1193. firstfloatreg:=RS_NO;
  1194. { save floating point registers? }
  1195. for r:=RS_F0 to RS_F7 do
  1196. if r in rg[R_FPUREGISTER].used_in_proc-paramanager.get_volatile_registers_fpu(pocall_stdcall) then
  1197. begin
  1198. if firstfloatreg=RS_NO then
  1199. firstfloatreg:=r;
  1200. lastfloatreg:=r;
  1201. end;
  1202. if firstfloatreg<>RS_NO then
  1203. begin
  1204. reference_reset(ref);
  1205. if tarmprocinfo(current_procinfo).floatregstart<=-1023 then
  1206. begin
  1207. a_load_const_reg(list,OS_ADDR,-tarmprocinfo(current_procinfo).floatregstart,NR_R12);
  1208. list.concat(taicpu.op_reg_reg_reg(A_SUB,NR_R12,NR_FRAME_POINTER_REG,NR_R12));
  1209. ref.base:=NR_R12;
  1210. end
  1211. else
  1212. begin
  1213. ref.base:=NR_FRAME_POINTER_REG;
  1214. ref.offset:=tarmprocinfo(current_procinfo).floatregstart;
  1215. end;
  1216. list.concat(taicpu.op_reg_const_ref(A_LFM,newreg(R_FPUREGISTER,firstfloatreg,R_SUBWHOLE),
  1217. lastfloatreg-firstfloatreg+1,ref));
  1218. end;
  1219. if (current_procinfo.framepointer=NR_STACK_POINTER_REG) then
  1220. begin
  1221. LocalSize:=current_procinfo.calc_stackframe_size;
  1222. if LocalSize<>0 then
  1223. if not(is_shifter_const(LocalSize,shift)) then
  1224. begin
  1225. a_reg_alloc(list,NR_R12);
  1226. a_load_const_reg(list,OS_ADDR,LocalSize,NR_R12);
  1227. list.concat(taicpu.op_reg_reg_reg(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,NR_R12));
  1228. a_reg_dealloc(list,NR_R12);
  1229. end
  1230. else
  1231. begin
  1232. list.concat(taicpu.op_reg_reg_const(A_ADD,NR_STACK_POINTER_REG,NR_STACK_POINTER_REG,LocalSize));
  1233. end;
  1234. regs:=rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall);
  1235. if (pi_do_call in current_procinfo.flags) or (regs<>[]) then
  1236. begin
  1237. exclude(regs,RS_R14);
  1238. include(regs,RS_R15);
  1239. end;
  1240. if regs=[] then
  1241. list.concat(taicpu.op_reg_reg(A_MOV,NR_R15,NR_R14))
  1242. else
  1243. begin
  1244. reference_reset(ref);
  1245. ref.index:=NR_STACK_POINTER_REG;
  1246. ref.addressmode:=AM_PREINDEXED;
  1247. list.concat(setoppostfix(taicpu.op_ref_regset(A_LDM,ref,regs),PF_FD));
  1248. end;
  1249. end
  1250. else
  1251. begin
  1252. { restore int registers and return }
  1253. reference_reset(ref);
  1254. ref.index:=NR_FRAME_POINTER_REG;
  1255. list.concat(setoppostfix(taicpu.op_ref_regset(A_LDM,ref,rg[R_INTREGISTER].used_in_proc-paramanager.get_volatile_registers_int(pocall_stdcall)+[RS_R11,RS_R13,RS_R15]),PF_EA));
  1256. end;
  1257. end
  1258. else
  1259. list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R14));
  1260. end;
  1261. procedure tcgarm.a_loadaddr_ref_reg(list : TAsmList;const ref : treference;r : tregister);
  1262. var
  1263. b : byte;
  1264. tmpref : treference;
  1265. instr : taicpu;
  1266. begin
  1267. if ref.addressmode<>AM_OFFSET then
  1268. internalerror(200309071);
  1269. tmpref:=ref;
  1270. { Be sure to have a base register }
  1271. if (tmpref.base=NR_NO) then
  1272. begin
  1273. if tmpref.shiftmode<>SM_None then
  1274. internalerror(200308294);
  1275. if tmpref.signindex<0 then
  1276. internalerror(200312023);
  1277. tmpref.base:=tmpref.index;
  1278. tmpref.index:=NR_NO;
  1279. end;
  1280. if assigned(tmpref.symbol) or
  1281. not((is_shifter_const(tmpref.offset,b)) or
  1282. (is_shifter_const(-tmpref.offset,b))
  1283. ) then
  1284. fixref(list,tmpref);
  1285. { expect a base here if there is an index }
  1286. if (tmpref.base=NR_NO) and (tmpref.index<>NR_NO) then
  1287. internalerror(200312022);
  1288. if tmpref.index<>NR_NO then
  1289. begin
  1290. if tmpref.shiftmode<>SM_None then
  1291. internalerror(200312021);
  1292. if tmpref.signindex<0 then
  1293. a_op_reg_reg_reg(list,OP_SUB,OS_ADDR,tmpref.base,tmpref.index,r)
  1294. else
  1295. a_op_reg_reg_reg(list,OP_ADD,OS_ADDR,tmpref.base,tmpref.index,r);
  1296. if tmpref.offset<>0 then
  1297. a_op_const_reg_reg(list,OP_ADD,OS_ADDR,tmpref.offset,r,r);
  1298. end
  1299. else
  1300. begin
  1301. if tmpref.offset<>0 then
  1302. begin
  1303. if tmpref.base<>NR_NO then
  1304. a_op_const_reg_reg(list,OP_ADD,OS_ADDR,tmpref.offset,tmpref.base,r)
  1305. else
  1306. a_load_const_reg(list,OS_ADDR,tmpref.offset,r);
  1307. end
  1308. else
  1309. begin
  1310. instr:=taicpu.op_reg_reg(A_MOV,r,tmpref.base);
  1311. list.concat(instr);
  1312. add_move_instruction(instr);
  1313. end;
  1314. end;
  1315. end;
  1316. procedure tcgarm.fixref(list : TAsmList;var ref : treference);
  1317. var
  1318. tmpreg : tregister;
  1319. tmpref : treference;
  1320. l : tasmlabel;
  1321. begin
  1322. { absolute symbols can't be handled directly, we've to store the symbol reference
  1323. in the text segment and access it pc relative
  1324. For now, we assume that references where base or index equals to PC are already
  1325. relative, all other references are assumed to be absolute and thus they need
  1326. to be handled extra.
  1327. A proper solution would be to change refoptions to a set and store the information
  1328. if the symbol is absolute or relative there.
  1329. }
  1330. { create consts entry }
  1331. reference_reset(tmpref);
  1332. current_asmdata.getjumplabel(l);
  1333. cg.a_label(current_procinfo.aktlocaldata,l);
  1334. tmpref.symboldata:=current_procinfo.aktlocaldata.last;
  1335. if assigned(ref.symbol) then
  1336. current_procinfo.aktlocaldata.concat(tai_const.create_sym_offset(ref.symbol,ref.offset))
  1337. else
  1338. current_procinfo.aktlocaldata.concat(tai_const.Create_32bit(ref.offset));
  1339. { load consts entry }
  1340. tmpreg:=getintregister(list,OS_INT);
  1341. tmpref.symbol:=l;
  1342. tmpref.base:=NR_PC;
  1343. list.concat(taicpu.op_reg_ref(A_LDR,tmpreg,tmpref));
  1344. if (ref.base<>NR_NO) then
  1345. begin
  1346. if ref.index<>NR_NO then
  1347. begin
  1348. list.concat(taicpu.op_reg_reg_reg(A_ADD,tmpreg,ref.base,tmpreg));
  1349. ref.base:=tmpreg;
  1350. end
  1351. else
  1352. if ref.base<>NR_PC then
  1353. begin
  1354. ref.index:=tmpreg;
  1355. ref.shiftimm:=0;
  1356. ref.signindex:=1;
  1357. ref.shiftmode:=SM_None;
  1358. end
  1359. else
  1360. ref.base:=tmpreg;
  1361. end
  1362. else
  1363. ref.base:=tmpreg;
  1364. ref.offset:=0;
  1365. ref.symbol:=nil;
  1366. end;
  1367. procedure tcgarm.g_concatcopy_move(list : TAsmList;const source,dest : treference;len : aint);
  1368. var
  1369. paraloc1,paraloc2,paraloc3 : TCGPara;
  1370. begin
  1371. paraloc1.init;
  1372. paraloc2.init;
  1373. paraloc3.init;
  1374. paramanager.getintparaloc(pocall_default,1,paraloc1);
  1375. paramanager.getintparaloc(pocall_default,2,paraloc2);
  1376. paramanager.getintparaloc(pocall_default,3,paraloc3);
  1377. paramanager.allocparaloc(list,paraloc3);
  1378. a_param_const(list,OS_INT,len,paraloc3);
  1379. paramanager.allocparaloc(list,paraloc2);
  1380. a_paramaddr_ref(list,dest,paraloc2);
  1381. paramanager.allocparaloc(list,paraloc2);
  1382. a_paramaddr_ref(list,source,paraloc1);
  1383. paramanager.freeparaloc(list,paraloc3);
  1384. paramanager.freeparaloc(list,paraloc2);
  1385. paramanager.freeparaloc(list,paraloc1);
  1386. alloccpuregisters(list,R_INTREGISTER,paramanager.get_volatile_registers_int(pocall_default));
  1387. alloccpuregisters(list,R_FPUREGISTER,paramanager.get_volatile_registers_fpu(pocall_default));
  1388. a_call_name(list,'FPC_MOVE');
  1389. dealloccpuregisters(list,R_FPUREGISTER,paramanager.get_volatile_registers_fpu(pocall_default));
  1390. dealloccpuregisters(list,R_INTREGISTER,paramanager.get_volatile_registers_int(pocall_default));
  1391. paraloc3.done;
  1392. paraloc2.done;
  1393. paraloc1.done;
  1394. end;
  1395. procedure tcgarm.g_concatcopy_internal(list : TAsmList;const source,dest : treference;len : aint;aligned : boolean);
  1396. const
  1397. maxtmpreg=10;{roozbeh: can be reduced to 8 or lower if might conflick with reserved ones,also +2 is used becouse of regs required for referencing}
  1398. var
  1399. srcref,dstref,usedtmpref,usedtmpref2:treference;
  1400. srcreg,destreg,countreg,r,tmpreg,tmpreg2:tregister;
  1401. helpsize:aword;
  1402. copysize:byte;
  1403. cgsize:Tcgsize;
  1404. so:tshifterop;
  1405. tmpregisters:array[1..maxtmpreg]of tregister;
  1406. tmpregi,tmpregi2:byte;
  1407. { will never be called with count<=4 }
  1408. procedure genloop(count : aword;size : byte);
  1409. const
  1410. size2opsize : array[1..4] of tcgsize = (OS_8,OS_16,OS_NO,OS_32);
  1411. var
  1412. l : tasmlabel;
  1413. begin
  1414. current_asmdata.getjumplabel(l);
  1415. if count<size then size:=1;
  1416. a_load_const_reg(list,OS_INT,count div size,countreg);
  1417. cg.a_label(list,l);
  1418. srcref.addressmode:=AM_POSTINDEXED;
  1419. dstref.addressmode:=AM_POSTINDEXED;
  1420. srcref.offset:=size;
  1421. dstref.offset:=size;
  1422. r:=getintregister(list,size2opsize[size]);
  1423. a_load_ref_reg(list,size2opsize[size],size2opsize[size],srcref,r);
  1424. list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,countreg,countreg,1),PF_S));
  1425. a_load_reg_ref(list,size2opsize[size],size2opsize[size],r,dstref);
  1426. a_jmp_flags(list,F_NE,l);
  1427. srcref.offset:=1;
  1428. dstref.offset:=1;
  1429. case count mod size of
  1430. 1:
  1431. begin
  1432. a_load_ref_reg(list,OS_8,OS_8,srcref,r);
  1433. a_load_reg_ref(list,OS_8,OS_8,r,dstref);
  1434. end;
  1435. 2:
  1436. if aligned then
  1437. begin
  1438. a_load_ref_reg(list,OS_16,OS_16,srcref,r);
  1439. a_load_reg_ref(list,OS_16,OS_16,r,dstref);
  1440. end
  1441. else
  1442. begin
  1443. a_load_ref_reg(list,OS_8,OS_8,srcref,r);
  1444. a_load_reg_ref(list,OS_8,OS_8,r,dstref);
  1445. a_load_ref_reg(list,OS_8,OS_8,srcref,r);
  1446. a_load_reg_ref(list,OS_8,OS_8,r,dstref);
  1447. end;
  1448. 3:
  1449. if aligned then
  1450. begin
  1451. srcref.offset:=2;
  1452. dstref.offset:=2;
  1453. a_load_ref_reg(list,OS_16,OS_16,srcref,r);
  1454. a_load_reg_ref(list,OS_16,OS_16,r,dstref);
  1455. a_load_ref_reg(list,OS_8,OS_8,srcref,r);
  1456. a_load_reg_ref(list,OS_8,OS_8,r,dstref);
  1457. end
  1458. else
  1459. begin
  1460. a_load_ref_reg(list,OS_8,OS_8,srcref,r);
  1461. a_load_reg_ref(list,OS_8,OS_8,r,dstref);
  1462. a_load_ref_reg(list,OS_8,OS_8,srcref,r);
  1463. a_load_reg_ref(list,OS_8,OS_8,r,dstref);
  1464. a_load_ref_reg(list,OS_8,OS_8,srcref,r);
  1465. a_load_reg_ref(list,OS_8,OS_8,r,dstref);
  1466. end;
  1467. end;
  1468. { keep the registers alive }
  1469. list.concat(taicpu.op_reg_reg(A_MOV,countreg,countreg));
  1470. list.concat(taicpu.op_reg_reg(A_MOV,srcreg,srcreg));
  1471. list.concat(taicpu.op_reg_reg(A_MOV,destreg,destreg));
  1472. end;
  1473. begin
  1474. if len=0 then
  1475. exit;
  1476. helpsize:=12+maxtmpreg*4;//52 with maxtmpreg=10
  1477. dstref:=dest;
  1478. srcref:=source;
  1479. if cs_opt_size in current_settings.optimizerswitches then
  1480. helpsize:=8;
  1481. if (len<=helpsize) and aligned then
  1482. begin
  1483. tmpregi:=0;
  1484. srcreg:=getintregister(list,OS_ADDR);
  1485. { explicit pc relative addressing, could be
  1486. e.g. a floating point constant }
  1487. if source.base=NR_PC then
  1488. begin
  1489. { ... then we don't need a loadaddr }
  1490. srcref:=source;
  1491. end
  1492. else
  1493. begin
  1494. a_loadaddr_ref_reg(list,source,srcreg);
  1495. reference_reset_base(srcref,srcreg,0);
  1496. end;
  1497. while (len div 4 <> 0) and (tmpregi<=maxtmpreg) do
  1498. begin
  1499. inc(tmpregi);
  1500. tmpregisters[tmpregi]:=getintregister(list,OS_32);
  1501. a_load_ref_reg(list,OS_32,OS_32,srcref,tmpregisters[tmpregi]);
  1502. inc(srcref.offset,4);
  1503. dec(len,4);
  1504. end;
  1505. destreg:=getintregister(list,OS_ADDR);
  1506. a_loadaddr_ref_reg(list,dest,destreg);
  1507. reference_reset_base(dstref,destreg,0);
  1508. tmpregi2:=1;
  1509. while (tmpregi2<=tmpregi) do
  1510. begin
  1511. a_load_reg_ref(list,OS_32,OS_32,tmpregisters[tmpregi2],dstref);
  1512. inc(dstref.offset,4);
  1513. inc(tmpregi2);
  1514. end;
  1515. copysize:=4;
  1516. cgsize:=OS_32;
  1517. while len<>0 do
  1518. begin
  1519. if len<2 then
  1520. begin
  1521. copysize:=1;
  1522. cgsize:=OS_8;
  1523. end
  1524. else if len<4 then
  1525. begin
  1526. copysize:=2;
  1527. cgsize:=OS_16;
  1528. end;
  1529. dec(len,copysize);
  1530. r:=getintregister(list,cgsize);
  1531. a_load_ref_reg(list,cgsize,cgsize,srcref,r);
  1532. a_load_reg_ref(list,cgsize,cgsize,r,dstref);
  1533. inc(srcref.offset,copysize);
  1534. inc(dstref.offset,copysize);
  1535. end;{end of while}
  1536. end
  1537. else
  1538. begin
  1539. cgsize:=OS_32;
  1540. if (len<=4) then{len<=4 and not aligned}
  1541. begin
  1542. r:=getintregister(list,cgsize);
  1543. usedtmpref:=a_internal_load_ref_reg(list,OS_8,OS_8,srcref,r);
  1544. if Len=1 then
  1545. a_load_reg_ref(list,OS_8,OS_8,r,dstref)
  1546. else
  1547. begin
  1548. tmpreg:=getintregister(list,cgsize);
  1549. usedtmpref2:=a_internal_load_reg_ref(list,OS_8,OS_8,r,dstref);
  1550. inc(usedtmpref.offset,1);
  1551. a_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);
  1552. inc(usedtmpref2.offset,1);
  1553. a_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref2);
  1554. if len>2 then
  1555. begin
  1556. inc(usedtmpref.offset,1);
  1557. a_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);
  1558. inc(usedtmpref2.offset,1);
  1559. a_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref2);
  1560. if len>3 then
  1561. begin
  1562. inc(usedtmpref.offset,1);
  1563. a_load_ref_reg(list,OS_8,OS_8,usedtmpref,tmpreg);
  1564. inc(usedtmpref2.offset,1);
  1565. a_load_reg_ref(list,OS_8,OS_8,tmpreg,usedtmpref2);
  1566. end;
  1567. end;
  1568. end;
  1569. end{end of if len<=4}
  1570. else
  1571. begin{unaligned & 4<len<helpsize **or** aligned/unaligned & len>helpsize}
  1572. destreg:=getintregister(list,OS_ADDR);
  1573. a_loadaddr_ref_reg(list,dest,destreg);
  1574. reference_reset_base(dstref,destreg,0);
  1575. srcreg:=getintregister(list,OS_ADDR);
  1576. a_loadaddr_ref_reg(list,source,srcreg);
  1577. reference_reset_base(srcref,srcreg,0);
  1578. countreg:=getintregister(list,OS_32);
  1579. // if cs_opt_size in current_settings.optimizerswitches then
  1580. { roozbeh : it seems loading 1 byte is faster becouse of caching/fetching(?) }
  1581. {if aligned then
  1582. genloop(len,4)
  1583. else}
  1584. genloop(len,1);
  1585. end;
  1586. end;
  1587. end;
  1588. procedure tcgarm.g_concatcopy_unaligned(list : TAsmList;const source,dest : treference;len : aint);
  1589. begin
  1590. g_concatcopy_internal(list,source,dest,len,false);
  1591. end;
  1592. procedure tcgarm.g_concatcopy(list : TAsmList;const source,dest : treference;len : aint);
  1593. begin
  1594. if (source.alignment in [1..3]) or
  1595. (dest.alignment in [1..3]) then
  1596. g_concatcopy_internal(list,source,dest,len,false)
  1597. else
  1598. g_concatcopy_internal(list,source,dest,len,true);
  1599. end;
  1600. procedure tcgarm.g_overflowCheck(list : TAsmList;const l : tlocation;def : tdef);
  1601. var
  1602. ovloc : tlocation;
  1603. begin
  1604. ovloc.loc:=LOC_VOID;
  1605. g_overflowCheck_loc(list,l,def,ovloc);
  1606. end;
  1607. procedure tcgarm.g_overflowCheck_loc(List:TAsmList;const Loc:TLocation;def:TDef;ovloc : tlocation);
  1608. var
  1609. hl : tasmlabel;
  1610. ai:TAiCpu;
  1611. hflags : tresflags;
  1612. begin
  1613. if not(cs_check_overflow in current_settings.localswitches) then
  1614. exit;
  1615. current_asmdata.getjumplabel(hl);
  1616. case ovloc.loc of
  1617. LOC_VOID:
  1618. begin
  1619. ai:=taicpu.op_sym(A_B,hl);
  1620. ai.is_jmp:=true;
  1621. if not((def.typ=pointerdef) or
  1622. ((def.typ=orddef) and
  1623. (torddef(def).ordtype in [u64bit,u16bit,u32bit,u8bit,uchar,bool8bit,bool16bit,bool32bit]))) then
  1624. ai.SetCondition(C_VC)
  1625. else
  1626. if TAiCpu(List.Last).opcode in [A_RSB,A_RSC,A_SBC,A_SUB] then
  1627. ai.SetCondition(C_CS)
  1628. else
  1629. ai.SetCondition(C_CC);
  1630. list.concat(ai);
  1631. end;
  1632. LOC_FLAGS:
  1633. begin
  1634. hflags:=ovloc.resflags;
  1635. inverse_flags(hflags);
  1636. cg.a_jmp_flags(list,hflags,hl);
  1637. end;
  1638. else
  1639. internalerror(200409281);
  1640. end;
  1641. a_call_name(list,'FPC_OVERFLOW');
  1642. a_label(list,hl);
  1643. end;
  1644. procedure tcgarm.g_save_standard_registers(list : TAsmList);
  1645. begin
  1646. { this work is done in g_proc_entry }
  1647. end;
  1648. procedure tcgarm.g_restore_standard_registers(list : TAsmList);
  1649. begin
  1650. { this work is done in g_proc_exit }
  1651. end;
  1652. procedure tcgarm.a_jmp_cond(list : TAsmList;cond : TOpCmp;l: tasmlabel);
  1653. var
  1654. ai : taicpu;
  1655. begin
  1656. ai:=Taicpu.Op_sym(A_B,l);
  1657. ai.SetCondition(OpCmp2AsmCond[cond]);
  1658. ai.is_jmp:=true;
  1659. list.concat(ai);
  1660. end;
  1661. procedure tcgarm.g_intf_wrapper(list: TAsmList; procdef: tprocdef; const labelname: string; ioffset: longint);
  1662. procedure loadvmttor12;
  1663. var
  1664. href : treference;
  1665. begin
  1666. reference_reset_base(href,NR_R0,0);
  1667. cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R12);
  1668. end;
  1669. procedure op_onr12methodaddr;
  1670. var
  1671. href : treference;
  1672. begin
  1673. if (procdef.extnumber=$ffff) then
  1674. Internalerror(200006139);
  1675. { call/jmp vmtoffs(%eax) ; method offs }
  1676. reference_reset_base(href,NR_R12,procdef._class.vmtmethodoffset(procdef.extnumber));
  1677. cg.a_load_ref_reg(list,OS_ADDR,OS_ADDR,href,NR_R12);
  1678. list.concat(taicpu.op_reg_reg(A_MOV,NR_PC,NR_R12));
  1679. end;
  1680. var
  1681. lab : tasmsymbol;
  1682. make_global : boolean;
  1683. href : treference;
  1684. begin
  1685. if not(procdef.proctypeoption in [potype_function,potype_procedure]) then
  1686. Internalerror(200006137);
  1687. if not assigned(procdef._class) or
  1688. (procdef.procoptions*[po_classmethod, po_staticmethod,
  1689. po_methodpointer, po_interrupt, po_iocheck]<>[]) then
  1690. Internalerror(200006138);
  1691. if procdef.owner.symtabletype<>ObjectSymtable then
  1692. Internalerror(200109191);
  1693. make_global:=false;
  1694. if (not current_module.is_unit) or
  1695. (cs_create_smart in current_settings.moduleswitches) or
  1696. (procdef.owner.defowner.owner.symtabletype=globalsymtable) then
  1697. make_global:=true;
  1698. if make_global then
  1699. list.concat(Tai_symbol.Createname_global(labelname,AT_FUNCTION,0))
  1700. else
  1701. list.concat(Tai_symbol.Createname(labelname,AT_FUNCTION,0));
  1702. { set param1 interface to self }
  1703. g_adjust_self_value(list,procdef,ioffset);
  1704. { case 4 }
  1705. if po_virtualmethod in procdef.procoptions then
  1706. begin
  1707. loadvmttor12;
  1708. op_onr12methodaddr;
  1709. end
  1710. { case 0 }
  1711. else
  1712. list.concat(taicpu.op_sym(A_B,current_asmdata.RefAsmSymbol(procdef.mangledname)));
  1713. list.concat(Tai_symbol_end.Createname(labelname));
  1714. end;
  1715. procedure tcg64farm.a_op64_reg_reg(list : TAsmList;op:TOpCG;size : tcgsize;regsrc,regdst : tregister64);
  1716. var
  1717. tmpreg : tregister;
  1718. begin
  1719. case op of
  1720. OP_NEG:
  1721. begin
  1722. list.concat(setoppostfix(taicpu.op_reg_reg_const(A_RSB,regdst.reglo,regsrc.reglo,0),PF_S));
  1723. list.concat(taicpu.op_reg_reg_const(A_RSC,regdst.reghi,regsrc.reghi,0));
  1724. end;
  1725. OP_NOT:
  1726. begin
  1727. cg.a_op_reg_reg(list,OP_NOT,OS_INT,regsrc.reglo,regdst.reglo);
  1728. cg.a_op_reg_reg(list,OP_NOT,OS_INT,regsrc.reghi,regdst.reghi);
  1729. end;
  1730. else
  1731. a_op64_reg_reg_reg(list,op,size,regsrc,regdst,regdst);
  1732. end;
  1733. end;
  1734. procedure tcg64farm.a_op64_const_reg(list : TAsmList;op:TOpCG;size : tcgsize;value : int64;reg : tregister64);
  1735. begin
  1736. a_op64_const_reg_reg(list,op,size,value,reg,reg);
  1737. end;
  1738. procedure tcg64farm.a_op64_const_reg_reg(list: TAsmList;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64);
  1739. var
  1740. ovloc : tlocation;
  1741. begin
  1742. a_op64_const_reg_reg_checkoverflow(list,op,size,value,regsrc,regdst,false,ovloc);
  1743. end;
  1744. procedure tcg64farm.a_op64_reg_reg_reg(list: TAsmList;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64);
  1745. var
  1746. ovloc : tlocation;
  1747. begin
  1748. a_op64_reg_reg_reg_checkoverflow(list,op,size,regsrc1,regsrc2,regdst,false,ovloc);
  1749. end;
  1750. procedure tcg64farm.a_op64_const_reg_reg_checkoverflow(list: TAsmList;op:TOpCG;size : tcgsize;value : int64;regsrc,regdst : tregister64;setflags : boolean;var ovloc : tlocation);
  1751. var
  1752. tmpreg : tregister;
  1753. b : byte;
  1754. begin
  1755. ovloc.loc:=LOC_VOID;
  1756. case op of
  1757. OP_NEG,
  1758. OP_NOT :
  1759. internalerror(200306017);
  1760. end;
  1761. if (setflags or tcgarm(cg).cgsetflags) and (op in [OP_ADD,OP_SUB]) then
  1762. begin
  1763. case op of
  1764. OP_ADD:
  1765. begin
  1766. if is_shifter_const(lo(value),b) then
  1767. list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADD,regdst.reglo,regsrc.reglo,lo(value)),PF_S))
  1768. else
  1769. begin
  1770. tmpreg:=cg.getintregister(list,OS_32);
  1771. cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);
  1772. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc.reglo,tmpreg),PF_S));
  1773. end;
  1774. if is_shifter_const(hi(value),b) then
  1775. list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADC,regdst.reghi,regsrc.reghi,hi(value)),PF_S))
  1776. else
  1777. begin
  1778. tmpreg:=cg.getintregister(list,OS_32);
  1779. cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);
  1780. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc.reghi,tmpreg),PF_S));
  1781. end;
  1782. end;
  1783. OP_SUB:
  1784. begin
  1785. if is_shifter_const(lo(value),b) then
  1786. list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,regdst.reglo,regsrc.reglo,lo(value)),PF_S))
  1787. else
  1788. begin
  1789. tmpreg:=cg.getintregister(list,OS_32);
  1790. cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);
  1791. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc.reglo,tmpreg),PF_S));
  1792. end;
  1793. if is_shifter_const(hi(value),b) then
  1794. list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SBC,regdst.reghi,regsrc.reghi,hi(value)),PF_S))
  1795. else
  1796. begin
  1797. tmpreg:=cg.getintregister(list,OS_32);
  1798. cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);
  1799. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc.reghi,tmpreg),PF_S));
  1800. end;
  1801. end;
  1802. else
  1803. internalerror(200502131);
  1804. end;
  1805. if size=OS_64 then
  1806. begin
  1807. { the arm has an weired opinion how flags for SUB/ADD are handled }
  1808. ovloc.loc:=LOC_FLAGS;
  1809. case op of
  1810. OP_ADD:
  1811. ovloc.resflags:=F_CS;
  1812. OP_SUB:
  1813. ovloc.resflags:=F_CC;
  1814. end;
  1815. end;
  1816. end
  1817. else
  1818. begin
  1819. case op of
  1820. OP_AND,OP_OR,OP_XOR:
  1821. begin
  1822. cg.a_op_const_reg_reg(list,op,OS_32,lo(value),regsrc.reglo,regdst.reglo);
  1823. cg.a_op_const_reg_reg(list,op,OS_32,hi(value),regsrc.reghi,regdst.reghi);
  1824. end;
  1825. OP_ADD:
  1826. begin
  1827. if is_shifter_const(lo(value),b) then
  1828. list.concat(setoppostfix(taicpu.op_reg_reg_const(A_ADD,regdst.reglo,regsrc.reglo,lo(value)),PF_S))
  1829. else
  1830. begin
  1831. tmpreg:=cg.getintregister(list,OS_32);
  1832. cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);
  1833. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc.reglo,tmpreg),PF_S));
  1834. end;
  1835. if is_shifter_const(hi(value),b) then
  1836. list.concat(taicpu.op_reg_reg_const(A_ADC,regdst.reghi,regsrc.reghi,hi(value)))
  1837. else
  1838. begin
  1839. tmpreg:=cg.getintregister(list,OS_32);
  1840. cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);
  1841. list.concat(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc.reghi,tmpreg));
  1842. end;
  1843. end;
  1844. OP_SUB:
  1845. begin
  1846. if is_shifter_const(lo(value),b) then
  1847. list.concat(setoppostfix(taicpu.op_reg_reg_const(A_SUB,regdst.reglo,regsrc.reglo,lo(value)),PF_S))
  1848. else
  1849. begin
  1850. tmpreg:=cg.getintregister(list,OS_32);
  1851. cg.a_load_const_reg(list,OS_32,lo(value),tmpreg);
  1852. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc.reglo,tmpreg),PF_S));
  1853. end;
  1854. if is_shifter_const(hi(value),b) then
  1855. list.concat(taicpu.op_reg_reg_const(A_SBC,regdst.reghi,regsrc.reghi,hi(value)))
  1856. else
  1857. begin
  1858. tmpreg:=cg.getintregister(list,OS_32);
  1859. cg.a_load_const_reg(list,OS_32,hi(value),tmpreg);
  1860. list.concat(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc.reghi,tmpreg));
  1861. end;
  1862. end;
  1863. else
  1864. internalerror(2003083101);
  1865. end;
  1866. end;
  1867. end;
  1868. procedure tcg64farm.a_op64_reg_reg_reg_checkoverflow(list: TAsmList;op:TOpCG;size : tcgsize;regsrc1,regsrc2,regdst : tregister64;setflags : boolean;var ovloc : tlocation);
  1869. var
  1870. op1,op2:TAsmOp;
  1871. begin
  1872. ovloc.loc:=LOC_VOID;
  1873. case op of
  1874. OP_NEG,
  1875. OP_NOT :
  1876. internalerror(200306017);
  1877. end;
  1878. if (setflags or tcgarm(cg).cgsetflags) and (op in [OP_ADD,OP_SUB]) then
  1879. begin
  1880. case op of
  1881. OP_ADD:
  1882. begin
  1883. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc1.reglo,regsrc2.reglo),PF_S));
  1884. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc1.reghi,regsrc2.reghi),PF_S));
  1885. end;
  1886. OP_SUB:
  1887. begin
  1888. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc2.reglo,regsrc1.reglo),PF_S));
  1889. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc2.reghi,regsrc1.reghi),PF_S));
  1890. end;
  1891. else
  1892. internalerror(2003083101);
  1893. end;
  1894. if size=OS_64 then
  1895. begin
  1896. { the arm has an weired opinion how flags for SUB/ADD are handled }
  1897. ovloc.loc:=LOC_FLAGS;
  1898. case op of
  1899. OP_ADD:
  1900. ovloc.resflags:=F_CS;
  1901. OP_SUB:
  1902. ovloc.resflags:=F_CC;
  1903. end;
  1904. end;
  1905. end
  1906. else
  1907. begin
  1908. case op of
  1909. OP_AND,OP_OR,OP_XOR:
  1910. begin
  1911. cg.a_op_reg_reg_reg(list,op,OS_32,regsrc1.reglo,regsrc2.reglo,regdst.reglo);
  1912. cg.a_op_reg_reg_reg(list,op,OS_32,regsrc1.reghi,regsrc2.reghi,regdst.reghi);
  1913. end;
  1914. OP_ADD:
  1915. begin
  1916. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_ADD,regdst.reglo,regsrc1.reglo,regsrc2.reglo),PF_S));
  1917. list.concat(taicpu.op_reg_reg_reg(A_ADC,regdst.reghi,regsrc1.reghi,regsrc2.reghi));
  1918. end;
  1919. OP_SUB:
  1920. begin
  1921. list.concat(setoppostfix(taicpu.op_reg_reg_reg(A_SUB,regdst.reglo,regsrc2.reglo,regsrc1.reglo),PF_S));
  1922. list.concat(taicpu.op_reg_reg_reg(A_SBC,regdst.reghi,regsrc2.reghi,regsrc1.reghi));
  1923. end;
  1924. else
  1925. internalerror(2003083101);
  1926. end;
  1927. end;
  1928. end;
  1929. begin
  1930. cg:=tcgarm.create;
  1931. cg64:=tcg64farm.create;
  1932. end.