| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688 | {    Copyright (c) 1998-2002 by Florian Klaempfl    This unit does the parsing process    This program is free software; you can redistribute it and/or modify    it under the terms of the GNU General Public License as published by    the Free Software Foundation; either version 2 of the License, or    (at your option) any later version.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the    GNU General Public License for more details.    You should have received a copy of the GNU General Public License    along with this program; if not, write to the Free Software    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. ****************************************************************************}unit parser;{$i fpcdefs.inc}interface{$ifdef PREPROCWRITE}    procedure preprocess(const filename:string);{$endif PREPROCWRITE}    procedure compile(const filename:string);    procedure initparser;    procedure doneparser;implementation    uses{$IFNDEF MACOS_USE_FAKE_SYSUTILS}      sysutils,{$ENDIF MACOS_USE_FAKE_SYSUTILS}      cutils,cclasses,      globtype,version,tokens,systems,globals,verbose,      symbase,symtable,symsym,      finput,fmodule,fppu,      aasmbase,aasmtai,      cgbase,      script,gendef,{$ifdef BrowserCol}      browcol,{$endif BrowserCol}{$ifdef BrowserLog}      browlog,{$endif BrowserLog}{$ifdef GDB}      gdb,{$endif GDB}      comphook,      scanner,scandir,      pbase,ptype,psystem,pmodules,psub,      cresstr,cpuinfo,procinfo;    procedure initparser;      begin         { ^M means a string or a char, because we don't parse a }         { type declaration                                      }         ignore_equal:=false;         { we didn't parse a object or class declaration }         { and no function header                        }         testcurobject:=0;         { Current compiled module/proc }         objectlibrary:=nil;         current_module:=nil;         compiled_module:=nil;         current_procinfo:=nil;         SetCompileModule(nil);         loaded_units:=TLinkedList.Create;         usedunits:=TLinkedList.Create;         { global switches }         aktglobalswitches:=initglobalswitches;         aktsourcecodepage:=initsourcecodepage;         { initialize scanner }         InitScanner;         InitScannerDirectives;         { scanner }         c:=#0;         pattern:='';         orgpattern:='';         current_scanner:=nil;         { register all nodes and tais }         registernodes;         registertais;         { memory sizes }         if stacksize=0 then           stacksize:=target_info.stacksize;         { open assembler response }         if cs_link_on_target in aktglobalswitches then           GenerateAsmRes(outputexedir+inputfile+'_ppas')         else           GenerateAsmRes(outputexedir+'ppas');         { open deffile }         DefFile:=TDefFile.Create(outputexedir+inputfile+target_info.defext);         { list of generated .o files, so the linker can remove them }         SmartLinkOFiles:=TStringList.Create;         { codegen }         if paraprintnodetree<>0 then           printnode_reset;         { target specific stuff }         case target_info.system of           system_powerpc_morphos:             include(supported_calling_conventions,pocall_syscall);           system_m68k_amiga:             include(supported_calling_conventions,pocall_syscall);         end;      end;    procedure doneparser;      begin         { Reset current compiling info, so destroy routines can't           reference the data that might already be destroyed }         objectlibrary:=nil;         current_module:=nil;         compiled_module:=nil;         current_procinfo:=nil;         SetCompileModule(nil);         { unload units }         loaded_units.free;         usedunits.free;         { if there was an error in the scanner, the scanner is           still assinged }         if assigned(current_scanner) then          begin            current_scanner.free;            current_scanner:=nil;          end;         { close scanner }         DoneScanner;         { close ppas,deffile }         asmres.free;         deffile.free;         { free list of .o files }         SmartLinkOFiles.Free;      end;{$ifdef PREPROCWRITE}    procedure preprocess(const filename:string);      var        i : longint;      begin         new(preprocfile,init('pre'));       { initialize a module }         current_module:=new(pmodule,init(filename,false));         macrosymtablestack:= initialmacrosymtable;         current_module.localmacrosymtable:= tmacrosymtable.create(false);         current_module.localmacrosymtable.next:= initialmacrosymtable;         macrosymtablestack:= current_module.localmacrosymtable;         ConsolidateMode;         main_module:=current_module;       { startup scanner, and save in current_module }         current_scanner:=new(pscannerfile,Init(filename));         current_module.scanner:=current_scanner;       { loop until EOF is found }         repeat           current_scanner^.readtoken;           preprocfile^.AddSpace;           case token of             _ID :               begin                 preprocfile^.Add(orgpattern);               end;             _REALNUMBER,             _INTCONST :               preprocfile^.Add(pattern);             _CSTRING :               begin                 i:=0;                 while (i<length(pattern)) do                  begin                    inc(i);                    if pattern[i]='''' then                     begin                       insert('''',pattern,i);                       inc(i);                     end;                  end;                 preprocfile^.Add(''''+pattern+'''');               end;             _CCHAR :               begin                 case pattern[1] of                   #39 :                     pattern:='''''''';                   #0..#31,                   #128..#255 :                     begin                       str(ord(pattern[1]),pattern);                       pattern:='#'+pattern;                     end;                   else                     pattern:=''''+pattern[1]+'''';                 end;                 preprocfile^.Add(pattern);               end;             _EOF :               break;             else               preprocfile^.Add(tokeninfo^[token].str)           end;         until false;       { free scanner }         dispose(current_scanner,done);         current_scanner:=nil;       { close }         dispose(preprocfile,done);      end;{$endif PREPROCWRITE}{*****************************************************************************                      Create information for a new module*****************************************************************************}    procedure init_module;      begin         { Create assembler output lists for CG }         exprasmlist:=taasmoutput.create;         datasegment:=taasmoutput.create;         codesegment:=taasmoutput.create;         bsssegment:=taasmoutput.create;         debuglist:=taasmoutput.create;         withdebuglist:=taasmoutput.create;         consts:=taasmoutput.create;         rttilist:=taasmoutput.create;         picdata:=taasmoutput.create;         if target_info.system=system_powerpc_darwin then           picdata.concat(tai_simple.create(ait_non_lazy_symbol_pointer));         ResourceStringList:=Nil;         importssection:=nil;         exportssection:=nil;         resourcesection:=nil;         { Resource strings }         ResourceStrings:=TResourceStrings.Create;         { use the librarydata from current_module }         objectlibrary:=current_module.librarydata;      end;    procedure done_module;{$ifdef MEMDEBUG}      var        d : tmemdebug;{$endif}      begin{$ifdef MEMDEBUG}         d:=tmemdebug.create(current_module.modulename^+' - asmlists');{$endif}         exprasmlist.free;         codesegment.free;         bsssegment.free;         datasegment.free;         debuglist.free;         withdebuglist.free;         consts.free;         rttilist.free;         picdata.free;         if assigned(ResourceStringList) then          ResourceStringList.free;         if assigned(importssection) then          importssection.free;         if assigned(exportssection) then          exportssection.free;         if assigned(resourcesection) then          resourcesection.free;{$ifdef MEMDEBUG}         d.free;{$endif}         { resource strings }         ResourceStrings.free;         objectlibrary:=nil;      end;{*****************************************************************************                             Compile a source file*****************************************************************************}    procedure compile(const filename:string);      type        polddata=^tolddata;        tolddata=record        { scanner }          oldidtoken,          oldtoken       : ttoken;          oldtokenpos    : tfileposinfo;          oldc           : char;          oldpattern,          oldorgpattern  : string;          old_block_type : tblock_type;        { symtable }          oldrefsymtable,          olddefaultsymtablestack,          oldsymtablestack : tsymtable;          olddefaultmacrosymtablestack,          oldmacrosymtablestack : tsymtable;          oldaktprocsym    : tprocsym;        { cg }          oldparse_only  : boolean;        { asmlists }          oldimports,          oldexports,          oldresource,          oldrttilist,          oldpicdata,          oldresourcestringlist,          oldbsssegment,          olddatasegment,          oldcodesegment,          oldexprasmlist,          olddebuglist,          oldwithdebuglist,          oldconsts     : taasmoutput;          oldobjectlibrary : tasmlibrarydata;        { resourcestrings }          OldResourceStrings : tResourceStrings;        { akt.. things }          oldaktlocalswitches  : tlocalswitches;          oldaktmoduleswitches : tmoduleswitches;          oldaktfilepos      : tfileposinfo;          oldaktpackrecords,          oldaktpackenum       : shortint;          oldaktmaxfpuregisters : longint;          oldaktalignment  : talignmentinfo;          oldaktoutputformat : tasm;          oldaktspecificoptprocessor,          oldaktoptprocessor : tprocessors;          oldaktfputype      : tfputype;          oldaktasmmode      : tasmmode;          oldaktinterfacetype: tinterfacetypes;          oldaktmodeswitches : tmodeswitches;          old_compiled_module : tmodule;          oldcurrent_procinfo : tprocinfo;          oldaktdefproccall : tproccalloption;          oldsourcecodepage : tcodepagestring;{$ifdef GDB}          store_dbx : plongint;{$endif GDB}        end;      var         olddata : polddata;       begin         inc(compile_level);         parser_current_file:=filename;         { Uses heap memory instead of placing everything on the           stack. This is needed because compile() can be called           recursively }         new(olddata);         with olddata^ do          begin            old_compiled_module:=compiled_module;          { save symtable state }            oldsymtablestack:=symtablestack;            oldmacrosymtablestack:=macrosymtablestack;            olddefaultsymtablestack:=defaultsymtablestack;            olddefaultmacrosymtablestack:=defaultmacrosymtablestack;            oldrefsymtable:=refsymtable;            oldcurrent_procinfo:=current_procinfo;            oldaktdefproccall:=aktdefproccall;          { save scanner state }            oldc:=c;            oldpattern:=pattern;            oldorgpattern:=orgpattern;            oldtoken:=token;            oldidtoken:=idtoken;            old_block_type:=block_type;            oldtokenpos:=akttokenpos;            oldsourcecodepage:=aktsourcecodepage;          { save cg }            oldparse_only:=parse_only;          { save assembler lists }            olddatasegment:=datasegment;            oldbsssegment:=bsssegment;            oldcodesegment:=codesegment;            olddebuglist:=debuglist;            oldwithdebuglist:=withdebuglist;            oldconsts:=consts;            oldrttilist:=rttilist;            oldpicdata:=picdata;            oldexprasmlist:=exprasmlist;            oldimports:=importssection;            oldexports:=exportssection;            oldresource:=resourcesection;            oldresourcestringlist:=resourcestringlist;            oldobjectlibrary:=objectlibrary;            OldResourceStrings:=ResourceStrings;          { save akt... state }          { handle the postponed case first }           if localswitcheschanged then             begin               aktlocalswitches:=nextaktlocalswitches;               localswitcheschanged:=false;             end;            oldaktlocalswitches:=aktlocalswitches;            oldaktmoduleswitches:=aktmoduleswitches;            oldaktalignment:=aktalignment;            oldaktpackenum:=aktpackenum;            oldaktpackrecords:=aktpackrecords;            oldaktfputype:=aktfputype;            oldaktmaxfpuregisters:=aktmaxfpuregisters;            oldaktoutputformat:=aktoutputformat;            oldaktoptprocessor:=aktoptprocessor;            oldaktspecificoptprocessor:=aktspecificoptprocessor;            oldaktasmmode:=aktasmmode;            oldaktinterfacetype:=aktinterfacetype;            oldaktfilepos:=aktfilepos;            oldaktmodeswitches:=aktmodeswitches;{$ifdef GDB}            store_dbx:=dbx_counter;            dbx_counter:=nil;{$endif GDB}          end;       { reset parser, a previous fatal error could have left these variables in an unreliable state, this is         important for the IDE }         afterassignment:=false;         in_args:=false;         got_addrn:=false;         getprocvardef:=nil;       { show info }         Message1(parser_i_compiling,filename);       { reset symtable }         symtablestack:=nil;         macrosymtablestack:=nil;         defaultsymtablestack:=nil;         defaultmacrosymtablestack:=nil;         systemunit:=nil;         refsymtable:=nil;         aktdefproccall:=initdefproccall;         registerdef:=true;         aktexceptblock:=0;         exceptblockcounter:=0;         aktmaxfpuregisters:=-1;       { reset the unit or create a new program }         { a unit compiled at command line must be inside the loaded_unit list }         if (compile_level=1) then           begin             if assigned(current_module) then               internalerror(200501158);             current_module:=tppumodule.create(nil,filename,'',false);             addloadedunit(current_module);             main_module:=current_module;             current_module.state:=ms_compile;           end;         if not(assigned(current_module) and                (current_module.state in [ms_compile,ms_second_compile])) then           internalerror(200212281);         { Set the module to use for verbose }         compiled_module:=current_module;         SetCompileModule(current_module);         Fillchar(aktfilepos,0,sizeof(aktfilepos));         { Load current state from the init values }         aktlocalswitches:=initlocalswitches;         aktmoduleswitches:=initmoduleswitches;         aktmodeswitches:=initmodeswitches;         {$IFDEF Testvarsets}         aktsetalloc:=initsetalloc;         {$ENDIF}         aktalignment:=initalignment;         aktfputype:=initfputype;         aktpackenum:=initpackenum;         aktpackrecords:=0;         aktoutputformat:=initoutputformat;         set_target_asm(aktoutputformat);         aktoptprocessor:=initoptprocessor;         aktspecificoptprocessor:=initspecificoptprocessor;         aktasmmode:=initasmmode;         aktinterfacetype:=initinterfacetype;         { startup scanner and load the first file }         current_scanner:=tscannerfile.Create(filename);         current_scanner.firstfile;         current_module.scanner:=current_scanner;         { init macros before anything in the file is parsed.}         macrosymtablestack:= initialmacrosymtable;         current_module.localmacrosymtable:= tmacrosymtable.create(false);         current_module.localmacrosymtable.next:= initialmacrosymtable;         macrosymtablestack:= current_module.localmacrosymtable;         { read the first token }         current_scanner.readtoken;         { init code generator for a new module }         init_module;         { If the compile level > 1 we get a nice "unit expected" error           message if we are trying to use a program as unit.}         try           try             if (token=_UNIT) or (compile_level>1) then               begin                 current_module.is_unit:=true;                 proc_unit;               end             else               proc_program(token=_LIBRARY);           except             on ECompilerAbort do               raise;             on Exception do               begin                 { Increase errorcounter to prevent some                   checks during cleanup }                 inc(status.errorcount);                 raise;               end;           end;         finally           { restore old state }           done_module;           if assigned(current_module) then            begin              { module is now compiled }              tppumodule(current_module).state:=ms_compiled;              { free ppu }              if assigned(tppumodule(current_module).ppufile) then               begin                 tppumodule(current_module).ppufile.free;                 tppumodule(current_module).ppufile:=nil;               end;              { free scanner }              if assigned(current_module.scanner) then               begin                 if current_scanner=tscannerfile(current_module.scanner) then                   current_scanner:=nil;                 tscannerfile(current_module.scanner).free;                 current_module.scanner:=nil;               end;            end;           if (compile_level>1) then             begin                with olddata^ do                 begin                   { restore scanner }                   c:=oldc;                   pattern:=oldpattern;                   orgpattern:=oldorgpattern;                   token:=oldtoken;                   idtoken:=oldidtoken;                   akttokenpos:=oldtokenpos;                   block_type:=old_block_type;                   { restore cg }                   parse_only:=oldparse_only;                   { restore asmlists }                   exprasmlist:=oldexprasmlist;                   datasegment:=olddatasegment;                   bsssegment:=oldbsssegment;                   codesegment:=oldcodesegment;                   consts:=oldconsts;                   debuglist:=olddebuglist;                   withdebuglist:=oldwithdebuglist;                   importssection:=oldimports;                   exportssection:=oldexports;                   resourcesection:=oldresource;                   rttilist:=oldrttilist;                   picdata:=oldpicdata;                   resourcestringlist:=oldresourcestringlist;                   { object data }                   ResourceStrings:=OldResourceStrings;                   objectlibrary:=oldobjectlibrary;                   { restore previous scanner }                   if assigned(old_compiled_module) then                     current_scanner:=tscannerfile(old_compiled_module.scanner)                   else                     current_scanner:=nil;                   if assigned(current_scanner) then                     parser_current_file:=current_scanner.inputfile.name^;                   { restore symtable state }                   refsymtable:=oldrefsymtable;                   symtablestack:=oldsymtablestack;                   macrosymtablestack:=oldmacrosymtablestack;                   defaultsymtablestack:=olddefaultsymtablestack;                   defaultmacrosymtablestack:=olddefaultmacrosymtablestack;                   aktdefproccall:=oldaktdefproccall;                   current_procinfo:=oldcurrent_procinfo;                   aktsourcecodepage:=oldsourcecodepage;                   aktlocalswitches:=oldaktlocalswitches;                   aktmoduleswitches:=oldaktmoduleswitches;                   aktalignment:=oldaktalignment;                   aktpackenum:=oldaktpackenum;                   aktpackrecords:=oldaktpackrecords;                   aktmaxfpuregisters:=oldaktmaxfpuregisters;                   aktoutputformat:=oldaktoutputformat;                   set_target_asm(aktoutputformat);                   aktoptprocessor:=oldaktoptprocessor;                   aktspecificoptprocessor:=oldaktspecificoptprocessor;                   aktfputype:=oldaktfputype;                   aktasmmode:=oldaktasmmode;                   aktinterfacetype:=oldaktinterfacetype;                   aktfilepos:=oldaktfilepos;                   aktmodeswitches:=oldaktmodeswitches;                   aktexceptblock:=0;                   exceptblockcounter:=0;  {$ifdef GDB}                   dbx_counter:=store_dbx;  {$endif GDB}                 end;             end           else             begin               { Shut down things when the last file is compiled succesfull }               if (compile_level=1) and                  (status.errorcount=0) then                begin                  parser_current_file:='';                  { Close script }                  if (not AsmRes.Empty) then                   begin                     Message1(exec_i_closing_script,AsmRes.Fn);                     AsmRes.WriteToDisk;                   end;                  { do not create browsers on errors !! }                  if status.errorcount=0 then                   begin{$ifdef BrowserLog}                     { Write Browser Log }                     if (cs_browser_log in aktglobalswitches) and                        (cs_browser in aktmoduleswitches) then                      begin                        if browserlog.elements_to_list.empty then                         begin                           Message1(parser_i_writing_browser_log,browserlog.Fname);                           WriteBrowserLog;                         end                        else                         browserlog.list_elements;                      end;{$endif BrowserLog}                     { Write Browser Collections, also used by the TextMode IDE to                       retrieve a list of sourcefiles }                     do_extractsymbolinfo{$ifdef FPC}(){$endif};                   end;                end;             end;           dec(compile_level);           compiled_module:=olddata^.old_compiled_module;           SetCompileModule(compiled_module);           dispose(olddata);         end;    end;end.
 |