parser.pas 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574
  1. {
  2. $Id$
  3. Copyright (c) 1993-98 by Florian Klaempfl
  4. This unit does the parsing process
  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. {$ifdef tp}
  19. {$E+,N+,D+,F+}
  20. {$endif}
  21. unit parser;
  22. interface
  23. procedure compile(const filename:string;compile_system:boolean);
  24. procedure initparser;
  25. implementation
  26. uses
  27. cobjects,verbose,comphook,systems,globals,
  28. symtable,files,aasm,hcodegen,
  29. assemble,link,script,gendef,
  30. {$ifdef UseBrowser}
  31. browser,
  32. {$endif UseBrowser}
  33. scanner,pbase,pdecl,psystem,pmodules;
  34. procedure initparser;
  35. begin
  36. forwardsallowed:=false;
  37. { ^M means a string or a char, because we don't parse a }
  38. { type declaration }
  39. ignore_equal:=false;
  40. { we didn't parse a object or class declaration }
  41. { and no function header }
  42. testcurobject:=0;
  43. { a long time, this was forgotten }
  44. aktprocsym:=nil;
  45. current_module:=nil;
  46. loaded_units.init;
  47. usedunits.init;
  48. { global switches }
  49. aktglobalswitches:=initglobalswitches;
  50. { memory sizes }
  51. if heapsize=0 then
  52. heapsize:=target_info.heapsize;
  53. if stacksize=0 then
  54. stacksize:=target_info.stacksize;
  55. end;
  56. procedure default_macros;
  57. var
  58. hp : pstring_item;
  59. begin
  60. { commandline }
  61. hp:=pstring_item(initdefines.first);
  62. while assigned(hp) do
  63. begin
  64. def_macro(hp^.str^);
  65. hp:=pstring_item(hp^.next);
  66. end;
  67. { set macros for version checking }
  68. set_macro('FPC_VERSION',version_nr);
  69. set_macro('FPC_RELEASE',release_nr);
  70. set_macro('FPC_PATCH',patch_nr);
  71. end;
  72. procedure compile(const filename:string;compile_system:boolean);
  73. var
  74. { scanner }
  75. oldtoken : ttoken;
  76. oldtokenpos : tfileposinfo;
  77. oldc : char;
  78. oldpattern,
  79. oldorgpattern : string;
  80. old_block_type : tblock_type;
  81. oldcurrent_scanner : pscannerfile;
  82. { symtable }
  83. oldmacros,
  84. oldrefsymtable,
  85. oldsymtablestack : psymtable;
  86. oldprocprefix : string;
  87. oldaktprocsym : pprocsym;
  88. { cg }
  89. oldnextlabelnr : longint;
  90. oldparse_only : boolean;
  91. { asmlists }
  92. oldimports,
  93. oldexports,
  94. oldresource,
  95. oldrttilist,
  96. oldbsssegment,
  97. olddatasegment,
  98. oldcodesegment,
  99. oldexprasmlist,
  100. olddebuglist,
  101. oldinternals,
  102. oldexternals,
  103. oldconsts : paasmoutput;
  104. { akt.. things }
  105. oldaktlocalswitches : tlocalswitches;
  106. oldaktmoduleswitches : tmoduleswitches;
  107. oldaktfilepos : tfileposinfo;
  108. oldaktpackrecords : word;
  109. oldaktoutputformat : tasm;
  110. oldaktoptprocessor : tprocessors;
  111. oldaktasmmode : tasmmode;
  112. label
  113. done;
  114. begin
  115. inc(compile_level);
  116. { save symtable state }
  117. oldsymtablestack:=symtablestack;
  118. oldrefsymtable:=refsymtable;
  119. oldmacros:=macros;
  120. oldprocprefix:=procprefix;
  121. oldaktprocsym:=aktprocsym;
  122. { save scanner state }
  123. oldc:=c;
  124. oldpattern:=pattern;
  125. oldorgpattern:=orgpattern;
  126. oldtoken:=token;
  127. old_block_type:=block_type;
  128. oldtokenpos:=tokenpos;
  129. oldcurrent_scanner:=current_scanner;
  130. { save cg }
  131. oldnextlabelnr:=nextlabelnr;
  132. oldparse_only:=parse_only;
  133. { save assembler lists }
  134. olddatasegment:=datasegment;
  135. oldbsssegment:=bsssegment;
  136. oldcodesegment:=codesegment;
  137. olddebuglist:=debuglist;
  138. oldexternals:=externals;
  139. oldinternals:=internals;
  140. oldconsts:=consts;
  141. oldrttilist:=rttilist;
  142. oldexprasmlist:=exprasmlist;
  143. oldimports:=importssection;
  144. oldexports:=exportssection;
  145. oldresource:=resourcesection;
  146. { save akt... state }
  147. oldaktlocalswitches:=aktlocalswitches;
  148. oldaktmoduleswitches:=aktmoduleswitches;
  149. oldaktpackrecords:=aktpackrecords;
  150. oldaktoutputformat:=aktoutputformat;
  151. oldaktoptprocessor:=aktoptprocessor;
  152. oldaktasmmode:=aktasmmode;
  153. oldaktfilepos:=aktfilepos;
  154. { show info }
  155. Message1(parser_i_compiling,filename);
  156. { reset symtable }
  157. symtablestack:=nil;
  158. refsymtable:=nil;
  159. procprefix:='';
  160. aktprocsym:=nil;
  161. { macros }
  162. macros:=new(psymtable,init(macrosymtable));
  163. macros^.name:=stringdup('Conditionals for '+filename);
  164. default_macros;
  165. { reset the unit or create a new program }
  166. if assigned(current_module) then
  167. begin
  168. current_module^.sourcefiles.done;
  169. current_module^.sourcefiles.init;
  170. current_module^.used_units.done;
  171. current_module^.used_units.init;
  172. current_module^.linkofiles.done;
  173. current_module^.linkofiles.init;
  174. current_module^.linkstaticlibs.done;
  175. current_module^.linkstaticlibs.init;
  176. current_module^.linksharedlibs.done;
  177. current_module^.linksharedlibs.init;
  178. end
  179. else
  180. begin
  181. current_module:=new(pmodule,init(filename,false));
  182. main_module:=current_module;
  183. end;
  184. { Load current state from the init values }
  185. aktlocalswitches:=initlocalswitches;
  186. aktmoduleswitches:=initmoduleswitches;
  187. aktpackrecords:=initpackrecords;
  188. aktoutputformat:=initoutputformat;
  189. aktoptprocessor:=initoptprocessor;
  190. aktasmmode:=initasmmode;
  191. { we need this to make the system unit }
  192. if compile_system then
  193. aktmoduleswitches:=aktmoduleswitches+[cs_compilesystem];
  194. { startup scanner }
  195. current_scanner:=new(pscannerfile,Init(filename));
  196. token:=current_scanner^.yylex;
  197. { global switches are read, so further changes aren't allowed }
  198. current_module^.in_main:=true;
  199. { init code generator for a new module }
  200. codegen_newmodule;
  201. {$ifdef GDB}
  202. reset_gdb_info;
  203. {$endif GDB}
  204. { Handle things which need to be once }
  205. if (compile_level=1) then
  206. begin
  207. { open assembler response }
  208. AsmRes.Init('ppas');
  209. end;
  210. { load system unit always }
  211. loadsystemunit;
  212. registerdef:=true;
  213. make_ref:=true;
  214. { current return type is void }
  215. procinfo.retdef:=voiddef;
  216. { reset lexical level }
  217. lexlevel:=0;
  218. { parse source }
  219. if (token=_UNIT) or (compile_level>1) then
  220. begin
  221. current_module^.is_unit:=true;
  222. { If the compile level > 1 we get a nice "unit expected" error
  223. message if we are trying to use a program as unit.}
  224. proc_unit;
  225. if current_module^.compiled then
  226. goto done;
  227. end
  228. else
  229. begin
  230. proc_program(token=_LIBRARY);
  231. end;
  232. if status.errorcount=0 then
  233. begin
  234. GenerateAsm(filename);
  235. if (cs_smartlink in aktmoduleswitches) then
  236. begin
  237. Linker.SetLibName(current_module^.libfilename^);
  238. Linker.MakeStaticLibrary(SmartLinkPath(FileName),SmartLinkFilesCnt);
  239. end;
  240. { add the files for the linker from current_module, this must be
  241. after the makestaticlibrary, because it will add the library
  242. name (PFV) }
  243. addlinkerfiles(current_module);
  244. { Check linking => we are at first level in compile }
  245. if (compile_level=1) then
  246. begin
  247. if (cs_link_deffile in aktglobalswitches) then
  248. deffile.writefile;
  249. if (not current_module^.is_unit) then
  250. begin
  251. if Linker.ExeName='' then
  252. Linker.SetExeName(FileName);
  253. Linker.MakeExecutable;
  254. end;
  255. end;
  256. end
  257. else
  258. Message1(unit_f_errors_in_unit,tostr(status.errorcount));
  259. done:
  260. { clear memory }
  261. {$ifdef Splitheap}
  262. if testsplit then
  263. begin
  264. { temp heap should be empty after that !!!}
  265. codegen_donemodule;
  266. Releasetempheap;
  267. end;
  268. {$endif Splitheap}
  269. { restore old state, close trees }
  270. {$ifdef VER0_99_5}
  271. if dispose_asm_lists then
  272. {$else}
  273. {$ifndef go32v2}
  274. {$ifndef linux}
  275. if dispose_asm_lists then
  276. {$endif}
  277. {$endif}
  278. {$endif}
  279. codegen_donemodule;
  280. {$ifdef GDB}
  281. reset_gdb_info;
  282. {$endif GDB}
  283. { free ppu }
  284. if assigned(current_module^.ppufile) then
  285. begin
  286. dispose(current_module^.ppufile,done);
  287. current_module^.ppufile:=nil;
  288. end;
  289. { free scanner }
  290. dispose(current_scanner,done);
  291. { free macros }
  292. {!!! No check for unused macros yet !!! }
  293. dispose(macros,done);
  294. { restore scanner }
  295. c:=oldc;
  296. pattern:=oldpattern;
  297. orgpattern:=oldorgpattern;
  298. token:=oldtoken;
  299. tokenpos:=oldtokenpos;
  300. block_type:=old_block_type;
  301. current_scanner:=oldcurrent_scanner;
  302. { restore cg }
  303. nextlabelnr:=oldnextlabelnr;
  304. parse_only:=oldparse_only;
  305. { restore asmlists }
  306. exprasmlist:=oldexprasmlist;
  307. datasegment:=olddatasegment;
  308. bsssegment:=oldbsssegment;
  309. codesegment:=oldcodesegment;
  310. consts:=oldconsts;
  311. debuglist:=olddebuglist;
  312. externals:=oldexternals;
  313. internals:=oldinternals;
  314. importssection:=oldimports;
  315. exportssection:=oldexports;
  316. resourcesection:=oldresource;
  317. rttilist:=oldrttilist;
  318. { restore symtable state }
  319. if (compile_level>1) then
  320. begin
  321. refsymtable:=oldrefsymtable;
  322. symtablestack:=oldsymtablestack;
  323. end;
  324. macros:=oldmacros;
  325. aktprocsym:=oldaktprocsym;
  326. procprefix:=oldprocprefix;
  327. { restore current state }
  328. aktlocalswitches:=oldaktlocalswitches;
  329. aktmoduleswitches:=oldaktmoduleswitches;
  330. aktpackrecords:=oldaktpackrecords;
  331. aktoutputformat:=oldaktoutputformat;
  332. aktoptprocessor:=oldaktoptprocessor;
  333. aktasmmode:=oldaktasmmode;
  334. aktfilepos:=oldaktfilepos;
  335. { Shut down things when the last file is compiled }
  336. if (compile_level=1) then
  337. begin
  338. { Close script }
  339. if (not AsmRes.Empty) then
  340. begin
  341. Message1(exec_i_closing_script,AsmRes.Fn);
  342. AsmRes.WriteToDisk;
  343. end;
  344. {$ifdef UseBrowser}
  345. { Write Browser }
  346. if cs_browser in initswitches then
  347. begin
  348. Message1(parser_i_writing_browser_log,Browse.Fname);
  349. Browse.CreateLog;
  350. write_browser_log;
  351. Browse.CloseLog;
  352. end;
  353. {$endif UseBrowser}
  354. end;
  355. dec(compile_level);
  356. end;
  357. end.
  358. {
  359. $Log$
  360. Revision 1.35 1998-08-12 19:22:09 peter
  361. * reset also the link* lists when recompiling an existing unit
  362. Revision 1.34 1998/08/10 23:58:56 peter
  363. * fixed asmlist dispose for 0.99.5
  364. Revision 1.33 1998/08/10 14:50:07 peter
  365. + localswitches, moduleswitches, globalswitches splitting
  366. Revision 1.32 1998/08/10 10:18:28 peter
  367. + Compiler,Comphook unit which are the new interface units to the
  368. compiler
  369. Revision 1.31 1998/07/14 21:46:46 peter
  370. * updated messages file
  371. Revision 1.30 1998/07/14 14:46:49 peter
  372. * released NEWINPUT
  373. Revision 1.29 1998/07/07 11:19:59 peter
  374. + NEWINPUT for a better inputfile and scanner object
  375. Revision 1.28 1998/06/25 11:15:33 pierre
  376. * ppu files where not closed in newppu !!
  377. second compilation was impossible due to too many opened files
  378. (not visible in 'make cycle' as we remove all the ppu files)
  379. Revision 1.27 1998/06/17 14:10:15 peter
  380. * small os2 fixes
  381. * fixed interdependent units with newppu (remake3 under linux works now)
  382. Revision 1.26 1998/06/16 08:56:23 peter
  383. + targetcpu
  384. * cleaner pmodules for newppu
  385. Revision 1.25 1998/06/15 15:38:07 pierre
  386. * small bug in systems.pas corrected
  387. + operators in different units better hanlded
  388. Revision 1.24 1998/06/13 00:10:08 peter
  389. * working browser and newppu
  390. * some small fixes against crashes which occured in bp7 (but not in
  391. fpc?!)
  392. Revision 1.23 1998/06/08 22:59:48 peter
  393. * smartlinking works for win32
  394. * some defines to exclude some compiler parts
  395. Revision 1.22 1998/06/05 17:47:28 peter
  396. * some better uses clauses
  397. Revision 1.21 1998/06/04 23:51:49 peter
  398. * m68k compiles
  399. + .def file creation moved to gendef.pas so it could also be used
  400. for win32
  401. Revision 1.20 1998/06/03 22:48:55 peter
  402. + wordbool,longbool
  403. * rename bis,von -> high,low
  404. * moved some systemunit loading/creating to psystem.pas
  405. Revision 1.19 1998/05/27 19:45:04 peter
  406. * symtable.pas splitted into includefiles
  407. * symtable adapted for $ifdef NEWPPU
  408. Revision 1.18 1998/05/23 01:21:15 peter
  409. + aktasmmode, aktoptprocessor, aktoutputformat
  410. + smartlink per module $SMARTLINK-/+ (like MMX) and moved to aktswitches
  411. + $LIBNAME to set the library name where the unit will be put in
  412. * splitted cgi386 a bit (codeseg to large for bp7)
  413. * nasm, tasm works again. nasm moved to ag386nsm.pas
  414. Revision 1.17 1998/05/20 09:42:34 pierre
  415. + UseTokenInfo now default
  416. * unit in interface uses and implementation uses gives error now
  417. * only one error for unknown symbol (uses lastsymknown boolean)
  418. the problem came from the label code !
  419. + first inlined procedures and function work
  420. (warning there might be allowed cases were the result is still wrong !!)
  421. * UseBrower updated gives a global list of all position of all used symbols
  422. with switch -gb
  423. Revision 1.16 1998/05/12 10:47:00 peter
  424. * moved printstatus to verb_def
  425. + V_Normal which is between V_Error and V_Warning and doesn't have a
  426. prefix like error: warning: and is included in V_Default
  427. * fixed some messages
  428. * first time parameter scan is only for -v and -T
  429. - removed old style messages
  430. Revision 1.15 1998/05/11 13:07:54 peter
  431. + $ifdef NEWPPU for the new ppuformat
  432. + $define GDB not longer required
  433. * removed all warnings and stripped some log comments
  434. * no findfirst/findnext anymore to remove smartlink *.o files
  435. Revision 1.14 1998/05/06 18:36:53 peter
  436. * tai_section extended with code,data,bss sections and enumerated type
  437. * ident 'compiled by FPC' moved to pmodules
  438. * small fix for smartlink
  439. Revision 1.13 1998/05/06 08:38:42 pierre
  440. * better position info with UseTokenInfo
  441. UseTokenInfo greatly simplified
  442. + added check for changed tree after first time firstpass
  443. (if we could remove all the cases were it happen
  444. we could skip all firstpass if firstpasscount > 1)
  445. Only with ExtDebug
  446. Revision 1.12 1998/05/04 17:54:28 peter
  447. + smartlinking works (only case jumptable left todo)
  448. * redesign of systems.pas to support assemblers and linkers
  449. + Unitname is now also in the PPU-file, increased version to 14
  450. Revision 1.11 1998/05/01 16:38:45 florian
  451. * handling of private and protected fixed
  452. + change_keywords_to_tp implemented to remove
  453. keywords which aren't supported by tp
  454. * break and continue are now symbols of the system unit
  455. + widestring, longstring and ansistring type released
  456. Revision 1.10 1998/05/01 07:43:56 florian
  457. + basics for rtti implemented
  458. + switch $m (generate rtti for published sections)
  459. Revision 1.9 1998/04/30 15:59:40 pierre
  460. * GDB works again better :
  461. correct type info in one pass
  462. + UseTokenInfo for better source position
  463. * fixed one remaining bug in scanner for line counts
  464. * several little fixes
  465. Revision 1.8 1998/04/29 10:33:55 pierre
  466. + added some code for ansistring (not complete nor working yet)
  467. * corrected operator overloading
  468. * corrected nasm output
  469. + started inline procedures
  470. + added starstarn : use ** for exponentiation (^ gave problems)
  471. + started UseTokenInfo cond to get accurate positions
  472. Revision 1.7 1998/04/27 23:10:28 peter
  473. + new scanner
  474. * $makelib -> if smartlink
  475. * small filename fixes pmodule.setfilename
  476. * moved import from files.pas -> import.pas
  477. Revision 1.6 1998/04/21 10:16:48 peter
  478. * patches from strasbourg
  479. * objects is not used anymore in the fpc compiled version
  480. Revision 1.5 1998/04/10 14:41:43 peter
  481. * removed some Hints
  482. * small speed optimization for AsmLn
  483. Revision 1.4 1998/04/08 16:58:03 pierre
  484. * several bugfixes
  485. ADD ADC and AND are also sign extended
  486. nasm output OK (program still crashes at end
  487. and creates wrong assembler files !!)
  488. procsym types sym in tdef removed !!
  489. Revision 1.3 1998/04/07 22:45:04 florian
  490. * bug0092, bug0115 and bug0121 fixed
  491. + packed object/class/array
  492. }