pass_2.pas 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539
  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. allow_multi_pass2 : boolean;
  28. flowcontrol : tflowcontrol;
  29. { produces assembler for the expression in variable p }
  30. { and produces an assembler node at the end }
  31. procedure generatecode(var p : tnode);
  32. { produces the actual code }
  33. function do_secondpass(var p : tnode) : boolean;
  34. procedure secondpass(var p : tnode);
  35. implementation
  36. uses
  37. {$ifdef EXTDEBUG}
  38. cutils,
  39. {$endif}
  40. globtype,systems,verbose,
  41. cclasses,globals,
  42. symconst,symbase,symtype,symsym,paramgr,
  43. aasmbase,aasmtai,
  44. pass_1,cpubase,cginfo,cgbase,
  45. {$ifdef EXTDEBUG}
  46. cgobj,
  47. {$endif EXTDEBUG}
  48. regvars,nflw,rgobj;
  49. {*****************************************************************************
  50. SecondPass
  51. *****************************************************************************}
  52. {$ifdef EXTDEBUG}
  53. procedure logsecond(ht:tnodetype; entry: boolean);
  54. const
  55. secondnames: array[tnodetype] of string[13] =
  56. ('<emptynode>',
  57. 'add-addn', {addn}
  58. 'add-muln', {muln}
  59. 'add-subn', {subn}
  60. 'moddiv-divn', {divn}
  61. 'add-symdifn', {symdifn}
  62. 'moddiv-modn', {modn}
  63. 'assignment', {assignn}
  64. 'load', {loadn}
  65. 'nothing-range', {range}
  66. 'add-ltn', {ltn}
  67. 'add-lten', {lten}
  68. 'add-gtn', {gtn}
  69. 'add-gten', {gten}
  70. 'add-equaln', {equaln}
  71. 'add-unequaln', {unequaln}
  72. 'in', {inn}
  73. 'add-orn', {orn}
  74. 'add-xorn', {xorn}
  75. 'shlshr-shrn', {shrn}
  76. 'shlshr-shln', {shln}
  77. 'add-slashn', {slashn}
  78. 'add-andn', {andn}
  79. 'subscriptn', {subscriptn}
  80. 'dderef', {derefn}
  81. 'addr', {addrn}
  82. 'ordconst', {ordconstn}
  83. 'typeconv', {typeconvn}
  84. 'calln', {calln}
  85. 'noth-callpar',{callparan}
  86. 'realconst', {realconstn}
  87. 'unaryminus', {unaryminusn}
  88. 'asm', {asmn}
  89. 'vecn', {vecn}
  90. 'pointerconst',{pointerconstn}
  91. 'stringconst', {stringconstn}
  92. 'not', {notn}
  93. 'inline', {inlinen}
  94. 'niln', {niln}
  95. 'error', {errorn}
  96. 'nothing-typen', {typen}
  97. 'setelement', {setelementn}
  98. 'setconst', {setconstn}
  99. 'blockn', {blockn}
  100. 'statement', {statementn}
  101. 'ifn', {ifn}
  102. 'breakn', {breakn}
  103. 'continuen', {continuen}
  104. 'while_repeat', {whilerepeatn}
  105. 'for', {forn}
  106. 'exitn', {exitn}
  107. 'with', {withn}
  108. 'case', {casen}
  109. 'label', {labeln}
  110. 'goto', {goton}
  111. 'tryexcept', {tryexceptn}
  112. 'raise', {raisen}
  113. 'tryfinally', {tryfinallyn}
  114. 'on', {onn}
  115. 'is', {isn}
  116. 'as', {asn}
  117. 'error-caret', {caretn}
  118. 'add-starstar', {starstarn}
  119. 'arrayconstruc', {arrayconstructn}
  120. 'noth-arrcnstr', {arrayconstructrangen}
  121. 'tempcreaten',
  122. 'temprefn',
  123. 'tempdeleten',
  124. 'addoptn',
  125. 'nothing-nothg', {nothingn}
  126. 'loadvmt', {loadvmtn}
  127. 'guidconstn',
  128. 'rttin'
  129. );
  130. var
  131. p: pchar;
  132. begin
  133. if entry then
  134. p := strpnew('second '+secondnames[ht]+' (entry)')
  135. else
  136. p := strpnew('second '+secondnames[ht]+' (exit)');
  137. exprasmlist.concat(tai_comment.create(p));
  138. end;
  139. {$endif EXTDEBUG}
  140. procedure secondpass(var p : tnode);
  141. var
  142. oldcodegenerror : boolean;
  143. oldlocalswitches : tlocalswitches;
  144. oldpos : tfileposinfo;
  145. {$ifdef TEMPREGDEBUG}
  146. prevp : pptree;
  147. {$endif TEMPREGDEBUG}
  148. {$ifdef EXTDEBUG}
  149. {$endif EXTDEBUG}
  150. begin
  151. if not assigned(p) then
  152. internalerror(200208221);
  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. if (p.expectloc=LOC_INVALID) then
  169. Comment(V_Warning,'ExpectLoc is not set before secondpass: '+nodetype2str[p.nodetype]);
  170. if (not allow_multi_pass2) and
  171. (p.location.loc<>LOC_INVALID) then
  172. Comment(V_Warning,'Location.Loc is already set before secondpass: '+nodetype2str[p.nodetype]);
  173. if (cs_asm_nodes in aktglobalswitches) then
  174. logsecond(p.nodetype,true);
  175. {$endif EXTDEBUG}
  176. p.pass_2;
  177. {$ifdef EXTDEBUG}
  178. if (cs_asm_nodes in aktglobalswitches) then
  179. logsecond(p.nodetype,false);
  180. if (not codegenerror) then
  181. begin
  182. if (p.location.loc=LOC_INVALID) then
  183. Comment(V_Warning,'Location not set in secondpass: '+nodetype2str[p.nodetype])
  184. else if (p.location.loc<>p.expectloc) then
  185. Comment(V_Warning,'Location is different in secondpass: '+nodetype2str[p.nodetype]);
  186. end;
  187. {$ifdef i386}
  188. if rg.unusedregsint*([first_int_supreg..last_int_supreg] - [RS_FRAME_POINTER_REG,RS_STACK_POINTER_REG])<>
  189. ([first_int_supreg..last_int_supreg] - [RS_FRAME_POINTER_REG,RS_STACK_POINTER_REG]) then
  190. internalerror(200306171);
  191. {$endif i386}
  192. {$endif EXTDEBUG}
  193. if codegenerror then
  194. include(p.flags,nf_error);
  195. codegenerror:=codegenerror or oldcodegenerror;
  196. aktlocalswitches:=oldlocalswitches;
  197. aktfilepos:=oldpos;
  198. {$ifdef TEMPREGDEBUG}
  199. curptree:=prevp;
  200. {$endif TEMPREGDEBUG}
  201. {$ifdef EXTTEMPREGDEBUG}
  202. if p.usableregs-usablereg32>p.reallyusedregs then
  203. p.reallyusedregs:=p.usableregs-usablereg32;
  204. if p.reallyusedregs<p.registers32 then
  205. Comment(V_Debug,'registers32 overestimated '+tostr(p^.registers32)+
  206. '>'+tostr(p^.reallyusedregs));
  207. {$endif EXTTEMPREGDEBUG}
  208. end
  209. else
  210. codegenerror:=true;
  211. end;
  212. function do_secondpass(var p : tnode) : boolean;
  213. begin
  214. codegenerror:=false;
  215. if not(nf_error in p.flags) then
  216. secondpass(p);
  217. do_secondpass:=codegenerror;
  218. end;
  219. procedure clearrefs(p : tnamedindexitem;arg:pointer);
  220. begin
  221. if (tsym(p).typ=varsym) then
  222. if tvarsym(p).refs>1 then
  223. tvarsym(p).refs:=1;
  224. end;
  225. procedure generatecode(var p : tnode);
  226. begin
  227. flowcontrol:=[];
  228. { when size optimization only count occurrence }
  229. if cs_littlesize in aktglobalswitches then
  230. rg.t_times:=1
  231. else
  232. { reference for repetition is 100 }
  233. rg.t_times:=100;
  234. { clear register count }
  235. rg.clearregistercount;
  236. symtablestack.foreach_static({$ifdef FPCPROCVAR}@{$endif}clearrefs,nil);
  237. symtablestack.next.foreach_static({$ifdef FPCPROCVAR}@{$endif}clearrefs,nil);
  238. { firstpass everything }
  239. do_firstpass(p);
  240. { only do secondpass if there are no errors }
  241. if ErrorCount=0 then
  242. begin
  243. { assign parameter locations }
  244. current_procinfo.after_pass1;
  245. { caller paraloc info is also necessary in the stackframe_entry }
  246. { code of the ppc (and possibly other processors) }
  247. if not current_procinfo.procdef.has_paraloc_info then
  248. begin
  249. paramanager.create_paraloc_info(current_procinfo.procdef,callerside);
  250. current_procinfo.procdef.has_paraloc_info:=true;
  251. end;
  252. { process register variable stuff (JM) }
  253. assign_regvars(p);
  254. // load_regvars(current_procinfo.aktentrycode,p);
  255. { for the i386 it must be done in genexitcode because it has }
  256. { to add 'fstp' instructions when using fpu regvars and those }
  257. { must come after the "exitlabel" (JM) }
  258. {$ifndef i386}
  259. // cleanup_regvars(current_procinfo.aktexitcode);
  260. {$endif i386}
  261. current_procinfo.allocate_framepointer_reg;
  262. do_secondpass(p);
  263. if assigned(current_procinfo.procdef) then
  264. current_procinfo.procdef.fpu_used:=p.registersfpu;
  265. end;
  266. current_procinfo.aktproccode.concatlist(exprasmlist);
  267. end;
  268. end.
  269. {
  270. $Log$
  271. Revision 1.64 2003-09-03 15:55:01 peter
  272. * NEWRA branch merged
  273. Revision 1.63.2.3 2003/08/31 15:46:26 peter
  274. * more updates for tregister
  275. Revision 1.63.2.2 2003/08/31 13:50:16 daniel
  276. * Remove sorting and use pregenerated indexes
  277. * Some work on making things compile
  278. Revision 1.63.2.1 2003/08/29 17:28:59 peter
  279. * next batch of updates
  280. Revision 1.63 2003/08/11 21:18:20 peter
  281. * start of sparc support for newra
  282. Revision 1.62 2003/08/10 17:25:23 peter
  283. * fixed some reported bugs
  284. Revision 1.61 2003/07/06 17:58:22 peter
  285. * framepointer fixes for sparc
  286. * parent framepointer code more generic
  287. Revision 1.60 2003/07/06 15:31:20 daniel
  288. * Fixed register allocator. *Lots* of fixes.
  289. Revision 1.59 2003/07/06 10:18:47 jonas
  290. * also generate the caller paraloc info of a procedure if it doesn't exist
  291. yet at the start of pass_2
  292. Revision 1.58 2003/07/05 20:13:03 jonas
  293. * create_paraloc_info() is now called separately for the caller and
  294. callee info
  295. * fixed ppc cycle
  296. Revision 1.57 2003/06/13 21:19:30 peter
  297. * current_procdef removed, use current_procinfo.procdef instead
  298. Revision 1.56 2003/06/12 16:43:07 peter
  299. * newra compiles for sparc
  300. Revision 1.55 2003/06/09 12:23:30 peter
  301. * init/final of procedure data splitted from genentrycode
  302. * use asmnode getposition to insert final at the correct position
  303. als for the implicit try...finally
  304. Revision 1.54 2003/06/03 13:01:59 daniel
  305. * Register allocator finished
  306. Revision 1.53 2003/05/26 21:17:17 peter
  307. * procinlinenode removed
  308. * aktexit2label removed, fast exit removed
  309. + tcallnode.inlined_pass_2 added
  310. Revision 1.52 2003/05/15 18:58:53 peter
  311. * removed selfpointer_offset, vmtpointer_offset
  312. * tvarsym.adjusted_address
  313. * address in localsymtable is now in the real direction
  314. * removed some obsolete globals
  315. Revision 1.51 2003/05/13 19:14:41 peter
  316. * failn removed
  317. * inherited result code check moven to pexpr
  318. Revision 1.50 2003/05/09 17:47:02 peter
  319. * self moved to hidden parameter
  320. * removed hdisposen,hnewn,selfn
  321. Revision 1.49 2003/04/27 11:21:33 peter
  322. * aktprocdef renamed to current_procinfo.procdef
  323. * procinfo renamed to current_procinfo
  324. * procinfo will now be stored in current_module so it can be
  325. cleaned up properly
  326. * gen_main_procsym changed to create_main_proc and release_main_proc
  327. to also generate a tprocinfo structure
  328. * fixed unit implicit initfinal
  329. Revision 1.48 2003/04/27 07:29:50 peter
  330. * current_procinfo.procdef cleanup, current_procdef is now always nil when parsing
  331. a new procdef declaration
  332. * aktprocsym removed
  333. * lexlevel removed, use symtable.symtablelevel instead
  334. * implicit init/final code uses the normal genentry/genexit
  335. * funcret state checking updated for new funcret handling
  336. Revision 1.47 2003/04/25 20:59:33 peter
  337. * removed funcretn,funcretsym, function result is now in varsym
  338. and aliases for result and function name are added using absolutesym
  339. * vs_hidden parameter for funcret passed in parameter
  340. * vs_hidden fixes
  341. * writenode changed to printnode and released from extdebug
  342. * -vp option added to generate a tree.log with the nodetree
  343. * nicer printnode for statements, callnode
  344. Revision 1.46 2003/04/23 10:12:14 peter
  345. * allow multi pass2 changed to global boolean instead of node flag
  346. Revision 1.45 2003/04/22 23:50:23 peter
  347. * firstpass uses expectloc
  348. * checks if there are differences between the expectloc and
  349. location.loc from secondpass in EXTDEBUG
  350. Revision 1.44 2003/04/22 12:45:58 florian
  351. * fixed generic in operator code
  352. + added debug code to check if all scratch registers are released
  353. Revision 1.43 2003/01/09 20:40:59 daniel
  354. * Converted some code in cgx86.pas to new register numbering
  355. Revision 1.42 2003/01/09 15:49:56 daniel
  356. * Added register conversion
  357. Revision 1.41 2002/12/22 14:35:39 peter
  358. * removed Writeln
  359. Revision 1.40 2002/12/21 23:21:47 mazen
  360. + added support for the shift nodes
  361. + added debug output on screen with -an command line option
  362. Revision 1.39 2002/08/23 16:14:49 peter
  363. * tempgen cleanup
  364. * tt_noreuse temp type added that will be used in genentrycode
  365. Revision 1.38 2002/08/20 16:55:38 peter
  366. * don't write (stabs)line info when inlining a procedure
  367. Revision 1.37 2002/08/19 19:36:44 peter
  368. * More fixes for cross unit inlining, all tnodes are now implemented
  369. * Moved pocall_internconst to po_internconst because it is not a
  370. calling type at all and it conflicted when inlining of these small
  371. functions was requested
  372. Revision 1.36 2002/08/18 20:06:24 peter
  373. * inlining is now also allowed in interface
  374. * renamed write/load to ppuwrite/ppuload
  375. * tnode storing in ppu
  376. * nld,ncon,nbas are already updated for storing in ppu
  377. Revision 1.35 2002/08/17 09:23:38 florian
  378. * first part of procinfo rewrite
  379. Revision 1.34 2002/08/15 19:10:35 peter
  380. * first things tai,tnode storing in ppu
  381. Revision 1.33 2002/07/30 20:50:44 florian
  382. * the code generator knows now if parameters are in registers
  383. Revision 1.32 2002/07/19 11:41:36 daniel
  384. * State tracker work
  385. * The whilen and repeatn are now completely unified into whilerepeatn. This
  386. allows the state tracker to change while nodes automatically into
  387. repeat nodes.
  388. * Resulttypepass improvements to the notn. 'not not a' is optimized away and
  389. 'not(a>b)' is optimized into 'a<=b'.
  390. * Resulttypepass improvements to the whilerepeatn. 'while not a' is optimized
  391. by removing the notn and later switchting the true and falselabels. The
  392. same is done with 'repeat until not a'.
  393. Revision 1.31 2002/07/01 18:46:25 peter
  394. * internal linker
  395. * reorganized aasm layer
  396. Revision 1.30 2002/05/18 13:34:11 peter
  397. * readded missing revisions
  398. Revision 1.29 2002/05/16 19:46:42 carl
  399. + defines.inc -> fpcdefs.inc to avoid conflicts if compiling by hand
  400. + try to fix temp allocation (still in ifdef)
  401. + generic constructor calls
  402. + start of tassembler / tmodulebase class cleanup
  403. Revision 1.27 2002/05/12 16:53:08 peter
  404. * moved entry and exitcode to ncgutil and cgobj
  405. * foreach gets extra argument for passing local data to the
  406. iterator function
  407. * -CR checks also class typecasts at runtime by changing them
  408. into as
  409. * fixed compiler to cycle with the -CR option
  410. * fixed stabs with elf writer, finally the global variables can
  411. be watched
  412. * removed a lot of routines from cga unit and replaced them by
  413. calls to cgobj
  414. * u32bit-s32bit updates for and,or,xor nodes. When one element is
  415. u32bit then the other is typecasted also to u32bit without giving
  416. a rangecheck warning/error.
  417. * fixed pascal calling method with reversing also the high tree in
  418. the parast, detected by tcalcst3 test
  419. Revision 1.26 2002/04/21 19:02:04 peter
  420. * removed newn and disposen nodes, the code is now directly
  421. inlined from pexpr
  422. * -an option that will write the secondpass nodes to the .s file, this
  423. requires EXTDEBUG define to actually write the info
  424. * fixed various internal errors and crashes due recent code changes
  425. Revision 1.25 2002/04/20 21:32:24 carl
  426. + generic FPC_CHECKPOINTER
  427. + first parameter offset in stack now portable
  428. * rename some constants
  429. + move some cpu stuff to other units
  430. - remove unused constents
  431. * fix stacksize for some targets
  432. * fix generic size problems which depend now on EXTEND_SIZE constant
  433. Revision 1.24 2002/04/07 13:30:13 carl
  434. - removed unused variable
  435. Revision 1.23 2002/04/02 17:11:29 peter
  436. * tlocation,treference update
  437. * LOC_CONSTANT added for better constant handling
  438. * secondadd splitted in multiple routines
  439. * location_force_reg added for loading a location to a register
  440. of a specified size
  441. * secondassignment parses now first the right and then the left node
  442. (this is compatible with Kylix). This saves a lot of push/pop especially
  443. with string operations
  444. * adapted some routines to use the new cg methods
  445. Revision 1.22 2002/03/31 20:26:35 jonas
  446. + a_loadfpu_* and a_loadmm_* methods in tcg
  447. * register allocation is now handled by a class and is mostly processor
  448. independent (+rgobj.pas and i386/rgcpu.pas)
  449. * temp allocation is now handled by a class (+tgobj.pas, -i386\tgcpu.pas)
  450. * some small improvements and fixes to the optimizer
  451. * some register allocation fixes
  452. * some fpuvaroffset fixes in the unary minus node
  453. * push/popusedregisters is now called rg.save/restoreusedregisters and
  454. (for i386) uses temps instead of push/pop's when using -Op3 (that code is
  455. also better optimizable)
  456. * fixed and optimized register saving/restoring for new/dispose nodes
  457. * LOC_FPU locations now also require their "register" field to be set to
  458. R_ST, not R_ST0 (the latter is used for LOC_CFPUREGISTER locations only)
  459. - list field removed of the tnode class because it's not used currently
  460. and can cause hard-to-find bugs
  461. }