regvars.pas 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549
  1. {
  2. $Id$
  3. Copyright (c) 1998-2000 by Florian Klaempfl and Jonas Maebe
  4. This unit handles register variable allocation
  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 regvars;
  19. {$i defines.inc}
  20. interface
  21. uses
  22. aasm,
  23. node,
  24. symsym,
  25. cpubase, tgcpu;
  26. procedure assign_regvars(p: tnode);
  27. procedure load_regvars(asml: TAAsmoutput; p: tnode);
  28. procedure cleanup_regvars(asml: TAAsmoutput);
  29. procedure store_regvar(asml: TAAsmoutput; reg: tregister);
  30. procedure load_regvar(asml: TAAsmoutput; vsym: tvarsym);
  31. procedure load_regvar_reg(asml: TAAsmoutput; reg: tregister);
  32. procedure load_all_regvars(asml: TAAsmoutput);
  33. procedure sync_regvars(list1, list2: taasmoutput; const regvarsloaded1,
  34. regvarsloaded2: regvar_booleanarray);
  35. implementation
  36. uses
  37. globtype,systems,comphook,
  38. cutils,cclasses,verbose,globals,
  39. symconst,symbase,symtype,symdef,types,
  40. cgbase,cpuasm,cgobj,cgcpu,cga;
  41. var
  42. parasym : boolean;
  43. procedure searchregvars(p : tnamedindexitem);
  44. var
  45. i,j,k : longint;
  46. begin
  47. if (tsym(p).typ=varsym) and (vo_regable in tvarsym(p).varoptions) then
  48. begin
  49. j:=tvarsym(p).refs;
  50. { parameter get a less value }
  51. if parasym then
  52. begin
  53. if cs_littlesize in aktglobalswitches then
  54. dec(j,1)
  55. else
  56. dec(j,100);
  57. end;
  58. { walk through all momentary register variables }
  59. for i:=1 to maxvarregs do
  60. begin
  61. with pregvarinfo(aktprocdef.regvarinfo)^ do
  62. if ((regvars[i]=nil) or (j>regvars_refs[i])) and (j>0) then
  63. begin
  64. for k:=maxvarregs-1 downto i do
  65. begin
  66. regvars[k+1]:=regvars[k];
  67. regvars_para[k+1]:=regvars_para[k];
  68. regvars_refs[k+1]:=regvars_refs[k];
  69. end;
  70. { calc the new refs
  71. tvarsym(p).refs:=j; }
  72. regvars[i]:=tvarsym(p);
  73. regvars_para[i]:=parasym;
  74. regvars_refs[i]:=j;
  75. break;
  76. end;
  77. end;
  78. end;
  79. end;
  80. procedure searchfpuregvars(p : tnamedindexitem);
  81. var
  82. i,j,k : longint;
  83. begin
  84. if (tsym(p).typ=varsym) and (vo_fpuregable in tvarsym(p).varoptions) then
  85. begin
  86. j:=tvarsym(p).refs;
  87. { parameter get a less value }
  88. if parasym then
  89. begin
  90. if cs_littlesize in aktglobalswitches then
  91. dec(j,1)
  92. else
  93. dec(j,100);
  94. end;
  95. { walk through all momentary register variables }
  96. for i:=1 to maxfpuvarregs do
  97. begin
  98. with pregvarinfo(aktprocdef.regvarinfo)^ do
  99. if ((fpuregvars[i]=nil) or (j>fpuregvars_refs[i])) and (j>0) then
  100. begin
  101. for k:=maxfpuvarregs-1 downto i do
  102. begin
  103. fpuregvars[k+1]:=fpuregvars[k];
  104. fpuregvars_para[k+1]:=fpuregvars_para[k];
  105. fpuregvars_refs[k+1]:=fpuregvars_refs[k];
  106. end;
  107. { calc the new refs
  108. tvarsym(p).refs:=j; }
  109. fpuregvars[i]:=tvarsym(p);
  110. fpuregvars_para[i]:=parasym;
  111. fpuregvars_refs[i]:=j;
  112. break;
  113. end;
  114. end;
  115. end;
  116. end;
  117. procedure assign_regvars(p: tnode);
  118. { register variables }
  119. var
  120. regvarinfo: pregvarinfo;
  121. i: longint;
  122. begin
  123. { max. optimizations }
  124. { only if no asm is used }
  125. { and no try statement }
  126. if (cs_regalloc in aktglobalswitches) and
  127. ((procinfo^.flags and (pi_uses_asm or pi_uses_exceptions))=0) then
  128. begin
  129. new(regvarinfo);
  130. fillchar(regvarinfo^,sizeof(regvarinfo^),0);
  131. aktprocdef.regvarinfo := regvarinfo;
  132. if (p.registers32<4) then
  133. begin
  134. parasym:=false;
  135. symtablestack.foreach_static({$ifdef FPCPROCVAR}@{$endif}searchregvars);
  136. { copy parameter into a register ? }
  137. parasym:=true;
  138. symtablestack.next.foreach_static({$ifdef FPCPROCVAR}@{$endif}searchregvars);
  139. { hold needed registers free }
  140. for i:=maxvarregs downto maxvarregs-p.registers32+1 do
  141. begin
  142. regvarinfo^.regvars[i]:=nil;
  143. regvarinfo^.regvars_para[i] := false;
  144. end;
  145. { now assign register }
  146. for i:=1 to maxvarregs-p.registers32 do
  147. begin
  148. if assigned(regvarinfo^.regvars[i]) and
  149. (reg_pushes[varregs[i]] < regvarinfo^.regvars[i].refs) then
  150. begin
  151. { register is no longer available for }
  152. { expressions }
  153. { search the register which is the most }
  154. { unused }
  155. usableregs:=usableregs-[varregs[i]];
  156. is_reg_var[varregs[i]]:=true;
  157. dec(c_usableregs);
  158. { possibly no 32 bit register are needed }
  159. { call by reference/const ? }
  160. if (regvarinfo^.regvars[i].varspez in [vs_var,vs_out]) or
  161. ((regvarinfo^.regvars[i].varspez=vs_const) and
  162. push_addr_param(regvarinfo^.regvars[i].vartype.def)) then
  163. begin
  164. regvarinfo^.regvars[i].reg:=varregs[i];
  165. end
  166. else
  167. {$ifdef i386}
  168. if (regvarinfo^.regvars[i].vartype.def.deftype in [orddef,enumdef]) and
  169. (torddef(regvarinfo^.regvars[i].vartype.def).size=1) then
  170. begin
  171. regvarinfo^.regvars[i].reg:=reg32toreg8(varregs[i]);
  172. end
  173. else
  174. if (regvarinfo^.regvars[i].vartype.def.deftype in [orddef,enumdef]) and
  175. (torddef(regvarinfo^.regvars[i].vartype.def).size=2) then
  176. begin
  177. regvarinfo^.regvars[i].reg:=reg32toreg16(varregs[i]);
  178. end
  179. else
  180. {$endif i386}
  181. begin
  182. regvarinfo^.regvars[i].reg:=varregs[i];
  183. end;
  184. if regvarinfo^.regvars_para[i] then
  185. unused:=unused - [regvarinfo^.regvars[i].reg];
  186. { procedure uses this register }
  187. {$ifdef i386}
  188. usedinproc:=usedinproc or ($80 shr byte(varregs[i]));
  189. {$else i386}
  190. usedinproc:=usedinproc + [varregs[i]];
  191. {$endif i386}
  192. end
  193. else
  194. begin
  195. regvarinfo^.regvars[i] := nil;
  196. regvarinfo^.regvars_para[i] := false;
  197. end;
  198. end;
  199. end;
  200. if ((p.registersfpu+1)<maxfpuvarregs) then
  201. begin
  202. parasym:=false;
  203. symtablestack.foreach_static({$ifdef FPCPROCVAR}@{$endif}searchfpuregvars);
  204. {$ifdef dummy}
  205. { copy parameter into a register ? }
  206. parasym:=true;
  207. symtablestack.next.foreach_static({$ifdef FPCPROCVAR}@{$endif}searchregvars);
  208. {$endif dummy}
  209. { hold needed registers free }
  210. { in non leaf procedures we must be very careful }
  211. { with assigning registers }
  212. if aktmaxfpuregisters=-1 then
  213. begin
  214. if (procinfo^.flags and pi_do_call)<>0 then
  215. begin
  216. for i:=maxfpuvarregs downto 2 do
  217. regvarinfo^.fpuregvars[i]:=nil;
  218. end
  219. else
  220. begin
  221. for i:=maxfpuvarregs downto maxfpuvarregs-p.registersfpu do
  222. regvarinfo^.fpuregvars[i]:=nil;
  223. end;
  224. end
  225. else
  226. begin
  227. for i:=aktmaxfpuregisters+1 to maxfpuvarregs do
  228. regvarinfo^.fpuregvars[i]:=nil;
  229. end;
  230. { now assign register }
  231. for i:=1 to maxfpuvarregs do
  232. begin
  233. if assigned(regvarinfo^.fpuregvars[i]) then
  234. begin
  235. {$ifdef i386}
  236. { reserve place on the FPU stack }
  237. regvarinfo^.fpuregvars[i].reg:=correct_fpuregister(R_ST0,i-1);
  238. {$else i386}
  239. regvarinfo^.fpuregvars[i].reg:=fpuvarregs[i];
  240. {$endif i386}
  241. end;
  242. end;
  243. end;
  244. end;
  245. end;
  246. procedure store_regvar(asml: TAAsmoutput; reg: tregister);
  247. var
  248. i: longint;
  249. hr: treference;
  250. regvarinfo: pregvarinfo;
  251. vsym: tvarsym;
  252. begin
  253. regvarinfo := pregvarinfo(aktprocdef.regvarinfo);
  254. if not assigned(regvarinfo) then
  255. exit;
  256. for i := 1 to maxvarregs do
  257. if assigned(regvarinfo^.regvars[i]) and
  258. (makereg32(regvarinfo^.regvars[i].reg) = reg) then
  259. begin
  260. if regvar_loaded[makereg32(reg)] then
  261. begin
  262. vsym := tvarsym(regvarinfo^.regvars[i]);
  263. { we only have to store the regvar back to memory if it's }
  264. { possible that it's been modified (JM) }
  265. if not(vsym.varspez in [vs_const,vs_var,vs_out]) then
  266. begin
  267. reset_reference(hr);
  268. if vsym.owner.symtabletype in [inlinelocalsymtable,localsymtable] then
  269. hr.offset:=-vsym.address+vsym.owner.address_fixup
  270. else hr.offset:=vsym.address+vsym.owner.address_fixup;
  271. hr.base:=procinfo^.framepointer;
  272. cg.a_load_reg_ref(asml,def_cgsize(vsym.vartype.def),vsym.reg,hr);
  273. { asml.concat(Taicpu.op_reg_ref(A_MOV,regsize(vsym.reg),vsym.reg,hr)); }
  274. end;
  275. asml.concat(Tairegalloc.dealloc(makereg32(reg)));
  276. regvar_loaded[makereg32(reg)] := false;
  277. end;
  278. break;
  279. end;
  280. end;
  281. procedure load_regvar(asml: TAAsmoutput; vsym: tvarsym);
  282. var
  283. hr: treference;
  284. opsize: tcgsize;
  285. begin
  286. if not regvar_loaded[makereg32(vsym.reg)] then
  287. begin
  288. asml.concat(Tairegalloc.alloc(makereg32(vsym.reg)));
  289. reset_reference(hr);
  290. if vsym.owner.symtabletype in [inlinelocalsymtable,localsymtable] then
  291. hr.offset:=-vsym.address+vsym.owner.address_fixup
  292. else hr.offset:=vsym.address+vsym.owner.address_fixup;
  293. hr.base:=procinfo^.framepointer;
  294. if (vsym.varspez in [vs_var,vs_out]) or
  295. ((vsym.varspez=vs_const) and
  296. push_addr_param(vsym.vartype.def)) then
  297. {FIXME!!! Needs to be OS_SIZE_OF_POINTER (JM) }
  298. opsize := OS_32
  299. else
  300. opsize := def_cgsize(vsym.vartype.def);
  301. cg.a_load_ref_reg(asml,opsize,hr,makereg32(vsym.reg));
  302. { asml.concat(Taicpu.op_ref_reg(opcode,opsize,hr,makereg32(vsym.reg))); }
  303. regvar_loaded[makereg32(vsym.reg)] := true;
  304. end;
  305. end;
  306. procedure load_regvar_reg(asml: TAAsmoutput; reg: tregister);
  307. var
  308. i: longint;
  309. regvarinfo: pregvarinfo;
  310. begin
  311. regvarinfo := pregvarinfo(aktprocdef.regvarinfo);
  312. if not assigned(regvarinfo) then
  313. exit;
  314. reg := makereg32(reg);
  315. for i := 1 to maxvarregs do
  316. if assigned(regvarinfo^.regvars[i]) and
  317. (makereg32(regvarinfo^.regvars[i].reg) = reg) then
  318. load_regvar(asml,tvarsym(regvarinfo^.regvars[i]))
  319. end;
  320. procedure load_all_regvars(asml: TAAsmoutput);
  321. var
  322. i: longint;
  323. regvarinfo: pregvarinfo;
  324. begin
  325. regvarinfo := pregvarinfo(aktprocdef.regvarinfo);
  326. if not assigned(regvarinfo) then
  327. exit;
  328. for i := 1 to maxvarregs do
  329. if assigned(regvarinfo^.regvars[i]) and
  330. (makereg32(regvarinfo^.regvars[i].reg) in [R_EAX,R_EBX,R_ECX,R_EDX]) then
  331. load_regvar(asml,tvarsym(regvarinfo^.regvars[i]))
  332. end;
  333. procedure load_regvars(asml: TAAsmoutput; p: tnode);
  334. var
  335. i: longint;
  336. regvarinfo: pregvarinfo;
  337. begin
  338. if (cs_regalloc in aktglobalswitches) and
  339. ((procinfo^.flags and (pi_uses_asm or pi_uses_exceptions))=0) then
  340. begin
  341. regvarinfo := pregvarinfo(aktprocdef.regvarinfo);
  342. { can happen when inlining assembler procedures (JM) }
  343. if not assigned(regvarinfo) then
  344. exit;
  345. for i:=1 to maxvarregs do
  346. begin
  347. if assigned(regvarinfo^.regvars[i]) then
  348. begin
  349. if cs_asm_source in aktglobalswitches then
  350. asml.insert(Tai_asm_comment.Create(strpnew(regvarinfo^.regvars[i].name+
  351. ' with weight '+tostr(regvarinfo^.regvars[i].refs)+' assigned to register '+
  352. reg2str(regvarinfo^.regvars[i].reg))));
  353. if (status.verbosity and v_debug)=v_debug then
  354. Message3(cg_d_register_weight,reg2str(regvarinfo^.regvars[i].reg),
  355. tostr(regvarinfo^.regvars[i].refs),regvarinfo^.regvars[i].name);
  356. end;
  357. end;
  358. for i:=1 to maxfpuvarregs do
  359. begin
  360. if assigned(regvarinfo^.fpuregvars[i]) then
  361. begin
  362. {$ifdef i386}
  363. { reserve place on the FPU stack }
  364. regvarinfo^.fpuregvars[i].reg:=correct_fpuregister(R_ST0,i-1);
  365. asml.concat(Taicpu.op_none(A_FLDZ,S_NO));
  366. {$endif i386}
  367. end;
  368. end;
  369. {$ifdef i386}
  370. if assigned(p) then
  371. if cs_asm_source in aktglobalswitches then
  372. asml.insert(Tai_asm_comment.Create(strpnew(tostr(p.registersfpu)+
  373. ' registers on FPU stack used by temp. expressions')));
  374. {$endif i386}
  375. for i:=1 to maxfpuvarregs do
  376. begin
  377. if assigned(regvarinfo^.fpuregvars[i]) then
  378. begin
  379. if cs_asm_source in aktglobalswitches then
  380. asml.insert(Tai_asm_comment.Create(strpnew(regvarinfo^.fpuregvars[i].name+
  381. ' with weight '+tostr(regvarinfo^.fpuregvars[i].refs)+' assigned to register '+
  382. reg2str(regvarinfo^.fpuregvars[i].reg))));
  383. if (status.verbosity and v_debug)=v_debug then
  384. Message3(cg_d_register_weight,reg2str(regvarinfo^.fpuregvars[i].reg),
  385. tostr(regvarinfo^.fpuregvars[i].refs),regvarinfo^.fpuregvars[i].name);
  386. end;
  387. end;
  388. if cs_asm_source in aktglobalswitches then
  389. asml.insert(Tai_asm_comment.Create(strpnew('Register variable assignment:')));
  390. end;
  391. end;
  392. procedure sync_regvars(list1, list2: taasmoutput; const regvarsloaded1,
  393. regvarsloaded2: regvar_booleanarray);
  394. var
  395. counter: tregister;
  396. begin
  397. for counter := low(regvar_loaded) to high(regvar_loaded) do
  398. begin
  399. regvar_loaded[counter] := regvarsloaded1[counter] and
  400. regvarsloaded2[counter];
  401. if regvarsloaded1[counter] xor regvarsloaded2[counter] then
  402. if regvarsloaded1[counter] then
  403. load_regvar_reg(list2,counter)
  404. else
  405. load_regvar_reg(list1,counter);
  406. end;
  407. end;
  408. procedure cleanup_regvars(asml: TAAsmoutput);
  409. var
  410. i: longint;
  411. begin
  412. { can happen when inlining assembler procedures (JM) }
  413. if not assigned(aktprocdef.regvarinfo) then
  414. exit;
  415. if (cs_regalloc in aktglobalswitches) and
  416. ((procinfo^.flags and (pi_uses_asm or pi_uses_exceptions))=0) then
  417. with pregvarinfo(aktprocdef.regvarinfo)^ do
  418. begin
  419. {$ifdef i386}
  420. for i:=1 to maxfpuvarregs do
  421. if assigned(fpuregvars[i]) then
  422. { ... and clean it up }
  423. asml.concat(Taicpu.op_reg(A_FSTP,S_NO,R_ST0));
  424. {$endif i386}
  425. for i := 1 to maxvarregs do
  426. if assigned(regvars[i]) and
  427. (regvar_loaded[makereg32(regvars[i].reg)]) then
  428. asml.concat(Tairegalloc.dealloc(makereg32(regvars[i].reg)));
  429. end;
  430. end;
  431. end.
  432. {
  433. $Log$
  434. Revision 1.21 2001-12-03 12:17:02 jonas
  435. * forgot to commit yesterday :( (less unnecessary loading of regvars with
  436. if-statements)
  437. Revision 1.20 2001/11/05 16:49:32 jonas
  438. * constant regvars (addresses of var/out para's and const para's) aren't
  439. saved to memory anymore when their register will be destroyed
  440. * unit has been made mostly processor independent
  441. Revision 1.19 2001/11/02 22:58:06 peter
  442. * procsym definition rewrite
  443. Revision 1.18 2001/08/26 13:36:49 florian
  444. * some cg reorganisation
  445. * some PPC updates
  446. Revision 1.17 2001/04/21 12:03:12 peter
  447. * m68k updates merged from fixes branch
  448. Revision 1.16 2001/04/13 01:22:13 peter
  449. * symtable change to classes
  450. * range check generation and errors fixed, make cycle DEBUG=1 works
  451. * memory leaks fixed
  452. Revision 1.15 2000/12/25 00:07:28 peter
  453. + new tlinkedlist class (merge of old tstringqueue,tcontainer and
  454. tlinkedlist objects)
  455. Revision 1.14 2000/12/05 11:44:32 jonas
  456. + new integer regvar handling, should be much more efficient
  457. Revision 1.13 2000/11/29 00:30:39 florian
  458. * unused units removed from uses clause
  459. * some changes for widestrings
  460. Revision 1.12 2000/11/04 14:25:21 florian
  461. + merged Attila's changes for interfaces, not tested yet
  462. Revision 1.11 2000/10/31 22:02:51 peter
  463. * symtable splitted, no real code changes
  464. Revision 1.10 2000/10/14 10:14:52 peter
  465. * moehrendorf oct 2000 rewrite
  466. Revision 1.9 2000/10/01 19:48:25 peter
  467. * lot of compile updates for cg11
  468. Revision 1.8 2000/09/30 16:08:45 peter
  469. * more cg11 updates
  470. Revision 1.7 2000/09/30 13:08:16 jonas
  471. * regvars are now zeroed at the start of their life if they contain an 8
  472. or 16bit var/parameter, because the full 32bits are used if they are
  473. necessary for a btrl instruction
  474. Revision 1.6 2000/09/24 15:06:27 peter
  475. * use defines.inc
  476. Revision 1.5 2000/08/27 16:11:52 peter
  477. * moved some util functions from globals,cobjects to cutils
  478. * splitted files into finput,fmodule
  479. Revision 1.4 2000/08/17 11:07:51 jonas
  480. * fixed crash when inlining assembler procedures with -Or
  481. Revision 1.3 2000/08/04 05:52:00 jonas
  482. * correct version (I also had a regvars.pp locally, which was used
  483. instead of the regvars.pas on CVS, so I didn't notice the errors :( )
  484. Revision 1.2 2000/08/03 14:36:47 jonas
  485. * fixed inserting of allocated register for regvars (only those for
  486. parameters were done, and sometimes even the wrong ones)
  487. Revision 1.1 2000/08/03 13:17:25 jonas
  488. + allow regvars to be used inside inlined procs, which required the
  489. following changes:
  490. + load regvars in genentrycode/free them in genexitcode (cgai386)
  491. * moved all regvar related code to new regvars unit
  492. + added pregvarinfo type to hcodegen
  493. + added regvarinfo field to tprocinfo (symdef/symdefh)
  494. * deallocate the regvars of the caller in secondprocinline before
  495. inlining the called procedure and reallocate them afterwards
  496. }