pstatmnt.pas 55 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605
  1. {
  2. $Id$
  3. Copyright (c) 1998 by Florian Klaempfl
  4. Does the parsing of the statements
  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 pstatmnt;
  19. interface
  20. uses tree;
  21. var
  22. { true, if we are in a except block }
  23. in_except_block : boolean;
  24. { reads a block }
  25. function block(islibrary : boolean) : ptree;
  26. { reads an assembler block }
  27. function assembler_block : ptree;
  28. implementation
  29. uses
  30. globtype,systems,tokens,
  31. strings,cobjects,globals,files,verbose,
  32. symtable,aasm,pass_1,types,scanner,hcodegen,ppu
  33. ,pbase,pexpr,pdecl
  34. {$ifdef i386}
  35. {$ifndef OLDASM}
  36. ,i386base,i386asm
  37. {$else}
  38. ,i386
  39. {$endif}
  40. ,tgeni386
  41. {$ifndef NoRa386Int}
  42. ,ra386int
  43. {$endif NoRa386Int}
  44. {$ifndef NoRa386Att}
  45. ,ra386att
  46. {$endif NoRa386Att}
  47. {$ifndef NoRa386Dir}
  48. ,ra386dir
  49. {$endif NoRa386Dir}
  50. {$endif i386}
  51. {$ifdef m68k}
  52. ,m68k,tgen68k
  53. {$ifndef NoRa68kMot}
  54. ,ra68kmot
  55. {$endif NoRa68kMot}
  56. {$endif m68k}
  57. ;
  58. const
  59. statement_level : longint = 0;
  60. function statement : ptree;forward;
  61. function if_statement : ptree;
  62. var
  63. ex,if_a,else_a : ptree;
  64. begin
  65. consume(_IF);
  66. ex:=comp_expr(true);
  67. consume(_THEN);
  68. if token<>_ELSE then
  69. if_a:=statement
  70. else
  71. if_a:=nil;
  72. if try_to_consume(_ELSE) then
  73. else_a:=statement
  74. else
  75. else_a:=nil;
  76. if_statement:=genloopnode(ifn,ex,if_a,else_a,false);
  77. end;
  78. { creates a block (list) of statements, til the next END token }
  79. function statements_til_end : ptree;
  80. var
  81. first,last : ptree;
  82. begin
  83. first:=nil;
  84. while token<>_END do
  85. begin
  86. if first=nil then
  87. begin
  88. last:=gennode(statementn,nil,statement);
  89. first:=last;
  90. end
  91. else
  92. begin
  93. last^.left:=gennode(statementn,nil,statement);
  94. last:=last^.left;
  95. end;
  96. if not try_to_consume(SEMICOLON) then
  97. break;
  98. emptystats;
  99. end;
  100. consume(_END);
  101. statements_til_end:=gensinglenode(blockn,first);
  102. end;
  103. function case_statement : ptree;
  104. var
  105. { contains the label number of currently parsed case block }
  106. aktcaselabel : plabel;
  107. firstlabel : boolean;
  108. root : pcaserecord;
  109. { the typ of the case expression }
  110. casedef : pdef;
  111. procedure newcaselabel(l,h : longint;first:boolean);
  112. var
  113. hcaselabel : pcaserecord;
  114. procedure insertlabel(var p : pcaserecord);
  115. begin
  116. if p=nil then p:=hcaselabel
  117. else
  118. if (p^._low>hcaselabel^._low) and
  119. (p^._low>hcaselabel^._high) then
  120. if (hcaselabel^.statement = p^.statement) and
  121. (p^._low = hcaselabel^._high + 1) then
  122. begin
  123. p^._low := hcaselabel^._low;
  124. freelabel(hcaselabel^._at);
  125. dispose(hcaselabel);
  126. end
  127. else
  128. insertlabel(p^.less)
  129. else
  130. if (p^._high<hcaselabel^._low) and
  131. (p^._high<hcaselabel^._high) then
  132. if (hcaselabel^.statement = p^.statement) and
  133. (p^._high+1 = hcaselabel^._low) then
  134. begin
  135. p^._high := hcaselabel^._high;
  136. freelabel(hcaselabel^._at);
  137. dispose(hcaselabel);
  138. end
  139. else
  140. insertlabel(p^.greater)
  141. else Message(parser_e_double_caselabel);
  142. end;
  143. begin
  144. new(hcaselabel);
  145. hcaselabel^.less:=nil;
  146. hcaselabel^.greater:=nil;
  147. hcaselabel^.statement:=aktcaselabel;
  148. hcaselabel^.firstlabel:=first;
  149. getlabel(hcaselabel^._at);
  150. hcaselabel^._low:=l;
  151. hcaselabel^._high:=h;
  152. insertlabel(root);
  153. end;
  154. var
  155. code,caseexpr,p,instruc,elseblock : ptree;
  156. hl1,hl2 : longint;
  157. begin
  158. consume(_CASE);
  159. caseexpr:=comp_expr(true);
  160. { determines result type }
  161. cleartempgen;
  162. do_firstpass(caseexpr);
  163. casedef:=caseexpr^.resulttype;
  164. if not(is_ordinal(casedef)) then
  165. Message(type_e_ordinal_expr_expected);
  166. consume(_OF);
  167. inc(statement_level);
  168. root:=nil;
  169. instruc:=nil;
  170. repeat
  171. getlabel(aktcaselabel);
  172. firstlabel:=true;
  173. { may be an instruction has more case labels }
  174. repeat
  175. p:=expr;
  176. cleartempgen;
  177. do_firstpass(p);
  178. if (p^.treetype=rangen) then
  179. begin
  180. { type checking for case statements }
  181. if not is_subequal(casedef, p^.left^.resulttype) then
  182. Message(parser_e_case_mismatch);
  183. { type checking for case statements }
  184. if not is_subequal(casedef, p^.right^.resulttype) then
  185. Message(parser_e_case_mismatch);
  186. hl1:=get_ordinal_value(p^.left);
  187. hl2:=get_ordinal_value(p^.right);
  188. testrange(casedef,hl1);
  189. testrange(casedef,hl2);
  190. if hl1>hl2 then
  191. Message(parser_e_case_lower_less_than_upper_bound);
  192. newcaselabel(hl1,hl2,firstlabel);
  193. end
  194. else
  195. begin
  196. { type checking for case statements }
  197. if not is_subequal(casedef, p^.resulttype) then
  198. Message(parser_e_case_mismatch);
  199. hl1:=get_ordinal_value(p);
  200. testrange(casedef,hl1);
  201. newcaselabel(hl1,hl1,firstlabel);
  202. end;
  203. disposetree(p);
  204. if token=COMMA then
  205. consume(COMMA)
  206. else
  207. break;
  208. firstlabel:=false;
  209. until false;
  210. consume(COLON);
  211. { handles instruction block }
  212. p:=gensinglenode(labeln,statement);
  213. p^.labelnr:=aktcaselabel;
  214. { concats instruction }
  215. instruc:=gennode(statementn,instruc,p);
  216. if not((token=_ELSE) or (token=_OTHERWISE) or (token=_END)) then
  217. consume(SEMICOLON);
  218. until (token=_ELSE) or (token=_OTHERWISE) or (token=_END);
  219. if (token=_ELSE) or (token=_OTHERWISE) then
  220. begin
  221. if not try_to_consume(_ELSE) then
  222. consume(_OTHERWISE);
  223. elseblock:=statements_til_end;
  224. end
  225. else
  226. begin
  227. elseblock:=nil;
  228. consume(_END);
  229. end;
  230. dec(statement_level);
  231. code:=gencasenode(caseexpr,instruc,root);
  232. code^.elseblock:=elseblock;
  233. case_statement:=code;
  234. end;
  235. function repeat_statement : ptree;
  236. var
  237. first,last,p_e : ptree;
  238. begin
  239. consume(_REPEAT);
  240. first:=nil;
  241. inc(statement_level);
  242. while token<>_UNTIL do
  243. begin
  244. if first=nil then
  245. begin
  246. last:=gennode(statementn,nil,statement);
  247. first:=last;
  248. end
  249. else
  250. begin
  251. last^.left:=gennode(statementn,nil,statement);
  252. last:=last^.left;
  253. end;
  254. if not try_to_consume(SEMICOLON) then
  255. break;
  256. emptystats;
  257. end;
  258. consume(_UNTIL);
  259. dec(statement_level);
  260. first:=gensinglenode(blockn,first);
  261. p_e:=comp_expr(true);
  262. repeat_statement:=genloopnode(repeatn,p_e,first,nil,false);
  263. end;
  264. function while_statement : ptree;
  265. var
  266. p_e,p_a : ptree;
  267. begin
  268. consume(_WHILE);
  269. p_e:=comp_expr(true);
  270. consume(_DO);
  271. p_a:=statement;
  272. while_statement:=genloopnode(whilen,p_e,p_a,nil,false);
  273. end;
  274. function for_statement : ptree;
  275. var
  276. p_e,tovalue,p_a : ptree;
  277. backward : boolean;
  278. begin
  279. { parse loop header }
  280. consume(_FOR);
  281. p_e:=expr;
  282. if token=_DOWNTO then
  283. begin
  284. consume(_DOWNTO);
  285. backward:=true;
  286. end
  287. else
  288. begin
  289. consume(_TO);
  290. backward:=false;
  291. end;
  292. tovalue:=comp_expr(true);
  293. consume(_DO);
  294. { ... now the instruction }
  295. p_a:=statement;
  296. for_statement:=genloopnode(forn,p_e,tovalue,p_a,backward);
  297. end;
  298. function _with_statement : ptree;
  299. var
  300. right,hp,p : ptree;
  301. i,levelcount : longint;
  302. withsymtable,symtab : psymtable;
  303. obj : pobjectdef;
  304. begin
  305. Must_be_valid:=false;
  306. p:=comp_expr(true);
  307. do_firstpass(p);
  308. right:=nil;
  309. if (not codegenerror) and
  310. (p^.resulttype^.deftype in [objectdef,recorddef]) then
  311. begin
  312. case p^.resulttype^.deftype of
  313. objectdef : begin
  314. obj:=pobjectdef(p^.resulttype);
  315. withsymtable:=new(pwithsymtable,init);
  316. {$ifndef OLDPPU}
  317. withsymtable^.symsearch:=obj^.publicsyms^.symsearch;
  318. {$else}
  319. withsymtable^.searchroot:=obj^.publicsyms^.searchroot;
  320. {$endif}
  321. withsymtable^.defowner:=obj;
  322. symtab:=withsymtable;
  323. {$ifndef NODIRECTWITH}
  324. if (p^.treetype=loadn) and
  325. (p^.symtable=aktprocsym^.definition^.localst) then
  326. pwithsymtable(symtab)^.direct_with:=true;
  327. {symtab^.withnode:=p; not yet allocated !! }
  328. pwithsymtable(symtab)^.withrefnode:=p;
  329. {$endif ndef NODIRECTWITH}
  330. levelcount:=1;
  331. obj:=obj^.childof;
  332. while assigned(obj) do
  333. begin
  334. symtab^.next:=new(pwithsymtable,init);
  335. symtab:=symtab^.next;
  336. {$ifndef OLDPPU}
  337. symtab^.symsearch:=obj^.publicsyms^.symsearch;
  338. {$else}
  339. symtab^.searchroot:=obj^.publicsyms^.searchroot;
  340. {$endif}
  341. {$ifndef NODIRECTWITH}
  342. if (p^.treetype=loadn) and
  343. (p^.symtable=aktprocsym^.definition^.localst) then
  344. pwithsymtable(symtab)^.direct_with:=true;
  345. {symtab^.withnode:=p; not yet allocated !! }
  346. pwithsymtable(symtab)^.withrefnode:=p;
  347. {$endif ndef NODIRECTWITH}
  348. symtab^.defowner:=obj;
  349. obj:=obj^.childof;
  350. inc(levelcount);
  351. end;
  352. symtab^.next:=symtablestack;
  353. symtablestack:=withsymtable;
  354. end;
  355. recorddef : begin
  356. symtab:=precdef(p^.resulttype)^.symtable;
  357. levelcount:=1;
  358. withsymtable:=new(pwithsymtable,init);
  359. {$ifndef OLDPPU}
  360. withsymtable^.symsearch:=symtab^.symsearch;
  361. {$else}
  362. withsymtable^.searchroot:=symtab^.searchroot;
  363. {$endif}
  364. withsymtable^.next:=symtablestack;
  365. {$ifndef NODIRECTWITH}
  366. if (p^.treetype=loadn) and
  367. (p^.symtable=aktprocsym^.definition^.localst) then
  368. pwithsymtable(withsymtable)^.direct_with:=true;
  369. {symtab^.withnode:=p; not yet allocated !! }
  370. pwithsymtable(withsymtable)^.withrefnode:=p;
  371. {$endif ndef NODIRECTWITH}
  372. withsymtable^.defowner:=obj;
  373. symtablestack:=withsymtable;
  374. end;
  375. end;
  376. if token=COMMA then
  377. begin
  378. consume(COMMA);
  379. {$ifdef tp}
  380. right:=_with_statement;
  381. {$else}
  382. right:=_with_statement();
  383. {$endif}
  384. end
  385. else
  386. begin
  387. consume(_DO);
  388. if token<>SEMICOLON then
  389. right:=statement
  390. else
  391. right:=nil;
  392. end;
  393. for i:=1 to levelcount do
  394. symtablestack:=symtablestack^.next;
  395. _with_statement:=genwithnode(pwithsymtable(withsymtable),p,right,levelcount);
  396. end
  397. else
  398. begin
  399. Message(parser_e_false_with_expr);
  400. { try to recover from error }
  401. if token=COMMA then
  402. begin
  403. consume(COMMA);
  404. {$ifdef tp}
  405. hp:=_with_statement;
  406. {$else}
  407. hp:=_with_statement();
  408. {$endif}
  409. end
  410. else
  411. begin
  412. consume(_DO);
  413. { ignore all }
  414. if token<>SEMICOLON then
  415. statement;
  416. end;
  417. _with_statement:=nil;
  418. end;
  419. end;
  420. function with_statement : ptree;
  421. begin
  422. consume(_WITH);
  423. with_statement:=_with_statement;
  424. end;
  425. function raise_statement : ptree;
  426. var
  427. p1,p2 : ptree;
  428. begin
  429. p1:=nil;
  430. p2:=nil;
  431. consume(_RAISE);
  432. if token<>SEMICOLON then
  433. begin
  434. p1:=comp_expr(true);
  435. if (idtoken=_AT) then
  436. begin
  437. consume(ID);
  438. p2:=comp_expr(true);
  439. end;
  440. end
  441. else
  442. begin
  443. if not(in_except_block) then
  444. Message(parser_e_no_reraise_possible);
  445. end;
  446. raise_statement:=gennode(raisen,p1,p2);
  447. end;
  448. function try_statement : ptree;
  449. var
  450. p_try_block,p_finally_block,first,last,
  451. p_default,p_specific,hp : ptree;
  452. ot : pobjectdef;
  453. sym : pvarsym;
  454. old_in_except_block : boolean;
  455. exceptsymtable : psymtable;
  456. objname : stringid;
  457. begin
  458. procinfo.flags:=procinfo.flags or
  459. pi_uses_exceptions;
  460. p_default:=nil;
  461. p_specific:=nil;
  462. { read statements to try }
  463. consume(_TRY);
  464. first:=nil;
  465. inc(statement_level);
  466. while (token<>_FINALLY) and (token<>_EXCEPT) do
  467. begin
  468. if first=nil then
  469. begin
  470. last:=gennode(statementn,nil,statement);
  471. first:=last;
  472. end
  473. else
  474. begin
  475. last^.left:=gennode(statementn,nil,statement);
  476. last:=last^.left;
  477. end;
  478. if not try_to_consume(SEMICOLON) then
  479. break;
  480. emptystats;
  481. end;
  482. p_try_block:=gensinglenode(blockn,first);
  483. if try_to_consume(_FINALLY) then
  484. begin
  485. p_finally_block:=statements_til_end;
  486. try_statement:=gennode(tryfinallyn,p_try_block,p_finally_block);
  487. dec(statement_level);
  488. end
  489. else
  490. begin
  491. consume(_EXCEPT);
  492. old_in_except_block:=in_except_block;
  493. in_except_block:=true;
  494. p_specific:=nil;
  495. if token=_ON then
  496. { catch specific exceptions }
  497. begin
  498. repeat
  499. consume(_ON);
  500. if token=ID then
  501. begin
  502. getsym(pattern,false);
  503. objname:=pattern;
  504. consume(ID);
  505. { is a explicit name for the exception given ? }
  506. if try_to_consume(COLON) then
  507. begin
  508. getsym(pattern,true);
  509. consume(ID);
  510. if srsym^.typ=unitsym then
  511. begin
  512. consume(POINT);
  513. getsymonlyin(punitsym(srsym)^.unitsymtable,pattern);
  514. consume(ID);
  515. end;
  516. if (srsym^.typ=typesym) and
  517. (ptypesym(srsym)^.definition^.deftype=objectdef) and
  518. pobjectdef(ptypesym(srsym)^.definition)^.isclass then
  519. ot:=pobjectdef(ptypesym(srsym)^.definition)
  520. else
  521. begin
  522. message(type_e_class_type_expected);
  523. ot:=pobjectdef(generrordef);
  524. end;
  525. sym:=new(pvarsym,init(objname,ot));
  526. exceptsymtable:=new(psymtable,init(stt_exceptsymtable));
  527. exceptsymtable^.insert(sym);
  528. { insert the exception symtable stack }
  529. exceptsymtable^.next:=symtablestack;
  530. symtablestack:=exceptsymtable;
  531. end
  532. else
  533. begin
  534. { only exception type }
  535. if srsym^.typ=unitsym then
  536. begin
  537. consume(POINT);
  538. getsymonlyin(punitsym(srsym)^.unitsymtable,pattern);
  539. consume(ID);
  540. end;
  541. if (srsym^.typ=typesym) and
  542. (ptypesym(srsym)^.definition^.deftype=objectdef) and
  543. pobjectdef(ptypesym(srsym)^.definition)^.isclass then
  544. ot:=pobjectdef(ptypesym(srsym)^.definition)
  545. else
  546. begin
  547. message(type_e_class_type_expected);
  548. ot:=pobjectdef(generrordef);
  549. end;
  550. exceptsymtable:=nil;
  551. end;
  552. end
  553. else
  554. consume(ID);
  555. consume(_DO);
  556. hp:=gennode(onn,nil,statement);
  557. if ot^.deftype=errordef then
  558. begin
  559. disposetree(hp);
  560. hp:=genzeronode(errorn);
  561. end;
  562. if p_specific=nil then
  563. begin
  564. last:=hp;
  565. p_specific:=last;
  566. end
  567. else
  568. begin
  569. last^.left:=hp;
  570. last:=last^.left;
  571. end;
  572. { set the informations }
  573. last^.excepttype:=ot;
  574. last^.exceptsymtable:=exceptsymtable;
  575. last^.disposetyp:=dt_onn;
  576. { remove exception symtable }
  577. if assigned(exceptsymtable) then
  578. dellexlevel;
  579. if not try_to_consume(SEMICOLON) then
  580. break;
  581. emptystats;
  582. until (token=_END) or(token=_ELSE);
  583. if token=_ELSE then
  584. { catch the other exceptions }
  585. begin
  586. consume(_ELSE);
  587. p_default:=statements_til_end;
  588. end
  589. else
  590. consume(_END);
  591. end
  592. else
  593. { catch all exceptions }
  594. begin
  595. p_default:=statements_til_end;
  596. end;
  597. dec(statement_level);
  598. in_except_block:=old_in_except_block;
  599. try_statement:=genloopnode(tryexceptn,p_try_block,p_specific,p_default,false);
  600. end;
  601. end;
  602. function exit_statement : ptree;
  603. var
  604. p : ptree;
  605. begin
  606. consume(_EXIT);
  607. if try_to_consume(LKLAMMER) then
  608. begin
  609. p:=comp_expr(true);
  610. consume(RKLAMMER);
  611. if procinfo.retdef=pdef(voiddef) then
  612. Message(parser_e_void_function)
  613. else
  614. procinfo.funcret_is_valid:=true;
  615. end
  616. else
  617. p:=nil;
  618. p:=gensinglenode(exitn,p);
  619. p^.resulttype:=procinfo.retdef;
  620. exit_statement:=p;
  621. end;
  622. function _asm_statement : ptree;
  623. var
  624. asmstat : ptree;
  625. Marker : Pai;
  626. begin
  627. Inside_asm_statement:=true;
  628. case aktasmmode of
  629. {$ifdef i386}
  630. {$ifndef NoRA386Att}
  631. asmmode_i386_att:
  632. asmstat:=ra386att.assemble;
  633. {$endif NoRA386Att}
  634. {$ifndef NoRA386Int}
  635. asmmode_i386_intel:
  636. asmstat:=ra386int.assemble;
  637. {$endif NoRA386Int}
  638. {$ifndef NoRA386Dir}
  639. asmmode_i386_direct:
  640. begin
  641. if (aktprocsym^.definition^.options and poinline)<>0 then
  642. Begin
  643. Message1(parser_w_not_supported_for_inline,'direct asm');
  644. Message(parser_w_inlining_disabled);
  645. aktprocsym^.definition^.options:= aktprocsym^.definition^.options and not poinline;
  646. End;
  647. asmstat:=ra386dir.assemble;
  648. end;
  649. {$endif NoRA386Dir}
  650. {$endif}
  651. {$ifdef m68k}
  652. {$ifndef NoRA68kMot}
  653. asmmode_m68k_mot:
  654. asmstat:=ra68kmot.assemble;
  655. {$endif NoRA68kMot}
  656. {$endif}
  657. else
  658. Message(parser_f_assembler_reader_not_supported);
  659. end;
  660. { Read first the _ASM statement }
  661. consume(_ASM);
  662. { END is read }
  663. if try_to_consume(LECKKLAMMER) then
  664. begin
  665. { it's possible to specify the modified registers }
  666. asmstat^.object_preserved:=true;
  667. if token<>RECKKLAMMER then
  668. repeat
  669. { uppercase, because it's a CSTRING }
  670. uppervar(pattern);
  671. {$ifdef i386}
  672. if pattern='EAX' then
  673. usedinproc:=usedinproc or ($80 shr byte(R_EAX))
  674. else if pattern='EBX' then
  675. usedinproc:=usedinproc or ($80 shr byte(R_EBX))
  676. else if pattern='ECX' then
  677. usedinproc:=usedinproc or ($80 shr byte(R_ECX))
  678. else if pattern='EDX' then
  679. usedinproc:=usedinproc or ($80 shr byte(R_EDX))
  680. else if pattern='ESI' then
  681. begin
  682. usedinproc:=usedinproc or ($80 shr byte(R_ESI));
  683. asmstat^.object_preserved:=false;
  684. end
  685. else if pattern='EDI' then
  686. usedinproc:=usedinproc or ($80 shr byte(R_EDI))
  687. {$endif i386}
  688. {$ifdef m68k}
  689. if pattern='D0' then
  690. usedinproc:=usedinproc or ($800 shr word(R_D0))
  691. else if pattern='D1' then
  692. usedinproc:=usedinproc or ($800 shr word(R_D1))
  693. else if pattern='D6' then
  694. usedinproc:=usedinproc or ($800 shr word(R_D6))
  695. else if pattern='A0' then
  696. usedinproc:=usedinproc or ($800 shr word(R_A0))
  697. else if pattern='A1' then
  698. usedinproc:=usedinproc or ($800 shr word(R_A1))
  699. {$endif m68k}
  700. else consume(RECKKLAMMER);
  701. consume(CSTRING);
  702. if not try_to_consume(COMMA) then
  703. break;
  704. until false;
  705. consume(RECKKLAMMER);
  706. end
  707. else usedinproc:=$ff;
  708. { mark the start and the end of the assembler block for the optimizer }
  709. If Assigned(AsmStat^.p_asm) Then
  710. Begin
  711. Marker := New(Pai_Marker, Init(AsmBlockStart));
  712. AsmStat^.p_asm^.Insert(Marker);
  713. Marker := New(Pai_Marker, Init(AsmBlockEnd));
  714. AsmStat^.p_asm^.Concat(Marker);
  715. End;
  716. Inside_asm_statement:=false;
  717. _asm_statement:=asmstat;
  718. end;
  719. function new_dispose_statement : ptree;
  720. var
  721. p,p2 : ptree;
  722. ht : ttoken;
  723. again : boolean; { dummy for do_proc_call }
  724. destrukname : stringid;
  725. sym : psym;
  726. classh : pobjectdef;
  727. pd,pd2 : pdef;
  728. store_valid : boolean;
  729. tt : ttreetyp;
  730. begin
  731. ht:=token;
  732. if try_to_consume(_NEW) then
  733. tt:=hnewn
  734. else
  735. begin
  736. consume(_DISPOSE);
  737. tt:=hdisposen;
  738. end;
  739. consume(LKLAMMER);
  740. p:=comp_expr(true);
  741. { calc return type }
  742. cleartempgen;
  743. Store_valid := Must_be_valid;
  744. Must_be_valid := False;
  745. do_firstpass(p);
  746. Must_be_valid := Store_valid;
  747. {var o:Pobject;
  748. begin
  749. new(o,init); (*Also a valid new statement*)
  750. end;}
  751. if try_to_consume(COMMA) then
  752. begin
  753. { extended syntax of new and dispose }
  754. { function styled new is handled in factor }
  755. { destructors have no parameters }
  756. destrukname:=pattern;
  757. consume(ID);
  758. pd:=p^.resulttype;
  759. pd2:=pd;
  760. if (p^.resulttype = nil) or (pd^.deftype<>pointerdef) then
  761. begin
  762. Message(type_e_pointer_type_expected);
  763. p:=factor(false);
  764. consume(RKLAMMER);
  765. new_dispose_statement:=genzeronode(errorn);
  766. exit;
  767. end;
  768. { first parameter must be an object or class }
  769. if ppointerdef(pd)^.definition^.deftype<>objectdef then
  770. begin
  771. Message(parser_e_pointer_to_class_expected);
  772. new_dispose_statement:=factor(false);
  773. consume_all_until(RKLAMMER);
  774. consume(RKLAMMER);
  775. exit;
  776. end;
  777. { check, if the first parameter is a pointer to a _class_ }
  778. classh:=pobjectdef(ppointerdef(pd)^.definition);
  779. if (classh^.options and oo_is_class)<>0 then
  780. begin
  781. Message(parser_e_no_new_or_dispose_for_classes);
  782. new_dispose_statement:=factor(false);
  783. { while token<>RKLAMMER do
  784. consume(token); }
  785. consume_all_until(RKLAMMER);
  786. consume(RKLAMMER);
  787. exit;
  788. end;
  789. { search cons-/destructor, also in parent classes }
  790. sym:=search_class_member(classh,pattern);
  791. { the second parameter of new/dispose must be a call }
  792. { to a cons-/destructor }
  793. if (not assigned(sym)) or (sym^.typ<>procsym) then
  794. begin
  795. Message(parser_e_expr_have_to_be_destructor_call);
  796. new_dispose_statement:=genzeronode(errorn);
  797. end
  798. else
  799. begin
  800. p2:=gensinglenode(tt,p);
  801. if ht=_NEW then
  802. begin
  803. { Constructors can take parameters.}
  804. p2^.resulttype:=ppointerdef(pd)^.definition;
  805. do_member_read(false,sym,p2,pd,again);
  806. end
  807. else
  808. { destructors can't.}
  809. p2:=genmethodcallnode(pprocsym(sym),srsymtable,p2);
  810. { we need the real called method }
  811. cleartempgen;
  812. do_firstpass(p2);
  813. if not codegenerror then
  814. begin
  815. if (ht=_NEW) and ((p2^.procdefinition^.options and poconstructor)=0) then
  816. Message(parser_e_expr_have_to_be_constructor_call);
  817. if (ht=_DISPOSE) and ((p2^.procdefinition^.options and podestructor)=0) then
  818. Message(parser_e_expr_have_to_be_destructor_call);
  819. if ht=_NEW then
  820. begin
  821. p2:=gennode(assignn,getcopy(p),gensinglenode(newn,p2));
  822. p2^.right^.resulttype:=pd2;
  823. end;
  824. end;
  825. new_dispose_statement:=p2;
  826. end;
  827. end
  828. else
  829. begin
  830. if (p^.resulttype=nil) or (p^.resulttype^.deftype<>pointerdef) then
  831. Begin
  832. Message(type_e_pointer_type_expected);
  833. new_dispose_statement:=genzeronode(errorn);
  834. end
  835. else
  836. begin
  837. if (ppointerdef(p^.resulttype)^.definition^.deftype=objectdef) and
  838. ((pobjectdef(ppointerdef(p^.resulttype)^.definition)^.options and oo_hasvmt) <> 0) then
  839. Message(parser_w_use_extended_syntax_for_objects);
  840. if (ppointerdef(p^.resulttype)^.definition^.deftype=orddef) and
  841. (porddef(ppointerdef(p^.resulttype)^.definition)^.typ=uvoid) then
  842. if (m_tp in aktmodeswitches) or
  843. (m_delphi in aktmodeswitches) then
  844. Message(parser_w_no_new_dispose_on_void_pointers)
  845. else
  846. Message(parser_e_no_new_dispose_on_void_pointers);
  847. case ht of
  848. _NEW : new_dispose_statement:=gensinglenode(simplenewn,p);
  849. _DISPOSE : new_dispose_statement:=gensinglenode(simpledisposen,p);
  850. end;
  851. end;
  852. end;
  853. consume(RKLAMMER);
  854. end;
  855. function statement_block(starttoken : ttoken) : ptree;
  856. var
  857. first,last : ptree;
  858. filepos : tfileposinfo;
  859. begin
  860. first:=nil;
  861. filepos:=tokenpos;
  862. consume(starttoken);
  863. inc(statement_level);
  864. while not(token in [_END,_FINALIZATION]) do
  865. begin
  866. if first=nil then
  867. begin
  868. last:=gennode(statementn,nil,statement);
  869. first:=last;
  870. end
  871. else
  872. begin
  873. last^.left:=gennode(statementn,nil,statement);
  874. last:=last^.left;
  875. end;
  876. if (token in [_END,_FINALIZATION]) then
  877. break
  878. else
  879. begin
  880. { if no semicolon, then error and go on }
  881. if token<>SEMICOLON then
  882. begin
  883. consume(SEMICOLON);
  884. consume_all_until(SEMICOLON);
  885. end;
  886. consume(SEMICOLON);
  887. end;
  888. emptystats;
  889. end;
  890. { don't consume the finalization token, it is consumed when
  891. reading the finalization block, but allow it only after
  892. an initalization ! }
  893. if (starttoken<>_INITIALIZATION) or (token<>_FINALIZATION) then
  894. consume(_END);
  895. dec(statement_level);
  896. last:=gensinglenode(blockn,first);
  897. set_tree_filepos(last,filepos);
  898. statement_block:=last;
  899. end;
  900. function statement : ptree;
  901. var
  902. p : ptree;
  903. code : ptree;
  904. labelnr : plabel;
  905. filepos : tfileposinfo;
  906. label
  907. ready;
  908. begin
  909. filepos:=tokenpos;
  910. case token of
  911. _GOTO : begin
  912. if not(cs_support_goto in aktmoduleswitches)then
  913. Message(sym_e_goto_and_label_not_supported);
  914. consume(_GOTO);
  915. if (token<>INTCONST) and (token<>ID) then
  916. begin
  917. Message(sym_e_label_not_found);
  918. code:=genzeronode(errorn);
  919. end
  920. else
  921. begin
  922. getsym(pattern,true);
  923. consume(token);
  924. if srsym^.typ<>labelsym then
  925. begin
  926. Message(sym_e_id_is_no_label_id);
  927. code:=genzeronode(errorn);
  928. end
  929. else
  930. code:=genlabelnode(goton,
  931. plabelsym(srsym)^.number);
  932. end;
  933. end;
  934. _BEGIN : code:=statement_block(_BEGIN);
  935. _IF : code:=if_statement;
  936. _CASE : code:=case_statement;
  937. _REPEAT : code:=repeat_statement;
  938. _WHILE : code:=while_statement;
  939. _FOR : code:=for_statement;
  940. _NEW,_DISPOSE : code:=new_dispose_statement;
  941. _WITH : code:=with_statement;
  942. _TRY : code:=try_statement;
  943. _RAISE : code:=raise_statement;
  944. { semicolons,else until and end are ignored }
  945. SEMICOLON,
  946. _ELSE,
  947. _UNTIL,
  948. _END:
  949. code:=genzeronode(niln);
  950. _FAIL : begin
  951. { internalerror(100); }
  952. if (aktprocsym^.definition^.options and poconstructor)=0 then
  953. Message(parser_e_fail_only_in_constructor);
  954. consume(_FAIL);
  955. code:=genzeronode(failn);
  956. end;
  957. _EXIT : code:=exit_statement;
  958. _ASM : begin
  959. code:=_asm_statement;
  960. end;
  961. _EOF : begin
  962. Message(scan_f_end_of_file);
  963. end;
  964. else
  965. begin
  966. if (token=INTCONST) or
  967. ((token=ID) and not((m_result in aktmodeswitches) and (idtoken=_RESULT))) then
  968. begin
  969. getsym(pattern,true);
  970. lastsymknown:=true;
  971. lastsrsym:=srsym;
  972. { it is NOT necessarily the owner
  973. it can be a withsymtable !!! }
  974. lastsrsymtable:=srsymtable;
  975. if assigned(srsym) and (srsym^.typ=labelsym) then
  976. begin
  977. consume(token);
  978. consume(COLON);
  979. if plabelsym(srsym)^.defined then
  980. Message(sym_e_label_already_defined);
  981. plabelsym(srsym)^.defined:=true;
  982. { statement modifies srsym }
  983. labelnr:=plabelsym(srsym)^.number;
  984. lastsymknown:=false;
  985. { the pointer to the following instruction }
  986. { isn't a very clean way }
  987. {$ifdef tp}
  988. code:=gensinglenode(labeln,statement);
  989. {$else}
  990. code:=gensinglenode(labeln,statement());
  991. {$endif}
  992. code^.labelnr:=labelnr;
  993. { sorry, but there is a jump the easiest way }
  994. goto ready;
  995. end;
  996. end;
  997. p:=expr;
  998. if not(p^.treetype in [calln,assignn,breakn,inlinen,
  999. continuen]) then
  1000. Message(cg_e_illegal_expression);
  1001. { specify that we don't use the value returned by the call }
  1002. { Question : can this be also improtant
  1003. for inlinen ??
  1004. it is used for :
  1005. - dispose of temp stack space
  1006. - dispose on FPU stack }
  1007. if p^.treetype=calln then
  1008. p^.return_value_used:=false;
  1009. code:=p;
  1010. end;
  1011. end;
  1012. ready:
  1013. if assigned(code) then
  1014. set_tree_filepos(code,filepos);
  1015. statement:=code;
  1016. end;
  1017. function block(islibrary : boolean) : ptree;
  1018. var
  1019. funcretsym : pfuncretsym;
  1020. begin
  1021. if procinfo.retdef<>pdef(voiddef) then
  1022. begin
  1023. { if the current is a function aktprocsym is non nil }
  1024. { and there is a local symtable set }
  1025. funcretsym:=new(pfuncretsym,init(aktprocsym^.name,@procinfo));
  1026. { insert in local symtable }
  1027. symtablestack^.insert(funcretsym);
  1028. if ret_in_acc(procinfo.retdef) or (procinfo.retdef^.deftype=floatdef) then
  1029. procinfo.retoffset:=-funcretsym^.address;
  1030. procinfo.funcretsym:=funcretsym;
  1031. end;
  1032. read_declarations(islibrary);
  1033. { temporary space is set, while the BEGIN of the procedure }
  1034. if (symtablestack^.symtabletype=localsymtable) then
  1035. procinfo.firsttemp := -symtablestack^.datasize
  1036. else procinfo.firsttemp := 0;
  1037. { space for the return value }
  1038. { !!!!! this means that we can not set the return value
  1039. in a subfunction !!!!! }
  1040. { because we don't know yet where the address is }
  1041. if procinfo.retdef<>pdef(voiddef) then
  1042. begin
  1043. if ret_in_acc(procinfo.retdef) or (procinfo.retdef^.deftype=floatdef) then
  1044. { if (procinfo.retdef^.deftype=orddef) or
  1045. (procinfo.retdef^.deftype=pointerdef) or
  1046. (procinfo.retdef^.deftype=enumdef) or
  1047. (procinfo.retdef^.deftype=procvardef) or
  1048. (procinfo.retdef^.deftype=floatdef) or
  1049. (
  1050. (procinfo.retdef^.deftype=setdef) and
  1051. (psetdef(procinfo.retdef)^.settype=smallset)
  1052. ) then }
  1053. begin
  1054. { the space has been set in the local symtable }
  1055. procinfo.retoffset:=-funcretsym^.address;
  1056. if (procinfo.flags and pi_operator)<>0 then
  1057. {opsym^.address:=procinfo.call_offset; is wrong PM }
  1058. opsym^.address:=-procinfo.retoffset;
  1059. { eax is modified by a function }
  1060. {$ifdef i386}
  1061. usedinproc:=usedinproc or ($80 shr byte(R_EAX));
  1062. if is_64bitint(procinfo.retdef) then
  1063. usedinproc:=usedinproc or ($80 shr byte(R_EDX))
  1064. {$endif}
  1065. {$ifdef m68k}
  1066. usedinproc:=usedinproc or ($800 shr word(R_D0))
  1067. if is_64bitint(procinfo.retdef) then
  1068. usedinproc:=usedinproc or ($800 shr byte(R_D1))
  1069. {$endif}
  1070. end;
  1071. end;
  1072. {Unit initialization?.}
  1073. if (lexlevel=unit_init_level) and (current_module^.is_unit) then
  1074. if (token=_END) then
  1075. begin
  1076. consume(_END);
  1077. block:=nil;
  1078. end
  1079. else
  1080. begin
  1081. if token=_INITIALIZATION then
  1082. begin
  1083. current_module^.flags:=current_module^.flags or uf_init;
  1084. block:=statement_block(_INITIALIZATION);
  1085. end
  1086. else if (token=_FINALIZATION) then
  1087. begin
  1088. if (current_module^.flags and uf_finalize)<>0 then
  1089. block:=statement_block(_FINALIZATION)
  1090. else
  1091. begin
  1092. block:=nil;
  1093. exit;
  1094. end;
  1095. end
  1096. else
  1097. begin
  1098. current_module^.flags:=current_module^.flags or uf_init;
  1099. block:=statement_block(_BEGIN);
  1100. end;
  1101. end
  1102. else
  1103. block:=statement_block(_BEGIN);
  1104. end;
  1105. function assembler_block : ptree;
  1106. begin
  1107. read_declarations(false);
  1108. { temporary space is set, while the BEGIN of the procedure }
  1109. if symtablestack^.symtabletype=localsymtable then
  1110. procinfo.firsttemp := -symtablestack^.datasize
  1111. else
  1112. procinfo.firsttemp := 0;
  1113. { assembler code does not allocate }
  1114. { space for the return value }
  1115. if procinfo.retdef<>pdef(voiddef) then
  1116. begin
  1117. if ret_in_acc(procinfo.retdef) then
  1118. begin
  1119. { in assembler code the result should be directly in %eax
  1120. procinfo.retoffset:=procinfo.firsttemp-procinfo.retdef^.size;
  1121. procinfo.firsttemp:=procinfo.retoffset; }
  1122. {$ifdef i386}
  1123. usedinproc:=usedinproc or ($80 shr byte(R_EAX))
  1124. {$endif}
  1125. {$ifdef m68k}
  1126. usedinproc:=usedinproc or ($800 shr word(R_D0))
  1127. {$endif}
  1128. end
  1129. {
  1130. else if not is_fpu(procinfo.retdef) then
  1131. should we allow assembler functions of big elements ?
  1132. YES (FK)!!
  1133. Message(parser_e_asm_incomp_with_function_return);
  1134. }
  1135. end;
  1136. { set the framepointer to esp for assembler functions }
  1137. { but only if the are no local variables }
  1138. { added no parameter also (PM) }
  1139. if ((aktprocsym^.definition^.options and poassembler)<>0) and
  1140. (aktprocsym^.definition^.localst^.datasize=0) and
  1141. (aktprocsym^.definition^.parast^.datasize=0) and
  1142. not(ret_in_param(aktprocsym^.definition^.retdef)) then
  1143. begin
  1144. procinfo.framepointer:=stack_pointer;
  1145. { set the right value for parameters }
  1146. dec(aktprocsym^.definition^.parast^.address_fixup,target_os.size_of_pointer);
  1147. dec(procinfo.call_offset,target_os.size_of_pointer);
  1148. end;
  1149. { force the asm statement }
  1150. if token<>_ASM then
  1151. consume(_ASM);
  1152. Procinfo.Flags := ProcInfo.Flags Or pi_is_assembler;
  1153. assembler_block:=_asm_statement;
  1154. { becuase the END is already read we need to get the
  1155. last_endtoken_filepos here (PFV) }
  1156. last_endtoken_filepos:=tokenpos;
  1157. end;
  1158. end.
  1159. {
  1160. $Log$
  1161. Revision 1.82 1999-05-01 13:24:35 peter
  1162. * merged nasm compiler
  1163. * old asm moved to oldasm/
  1164. Revision 1.81 1999/04/26 13:31:42 peter
  1165. * release storenumber,double_checksum
  1166. Revision 1.80 1999/04/21 09:43:48 peter
  1167. * storenumber works
  1168. * fixed some typos in double_checksum
  1169. + incompatible types type1 and type2 message (with storenumber)
  1170. Revision 1.79 1999/04/16 12:14:49 pierre
  1171. * void pointer accepted with warning in tp and delphi mode
  1172. Revision 1.78 1999/04/15 12:58:14 pierre
  1173. * fix for bug0234
  1174. Revision 1.77 1999/04/15 09:01:33 peter
  1175. * fixed set loading
  1176. * object inheritance support for browser
  1177. Revision 1.76 1999/04/14 18:41:25 daniel
  1178. * Better use of routines in pbase and symtable. 4k code removed.
  1179. Revision 1.75 1999/04/14 09:14:53 peter
  1180. * first things to store the symbol/def number in the ppu
  1181. Revision 1.74 1999/04/09 12:22:06 pierre
  1182. * bug found by Peter for DirectWith code fixed
  1183. Revision 1.73 1999/04/06 11:21:57 peter
  1184. * more use of ttoken
  1185. Revision 1.72 1999/03/31 13:55:15 peter
  1186. * assembler inlining working for ag386bin
  1187. Revision 1.71 1999/03/10 11:23:29 pierre
  1188. * typecheck for exit(value) : resulttype was not set
  1189. Revision 1.70 1999/03/04 13:55:45 pierre
  1190. * some m68k fixes (still not compilable !)
  1191. * new(tobj) does not give warning if tobj has no VMT !
  1192. Revision 1.69 1999/03/02 02:56:15 peter
  1193. + stabs support for binary writers
  1194. * more fixes and missing updates from the previous commit :(
  1195. Revision 1.68 1999/02/26 00:48:23 peter
  1196. * assembler writers fixed for ag386bin
  1197. Revision 1.67 1999/02/22 13:07:01 pierre
  1198. + -b and -bl options work !
  1199. + cs_local_browser ($L+) is disabled if cs_browser ($Y+)
  1200. is not enabled when quitting global section
  1201. * local vars and procedures are not yet stored into PPU
  1202. Revision 1.66 1999/02/22 02:15:31 peter
  1203. * updates for ag386bin
  1204. Revision 1.65 1999/02/15 13:13:15 pierre
  1205. * fix for bug0216
  1206. Revision 1.64 1999/02/11 09:46:26 pierre
  1207. * fix for normal method calls inside static methods :
  1208. WARNING there were both parser and codegen errors !!
  1209. added static_call boolean to calln tree
  1210. Revision 1.63 1999/02/09 15:45:47 florian
  1211. + complex results for assembler functions, fixes bug0155
  1212. Revision 1.62 1999/01/27 13:06:57 pierre
  1213. * memory leak in case optimization fixed
  1214. Revision 1.61 1999/01/25 22:49:09 peter
  1215. * more fixes for the on bug with unknown id
  1216. Revision 1.60 1999/01/23 23:29:38 florian
  1217. * first running version of the new code generator
  1218. * when compiling exceptions under Linux fixed
  1219. Revision 1.59 1999/01/21 16:41:02 pierre
  1220. * fix for constructor inside with statements
  1221. Revision 1.58 1999/01/05 08:20:07 florian
  1222. * mainly problem with invalid case ranges fixed (reported by Jonas)
  1223. Revision 1.57 1998/12/29 18:48:15 jonas
  1224. + optimize pascal code surrounding assembler blocks
  1225. Revision 1.56 1998/12/23 22:52:56 peter
  1226. * fixed new(x) crash if x contains an error
  1227. Revision 1.55 1998/12/16 12:30:59 jonas
  1228. * released CaseRange
  1229. Revision 1.54 1998/12/15 22:32:24 jonas
  1230. + convert consecutive case labels to a single range (-dCaseRange)
  1231. Revision 1.53 1998/12/15 11:52:18 peter
  1232. * fixed dup release of statement label in case
  1233. Revision 1.52 1998/12/11 00:03:37 peter
  1234. + globtype,tokens,version unit splitted from globals
  1235. Revision 1.51 1998/12/10 09:47:24 florian
  1236. + basic operations with int64/qord (compiler with -dint64)
  1237. + rtti of enumerations extended: names are now written
  1238. Revision 1.50 1998/11/13 15:40:25 pierre
  1239. + added -Se in Makefile cvstest target
  1240. + lexlevel cleanup
  1241. normal_function_level main_program_level and unit_init_level defined
  1242. * tins_cache grown to A_EMMS (gave range check error in asm readers)
  1243. (test added in code !)
  1244. * -Un option was wrong
  1245. * _FAIL and _SELF only keyword inside
  1246. constructors and methods respectively
  1247. Revision 1.49 1998/11/12 12:55:17 pierre
  1248. * fix for bug0176 and bug0177
  1249. Revision 1.48 1998/11/05 23:43:24 peter
  1250. * fixed assembler directive and then not an ASM statement
  1251. Revision 1.47 1998/10/30 16:20:22 peter
  1252. * fixed dispose(destructor) crash when destructor didn't exists
  1253. Revision 1.46 1998/10/20 08:06:53 pierre
  1254. * several memory corruptions due to double freemem solved
  1255. => never use p^.loc.location:=p^.left^.loc.location;
  1256. + finally I added now by default
  1257. that ra386dir translates global and unit symbols
  1258. + added a first field in tsymtable and
  1259. a nextsym field in tsym
  1260. (this allows to obtain ordered type info for
  1261. records and objects in gdb !)
  1262. Revision 1.45 1998/10/19 08:55:01 pierre
  1263. * wrong stabs info corrected once again !!
  1264. + variable vmt offset with vmt field only if required
  1265. implemented now !!!
  1266. Revision 1.44 1998/10/13 13:10:27 peter
  1267. * new style for m68k/i386 infos and enums
  1268. Revision 1.43 1998/10/08 13:46:22 peter
  1269. * added eof message
  1270. * fixed unit init section parsing with finalize
  1271. Revision 1.42 1998/09/26 17:45:38 peter
  1272. + idtoken and only one token table
  1273. Revision 1.41 1998/09/24 23:49:15 peter
  1274. + aktmodeswitches
  1275. Revision 1.40 1998/09/23 21:53:04 florian
  1276. * the following doesn't work: on texception do, was a parser error, fixed
  1277. Revision 1.39 1998/09/21 10:26:07 peter
  1278. * merged fix
  1279. Revision 1.38.2.1 1998/09/21 10:24:43 peter
  1280. * fixed error recovery with with
  1281. Revision 1.38 1998/09/04 08:42:04 peter
  1282. * updated some error messages
  1283. Revision 1.37 1998/08/21 14:08:52 pierre
  1284. + TEST_FUNCRET now default (old code removed)
  1285. works also for m68k (at least compiles)
  1286. Revision 1.36 1998/08/20 21:36:41 peter
  1287. * fixed 'with object do' bug
  1288. Revision 1.35 1998/08/20 09:26:42 pierre
  1289. + funcret setting in underproc testing
  1290. compile with _dTEST_FUNCRET
  1291. Revision 1.34 1998/08/17 10:10:09 peter
  1292. - removed OLDPPU
  1293. Revision 1.33 1998/08/12 19:39:30 peter
  1294. * fixed some crashes
  1295. Revision 1.32 1998/08/10 14:50:17 peter
  1296. + localswitches, moduleswitches, globalswitches splitting
  1297. Revision 1.31 1998/08/02 16:41:59 florian
  1298. * on o : tobject do should also work now, the exceptsymtable shouldn't be
  1299. disposed by dellexlevel
  1300. Revision 1.30 1998/07/30 16:07:10 florian
  1301. * try ... expect <statement> end; works now
  1302. Revision 1.29 1998/07/30 13:30:37 florian
  1303. * final implemenation of exception support, maybe it needs
  1304. some fixes :)
  1305. Revision 1.28 1998/07/30 11:18:18 florian
  1306. + first implementation of try ... except on .. do end;
  1307. * limitiation of 65535 bytes parameters for cdecl removed
  1308. Revision 1.27 1998/07/28 21:52:55 florian
  1309. + implementation of raise and try..finally
  1310. + some misc. exception stuff
  1311. Revision 1.26 1998/07/27 21:57:14 florian
  1312. * fix to allow tv like stream registration:
  1313. @tmenu.load doesn't work if load had parameters or if load was only
  1314. declared in an anchestor class of tmenu
  1315. Revision 1.25 1998/07/14 21:46:53 peter
  1316. * updated messages file
  1317. Revision 1.24 1998/07/10 10:48:42 peter
  1318. * fixed realnumber scanning
  1319. * [] after asmblock was not uppercased anymore
  1320. Revision 1.23 1998/06/25 08:48:18 florian
  1321. * first version of rtti support
  1322. Revision 1.22 1998/06/24 14:48:36 peter
  1323. * ifdef newppu -> ifndef oldppu
  1324. Revision 1.21 1998/06/24 14:06:34 peter
  1325. * fixed the name changes
  1326. Revision 1.20 1998/06/23 14:00:16 peter
  1327. * renamed RA* units
  1328. Revision 1.19 1998/06/08 22:59:50 peter
  1329. * smartlinking works for win32
  1330. * some defines to exclude some compiler parts
  1331. Revision 1.18 1998/06/05 14:37:35 pierre
  1332. * fixes for inline for operators
  1333. * inline procedure more correctly restricted
  1334. Revision 1.17 1998/06/04 09:55:43 pierre
  1335. * demangled name of procsym reworked to become independant of the mangling scheme
  1336. Revision 1.16 1998/06/02 17:03:04 pierre
  1337. * with node corrected for objects
  1338. * small bugs for SUPPORT_MMX fixed
  1339. Revision 1.15 1998/05/30 14:31:06 peter
  1340. + $ASMMODE
  1341. Revision 1.14 1998/05/29 09:58:14 pierre
  1342. * OPR_REGISTER for 1 arg was missing in ratti386.pas
  1343. (probably a merging problem)
  1344. * errors at start of line were lost
  1345. Revision 1.13 1998/05/28 17:26:50 peter
  1346. * fixed -R switch, it didn't work after my previous akt/init patch
  1347. * fixed bugs 110,130,136
  1348. Revision 1.12 1998/05/21 19:33:33 peter
  1349. + better procedure directive handling and only one table
  1350. Revision 1.11 1998/05/20 09:42:35 pierre
  1351. + UseTokenInfo now default
  1352. * unit in interface uses and implementation uses gives error now
  1353. * only one error for unknown symbol (uses lastsymknown boolean)
  1354. the problem came from the label code !
  1355. + first inlined procedures and function work
  1356. (warning there might be allowed cases were the result is still wrong !!)
  1357. * UseBrower updated gives a global list of all position of all used symbols
  1358. with switch -gb
  1359. Revision 1.10 1998/05/11 13:07:56 peter
  1360. + $ifdef NEWPPU for the new ppuformat
  1361. + $define GDB not longer required
  1362. * removed all warnings and stripped some log comments
  1363. * no findfirst/findnext anymore to remove smartlink *.o files
  1364. Revision 1.9 1998/05/06 08:38:46 pierre
  1365. * better position info with UseTokenInfo
  1366. UseTokenInfo greatly simplified
  1367. + added check for changed tree after first time firstpass
  1368. (if we could remove all the cases were it happen
  1369. we could skip all firstpass if firstpasscount > 1)
  1370. Only with ExtDebug
  1371. Revision 1.8 1998/05/05 12:05:42 florian
  1372. * problems with properties fixed
  1373. * crash fixed: i:=l when i and l are undefined, was a problem with
  1374. implementation of private/protected
  1375. Revision 1.7 1998/05/01 16:38:46 florian
  1376. * handling of private and protected fixed
  1377. + change_keywords_to_tp implemented to remove
  1378. keywords which aren't supported by tp
  1379. * break and continue are now symbols of the system unit
  1380. + widestring, longstring and ansistring type released
  1381. Revision 1.6 1998/04/30 15:59:42 pierre
  1382. * GDB works again better :
  1383. correct type info in one pass
  1384. + UseTokenInfo for better source position
  1385. * fixed one remaining bug in scanner for line counts
  1386. * several little fixes
  1387. Revision 1.5 1998/04/29 10:33:59 pierre
  1388. + added some code for ansistring (not complete nor working yet)
  1389. * corrected operator overloading
  1390. * corrected nasm output
  1391. + started inline procedures
  1392. + added starstarn : use ** for exponentiation (^ gave problems)
  1393. + started UseTokenInfo cond to get accurate positions
  1394. Revision 1.4 1998/04/08 16:58:05 pierre
  1395. * several bugfixes
  1396. ADD ADC and AND are also sign extended
  1397. nasm output OK (program still crashes at end
  1398. and creates wrong assembler files !!)
  1399. procsym types sym in tdef removed !!
  1400. }