pass_2.pas 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515
  1. {
  2. $Id$
  3. Copyright (c) 1993-98 by Florian Klaempfl
  4. This unit handles the codegeneration pass
  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. {$ifdef TP}
  19. {$E+,F+,N+}
  20. {$endif}
  21. unit pass_2;
  22. interface
  23. uses
  24. tree;
  25. { produces assembler for the expression in variable p }
  26. { and produces an assembler node at the end }
  27. procedure generatecode(var p : ptree);
  28. { produces the actual code }
  29. function do_secondpass(var p : ptree) : boolean;
  30. procedure secondpass(var p : ptree);
  31. implementation
  32. uses
  33. cobjects,verbose,comphook,systems,globals,files,
  34. symtable,types,aasm,scanner,
  35. pass_1,hcodegen,temp_gen
  36. {$ifdef GDB}
  37. ,gdb
  38. {$endif}
  39. {$ifdef i386}
  40. ,i386,tgeni386,cgai386
  41. ,cg386con,cg386mat,cg386cnv,cg386set,cg386add
  42. ,cg386mem,cg386cal,cg386ld,cg386flw,cg386inl
  43. {$endif}
  44. {$ifdef m68k}
  45. ,m68k,tgen68k,cga68k
  46. ,cg68kcon,cg68kmat,cg68kcnv,cg68kset,cg68kadd
  47. ,cg68kmem,cg68kcal,cg68kld,cg68kflw,cg68kinl
  48. {$endif}
  49. ;
  50. {*****************************************************************************
  51. SecondPass
  52. *****************************************************************************}
  53. type
  54. secondpassproc = procedure(var p : ptree);
  55. procedure secondnothing(var p : ptree);
  56. begin
  57. end;
  58. procedure seconderror(var p : ptree);
  59. begin
  60. p^.error:=true;
  61. codegenerror:=true;
  62. end;
  63. procedure secondstatement(var p : ptree);
  64. var
  65. hp : ptree;
  66. begin
  67. hp:=p;
  68. while assigned(hp) do
  69. begin
  70. if assigned(hp^.right) then
  71. begin
  72. cleartempgen;
  73. secondpass(hp^.right);
  74. end;
  75. hp:=hp^.left;
  76. end;
  77. end;
  78. procedure secondblockn(var p : ptree);
  79. begin
  80. { do second pass on left node }
  81. if assigned(p^.left) then
  82. secondpass(p^.left);
  83. end;
  84. procedure secondasm(var p : ptree);
  85. begin
  86. exprasmlist^.concatlist(p^.p_asm);
  87. if not p^.object_preserved then
  88. begin
  89. {$ifdef i386}
  90. maybe_loadesi;
  91. {$endif}
  92. {$ifdef m68k}
  93. maybe_loada5;
  94. {$endif}
  95. end;
  96. end;
  97. procedure secondpass(var p : ptree);
  98. const
  99. procedures : array[ttreetyp] of secondpassproc =
  100. (secondadd, {addn}
  101. secondadd, {muln}
  102. secondadd, {subn}
  103. secondmoddiv, {divn}
  104. secondadd, {symdifn}
  105. secondmoddiv, {modn}
  106. secondassignment, {assignn}
  107. secondload, {loadn}
  108. secondnothing, {range}
  109. secondadd, {ltn}
  110. secondadd, {lten}
  111. secondadd, {gtn}
  112. secondadd, {gten}
  113. secondadd, {equaln}
  114. secondadd, {unequaln}
  115. secondin, {inn}
  116. secondadd, {orn}
  117. secondadd, {xorn}
  118. secondshlshr, {shrn}
  119. secondshlshr, {shln}
  120. secondadd, {slashn}
  121. secondadd, {andn}
  122. secondsubscriptn, {subscriptn}
  123. secondderef, {derefn}
  124. secondaddr, {addrn}
  125. seconddoubleaddr, {doubleaddrn}
  126. secondordconst, {ordconstn}
  127. secondtypeconv, {typeconvn}
  128. secondcalln, {calln}
  129. secondnothing, {callparan}
  130. secondrealconst, {realconstn}
  131. secondfixconst, {fixconstn}
  132. secondumminus, {umminusn}
  133. secondasm, {asmn}
  134. secondvecn, {vecn}
  135. secondstringconst, {stringconstn}
  136. secondfuncret, {funcretn}
  137. secondselfn, {selfn}
  138. secondnot, {notn}
  139. secondinline, {inlinen}
  140. secondniln, {niln}
  141. seconderror, {errorn}
  142. secondnothing, {typen}
  143. secondhnewn, {hnewn}
  144. secondhdisposen, {hdisposen}
  145. secondnewn, {newn}
  146. secondsimplenewdispose, {simpledisposen}
  147. secondsetelement, {setelementn}
  148. secondsetconst, {setconstn}
  149. secondblockn, {blockn}
  150. secondstatement, {statementn}
  151. secondnothing, {loopn}
  152. secondifn, {ifn}
  153. secondbreakn, {breakn}
  154. secondcontinuen, {continuen}
  155. second_while_repeatn, {repeatn}
  156. second_while_repeatn, {whilen}
  157. secondfor, {forn}
  158. secondexitn, {exitn}
  159. secondwith, {withn}
  160. secondcase, {casen}
  161. secondlabel, {labeln}
  162. secondgoto, {goton}
  163. secondsimplenewdispose, {simplenewn}
  164. secondtryexcept, {tryexceptn}
  165. secondraise, {raisen}
  166. secondnothing, {switchesn}
  167. secondtryfinally, {tryfinallyn}
  168. secondon, {onn}
  169. secondis, {isn}
  170. secondas, {asn}
  171. seconderror, {caretn}
  172. secondfail, {failn}
  173. secondadd, {starstarn}
  174. secondprocinline, {procinlinen}
  175. secondnothing, {nothingn}
  176. secondloadvmt {loadvmtn}
  177. );
  178. var
  179. oldcodegenerror : boolean;
  180. oldlocalswitches : tlocalswitches;
  181. oldpos : tfileposinfo;
  182. begin
  183. if not(p^.error) then
  184. begin
  185. oldcodegenerror:=codegenerror;
  186. oldlocalswitches:=aktlocalswitches;
  187. oldpos:=aktfilepos;
  188. aktfilepos:=p^.fileinfo;
  189. aktlocalswitches:=p^.localswitches;
  190. codegenerror:=false;
  191. procedures[p^.treetype](p);
  192. p^.error:=codegenerror;
  193. codegenerror:=codegenerror or oldcodegenerror;
  194. aktlocalswitches:=oldlocalswitches;
  195. aktfilepos:=oldpos;
  196. end
  197. else
  198. codegenerror:=true;
  199. end;
  200. function do_secondpass(var p : ptree) : boolean;
  201. begin
  202. codegenerror:=false;
  203. if not(p^.error) then
  204. secondpass(p);
  205. do_secondpass:=codegenerror;
  206. end;
  207. var
  208. regvars : array[1..maxvarregs] of pvarsym;
  209. regvars_para : array[1..maxvarregs] of boolean;
  210. regvars_refs : array[1..maxvarregs] of longint;
  211. parasym : boolean;
  212. procedure searchregvars(p : psym);
  213. var
  214. i,j,k : longint;
  215. begin
  216. if (p^.typ=varsym) and ((pvarsym(p)^.var_options and vo_regable)<>0) then
  217. begin
  218. { walk through all momentary register variables }
  219. for i:=1 to maxvarregs do
  220. begin
  221. { free register ? }
  222. if regvars[i]=nil then
  223. begin
  224. regvars[i]:=pvarsym(p);
  225. regvars_para[i]:=parasym;
  226. break;
  227. end;
  228. { else throw out a variable ? }
  229. j:=pvarsym(p)^.refs;
  230. { parameter get a less value }
  231. if parasym then
  232. begin
  233. if cs_littlesize in aktglobalswitches then
  234. dec(j,1)
  235. else
  236. dec(j,100);
  237. end;
  238. if (j>regvars_refs[i]) and (j>0) then
  239. begin
  240. for k:=maxvarregs-1 downto i do
  241. begin
  242. regvars[k+1]:=regvars[k];
  243. regvars_para[k+1]:=regvars_para[k];
  244. end;
  245. { calc the new refs
  246. pvarsym(p)^.refs:=j; }
  247. regvars[i]:=pvarsym(p);
  248. regvars_para[i]:=parasym;
  249. regvars_refs[i]:=j;
  250. break;
  251. end;
  252. end;
  253. end;
  254. end;
  255. procedure generatecode(var p : ptree);
  256. var
  257. i : longint;
  258. regsize : topsize;
  259. hr : preference;
  260. label
  261. nextreg;
  262. begin
  263. cleartempgen;
  264. { when size optimization only count occurrence }
  265. if cs_littlesize in aktglobalswitches then
  266. t_times:=1
  267. else
  268. { reference for repetition is 100 }
  269. t_times:=100;
  270. { clear register count }
  271. clearregistercount;
  272. use_esp_stackframe:=false;
  273. if not(do_firstpass(p)) then
  274. begin
  275. { max. optimizations }
  276. { only if no asm is used }
  277. { and no try statement }
  278. if (cs_regalloc in aktglobalswitches) and
  279. ((procinfo.flags and (pi_uses_asm or pi_uses_exceptions))=0) then
  280. begin
  281. { can we omit the stack frame ? }
  282. { conditions:
  283. 1. procedure (not main block)
  284. 2. no constructor or destructor
  285. 3. no call to other procedures
  286. 4. no interrupt handler
  287. }
  288. if assigned(aktprocsym) then
  289. begin
  290. if (aktprocsym^.definition^.options and
  291. (poconstructor+podestructor{+poinline}+pointerrupt)=0) and
  292. ((procinfo.flags and pi_do_call)=0) and (lexlevel>1) then
  293. begin
  294. { use ESP as frame pointer }
  295. procinfo.framepointer:=stack_pointer;
  296. use_esp_stackframe:=true;
  297. { calc parameter distance new }
  298. dec(procinfo.framepointer_offset,4);
  299. dec(procinfo.ESI_offset,4);
  300. { is this correct ???}
  301. { retoffset can be negativ for results in eax !! }
  302. { the value should be decreased only if positive }
  303. if procinfo.retoffset>=0 then
  304. dec(procinfo.retoffset,4);
  305. dec(procinfo.call_offset,4);
  306. aktprocsym^.definition^.parast^.call_offset:=procinfo.call_offset;
  307. end;
  308. end;
  309. if (p^.registers32<4) then
  310. begin
  311. for i:=1 to maxvarregs do
  312. regvars[i]:=nil;
  313. parasym:=false;
  314. {$ifdef tp}
  315. symtablestack^.foreach(searchregvars);
  316. {$else}
  317. symtablestack^.foreach(@searchregvars);
  318. {$endif}
  319. { copy parameter into a register ? }
  320. parasym:=true;
  321. {$ifdef tp}
  322. symtablestack^.next^.foreach(searchregvars);
  323. {$else}
  324. symtablestack^.next^.foreach(@searchregvars);
  325. {$endif}
  326. { hold needed registers free }
  327. for i:=maxvarregs downto maxvarregs-p^.registers32+1 do
  328. regvars[i]:=nil;
  329. { now assign register }
  330. for i:=1 to maxvarregs-p^.registers32 do
  331. begin
  332. if assigned(regvars[i]) then
  333. begin
  334. { it is nonsens, to copy the variable to }
  335. { a register because we need then much }
  336. { pushes ? }
  337. if reg_pushes[varregs[i]]>=regvars[i]^.refs then
  338. begin
  339. regvars[i]:=nil;
  340. goto nextreg;
  341. end;
  342. { register is no longer available for }
  343. { expressions }
  344. { search the register which is the most }
  345. { unused }
  346. usableregs:=usableregs-[varregs[i]];
  347. is_reg_var[varregs[i]]:=true;
  348. dec(c_usableregs);
  349. { possibly no 32 bit register are needed }
  350. { call by reference/const ? }
  351. if (regvars[i]^.varspez=vs_var) or
  352. ((regvars[i]^.varspez=vs_const) and
  353. dont_copy_const_param(regvars[i]^.definition)) then
  354. begin
  355. regvars[i]^.reg:=varregs[i];
  356. regsize:=S_L;
  357. end
  358. else
  359. if (regvars[i]^.definition^.deftype=orddef) and
  360. (porddef(regvars[i]^.definition)^.typ in [bool8bit,uchar,u8bit,s8bit]) then
  361. begin
  362. {$ifdef i386}
  363. regvars[i]^.reg:=reg32toreg8(varregs[i]);
  364. {$endif}
  365. regsize:=S_B;
  366. end
  367. else if (regvars[i]^.definition^.deftype=orddef) and
  368. (porddef(regvars[i]^.definition)^.typ in [bool16bit,u16bit,s16bit]) then
  369. begin
  370. {$ifdef i386}
  371. regvars[i]^.reg:=reg32toreg16(varregs[i]);
  372. {$endif}
  373. regsize:=S_W;
  374. end
  375. else
  376. begin
  377. regvars[i]^.reg:=varregs[i];
  378. regsize:=S_L;
  379. end;
  380. { parameter must be load }
  381. if regvars_para[i] then
  382. begin
  383. { procinfo is there actual, }
  384. { because we can't never be in a }
  385. { nested procedure }
  386. { when loading parameter to reg }
  387. new(hr);
  388. reset_reference(hr^);
  389. hr^.offset:=pvarsym(regvars[i])^.address+procinfo.call_offset;
  390. hr^.base:=procinfo.framepointer;
  391. {$ifdef i386}
  392. procinfo.aktentrycode^.concat(new(pai386,op_ref_reg(A_MOV,regsize,
  393. hr,regvars[i]^.reg)));
  394. {$endif i386}
  395. {$ifdef m68k}
  396. procinfo.aktentrycode^.concat(new(pai68k,op_ref_reg(A_MOVE,regsize,
  397. hr,regvars[i]^.reg)));
  398. {$endif m68k}
  399. unused:=unused - [regvars[i]^.reg];
  400. end;
  401. { procedure uses this register }
  402. {$ifdef i386}
  403. usedinproc:=usedinproc or ($80 shr byte(varregs[i]));
  404. {$endif i386}
  405. {$ifdef m68k}
  406. usedinproc:=usedinproc or ($800 shr word(varregs[i]));
  407. {$endif m68k}
  408. end;
  409. nextreg:
  410. { dummy }
  411. regsize:=S_W;
  412. end;
  413. if (status.verbosity and v_debug)=v_debug then
  414. begin
  415. for i:=1 to maxvarregs do
  416. begin
  417. if assigned(regvars[i]) then
  418. Message3(cg_d_register_weight,reg2str(regvars[i]^.reg),
  419. tostr(regvars[i]^.refs),regvars[i]^.name);
  420. end;
  421. end;
  422. end;
  423. end;
  424. if assigned(aktprocsym) and
  425. ((aktprocsym^.definition^.options and poinline)<>0) then
  426. make_const_global:=true;
  427. do_secondpass(p);
  428. procinfo.def^.fpu_used:=p^.registersfpu;
  429. { all registers can be used again }
  430. resetusableregisters;
  431. end;
  432. procinfo.aktproccode^.concatlist(exprasmlist);
  433. make_const_global:=false;
  434. end;
  435. end.
  436. {
  437. $Log$
  438. Revision 1.4 1998-09-21 08:45:16 pierre
  439. + added vmt_offset in tobjectdef.write for fututre use
  440. (first steps to have objects without vmt if no virtual !!)
  441. + added fpu_used field for tabstractprocdef :
  442. sets this level to 2 if the functions return with value in FPU
  443. (is then set to correct value at parsing of implementation)
  444. THIS MIGHT refuse some code with FPU expression too complex
  445. that were accepted before and even in some cases
  446. that don't overflow in fact
  447. ( like if f : float; is a forward that finally in implementation
  448. only uses one fpu register !!)
  449. Nevertheless I think that it will improve security on
  450. FPU operations !!
  451. * most other changes only for UseBrowser code
  452. (added symtable references for record and objects)
  453. local switch for refs to args and local of each function
  454. (static symtable still missing)
  455. UseBrowser still not stable and probably broken by
  456. the definition hash array !!
  457. Revision 1.3 1998/09/17 09:42:40 peter
  458. + pass_2 for cg386
  459. * Message() -> CGMessage() for pass_1/pass_2
  460. Revision 1.2 1998/09/07 18:46:07 peter
  461. * update smartlinking, uses getdatalabel
  462. * renamed ptree.value vars to value_str,value_real,value_set
  463. Revision 1.1 1998/09/01 09:07:12 peter
  464. * m68k fixes, splitted cg68k like cgi386
  465. }