pass_2.pas 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. {
  2. $Id$
  3. Copyright (c) 1998-2002 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. unit pass_2;
  19. {$i fpcdefs.inc}
  20. interface
  21. uses
  22. node;
  23. type
  24. tenumflowcontrol = (fc_exit,fc_break,fc_continue);
  25. tflowcontrol = set of tenumflowcontrol;
  26. var
  27. flowcontrol : tflowcontrol;
  28. { produces assembler for the expression in variable p }
  29. { and produces an assembler node at the end }
  30. procedure generatecode(var p : tnode);
  31. { produces the actual code }
  32. function do_secondpass(var p : tnode) : boolean;
  33. procedure secondpass(var p : tnode);
  34. implementation
  35. uses
  36. {$ifdef EXTDEBUG}
  37. cutils,
  38. {$endif}
  39. globtype,systems,verbose,
  40. cclasses,globals,
  41. symconst,symbase,symtype,symsym,paramgr,
  42. aasmbase,aasmtai,
  43. pass_1,cpubase,cgbase,regvars,nflw,rgobj;
  44. {*****************************************************************************
  45. SecondPass
  46. *****************************************************************************}
  47. {$ifdef EXTDEBUG}
  48. procedure logsecond(ht:tnodetype; entry: boolean);
  49. const
  50. secondnames: array[tnodetype] of string[13] =
  51. ('add-addn', {addn}
  52. 'add-muln', {muln}
  53. 'add-subn', {subn}
  54. 'moddiv-divn', {divn}
  55. 'add-symdifn', {symdifn}
  56. 'moddiv-modn', {modn}
  57. 'assignment', {assignn}
  58. 'load', {loadn}
  59. 'nothing-range', {range}
  60. 'add-ltn', {ltn}
  61. 'add-lten', {lten}
  62. 'add-gtn', {gtn}
  63. 'add-gten', {gten}
  64. 'add-equaln', {equaln}
  65. 'add-unequaln', {unequaln}
  66. 'in', {inn}
  67. 'add-orn', {orn}
  68. 'add-xorn', {xorn}
  69. 'shlshr-shrn', {shrn}
  70. 'shlshr-shln', {shln}
  71. 'add-slashn', {slashn}
  72. 'add-andn', {andn}
  73. 'subscriptn', {subscriptn}
  74. 'dderef', {derefn}
  75. 'addr', {addrn}
  76. 'doubleaddr', {doubleaddrn}
  77. 'ordconst', {ordconstn}
  78. 'typeconv', {typeconvn}
  79. 'calln', {calln}
  80. 'noth-callpar',{callparan}
  81. 'realconst', {realconstn}
  82. 'unaryminus', {unaryminusn}
  83. 'asm', {asmn}
  84. 'vecn', {vecn}
  85. 'pointerconst',{pointerconstn}
  86. 'stringconst', {stringconstn}
  87. 'funcret', {funcretn}
  88. 'selfn', {selfn}
  89. 'not', {notn}
  90. 'inline', {inlinen}
  91. 'niln', {niln}
  92. 'error', {errorn}
  93. 'nothing-typen', {typen}
  94. 'hnewn', {hnewn}
  95. 'hdisposen', {hdisposen}
  96. 'setelement', {setelementn}
  97. 'setconst', {setconstn}
  98. 'blockn', {blockn}
  99. 'statement', {statementn}
  100. 'ifn', {ifn}
  101. 'breakn', {breakn}
  102. 'continuen', {continuen}
  103. 'while_repeat', {whilerepeatn}
  104. 'for', {forn}
  105. 'exitn', {exitn}
  106. 'with', {withn}
  107. 'case', {casen}
  108. 'label', {labeln}
  109. 'goto', {goton}
  110. 'tryexcept', {tryexceptn}
  111. 'raise', {raisen}
  112. 'tryfinally', {tryfinallyn}
  113. 'on', {onn}
  114. 'is', {isn}
  115. 'as', {asn}
  116. 'error-caret', {caretn}
  117. 'fail', {failn}
  118. 'add-starstar', {starstarn}
  119. 'procinline', {procinlinen}
  120. 'arrayconstruc', {arrayconstructn}
  121. 'noth-arrcnstr', {arrayconstructrangen}
  122. 'tempcreaten',
  123. 'temprefn',
  124. 'tempdeleten',
  125. 'addoptn',
  126. 'nothing-nothg', {nothingn}
  127. 'loadvmt', {loadvmtn}
  128. 'guidconstn',
  129. 'rttin'
  130. );
  131. var
  132. p: pchar;
  133. begin
  134. if entry then
  135. p := strpnew('second'+secondnames[ht]+' (entry)')
  136. else
  137. p := strpnew('second'+secondnames[ht]+' (exit)');
  138. exprasmlist.concat(tai_asm_comment.create(p));
  139. end;
  140. {$endif EXTDEBUG}
  141. procedure secondpass(var p : tnode);
  142. var
  143. oldcodegenerror : boolean;
  144. oldlocalswitches : tlocalswitches;
  145. oldpos : tfileposinfo;
  146. {$ifdef TEMPREGDEBUG}
  147. prevp : pptree;
  148. {$endif TEMPREGDEBUG}
  149. {$ifdef EXTDEBUG}
  150. oldloc : tloc;
  151. {$endif EXTDEBUG}
  152. begin
  153. if not(nf_error in p.flags) then
  154. begin
  155. oldcodegenerror:=codegenerror;
  156. oldlocalswitches:=aktlocalswitches;
  157. oldpos:=aktfilepos;
  158. {$ifdef TEMPREGDEBUG}
  159. testregisters32;
  160. prevp:=curptree;
  161. curptree:=@p;
  162. p^.usableregs:=usablereg32;
  163. {$endif TEMPREGDEBUG}
  164. aktfilepos:=p.fileinfo;
  165. aktlocalswitches:=p.localswitches;
  166. codegenerror:=false;
  167. {$ifdef EXTDEBUG}
  168. oldloc:=p.location.loc;
  169. p.location.loc:=LOC_INVALID;
  170. if (cs_asm_nodes in aktglobalswitches) then
  171. logsecond(p.nodetype,true);
  172. {$endif EXTDEBUG}
  173. p.pass_2;
  174. {$ifdef EXTDEBUG}
  175. if (cs_asm_nodes in aktglobalswitches) then
  176. logsecond(p.nodetype,false);
  177. if (not codegenerror) and
  178. (oldloc<>LOC_INVALID) and
  179. (p.location.loc=LOC_INVALID) then
  180. Comment(V_Fatal,'Location not set in secondpass: '+nodetype2str[p.nodetype]);
  181. {$endif EXTDEBUG}
  182. if codegenerror then
  183. include(p.flags,nf_error);
  184. codegenerror:=codegenerror or oldcodegenerror;
  185. aktlocalswitches:=oldlocalswitches;
  186. aktfilepos:=oldpos;
  187. {$ifdef TEMPREGDEBUG}
  188. curptree:=prevp;
  189. {$endif TEMPREGDEBUG}
  190. {$ifdef EXTTEMPREGDEBUG}
  191. if p.usableregs-usablereg32>p.reallyusedregs then
  192. p.reallyusedregs:=p.usableregs-usablereg32;
  193. if p.reallyusedregs<p.registers32 then
  194. Comment(V_Debug,'registers32 overestimated '+tostr(p^.registers32)+
  195. '>'+tostr(p^.reallyusedregs));
  196. {$endif EXTTEMPREGDEBUG}
  197. end
  198. else
  199. codegenerror:=true;
  200. end;
  201. function do_secondpass(var p : tnode) : boolean;
  202. begin
  203. codegenerror:=false;
  204. if not(nf_error in p.flags) then
  205. secondpass(p);
  206. do_secondpass:=codegenerror;
  207. end;
  208. procedure clearrefs(p : tnamedindexitem;arg:pointer);
  209. begin
  210. if (tsym(p).typ=varsym) then
  211. if tvarsym(p).refs>1 then
  212. tvarsym(p).refs:=1;
  213. end;
  214. procedure generatecode(var p : tnode);
  215. begin
  216. rg.cleartempgen;
  217. flowcontrol:=[];
  218. { when size optimization only count occurrence }
  219. if cs_littlesize in aktglobalswitches then
  220. rg.t_times:=1
  221. else
  222. { reference for repetition is 100 }
  223. rg.t_times:=100;
  224. { clear register count }
  225. rg.clearregistercount;
  226. use_esp_stackframe:=false;
  227. symtablestack.foreach_static({$ifdef FPCPROCVAR}@{$endif}clearrefs,nil);
  228. symtablestack.next.foreach_static({$ifdef FPCPROCVAR}@{$endif}clearrefs,nil);
  229. { firstpass everything }
  230. do_firstpass(p);
  231. { only do secondpass if there are no errors }
  232. if ErrorCount=0 then
  233. begin
  234. {$ifdef OMITSTACKFRAME}
  235. if (cs_regalloc in aktglobalswitches) and
  236. ((procinfo^.flags and (pi_uses_asm or pi_uses_exceptions))=0) then
  237. begin
  238. { can we omit the stack frame ? }
  239. { conditions:
  240. 1. procedure (not main block)
  241. 2. no constructor or destructor
  242. 3. no call to other procedures
  243. 4. no interrupt handler
  244. }
  245. {!!!!!! this doesn work yet, because of problems with
  246. with linux and windows
  247. }
  248. (*
  249. if assigned(aktprocsym) then
  250. begin
  251. if not(assigned(procinfo^._class)) and
  252. not(aktprocdef.proctypeoption in [potype_constructor,potype_destructor]) and
  253. not(po_interrupt in aktprocdef.procoptions) and
  254. ((procinfo^.flags and pi_do_call)=0) and
  255. (lexlevel>=normal_function_level) then
  256. begin
  257. { use ESP as frame pointer }
  258. procinfo^.framepointer:=STACK_POINTER_REG;
  259. use_esp_stackframe:=true;
  260. { calc parameter distance new }
  261. dec(procinfo^.framepointer_offset,4);
  262. dec(procinfo^.selfpointer_offset,4);
  263. { is this correct ???}
  264. { retoffset can be negativ for results in eax !! }
  265. { the value should be decreased only if positive }
  266. if procinfo^.retoffset>=0 then
  267. dec(procinfo^.retoffset,4);
  268. dec(procinfo^.para_offset,4);
  269. aktprocdef.parast.address_fixup:=procinfo^.para_offset;
  270. end;
  271. end;
  272. *)
  273. end;
  274. {$endif OMITSTACKFRAME}
  275. { assign parameter locations }
  276. {$ifndef i386}
  277. setparalocs(procinfo^.procdef);
  278. {$endif i386}
  279. { process register variable stuff (JM) }
  280. assign_regvars(p);
  281. load_regvars(procinfo^.aktentrycode,p);
  282. { for the i386 it must be done in genexitcode because it has }
  283. { to add 'fstp' instructions when using fpu regvars and those }
  284. { must come after the "exitlabel" (JM) }
  285. {$ifndef i386}
  286. cleanup_regvars(procinfo^.aktexitcode);
  287. {$endif i386}
  288. do_secondpass(p);
  289. if assigned(procinfo^.procdef) then
  290. procinfo^.procdef.fpu_used:=p.registersfpu;
  291. end;
  292. procinfo^.aktproccode.concatlist(exprasmlist);
  293. end;
  294. end.
  295. {
  296. $Log$
  297. Revision 1.34 2002-08-15 19:10:35 peter
  298. * first things tai,tnode storing in ppu
  299. Revision 1.33 2002/07/30 20:50:44 florian
  300. * the code generator knows now if parameters are in registers
  301. Revision 1.32 2002/07/19 11:41:36 daniel
  302. * State tracker work
  303. * The whilen and repeatn are now completely unified into whilerepeatn. This
  304. allows the state tracker to change while nodes automatically into
  305. repeat nodes.
  306. * Resulttypepass improvements to the notn. 'not not a' is optimized away and
  307. 'not(a>b)' is optimized into 'a<=b'.
  308. * Resulttypepass improvements to the whilerepeatn. 'while not a' is optimized
  309. by removing the notn and later switchting the true and falselabels. The
  310. same is done with 'repeat until not a'.
  311. Revision 1.31 2002/07/01 18:46:25 peter
  312. * internal linker
  313. * reorganized aasm layer
  314. Revision 1.30 2002/05/18 13:34:11 peter
  315. * readded missing revisions
  316. Revision 1.29 2002/05/16 19:46:42 carl
  317. + defines.inc -> fpcdefs.inc to avoid conflicts if compiling by hand
  318. + try to fix temp allocation (still in ifdef)
  319. + generic constructor calls
  320. + start of tassembler / tmodulebase class cleanup
  321. Revision 1.27 2002/05/12 16:53:08 peter
  322. * moved entry and exitcode to ncgutil and cgobj
  323. * foreach gets extra argument for passing local data to the
  324. iterator function
  325. * -CR checks also class typecasts at runtime by changing them
  326. into as
  327. * fixed compiler to cycle with the -CR option
  328. * fixed stabs with elf writer, finally the global variables can
  329. be watched
  330. * removed a lot of routines from cga unit and replaced them by
  331. calls to cgobj
  332. * u32bit-s32bit updates for and,or,xor nodes. When one element is
  333. u32bit then the other is typecasted also to u32bit without giving
  334. a rangecheck warning/error.
  335. * fixed pascal calling method with reversing also the high tree in
  336. the parast, detected by tcalcst3 test
  337. Revision 1.26 2002/04/21 19:02:04 peter
  338. * removed newn and disposen nodes, the code is now directly
  339. inlined from pexpr
  340. * -an option that will write the secondpass nodes to the .s file, this
  341. requires EXTDEBUG define to actually write the info
  342. * fixed various internal errors and crashes due recent code changes
  343. Revision 1.25 2002/04/20 21:32:24 carl
  344. + generic FPC_CHECKPOINTER
  345. + first parameter offset in stack now portable
  346. * rename some constants
  347. + move some cpu stuff to other units
  348. - remove unused constents
  349. * fix stacksize for some targets
  350. * fix generic size problems which depend now on EXTEND_SIZE constant
  351. Revision 1.24 2002/04/07 13:30:13 carl
  352. - removed unused variable
  353. Revision 1.23 2002/04/02 17:11:29 peter
  354. * tlocation,treference update
  355. * LOC_CONSTANT added for better constant handling
  356. * secondadd splitted in multiple routines
  357. * location_force_reg added for loading a location to a register
  358. of a specified size
  359. * secondassignment parses now first the right and then the left node
  360. (this is compatible with Kylix). This saves a lot of push/pop especially
  361. with string operations
  362. * adapted some routines to use the new cg methods
  363. Revision 1.22 2002/03/31 20:26:35 jonas
  364. + a_loadfpu_* and a_loadmm_* methods in tcg
  365. * register allocation is now handled by a class and is mostly processor
  366. independent (+rgobj.pas and i386/rgcpu.pas)
  367. * temp allocation is now handled by a class (+tgobj.pas, -i386\tgcpu.pas)
  368. * some small improvements and fixes to the optimizer
  369. * some register allocation fixes
  370. * some fpuvaroffset fixes in the unary minus node
  371. * push/popusedregisters is now called rg.save/restoreusedregisters and
  372. (for i386) uses temps instead of push/pop's when using -Op3 (that code is
  373. also better optimizable)
  374. * fixed and optimized register saving/restoring for new/dispose nodes
  375. * LOC_FPU locations now also require their "register" field to be set to
  376. R_ST, not R_ST0 (the latter is used for LOC_CFPUREGISTER locations only)
  377. - list field removed of the tnode class because it's not used currently
  378. and can cause hard-to-find bugs
  379. }