nflw.pas 52 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743
  1. {
  2. $Id$
  3. Copyright (c) 1998-2002 by Florian Klaempfl
  4. Type checking and register allocation for nodes that influence
  5. the flow
  6. This program is free software; you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation; either version 2 of the License, or
  9. (at your option) any later version.
  10. This program is distributed in the hope that it will be useful,
  11. but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. GNU General Public License for more details.
  14. You should have received a copy of the GNU General Public License
  15. along with this program; if not, write to the Free Software
  16. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  17. ****************************************************************************
  18. }
  19. unit nflw;
  20. {$i fpcdefs.inc}
  21. interface
  22. uses
  23. node,cpubase,
  24. aasmbase,aasmtai,aasmcpu,symnot,
  25. symtype,symbase,symdef,symsym;
  26. type
  27. { flags used by loop nodes }
  28. tloopflag = (
  29. { set if it is a for ... downto ... do loop }
  30. lnf_backward,
  31. { Do we need to parse childs to set var state? }
  32. lnf_varstate,
  33. { Do a test at the begin of the loop?}
  34. lnf_testatbegin,
  35. { Negate the loop test? }
  36. lnf_checknegate,
  37. { Should the value of the loop variable on exit be correct. }
  38. lnf_dont_mind_loopvar_on_exit);
  39. tloopflags = set of tloopflag;
  40. const
  41. { loop flags which must match to consider loop nodes equal regarding the flags }
  42. loopflagsequal = [lnf_backward];
  43. type
  44. tloopnode = class(tbinarynode)
  45. t1,t2 : tnode;
  46. loopflags : tloopflags;
  47. constructor create(tt : tnodetype;l,r,_t1,_t2 : tnode);virtual;
  48. destructor destroy;override;
  49. function getcopy : tnode;override;
  50. constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;
  51. procedure ppuwrite(ppufile:tcompilerppufile);override;
  52. procedure buildderefimpl;override;
  53. procedure derefimpl;override;
  54. procedure insertintolist(l : tnodelist);override;
  55. procedure printnodetree(var t:text);override;
  56. function docompare(p: tnode): boolean; override;
  57. end;
  58. twhilerepeatnode = class(tloopnode)
  59. constructor create(l,r,_t1:Tnode;tab,cn:boolean);virtual;
  60. function det_resulttype:tnode;override;
  61. function pass_1 : tnode;override;
  62. {$ifdef state_tracking}
  63. function track_state_pass(exec_known:boolean):boolean;override;
  64. {$endif}
  65. end;
  66. twhilerepeatnodeclass = class of twhilerepeatnode;
  67. tifnode = class(tloopnode)
  68. constructor create(l,r,_t1 : tnode);virtual;
  69. function det_resulttype:tnode;override;
  70. function pass_1 : tnode;override;
  71. end;
  72. tifnodeclass = class of tifnode;
  73. tfornode = class(tloopnode)
  74. loopvar_notid:cardinal;
  75. constructor create(l,r,_t1,_t2 : tnode;back : boolean);virtual;
  76. procedure loop_var_access(not_type:Tnotification_flag;symbol:Tsym);
  77. function det_resulttype:tnode;override;
  78. function pass_1 : tnode;override;
  79. end;
  80. tfornodeclass = class of tfornode;
  81. texitnode = class(tunarynode)
  82. constructor create(l:tnode);virtual;
  83. constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;
  84. procedure ppuwrite(ppufile:tcompilerppufile);override;
  85. function det_resulttype:tnode;override;
  86. function pass_1 : tnode;override;
  87. end;
  88. texitnodeclass = class of texitnode;
  89. tbreaknode = class(tnode)
  90. constructor create;virtual;
  91. function det_resulttype:tnode;override;
  92. function pass_1 : tnode;override;
  93. end;
  94. tbreaknodeclass = class of tbreaknode;
  95. tcontinuenode = class(tnode)
  96. constructor create;virtual;
  97. function det_resulttype:tnode;override;
  98. function pass_1 : tnode;override;
  99. end;
  100. tcontinuenodeclass = class of tcontinuenode;
  101. tgotonode = class(tnode)
  102. labsym : tlabelsym;
  103. labsymderef : tderef;
  104. exceptionblock : integer;
  105. { internlab : tinterngotolabel;}
  106. constructor create(p : tlabelsym);virtual;
  107. { constructor createintern(g:tinterngotolabel);}
  108. constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;
  109. procedure ppuwrite(ppufile:tcompilerppufile);override;
  110. procedure buildderefimpl;override;
  111. procedure derefimpl;override;
  112. function getcopy : tnode;override;
  113. function det_resulttype:tnode;override;
  114. function pass_1 : tnode;override;
  115. function docompare(p: tnode): boolean; override;
  116. end;
  117. tgotonodeclass = class of tgotonode;
  118. tlabelnode = class(tunarynode)
  119. labelnr : tasmlabel;
  120. labsym : tlabelsym;
  121. labsymderef : tderef;
  122. exceptionblock : integer;
  123. constructor createcase(p : tasmlabel;l:tnode);virtual;
  124. constructor create(p : tlabelsym;l:tnode);virtual;
  125. constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;
  126. procedure ppuwrite(ppufile:tcompilerppufile);override;
  127. procedure buildderefimpl;override;
  128. procedure derefimpl;override;
  129. function getcopy : tnode;override;
  130. function det_resulttype:tnode;override;
  131. function pass_1 : tnode;override;
  132. function docompare(p: tnode): boolean; override;
  133. end;
  134. tlabelnodeclass = class of tlabelnode;
  135. traisenode = class(tbinarynode)
  136. frametree : tnode;
  137. constructor create(l,taddr,tframe:tnode);virtual;
  138. constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;
  139. procedure ppuwrite(ppufile:tcompilerppufile);override;
  140. procedure buildderefimpl;override;
  141. procedure derefimpl;override;
  142. function getcopy : tnode;override;
  143. procedure insertintolist(l : tnodelist);override;
  144. function det_resulttype:tnode;override;
  145. function pass_1 : tnode;override;
  146. function docompare(p: tnode): boolean; override;
  147. end;
  148. traisenodeclass = class of traisenode;
  149. ttryexceptnode = class(tloopnode)
  150. constructor create(l,r,_t1 : tnode);virtual;
  151. function det_resulttype:tnode;override;
  152. function pass_1 : tnode;override;
  153. end;
  154. ttryexceptnodeclass = class of ttryexceptnode;
  155. ttryfinallynode = class(tloopnode)
  156. implicitframe : boolean;
  157. constructor create(l,r:tnode);virtual;
  158. constructor create_implicit(l,r,_t1:tnode);virtual;
  159. function det_resulttype:tnode;override;
  160. function pass_1 : tnode;override;
  161. end;
  162. ttryfinallynodeclass = class of ttryfinallynode;
  163. tonnode = class(tbinarynode)
  164. exceptsymtable : tsymtable;
  165. excepttype : tobjectdef;
  166. constructor create(l,r:tnode);virtual;
  167. destructor destroy;override;
  168. constructor ppuload(t:tnodetype;ppufile:tcompilerppufile);override;
  169. function det_resulttype:tnode;override;
  170. function pass_1 : tnode;override;
  171. function getcopy : tnode;override;
  172. function docompare(p: tnode): boolean; override;
  173. end;
  174. tonnodeclass = class of tonnode;
  175. { for compatibilty }
  176. function genloopnode(t : tnodetype;l,r,n1 : tnode;back : boolean) : tnode;
  177. var
  178. cwhilerepeatnode : twhilerepeatnodeclass;
  179. cifnode : tifnodeclass;
  180. cfornode : tfornodeclass;
  181. cexitnode : texitnodeclass;
  182. cbreaknode : tbreaknodeclass;
  183. ccontinuenode : tcontinuenodeclass;
  184. cgotonode : tgotonodeclass;
  185. clabelnode : tlabelnodeclass;
  186. craisenode : traisenodeclass;
  187. ctryexceptnode : ttryexceptnodeclass;
  188. ctryfinallynode : ttryfinallynodeclass;
  189. connode : tonnodeclass;
  190. implementation
  191. uses
  192. globtype,systems,
  193. cutils,verbose,globals,
  194. symconst,paramgr,defcmp,defutil,htypechk,pass_1,
  195. ncal,nadd,ncon,nmem,nld,ncnv,nbas,cgobj,
  196. {$ifdef state_tracking}
  197. nstate,
  198. {$endif}
  199. cgbase,procinfo
  200. ;
  201. function genloopnode(t : tnodetype;l,r,n1 : tnode;back : boolean) : tnode;
  202. var
  203. p : tnode;
  204. begin
  205. case t of
  206. ifn:
  207. p:=cifnode.create(l,r,n1);
  208. whilerepeatn:
  209. if back then
  210. {Repeat until.}
  211. p:=cwhilerepeatnode.create(l,r,n1,false,true)
  212. else
  213. {While do.}
  214. p:=cwhilerepeatnode.create(l,r,n1,true,false);
  215. forn:
  216. p:=cfornode.create(l,r,n1,nil,back);
  217. end;
  218. resulttypepass(p);
  219. genloopnode:=p;
  220. end;
  221. {****************************************************************************
  222. TLOOPNODE
  223. *****************************************************************************}
  224. constructor tloopnode.create(tt : tnodetype;l,r,_t1,_t2 : tnode);
  225. begin
  226. inherited create(tt,l,r);
  227. t1:=_t1;
  228. t2:=_t2;
  229. set_file_line(l);
  230. end;
  231. destructor tloopnode.destroy;
  232. begin
  233. t1.free;
  234. t2.free;
  235. inherited destroy;
  236. end;
  237. constructor tloopnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);
  238. begin
  239. inherited ppuload(t,ppufile);
  240. t1:=ppuloadnode(ppufile);
  241. t2:=ppuloadnode(ppufile);
  242. end;
  243. procedure tloopnode.ppuwrite(ppufile:tcompilerppufile);
  244. begin
  245. inherited ppuwrite(ppufile);
  246. ppuwritenode(ppufile,t1);
  247. ppuwritenode(ppufile,t2);
  248. end;
  249. procedure tloopnode.buildderefimpl;
  250. begin
  251. inherited buildderefimpl;
  252. if assigned(t1) then
  253. t1.buildderefimpl;
  254. if assigned(t2) then
  255. t2.buildderefimpl;
  256. end;
  257. procedure tloopnode.derefimpl;
  258. begin
  259. inherited derefimpl;
  260. if assigned(t1) then
  261. t1.derefimpl;
  262. if assigned(t2) then
  263. t2.derefimpl;
  264. end;
  265. function tloopnode.getcopy : tnode;
  266. var
  267. p : tloopnode;
  268. begin
  269. p:=tloopnode(inherited getcopy);
  270. if assigned(t1) then
  271. p.t1:=t1.getcopy
  272. else
  273. p.t1:=nil;
  274. if assigned(t2) then
  275. p.t2:=t2.getcopy
  276. else
  277. p.t2:=nil;
  278. p.loopflags:=loopflags;
  279. getcopy:=p;
  280. end;
  281. procedure tloopnode.insertintolist(l : tnodelist);
  282. begin
  283. end;
  284. procedure tloopnode.printnodetree(var t:text);
  285. begin
  286. write(t,printnodeindention,'(');
  287. printnodeindent;
  288. printnodeinfo(t);
  289. printnode(t,left);
  290. printnode(t,right);
  291. printnode(t,t1);
  292. printnode(t,t2);
  293. printnodeunindent;
  294. writeln(t,printnodeindention,')');
  295. end;
  296. function tloopnode.docompare(p: tnode): boolean;
  297. begin
  298. docompare :=
  299. inherited docompare(p) and
  300. (loopflags*loopflagsequal=tloopnode(p).loopflags*loopflagsequal) and
  301. t1.isequal(tloopnode(p).t1) and
  302. t2.isequal(tloopnode(p).t2);
  303. end;
  304. {****************************************************************************
  305. TWHILEREPEATNODE
  306. *****************************************************************************}
  307. constructor Twhilerepeatnode.create(l,r,_t1:Tnode;tab,cn:boolean);
  308. begin
  309. inherited create(whilerepeatn,l,r,_t1,nil);
  310. if tab then
  311. include(loopflags, lnf_testatbegin);
  312. if cn then
  313. include(loopflags,lnf_checknegate);
  314. end;
  315. function twhilerepeatnode.det_resulttype:tnode;
  316. var
  317. t:Tunarynode;
  318. begin
  319. result:=nil;
  320. resulttype:=voidtype;
  321. resulttypepass(left);
  322. {A not node can be removed.}
  323. if left.nodetype=notn then
  324. begin
  325. t:=Tunarynode(left);
  326. left:=Tunarynode(left).left;
  327. t.left:=nil;
  328. t.destroy;
  329. {$ifdef Delphi}
  330. { How can this be handled in Delphi ? }
  331. RunError(255);
  332. {$else}
  333. {Symdif operator, in case you are wondering:}
  334. loopflags:=loopflags >< [lnf_checknegate];
  335. {$endif}
  336. end;
  337. { loop instruction }
  338. if assigned(right) then
  339. resulttypepass(right);
  340. set_varstate(left,vs_used,true);
  341. if codegenerror then
  342. exit;
  343. if not is_boolean(left.resulttype.def) then
  344. begin
  345. CGMessage(type_e_mismatch);
  346. exit;
  347. end;
  348. end;
  349. function twhilerepeatnode.pass_1 : tnode;
  350. var
  351. old_t_times : longint;
  352. begin
  353. result:=nil;
  354. expectloc:=LOC_VOID;
  355. old_t_times:=cg.t_times;
  356. { calc register weight }
  357. if not(cs_littlesize in aktglobalswitches ) then
  358. cg.t_times:=cg.t_times*8;
  359. firstpass(left);
  360. if codegenerror then
  361. exit;
  362. registers32:=left.registers32;
  363. registersfpu:=left.registersfpu;
  364. {$ifdef SUPPORT_MMX}
  365. registersmmx:=left.registersmmx;
  366. {$endif SUPPORT_MMX}
  367. { loop instruction }
  368. if assigned(right) then
  369. begin
  370. firstpass(right);
  371. if codegenerror then
  372. exit;
  373. if registers32<right.registers32 then
  374. registers32:=right.registers32;
  375. if registersfpu<right.registersfpu then
  376. registersfpu:=right.registersfpu;
  377. {$ifdef SUPPORT_MMX}
  378. if registersmmx<right.registersmmx then
  379. registersmmx:=right.registersmmx;
  380. {$endif SUPPORT_MMX}
  381. end;
  382. cg.t_times:=old_t_times;
  383. end;
  384. {$ifdef state_tracking}
  385. function Twhilerepeatnode.track_state_pass(exec_known:boolean):boolean;
  386. var condition:Tnode;
  387. code:Tnode;
  388. done:boolean;
  389. value:boolean;
  390. change:boolean;
  391. firsttest:boolean;
  392. factval:Tnode;
  393. begin
  394. track_state_pass:=false;
  395. done:=false;
  396. firsttest:=true;
  397. {For repeat until statements, first do a pass through the code.}
  398. if not(lnf_testatbegin in flags) then
  399. begin
  400. code:=right.getcopy;
  401. if code.track_state_pass(exec_known) then
  402. track_state_pass:=true;
  403. code.destroy;
  404. end;
  405. repeat
  406. condition:=left.getcopy;
  407. code:=right.getcopy;
  408. change:=condition.track_state_pass(exec_known);
  409. factval:=aktstate.find_fact(left);
  410. if factval<>nil then
  411. begin
  412. condition.destroy;
  413. condition:=factval.getcopy;
  414. change:=true;
  415. end;
  416. if change then
  417. begin
  418. track_state_pass:=true;
  419. {Force new resulttype pass.}
  420. condition.resulttype.def:=nil;
  421. do_resulttypepass(condition);
  422. end;
  423. if is_constboolnode(condition) then
  424. begin
  425. {Try to turn a while loop into a repeat loop.}
  426. if firsttest then
  427. exclude(flags,testatbegin);
  428. value:=(Tordconstnode(condition).value<>0) xor checknegate;
  429. if value then
  430. begin
  431. if code.track_state_pass(exec_known) then
  432. track_state_pass:=true;
  433. end
  434. else
  435. done:=true;
  436. end
  437. else
  438. begin
  439. {Remove any modified variables from the state.}
  440. code.track_state_pass(false);
  441. done:=true;
  442. end;
  443. code.destroy;
  444. condition.destroy;
  445. firsttest:=false;
  446. until done;
  447. {The loop condition is also known, for example:
  448. while i<10 do
  449. begin
  450. ...
  451. end;
  452. When the loop is done, we do know that i<10 = false.
  453. }
  454. condition:=left.getcopy;
  455. if condition.track_state_pass(exec_known) then
  456. begin
  457. track_state_pass:=true;
  458. {Force new resulttype pass.}
  459. condition.resulttype.def:=nil;
  460. do_resulttypepass(condition);
  461. end;
  462. if not is_constboolnode(condition) then
  463. aktstate.store_fact(condition,
  464. cordconstnode.create(byte(checknegate),booltype,true))
  465. else
  466. condition.destroy;
  467. end;
  468. {$endif}
  469. {*****************************************************************************
  470. TIFNODE
  471. *****************************************************************************}
  472. constructor tifnode.create(l,r,_t1 : tnode);
  473. begin
  474. inherited create(ifn,l,r,_t1,nil);
  475. end;
  476. function tifnode.det_resulttype:tnode;
  477. begin
  478. result:=nil;
  479. resulttype:=voidtype;
  480. resulttypepass(left);
  481. { if path }
  482. if assigned(right) then
  483. resulttypepass(right);
  484. { else path }
  485. if assigned(t1) then
  486. resulttypepass(t1);
  487. set_varstate(left,vs_used,true);
  488. if codegenerror then
  489. exit;
  490. if not is_boolean(left.resulttype.def) then
  491. Message1(type_e_boolean_expr_expected,left.resulttype.def.typename);
  492. end;
  493. function tifnode.pass_1 : tnode;
  494. var
  495. old_t_times : longint;
  496. hp : tnode;
  497. begin
  498. result:=nil;
  499. expectloc:=LOC_VOID;
  500. old_t_times:=cg.t_times;
  501. firstpass(left);
  502. registers32:=left.registers32;
  503. registersfpu:=left.registersfpu;
  504. {$ifdef SUPPORT_MMX}
  505. registersmmx:=left.registersmmx;
  506. {$endif SUPPORT_MMX}
  507. { determines registers weigths }
  508. if not(cs_littlesize in aktglobalswitches) then
  509. cg.t_times:=cg.t_times div 2;
  510. if cg.t_times=0 then
  511. cg.t_times:=1;
  512. { if path }
  513. if assigned(right) then
  514. begin
  515. firstpass(right);
  516. if registers32<right.registers32 then
  517. registers32:=right.registers32;
  518. if registersfpu<right.registersfpu then
  519. registersfpu:=right.registersfpu;
  520. {$ifdef SUPPORT_MMX}
  521. if registersmmx<right.registersmmx then
  522. registersmmx:=right.registersmmx;
  523. {$endif SUPPORT_MMX}
  524. end;
  525. { else path }
  526. if assigned(t1) then
  527. begin
  528. firstpass(t1);
  529. if registers32<t1.registers32 then
  530. registers32:=t1.registers32;
  531. if registersfpu<t1.registersfpu then
  532. registersfpu:=t1.registersfpu;
  533. {$ifdef SUPPORT_MMX}
  534. if registersmmx<t1.registersmmx then
  535. registersmmx:=t1.registersmmx;
  536. {$endif SUPPORT_MMX}
  537. end;
  538. { leave if we've got an error in one of the paths }
  539. if codegenerror then
  540. exit;
  541. if left.nodetype=ordconstn then
  542. begin
  543. { optimize }
  544. if tordconstnode(left).value=1 then
  545. begin
  546. hp:=right;
  547. right:=nil;
  548. { we cannot set p to nil !!! }
  549. if assigned(hp) then
  550. result:=hp
  551. else
  552. result:=cnothingnode.create;
  553. end
  554. else
  555. begin
  556. hp:=t1;
  557. t1:=nil;
  558. { we cannot set p to nil !!! }
  559. if assigned(hp) then
  560. result:=hp
  561. else
  562. result:=cnothingnode.create;
  563. end;
  564. end;
  565. cg.t_times:=old_t_times;
  566. end;
  567. {*****************************************************************************
  568. TFORNODE
  569. *****************************************************************************}
  570. constructor tfornode.create(l,r,_t1,_t2 : tnode;back : boolean);
  571. begin
  572. inherited create(forn,l,r,_t1,_t2);
  573. if back then
  574. include(loopflags,lnf_backward);
  575. include(loopflags,lnf_testatbegin);
  576. end;
  577. procedure Tfornode.loop_var_access(not_type:Tnotification_flag;
  578. symbol:Tsym);
  579. begin
  580. {If there is a read access, the value of the loop counter is important;
  581. at the end of the loop the loop variable should contain the value it
  582. had in the last iteration.}
  583. if not_type=vn_onwrite then
  584. begin
  585. writeln('Loopvar does not matter on exit');
  586. end
  587. else
  588. begin
  589. exclude(loopflags,lnf_dont_mind_loopvar_on_exit);
  590. writeln('Loopvar does matter on exit');
  591. end;
  592. Tvarsym(symbol).unregister_notification(loopvar_notid);
  593. end;
  594. function tfornode.det_resulttype:tnode;
  595. var
  596. hp : tnode;
  597. begin
  598. result:=nil;
  599. resulttype:=voidtype;
  600. if left.nodetype<>assignn then
  601. begin
  602. CGMessage(cg_e_illegal_expression);
  603. exit;
  604. end;
  605. {Can we spare the first comparision?}
  606. if (right.nodetype=ordconstn) and (Tassignmentnode(left).right.nodetype=ordconstn) then
  607. if (
  608. (lnf_backward in loopflags) and
  609. (Tordconstnode(Tassignmentnode(left).right).value>=Tordconstnode(right).value)
  610. )
  611. or not(
  612. (lnf_backward in loopflags) and
  613. (Tordconstnode(Tassignmentnode(left).right).value<=Tordconstnode(right).value)
  614. ) then
  615. exclude(loopflags,lnf_testatbegin);
  616. { save counter var }
  617. t2:=tassignmentnode(left).left.getcopy;
  618. resulttypepass(left);
  619. set_varstate(left,vs_used,true);
  620. if assigned(t1) then
  621. begin
  622. resulttypepass(t1);
  623. if codegenerror then
  624. exit;
  625. end;
  626. { process count var }
  627. resulttypepass(t2);
  628. set_varstate(t2,vs_used,false);
  629. if codegenerror then
  630. exit;
  631. { Check count var, record fields are also allowed in tp7 }
  632. hp:=t2;
  633. while (hp.nodetype=subscriptn) or
  634. ((hp.nodetype=vecn) and
  635. is_constintnode(tvecnode(hp).right)) or
  636. ((hp.nodetype=typeconvn) and
  637. (ttypeconvnode(hp).convtype=tc_equal)) do
  638. hp:=tunarynode(hp).left;
  639. { we need a simple loadn, but the load must be in a global symtable or
  640. in the same level as the para of the current proc }
  641. if (
  642. (hp.nodetype=loadn) and
  643. (
  644. (tloadnode(hp).symtable.symtablelevel=main_program_level) or
  645. (tloadnode(hp).symtable.symtablelevel=current_procinfo.procdef.parast.symtablelevel)
  646. ) and
  647. not(
  648. (tloadnode(hp).symtableentry.typ=varsym) and
  649. ((tvarsym(tloadnode(hp).symtableentry).varspez in [vs_var,vs_out]) or
  650. (vo_is_thread_var in tvarsym(tloadnode(hp).symtableentry).varoptions))
  651. )
  652. ) then
  653. begin
  654. if (hp.nodetype=loadn) and
  655. (tloadnode(hp).symtableentry.typ=varsym) then
  656. tvarsym(tloadnode(hp).symtableentry).varstate:=vs_used;
  657. if not(is_ordinal(t2.resulttype.def))
  658. {$ifndef cpu64bit}
  659. or is_64bitint(t2.resulttype.def)
  660. {$endif cpu64bit}
  661. then
  662. CGMessagePos(hp.fileinfo,type_e_ordinal_expr_expected);
  663. end
  664. else
  665. CGMessagePos(hp.fileinfo,cg_e_illegal_count_var);
  666. resulttypepass(right);
  667. set_varstate(right,vs_used,true);
  668. inserttypeconv(right,t2.resulttype);
  669. end;
  670. function tfornode.pass_1 : tnode;
  671. var
  672. old_t_times : longint;
  673. {$ifdef loopvar_dont_mind}
  674. hp : Tnode;
  675. {$endif loopvar_dont_mind}
  676. begin
  677. result:=nil;
  678. expectloc:=LOC_VOID;
  679. { Calc register weight }
  680. old_t_times:=cg.t_times;
  681. if not(cs_littlesize in aktglobalswitches) then
  682. cg.t_times:=cg.t_times*8;
  683. firstpass(left);
  684. if assigned(t1) then
  685. begin
  686. firstpass(t1);
  687. if codegenerror then
  688. exit;
  689. end;
  690. registers32:=t1.registers32;
  691. registersfpu:=t1.registersfpu;
  692. {$ifdef SUPPORT_MMX}
  693. registersmmx:=left.registersmmx;
  694. {$endif SUPPORT_MMX}
  695. if left.registers32>registers32 then
  696. registers32:=left.registers32;
  697. if left.registersfpu>registersfpu then
  698. registersfpu:=left.registersfpu;
  699. {$ifdef SUPPORT_MMX}
  700. if left.registersmmx>registersmmx then
  701. registersmmx:=left.registersmmx;
  702. {$endif SUPPORT_MMX}
  703. { process count var }
  704. firstpass(t2);
  705. if codegenerror then
  706. exit;
  707. if t2.registers32>registers32 then
  708. registers32:=t2.registers32;
  709. if t2.registersfpu>registersfpu then
  710. registersfpu:=t2.registersfpu;
  711. {$ifdef SUPPORT_MMX}
  712. if t2.registersmmx>registersmmx then
  713. registersmmx:=t2.registersmmx;
  714. {$endif SUPPORT_MMX}
  715. firstpass(right);
  716. {$ifdef loopvar_dont_mind}
  717. { Check count var, record fields are also allowed in tp7 }
  718. include(loopflags,lnf_dont_mind_loopvar_on_exit);
  719. hp:=t2;
  720. while (hp.nodetype=subscriptn) or
  721. ((hp.nodetype=vecn) and
  722. is_constintnode(tvecnode(hp).right)) do
  723. hp:=tunarynode(hp).left;
  724. if (hp.nodetype=loadn) and (Tloadnode(hp).symtableentry.typ=varsym) then
  725. loopvar_notid:=Tvarsym(Tloadnode(hp).symtableentry).
  726. register_notification([vn_onread,vn_onwrite],@loop_var_access);
  727. {$endif}
  728. if right.registers32>registers32 then
  729. registers32:=right.registers32;
  730. if right.registersfpu>registersfpu then
  731. registersfpu:=right.registersfpu;
  732. {$ifdef SUPPORT_MMX}
  733. if right.registersmmx>registersmmx then
  734. registersmmx:=right.registersmmx;
  735. {$endif SUPPORT_MMX}
  736. { we need at least one register for comparisons PM }
  737. if registers32=0 then
  738. inc(registers32);
  739. cg.t_times:=old_t_times;
  740. end;
  741. {*****************************************************************************
  742. TEXITNODE
  743. *****************************************************************************}
  744. constructor texitnode.create(l:tnode);
  745. begin
  746. inherited create(exitn,l);
  747. end;
  748. constructor texitnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);
  749. begin
  750. inherited ppuload(t,ppufile);
  751. end;
  752. procedure texitnode.ppuwrite(ppufile:tcompilerppufile);
  753. begin
  754. inherited ppuwrite(ppufile);
  755. end;
  756. function texitnode.det_resulttype:tnode;
  757. begin
  758. result:=nil;
  759. if assigned(left) then
  760. begin
  761. { add assignment to funcretsym }
  762. inserttypeconv(left,current_procinfo.procdef.rettype);
  763. left:=cassignmentnode.create(
  764. cloadnode.create(current_procinfo.procdef.funcretsym,current_procinfo.procdef.funcretsym.owner),
  765. left);
  766. resulttypepass(left);
  767. set_varstate(left,vs_used,true);
  768. end;
  769. resulttype:=voidtype;
  770. end;
  771. function texitnode.pass_1 : tnode;
  772. begin
  773. result:=nil;
  774. expectloc:=LOC_VOID;
  775. if assigned(left) then
  776. begin
  777. firstpass(left);
  778. if codegenerror then
  779. exit;
  780. registers32:=left.registers32;
  781. registersfpu:=left.registersfpu;
  782. {$ifdef SUPPORT_MMX}
  783. registersmmx:=left.registersmmx;
  784. {$endif SUPPORT_MMX}
  785. end;
  786. end;
  787. {*****************************************************************************
  788. TBREAKNODE
  789. *****************************************************************************}
  790. constructor tbreaknode.create;
  791. begin
  792. inherited create(breakn);
  793. end;
  794. function tbreaknode.det_resulttype:tnode;
  795. begin
  796. result:=nil;
  797. resulttype:=voidtype;
  798. end;
  799. function tbreaknode.pass_1 : tnode;
  800. begin
  801. result:=nil;
  802. expectloc:=LOC_VOID;
  803. end;
  804. {*****************************************************************************
  805. TCONTINUENODE
  806. *****************************************************************************}
  807. constructor tcontinuenode.create;
  808. begin
  809. inherited create(continuen);
  810. end;
  811. function tcontinuenode.det_resulttype:tnode;
  812. begin
  813. result:=nil;
  814. resulttype:=voidtype;
  815. end;
  816. function tcontinuenode.pass_1 : tnode;
  817. begin
  818. result:=nil;
  819. expectloc:=LOC_VOID;
  820. end;
  821. {*****************************************************************************
  822. TGOTONODE
  823. *****************************************************************************}
  824. constructor tgotonode.create(p : tlabelsym);
  825. begin
  826. inherited create(goton);
  827. exceptionblock:=aktexceptblock;
  828. labsym:=p;
  829. end;
  830. constructor tgotonode.ppuload(t:tnodetype;ppufile:tcompilerppufile);
  831. begin
  832. inherited ppuload(t,ppufile);
  833. ppufile.getderef(labsymderef);
  834. exceptionblock:=ppufile.getbyte;
  835. end;
  836. procedure tgotonode.ppuwrite(ppufile:tcompilerppufile);
  837. begin
  838. inherited ppuwrite(ppufile);
  839. ppufile.putderef(labsymderef);
  840. ppufile.putbyte(exceptionblock);
  841. end;
  842. procedure tgotonode.buildderefimpl;
  843. begin
  844. inherited buildderefimpl;
  845. labsymderef.build(labsym);
  846. end;
  847. procedure tgotonode.derefimpl;
  848. begin
  849. inherited derefimpl;
  850. labsym:=tlabelsym(labsymderef.resolve);
  851. end;
  852. function tgotonode.det_resulttype:tnode;
  853. begin
  854. result:=nil;
  855. resulttype:=voidtype;
  856. end;
  857. function tgotonode.pass_1 : tnode;
  858. begin
  859. result:=nil;
  860. expectloc:=LOC_VOID;
  861. { check if }
  862. if assigned(labsym) and
  863. assigned(labsym.code) and
  864. (exceptionblock<>tlabelnode(labsym.code).exceptionblock) then
  865. begin
  866. writeln('goto exceptblock: ',exceptionblock);
  867. writeln('label exceptblock: ',tlabelnode(labsym.code).exceptionblock);
  868. CGMessage(cg_e_goto_inout_of_exception_block);
  869. end;
  870. end;
  871. function tgotonode.getcopy : tnode;
  872. var
  873. p : tgotonode;
  874. begin
  875. p:=tgotonode(inherited getcopy);
  876. p.labsym:=labsym;
  877. p.exceptionblock:=exceptionblock;
  878. result:=p;
  879. end;
  880. function tgotonode.docompare(p: tnode): boolean;
  881. begin
  882. docompare := false;
  883. end;
  884. {*****************************************************************************
  885. TLABELNODE
  886. *****************************************************************************}
  887. constructor tlabelnode.createcase(p : tasmlabel;l:tnode);
  888. begin
  889. inherited create(labeln,l);
  890. { it shouldn't be possible to jump to case labels using goto }
  891. exceptionblock:=-1;
  892. labsym:=nil;
  893. labelnr:=p;
  894. end;
  895. constructor tlabelnode.create(p : tlabelsym;l:tnode);
  896. begin
  897. inherited create(labeln,l);
  898. exceptionblock:=aktexceptblock;
  899. labsym:=p;
  900. labelnr:=p.lab;
  901. { save the current labelnode in the labelsym }
  902. p.code:=self;
  903. end;
  904. constructor tlabelnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);
  905. begin
  906. inherited ppuload(t,ppufile);
  907. ppufile.getderef(labsymderef);
  908. labelnr:=tasmlabel(ppufile.getasmsymbol);
  909. exceptionblock:=ppufile.getbyte;
  910. end;
  911. procedure tlabelnode.ppuwrite(ppufile:tcompilerppufile);
  912. begin
  913. inherited ppuwrite(ppufile);
  914. ppufile.putderef(labsymderef);
  915. ppufile.putasmsymbol(labelnr);
  916. ppufile.putbyte(exceptionblock);
  917. end;
  918. procedure tlabelnode.buildderefimpl;
  919. begin
  920. inherited buildderefimpl;
  921. labsymderef.build(labsym);
  922. end;
  923. procedure tlabelnode.derefimpl;
  924. begin
  925. inherited derefimpl;
  926. labsym:=tlabelsym(labsymderef.resolve);
  927. objectlibrary.derefasmsymbol(tasmsymbol(labelnr));
  928. end;
  929. function tlabelnode.det_resulttype:tnode;
  930. begin
  931. result:=nil;
  932. { left could still be unassigned }
  933. if assigned(left) then
  934. resulttypepass(left);
  935. resulttype:=voidtype;
  936. end;
  937. function tlabelnode.pass_1 : tnode;
  938. begin
  939. result:=nil;
  940. expectloc:=LOC_VOID;
  941. if assigned(left) then
  942. begin
  943. firstpass(left);
  944. registers32:=left.registers32;
  945. registersfpu:=left.registersfpu;
  946. {$ifdef SUPPORT_MMX}
  947. registersmmx:=left.registersmmx;
  948. {$endif SUPPORT_MMX}
  949. end;
  950. end;
  951. function tlabelnode.getcopy : tnode;
  952. var
  953. p : tlabelnode;
  954. begin
  955. p:=tlabelnode(inherited getcopy);
  956. p.labelnr:=labelnr;
  957. p.exceptionblock:=exceptionblock;
  958. p.labsym:=labsym;
  959. result:=p;
  960. end;
  961. function tlabelnode.docompare(p: tnode): boolean;
  962. begin
  963. docompare := false;
  964. end;
  965. {*****************************************************************************
  966. TRAISENODE
  967. *****************************************************************************}
  968. constructor traisenode.create(l,taddr,tframe:tnode);
  969. begin
  970. inherited create(raisen,l,taddr);
  971. frametree:=tframe;
  972. end;
  973. constructor traisenode.ppuload(t:tnodetype;ppufile:tcompilerppufile);
  974. begin
  975. inherited ppuload(t,ppufile);
  976. frametree:=ppuloadnode(ppufile);
  977. end;
  978. procedure traisenode.ppuwrite(ppufile:tcompilerppufile);
  979. begin
  980. inherited ppuwrite(ppufile);
  981. ppuwritenode(ppufile,frametree);
  982. end;
  983. procedure traisenode.buildderefimpl;
  984. begin
  985. inherited buildderefimpl;
  986. if assigned(frametree) then
  987. frametree.buildderefimpl;
  988. end;
  989. procedure traisenode.derefimpl;
  990. begin
  991. inherited derefimpl;
  992. if assigned(frametree) then
  993. frametree.derefimpl;
  994. end;
  995. function traisenode.getcopy : tnode;
  996. var
  997. n : traisenode;
  998. begin
  999. n:=traisenode(inherited getcopy);
  1000. if assigned(frametree) then
  1001. n.frametree:=frametree.getcopy
  1002. else
  1003. n.frametree:=nil;
  1004. getcopy:=n;
  1005. end;
  1006. procedure traisenode.insertintolist(l : tnodelist);
  1007. begin
  1008. end;
  1009. function traisenode.det_resulttype:tnode;
  1010. begin
  1011. result:=nil;
  1012. resulttype:=voidtype;
  1013. if assigned(left) then
  1014. begin
  1015. { first para must be a _class_ }
  1016. resulttypepass(left);
  1017. set_varstate(left,vs_used,true);
  1018. if codegenerror then
  1019. exit;
  1020. if not(is_class(left.resulttype.def)) then
  1021. CGMessage(type_e_mismatch);
  1022. { insert needed typeconvs for addr,frame }
  1023. if assigned(right) then
  1024. begin
  1025. { addr }
  1026. resulttypepass(right);
  1027. inserttypeconv(right,voidpointertype);
  1028. { frame }
  1029. if assigned(frametree) then
  1030. begin
  1031. resulttypepass(frametree);
  1032. inserttypeconv(frametree,voidpointertype);
  1033. end;
  1034. end;
  1035. end;
  1036. end;
  1037. function traisenode.pass_1 : tnode;
  1038. begin
  1039. result:=nil;
  1040. include(current_procinfo.flags,pi_do_call);
  1041. expectloc:=LOC_VOID;
  1042. if assigned(left) then
  1043. begin
  1044. { first para must be a _class_ }
  1045. firstpass(left);
  1046. { insert needed typeconvs for addr,frame }
  1047. if assigned(right) then
  1048. begin
  1049. { addr }
  1050. firstpass(right);
  1051. { frame }
  1052. if assigned(frametree) then
  1053. firstpass(frametree);
  1054. end;
  1055. left_right_max;
  1056. end;
  1057. end;
  1058. function traisenode.docompare(p: tnode): boolean;
  1059. begin
  1060. docompare := false;
  1061. end;
  1062. {*****************************************************************************
  1063. TTRYEXCEPTNODE
  1064. *****************************************************************************}
  1065. constructor ttryexceptnode.create(l,r,_t1 : tnode);
  1066. begin
  1067. inherited create(tryexceptn,l,r,_t1,nil);
  1068. end;
  1069. function ttryexceptnode.det_resulttype:tnode;
  1070. begin
  1071. result:=nil;
  1072. resulttypepass(left);
  1073. { on statements }
  1074. if assigned(right) then
  1075. resulttypepass(right);
  1076. { else block }
  1077. if assigned(t1) then
  1078. resulttypepass(t1);
  1079. resulttype:=voidtype;
  1080. end;
  1081. function ttryexceptnode.pass_1 : tnode;
  1082. begin
  1083. result:=nil;
  1084. include(current_procinfo.flags,pi_do_call);
  1085. expectloc:=LOC_VOID;
  1086. firstpass(left);
  1087. { on statements }
  1088. if assigned(right) then
  1089. begin
  1090. firstpass(right);
  1091. registers32:=max(registers32,right.registers32);
  1092. registersfpu:=max(registersfpu,right.registersfpu);
  1093. {$ifdef SUPPORT_MMX}
  1094. registersmmx:=max(registersmmx,right.registersmmx);
  1095. {$endif SUPPORT_MMX}
  1096. end;
  1097. { else block }
  1098. if assigned(t1) then
  1099. begin
  1100. firstpass(t1);
  1101. registers32:=max(registers32,t1.registers32);
  1102. registersfpu:=max(registersfpu,t1.registersfpu);
  1103. {$ifdef SUPPORT_MMX}
  1104. registersmmx:=max(registersmmx,t1.registersmmx);
  1105. {$endif SUPPORT_MMX}
  1106. end;
  1107. end;
  1108. {*****************************************************************************
  1109. TTRYFINALLYNODE
  1110. *****************************************************************************}
  1111. constructor ttryfinallynode.create(l,r:tnode);
  1112. begin
  1113. inherited create(tryfinallyn,l,r,nil,nil);
  1114. implicitframe:=false;
  1115. end;
  1116. constructor ttryfinallynode.create_implicit(l,r,_t1:tnode);
  1117. begin
  1118. inherited create(tryfinallyn,l,r,_t1,nil);
  1119. implicitframe:=true;
  1120. end;
  1121. function ttryfinallynode.det_resulttype:tnode;
  1122. begin
  1123. result:=nil;
  1124. include(current_procinfo.flags,pi_do_call);
  1125. resulttype:=voidtype;
  1126. resulttypepass(left);
  1127. set_varstate(left,vs_used,true);
  1128. resulttypepass(right);
  1129. set_varstate(right,vs_used,true);
  1130. { special finally block only executed when there was an exception }
  1131. if assigned(t1) then
  1132. begin
  1133. resulttypepass(t1);
  1134. set_varstate(t1,vs_used,true);
  1135. end;
  1136. end;
  1137. function ttryfinallynode.pass_1 : tnode;
  1138. begin
  1139. result:=nil;
  1140. expectloc:=LOC_VOID;
  1141. firstpass(left);
  1142. firstpass(right);
  1143. left_right_max;
  1144. if assigned(t1) then
  1145. begin
  1146. firstpass(t1);
  1147. registers32:=max(registers32,t1.registers32);
  1148. registersfpu:=max(registersfpu,t1.registersfpu);
  1149. {$ifdef SUPPORT_MMX}
  1150. registersmmx:=max(registersmmx,t1.registersmmx);
  1151. {$endif SUPPORT_MMX}
  1152. end;
  1153. end;
  1154. {*****************************************************************************
  1155. TONNODE
  1156. *****************************************************************************}
  1157. constructor tonnode.create(l,r:tnode);
  1158. begin
  1159. inherited create(onn,l,r);
  1160. exceptsymtable:=nil;
  1161. excepttype:=nil;
  1162. end;
  1163. destructor tonnode.destroy;
  1164. begin
  1165. { copied nodes don't need to release the symtable }
  1166. if assigned(exceptsymtable) then
  1167. exceptsymtable.free;
  1168. inherited destroy;
  1169. end;
  1170. constructor tonnode.ppuload(t:tnodetype;ppufile:tcompilerppufile);
  1171. begin
  1172. inherited ppuload(t,ppufile);
  1173. exceptsymtable:=nil;
  1174. excepttype:=nil;
  1175. end;
  1176. function tonnode.getcopy : tnode;
  1177. var
  1178. n : tonnode;
  1179. begin
  1180. n:=tonnode(inherited getcopy);
  1181. n.exceptsymtable:=exceptsymtable.getcopy;
  1182. n.excepttype:=excepttype;
  1183. result:=n;
  1184. end;
  1185. function tonnode.det_resulttype:tnode;
  1186. begin
  1187. result:=nil;
  1188. resulttype:=voidtype;
  1189. if not(is_class(excepttype)) then
  1190. CGMessage(type_e_mismatch);
  1191. if assigned(left) then
  1192. resulttypepass(left);
  1193. if assigned(right) then
  1194. resulttypepass(right);
  1195. end;
  1196. function tonnode.pass_1 : tnode;
  1197. begin
  1198. result:=nil;
  1199. include(current_procinfo.flags,pi_do_call);
  1200. expectloc:=LOC_VOID;
  1201. registers32:=0;
  1202. registersfpu:=0;
  1203. {$ifdef SUPPORT_MMX}
  1204. registersmmx:=0;
  1205. {$endif SUPPORT_MMX}
  1206. if assigned(left) then
  1207. begin
  1208. firstpass(left);
  1209. registers32:=left.registers32;
  1210. registersfpu:=left.registersfpu;
  1211. {$ifdef SUPPORT_MMX}
  1212. registersmmx:=left.registersmmx;
  1213. {$endif SUPPORT_MMX}
  1214. end;
  1215. if assigned(right) then
  1216. begin
  1217. firstpass(right);
  1218. registers32:=max(registers32,right.registers32);
  1219. registersfpu:=max(registersfpu,right.registersfpu);
  1220. {$ifdef SUPPORT_MMX}
  1221. registersmmx:=max(registersmmx,right.registersmmx);
  1222. {$endif SUPPORT_MMX}
  1223. end;
  1224. end;
  1225. function tonnode.docompare(p: tnode): boolean;
  1226. begin
  1227. docompare := false;
  1228. end;
  1229. begin
  1230. cwhilerepeatnode:=twhilerepeatnode;
  1231. cifnode:=tifnode;
  1232. cfornode:=tfornode;
  1233. cexitnode:=texitnode;
  1234. cgotonode:=tgotonode;
  1235. clabelnode:=tlabelnode;
  1236. craisenode:=traisenode;
  1237. ctryexceptnode:=ttryexceptnode;
  1238. ctryfinallynode:=ttryfinallynode;
  1239. connode:=tonnode;
  1240. end.
  1241. {
  1242. $Log$
  1243. Revision 1.92 2004-01-26 16:12:27 daniel
  1244. * reginfo now also only allocated during register allocation
  1245. * third round of gdb cleanups: kick out most of concatstabto
  1246. Revision 1.91 2003/12/28 22:51:18 florian
  1247. + except handling related nodes now include pi_do_call if necessary
  1248. Revision 1.90 2003/12/08 19:29:21 peter
  1249. * copy loopflags
  1250. Revision 1.89 2003/12/01 18:44:15 peter
  1251. * fixed some crashes
  1252. * fixed varargs and register calling probs
  1253. Revision 1.88 2003/11/23 17:39:16 peter
  1254. * don't release exceptsymtable for copied nodes
  1255. Revision 1.87 2003/11/12 15:48:27 peter
  1256. * fix set_varstate in for loops
  1257. * fix set_varstate from case statements
  1258. Revision 1.86 2003/10/28 15:36:01 peter
  1259. * absolute to object field supported, fixes tb0458
  1260. Revision 1.85 2003/10/23 14:44:07 peter
  1261. * splitted buildderef and buildderefimpl to fix interface crc
  1262. calculation
  1263. Revision 1.84 2003/10/22 20:40:00 peter
  1264. * write derefdata in a separate ppu entry
  1265. Revision 1.83 2003/10/09 21:31:37 daniel
  1266. * Register allocator splitted, ans abstract now
  1267. Revision 1.82 2003/10/08 19:19:45 peter
  1268. * set_varstate cleanup
  1269. Revision 1.81 2003/10/05 11:53:57 florian
  1270. * writing of loop nodes fixed
  1271. Revision 1.80 2003/10/01 20:34:48 peter
  1272. * procinfo unit contains tprocinfo
  1273. * cginfo renamed to cgbase
  1274. * moved cgmessage to verbose
  1275. * fixed ppc and sparc compiles
  1276. Revision 1.79 2003/07/06 15:31:20 daniel
  1277. * Fixed register allocator. *Lots* of fixes.
  1278. Revision 1.78 2003/06/13 21:19:30 peter
  1279. * current_procdef removed, use current_procinfo.procdef instead
  1280. Revision 1.77 2003/06/07 20:26:32 peter
  1281. * re-resolving added instead of reloading from ppu
  1282. * tderef object added to store deref info for resolving
  1283. Revision 1.76 2003/06/07 18:57:04 jonas
  1284. + added freeintparaloc
  1285. * ppc get/freeintparaloc now check whether the parameter regs are
  1286. properly allocated/deallocated (and get an extra list para)
  1287. * ppc a_call_* now internalerrors if pi_do_call is not yet set
  1288. * fixed lot of missing pi_do_call's
  1289. Revision 1.75 2003/05/26 21:17:17 peter
  1290. * procinlinenode removed
  1291. * aktexit2label removed, fast exit removed
  1292. + tcallnode.inlined_pass_2 added
  1293. Revision 1.74 2003/05/13 19:14:41 peter
  1294. * failn removed
  1295. * inherited result code check moven to pexpr
  1296. Revision 1.73 2003/05/11 21:37:03 peter
  1297. * moved implicit exception frame from ncgutil to psub
  1298. * constructor/destructor helpers moved from cobj/ncgutil to psub
  1299. Revision 1.72 2003/05/01 07:59:42 florian
  1300. * introduced defaultordconsttype to decribe the default size of ordinal constants
  1301. on 64 bit CPUs it's equal to cs64bitdef while on 32 bit CPUs it's equal to s32bitdef
  1302. + added defines CPU32 and CPU64 for 32 bit and 64 bit CPUs
  1303. * int64s/qwords are allowed as for loop counter on 64 bit CPUs
  1304. Revision 1.71 2003/04/27 11:21:33 peter
  1305. * aktprocdef renamed to current_procinfo.procdef
  1306. * procinfo renamed to current_procinfo
  1307. * procinfo will now be stored in current_module so it can be
  1308. cleaned up properly
  1309. * gen_main_procsym changed to create_main_proc and release_main_proc
  1310. to also generate a tprocinfo structure
  1311. * fixed unit implicit initfinal
  1312. Revision 1.70 2003/04/27 07:29:50 peter
  1313. * current_procinfo.procdef cleanup, current_procdef is now always nil when parsing
  1314. a new procdef declaration
  1315. * aktprocsym removed
  1316. * lexlevel removed, use symtable.symtablelevel instead
  1317. * implicit init/final code uses the normal genentry/genexit
  1318. * funcret state checking updated for new funcret handling
  1319. Revision 1.69 2003/04/26 00:28:41 peter
  1320. * removed load_funcret
  1321. Revision 1.68 2003/04/25 20:59:33 peter
  1322. * removed funcretn,funcretsym, function result is now in varsym
  1323. and aliases for result and function name are added using absolutesym
  1324. * vs_hidden parameter for funcret passed in parameter
  1325. * vs_hidden fixes
  1326. * writenode changed to printnode and released from extdebug
  1327. * -vp option added to generate a tree.log with the nodetree
  1328. * nicer printnode for statements, callnode
  1329. Revision 1.67 2003/04/25 08:25:26 daniel
  1330. * Ifdefs around a lot of calls to cleartempgen
  1331. * Fixed registers that are allocated but not freed in several nodes
  1332. * Tweak to register allocator to cause less spills
  1333. * 8-bit registers now interfere with esi,edi and ebp
  1334. Compiler can now compile rtl successfully when using new register
  1335. allocator
  1336. Revision 1.66 2003/04/22 23:50:23 peter
  1337. * firstpass uses expectloc
  1338. * checks if there are differences between the expectloc and
  1339. location.loc from secondpass in EXTDEBUG
  1340. Revision 1.65 2003/03/20 15:54:46 peter
  1341. * don't allow var and out parameters as for loop counter
  1342. Revision 1.64 2003/01/09 21:52:37 peter
  1343. * merged some verbosity options.
  1344. * V_LineInfo is a verbosity flag to include line info
  1345. Revision 1.63 2003/01/04 08:08:47 daniel
  1346. * Readded missing variable
  1347. Revision 1.62 2003/01/03 17:16:57 peter
  1348. * fixed warning about unset funcret
  1349. Revision 1.61 2003/01/03 12:15:56 daniel
  1350. * Removed ifdefs around notifications
  1351. ifdefs around for loop optimizations remain
  1352. Revision 1.60 2002/12/31 09:55:58 daniel
  1353. + Notification implementation complete
  1354. + Add for loop code optimization using notifications
  1355. results in 1.5-1.9% speed improvement in nestloop benchmark
  1356. Optimization incomplete, compiler does not cycle yet with
  1357. notifications enabled.
  1358. Revision 1.59 2002/12/30 22:44:53 daniel
  1359. * Some work on notifications
  1360. Revision 1.58 2002/12/27 15:25:40 peter
  1361. * do not allow threadvar as loop counter
  1362. Revision 1.57 2002/11/28 11:17:02 florian
  1363. * loop node flags from node flags splitted
  1364. Revision 1.56 2002/11/25 17:43:18 peter
  1365. * splitted defbase in defutil,symutil,defcmp
  1366. * merged isconvertable and is_equal into compare_defs(_ext)
  1367. * made operator search faster by walking the list only once
  1368. Revision 1.55 2002/11/18 17:31:56 peter
  1369. * pass proccalloption to ret_in_xxx and push_xxx functions
  1370. Revision 1.54 2002/10/20 15:31:49 peter
  1371. * set funcret state for exit(0)
  1372. Revision 1.53 2002/10/05 12:43:25 carl
  1373. * fixes for Delphi 6 compilation
  1374. (warning : Some features do not work under Delphi)
  1375. Revision 1.52 2002/09/07 15:25:03 peter
  1376. * old logs removed and tabs fixed
  1377. Revision 1.51 2002/09/07 12:16:04 carl
  1378. * second part bug report 1996 fix, testrange in cordconstnode
  1379. only called if option is set (also make parsing a tiny faster)
  1380. Revision 1.50 2002/09/01 18:47:00 peter
  1381. * assignn check in exitnode changed to use a separate boolean as the
  1382. assignn can be changed to a calln
  1383. Revision 1.49 2002/09/01 08:01:16 daniel
  1384. * Removed sets from Tcallnode.det_resulttype
  1385. + Added read/write notifications of variables. These will be usefull
  1386. for providing information for several optimizations. For example
  1387. the value of the loop variable of a for loop does matter is the
  1388. variable is read after the for loop, but if it's no longer used
  1389. or written, it doesn't matter and this can be used to optimize
  1390. the loop code generation.
  1391. Revision 1.48 2002/08/22 15:15:20 daniel
  1392. * Fixed the detection wether the first check of a for loop can be skipped
  1393. Revision 1.47 2002/08/19 19:36:43 peter
  1394. * More fixes for cross unit inlining, all tnodes are now implemented
  1395. * Moved pocall_internconst to po_internconst because it is not a
  1396. calling type at all and it conflicted when inlining of these small
  1397. functions was requested
  1398. Revision 1.46 2002/08/17 22:09:46 florian
  1399. * result type handling in tcgcal.pass_2 overhauled
  1400. * better tnode.dowrite
  1401. * some ppc stuff fixed
  1402. Revision 1.45 2002/08/17 09:23:37 florian
  1403. * first part of current_procinfo rewrite
  1404. Revision 1.44 2002/07/21 06:58:49 daniel
  1405. * Changed booleans into flags
  1406. Revision 1.43 2002/07/20 11:57:54 florian
  1407. * types.pas renamed to defbase.pas because D6 contains a types
  1408. unit so this would conflicts if D6 programms are compiled
  1409. + Willamette/SSE2 instructions to assembler added
  1410. Revision 1.42 2002/07/20 11:18:18 daniel
  1411. * Small mistake fixed; the skip test was done before we know the for node
  1412. is correct.
  1413. Revision 1.40 2002/07/20 08:19:31 daniel
  1414. * State tracker automatically changes while loops into repeat loops
  1415. Revision 1.39 2002/07/19 12:55:27 daniel
  1416. * Further developed state tracking in whilerepeatn
  1417. Revision 1.38 2002/07/19 11:41:35 daniel
  1418. * State tracker work
  1419. * The whilen and repeatn are now completely unified into whilerepeatn. This
  1420. allows the state tracker to change while nodes automatically into
  1421. repeat nodes.
  1422. * Resulttypepass improvements to the notn. 'not not a' is optimized away and
  1423. 'not(a>b)' is optimized into 'a<=b'.
  1424. * Resulttypepass improvements to the whilerepeatn. 'while not a' is optimized
  1425. by removing the notn and later switchting the true and falselabels. The
  1426. same is done with 'repeat until not a'.
  1427. Revision 1.37 2002/07/16 13:57:02 florian
  1428. * raise takes now a void pointer as at and frame address
  1429. instead of a longint
  1430. Revision 1.36 2002/07/15 18:03:15 florian
  1431. * readded removed changes
  1432. Revision 1.35 2002/07/14 18:00:44 daniel
  1433. + Added the beginning of a state tracker. This will track the values of
  1434. variables through procedures and optimize things away.
  1435. Revision 1.34 2002/07/11 14:41:28 florian
  1436. * start of the new generic parameter handling
  1437. Revision 1.33 2002/07/01 18:46:23 peter
  1438. * internal linker
  1439. * reorganized aasm layer
  1440. Revision 1.32 2002/05/18 13:34:10 peter
  1441. * readded missing revisions
  1442. Revision 1.31 2002/05/16 19:46:38 carl
  1443. + defines.inc -> fpcdefs.inc to avoid conflicts if compiling by hand
  1444. + try to fix temp allocation (still in ifdef)
  1445. + generic constructor calls
  1446. + start of tassembler / tmodulebase class cleanup
  1447. }