pass_2.pas 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554
  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. 'loadparentfpn'
  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_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. {$endif EXTDEBUG}
  151. begin
  152. if not assigned(p) then
  153. internalerror(200208221);
  154. if not(nf_error in p.flags) then
  155. begin
  156. oldcodegenerror:=codegenerror;
  157. oldlocalswitches:=aktlocalswitches;
  158. oldpos:=aktfilepos;
  159. {$ifdef TEMPREGDEBUG}
  160. testregisters32;
  161. prevp:=curptree;
  162. curptree:=@p;
  163. p^.usableregs:=usablereg32;
  164. {$endif TEMPREGDEBUG}
  165. aktfilepos:=p.fileinfo;
  166. aktlocalswitches:=p.localswitches;
  167. codegenerror:=false;
  168. {$ifdef EXTDEBUG}
  169. if (p.expectloc=LOC_INVALID) then
  170. Comment(V_Warning,'ExpectLoc is not set before secondpass: '+nodetype2str[p.nodetype]);
  171. if (not allow_multi_pass2) and
  172. (p.location.loc<>LOC_INVALID) then
  173. Comment(V_Warning,'Location.Loc is already set before secondpass: '+nodetype2str[p.nodetype]);
  174. if (cs_asm_nodes in aktglobalswitches) then
  175. logsecond(p.nodetype,true);
  176. {$endif EXTDEBUG}
  177. p.pass_2;
  178. {$ifdef EXTDEBUG}
  179. if (cs_asm_nodes in aktglobalswitches) then
  180. logsecond(p.nodetype,false);
  181. if (not codegenerror) then
  182. begin
  183. if (p.location.loc=LOC_INVALID) then
  184. Comment(V_Warning,'Location not set in secondpass: '+nodetype2str[p.nodetype])
  185. else if (p.location.loc<>p.expectloc) then
  186. Comment(V_Warning,'Location is different in secondpass: '+nodetype2str[p.nodetype]);
  187. end;
  188. {$ifdef i386}
  189. if rg.unusedregsint*([first_int_supreg..last_int_supreg] - [RS_FRAME_POINTER_REG,RS_STACK_POINTER_REG])<>
  190. ([first_int_supreg..last_int_supreg] - [RS_FRAME_POINTER_REG,RS_STACK_POINTER_REG]) then
  191. internalerror(200306171);
  192. {$endif i386}
  193. {$endif EXTDEBUG}
  194. if codegenerror then
  195. include(p.flags,nf_error);
  196. codegenerror:=codegenerror or oldcodegenerror;
  197. aktlocalswitches:=oldlocalswitches;
  198. aktfilepos:=oldpos;
  199. {$ifdef TEMPREGDEBUG}
  200. curptree:=prevp;
  201. {$endif TEMPREGDEBUG}
  202. {$ifdef EXTTEMPREGDEBUG}
  203. if p.usableregs-usablereg32>p.reallyusedregs then
  204. p.reallyusedregs:=p.usableregs-usablereg32;
  205. if p.reallyusedregs<p.registers32 then
  206. Comment(V_Debug,'registers32 overestimated '+tostr(p^.registers32)+
  207. '>'+tostr(p^.reallyusedregs));
  208. {$endif EXTTEMPREGDEBUG}
  209. end
  210. else
  211. codegenerror:=true;
  212. end;
  213. function do_secondpass(var p : tnode) : boolean;
  214. begin
  215. codegenerror:=false;
  216. if not(nf_error in p.flags) then
  217. secondpass(p);
  218. do_secondpass:=codegenerror;
  219. end;
  220. procedure clearrefs(p : tnamedindexitem;arg:pointer);
  221. begin
  222. if (tsym(p).typ=varsym) then
  223. if tvarsym(p).refs>1 then
  224. tvarsym(p).refs:=1;
  225. end;
  226. procedure generatecode(var p : tnode);
  227. {$ifdef EXTDEBUG}
  228. var
  229. sr : tsuperregister;
  230. {$endif EXTDEBUG}
  231. begin
  232. flowcontrol:=[];
  233. { when size optimization only count occurrence }
  234. if cs_littlesize in aktglobalswitches then
  235. rg.t_times:=1
  236. else
  237. { reference for repetition is 100 }
  238. rg.t_times:=100;
  239. { clear register count }
  240. rg.clearregistercount;
  241. symtablestack.foreach_static({$ifdef FPCPROCVAR}@{$endif}clearrefs,nil);
  242. symtablestack.next.foreach_static({$ifdef FPCPROCVAR}@{$endif}clearrefs,nil);
  243. { firstpass everything }
  244. do_firstpass(p);
  245. { only do secondpass if there are no errors }
  246. if ErrorCount=0 then
  247. begin
  248. { assign parameter locations }
  249. current_procinfo.after_pass1;
  250. { caller paraloc info is also necessary in the stackframe_entry }
  251. { code of the ppc (and possibly other processors) }
  252. if not current_procinfo.procdef.has_paraloc_info then
  253. begin
  254. paramanager.create_paraloc_info(current_procinfo.procdef,callerside);
  255. current_procinfo.procdef.has_paraloc_info:=true;
  256. end;
  257. { process register variable stuff (JM) }
  258. assign_regvars(p);
  259. // load_regvars(current_procinfo.aktentrycode,p);
  260. { for the i386 it must be done in genexitcode because it has }
  261. { to add 'fstp' instructions when using fpu regvars and those }
  262. { must come after the "exitlabel" (JM) }
  263. {$ifndef i386}
  264. // cleanup_regvars(current_procinfo.aktexitcode);
  265. {$endif i386}
  266. current_procinfo.allocate_framepointer_reg;
  267. do_secondpass(p);
  268. {$ifdef EXTDEBUG}
  269. for sr:=first_int_imreg to last_int_imreg do
  270. if not(sr in rg.unusedregsint) then
  271. Comment(V_Warning,'Register '+std_regname(newreg(R_INTREGISTER,sr,R_SUBWHOLE))+' not released');
  272. {$endif EXTDEBUG}
  273. if assigned(current_procinfo.procdef) then
  274. current_procinfo.procdef.fpu_used:=p.registersfpu;
  275. end;
  276. current_procinfo.aktproccode.concatlist(exprasmlist);
  277. end;
  278. end.
  279. {
  280. $Log$
  281. Revision 1.65 2003-09-28 17:55:04 peter
  282. * parent framepointer changed to hidden parameter
  283. * tloadparentfpnode added
  284. Revision 1.64 2003/09/03 15:55:01 peter
  285. * NEWRA branch merged
  286. Revision 1.63.2.3 2003/08/31 15:46:26 peter
  287. * more updates for tregister
  288. Revision 1.63.2.2 2003/08/31 13:50:16 daniel
  289. * Remove sorting and use pregenerated indexes
  290. * Some work on making things compile
  291. Revision 1.63.2.1 2003/08/29 17:28:59 peter
  292. * next batch of updates
  293. Revision 1.63 2003/08/11 21:18:20 peter
  294. * start of sparc support for newra
  295. Revision 1.62 2003/08/10 17:25:23 peter
  296. * fixed some reported bugs
  297. Revision 1.61 2003/07/06 17:58:22 peter
  298. * framepointer fixes for sparc
  299. * parent framepointer code more generic
  300. Revision 1.60 2003/07/06 15:31:20 daniel
  301. * Fixed register allocator. *Lots* of fixes.
  302. Revision 1.59 2003/07/06 10:18:47 jonas
  303. * also generate the caller paraloc info of a procedure if it doesn't exist
  304. yet at the start of pass_2
  305. Revision 1.58 2003/07/05 20:13:03 jonas
  306. * create_paraloc_info() is now called separately for the caller and
  307. callee info
  308. * fixed ppc cycle
  309. Revision 1.57 2003/06/13 21:19:30 peter
  310. * current_procdef removed, use current_procinfo.procdef instead
  311. Revision 1.56 2003/06/12 16:43:07 peter
  312. * newra compiles for sparc
  313. Revision 1.55 2003/06/09 12:23:30 peter
  314. * init/final of procedure data splitted from genentrycode
  315. * use asmnode getposition to insert final at the correct position
  316. als for the implicit try...finally
  317. Revision 1.54 2003/06/03 13:01:59 daniel
  318. * Register allocator finished
  319. Revision 1.53 2003/05/26 21:17:17 peter
  320. * procinlinenode removed
  321. * aktexit2label removed, fast exit removed
  322. + tcallnode.inlined_pass_2 added
  323. Revision 1.52 2003/05/15 18:58:53 peter
  324. * removed selfpointer_offset, vmtpointer_offset
  325. * tvarsym.adjusted_address
  326. * address in localsymtable is now in the real direction
  327. * removed some obsolete globals
  328. Revision 1.51 2003/05/13 19:14:41 peter
  329. * failn removed
  330. * inherited result code check moven to pexpr
  331. Revision 1.50 2003/05/09 17:47:02 peter
  332. * self moved to hidden parameter
  333. * removed hdisposen,hnewn,selfn
  334. Revision 1.49 2003/04/27 11:21:33 peter
  335. * aktprocdef renamed to current_procinfo.procdef
  336. * procinfo renamed to current_procinfo
  337. * procinfo will now be stored in current_module so it can be
  338. cleaned up properly
  339. * gen_main_procsym changed to create_main_proc and release_main_proc
  340. to also generate a tprocinfo structure
  341. * fixed unit implicit initfinal
  342. Revision 1.48 2003/04/27 07:29:50 peter
  343. * current_procinfo.procdef cleanup, current_procdef is now always nil when parsing
  344. a new procdef declaration
  345. * aktprocsym removed
  346. * lexlevel removed, use symtable.symtablelevel instead
  347. * implicit init/final code uses the normal genentry/genexit
  348. * funcret state checking updated for new funcret handling
  349. Revision 1.47 2003/04/25 20:59:33 peter
  350. * removed funcretn,funcretsym, function result is now in varsym
  351. and aliases for result and function name are added using absolutesym
  352. * vs_hidden parameter for funcret passed in parameter
  353. * vs_hidden fixes
  354. * writenode changed to printnode and released from extdebug
  355. * -vp option added to generate a tree.log with the nodetree
  356. * nicer printnode for statements, callnode
  357. Revision 1.46 2003/04/23 10:12:14 peter
  358. * allow multi pass2 changed to global boolean instead of node flag
  359. Revision 1.45 2003/04/22 23:50:23 peter
  360. * firstpass uses expectloc
  361. * checks if there are differences between the expectloc and
  362. location.loc from secondpass in EXTDEBUG
  363. Revision 1.44 2003/04/22 12:45:58 florian
  364. * fixed generic in operator code
  365. + added debug code to check if all scratch registers are released
  366. Revision 1.43 2003/01/09 20:40:59 daniel
  367. * Converted some code in cgx86.pas to new register numbering
  368. Revision 1.42 2003/01/09 15:49:56 daniel
  369. * Added register conversion
  370. Revision 1.41 2002/12/22 14:35:39 peter
  371. * removed Writeln
  372. Revision 1.40 2002/12/21 23:21:47 mazen
  373. + added support for the shift nodes
  374. + added debug output on screen with -an command line option
  375. Revision 1.39 2002/08/23 16:14:49 peter
  376. * tempgen cleanup
  377. * tt_noreuse temp type added that will be used in genentrycode
  378. Revision 1.38 2002/08/20 16:55:38 peter
  379. * don't write (stabs)line info when inlining a procedure
  380. Revision 1.37 2002/08/19 19:36:44 peter
  381. * More fixes for cross unit inlining, all tnodes are now implemented
  382. * Moved pocall_internconst to po_internconst because it is not a
  383. calling type at all and it conflicted when inlining of these small
  384. functions was requested
  385. Revision 1.36 2002/08/18 20:06:24 peter
  386. * inlining is now also allowed in interface
  387. * renamed write/load to ppuwrite/ppuload
  388. * tnode storing in ppu
  389. * nld,ncon,nbas are already updated for storing in ppu
  390. Revision 1.35 2002/08/17 09:23:38 florian
  391. * first part of procinfo rewrite
  392. Revision 1.34 2002/08/15 19:10:35 peter
  393. * first things tai,tnode storing in ppu
  394. Revision 1.33 2002/07/30 20:50:44 florian
  395. * the code generator knows now if parameters are in registers
  396. Revision 1.32 2002/07/19 11:41:36 daniel
  397. * State tracker work
  398. * The whilen and repeatn are now completely unified into whilerepeatn. This
  399. allows the state tracker to change while nodes automatically into
  400. repeat nodes.
  401. * Resulttypepass improvements to the notn. 'not not a' is optimized away and
  402. 'not(a>b)' is optimized into 'a<=b'.
  403. * Resulttypepass improvements to the whilerepeatn. 'while not a' is optimized
  404. by removing the notn and later switchting the true and falselabels. The
  405. same is done with 'repeat until not a'.
  406. Revision 1.31 2002/07/01 18:46:25 peter
  407. * internal linker
  408. * reorganized aasm layer
  409. Revision 1.30 2002/05/18 13:34:11 peter
  410. * readded missing revisions
  411. Revision 1.29 2002/05/16 19:46:42 carl
  412. + defines.inc -> fpcdefs.inc to avoid conflicts if compiling by hand
  413. + try to fix temp allocation (still in ifdef)
  414. + generic constructor calls
  415. + start of tassembler / tmodulebase class cleanup
  416. Revision 1.27 2002/05/12 16:53:08 peter
  417. * moved entry and exitcode to ncgutil and cgobj
  418. * foreach gets extra argument for passing local data to the
  419. iterator function
  420. * -CR checks also class typecasts at runtime by changing them
  421. into as
  422. * fixed compiler to cycle with the -CR option
  423. * fixed stabs with elf writer, finally the global variables can
  424. be watched
  425. * removed a lot of routines from cga unit and replaced them by
  426. calls to cgobj
  427. * u32bit-s32bit updates for and,or,xor nodes. When one element is
  428. u32bit then the other is typecasted also to u32bit without giving
  429. a rangecheck warning/error.
  430. * fixed pascal calling method with reversing also the high tree in
  431. the parast, detected by tcalcst3 test
  432. Revision 1.26 2002/04/21 19:02:04 peter
  433. * removed newn and disposen nodes, the code is now directly
  434. inlined from pexpr
  435. * -an option that will write the secondpass nodes to the .s file, this
  436. requires EXTDEBUG define to actually write the info
  437. * fixed various internal errors and crashes due recent code changes
  438. Revision 1.25 2002/04/20 21:32:24 carl
  439. + generic FPC_CHECKPOINTER
  440. + first parameter offset in stack now portable
  441. * rename some constants
  442. + move some cpu stuff to other units
  443. - remove unused constents
  444. * fix stacksize for some targets
  445. * fix generic size problems which depend now on EXTEND_SIZE constant
  446. Revision 1.24 2002/04/07 13:30:13 carl
  447. - removed unused variable
  448. Revision 1.23 2002/04/02 17:11:29 peter
  449. * tlocation,treference update
  450. * LOC_CONSTANT added for better constant handling
  451. * secondadd splitted in multiple routines
  452. * location_force_reg added for loading a location to a register
  453. of a specified size
  454. * secondassignment parses now first the right and then the left node
  455. (this is compatible with Kylix). This saves a lot of push/pop especially
  456. with string operations
  457. * adapted some routines to use the new cg methods
  458. Revision 1.22 2002/03/31 20:26:35 jonas
  459. + a_loadfpu_* and a_loadmm_* methods in tcg
  460. * register allocation is now handled by a class and is mostly processor
  461. independent (+rgobj.pas and i386/rgcpu.pas)
  462. * temp allocation is now handled by a class (+tgobj.pas, -i386\tgcpu.pas)
  463. * some small improvements and fixes to the optimizer
  464. * some register allocation fixes
  465. * some fpuvaroffset fixes in the unary minus node
  466. * push/popusedregisters is now called rg.save/restoreusedregisters and
  467. (for i386) uses temps instead of push/pop's when using -Op3 (that code is
  468. also better optimizable)
  469. * fixed and optimized register saving/restoring for new/dispose nodes
  470. * LOC_FPU locations now also require their "register" field to be set to
  471. R_ST, not R_ST0 (the latter is used for LOC_CFPUREGISTER locations only)
  472. - list field removed of the tnode class because it's not used currently
  473. and can cause hard-to-find bugs
  474. }