123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260 |
- {
- Copyright (c) 2012 by the FPC development team
- Contains functionality to save/restore the global compiler state when
- switching between the compilation of different units.
- 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 globstat;
- {$i fpcdefs.inc}
- interface
- uses
- globtype,tokens,globals,
- aasmdata,
- dbgbase,
- symbase,symsym,
- fmodule,
- scanner,scandir,
- procinfo;
- type
- tglobalstate = class
- { scanner }
- oldidtoken,
- oldtoken : ttoken;
- oldtokenpos : tfileposinfo;
- oldc : char;
- oldpattern,
- oldorgpattern : string;
- old_block_type : tblock_type;
- { symtable }
- oldsymtablestack,
- oldmacrosymtablestack : TSymtablestack;
- oldaktprocsym : tprocsym;
- { cg }
- oldparse_only : boolean;
- { akt.. things }
- oldcurrent_filepos : tfileposinfo;
- old_current_module : tmodule;
- oldcurrent_procinfo : tprocinfo;
- old_settings : tsettings;
- old_switchesstatestack : tswitchesstatestack;
- old_switchesstatestackpos : Integer;
- old_verbosity : longint;
- { only saved/restored if "full" is true }
- old_asmdata : tasmdata;
- old_debuginfo : tdebuginfo;
- old_scanner : tscannerfile;
- old_parser_file : string;
- constructor create(savefull : boolean);
- destructor destroy; override;
- procedure clearscanner;
- class procedure remove_scanner_from_states(scanner : tscannerfile); static;
- procedure save(full : boolean);
- procedure restore(full : boolean);
- end;
- procedure save_global_state(state:tglobalstate;full:boolean);
- procedure restore_global_state(state:tglobalstate;full:boolean);
- implementation
- uses
- switches, verbose, pbase,comphook;
- var
- states : array of tglobalstate;
- statecount : integer = 0;
- class procedure tglobalstate.remove_scanner_from_states(scanner : tscannerfile);
- var
- i : integer;
- begin
- for I:=0 to statecount-1 do
- if (states[i].old_scanner=scanner) then
- states[i].clearscanner;
- end;
- procedure addstate(astate : tglobalstate);
- var
- l : integer;
- begin
- l:=length(states);
- if l=statecount then
- setlength(states,l+10);
- states[statecount]:=astate;
- inc(statecount);
- end;
- procedure removestate(astate : tglobalstate);
- var
- l : integer;
- begin
- l:=statecount-1;
- While (l>=0) and (states[l]<>astate) do
- dec(l);
- if l<0 then
- exit;
- if l<>statecount-1 then
- states[l]:=states[statecount-1];
- states[statecount-1]:=Nil;
- Dec(Statecount);
- end;
- procedure save_global_state(state:tglobalstate;full:boolean);
- begin
- state.save(full);
- end;
- procedure restore_global_state(state:tglobalstate;full:boolean);
- begin
- state.restore(full);
- end;
- procedure tglobalstate.save(full: boolean);
- begin
- old_current_module:=current_module;
- { save symtable state }
- oldsymtablestack:=symtablestack;
- oldmacrosymtablestack:=macrosymtablestack;
- oldcurrent_procinfo:=current_procinfo;
- { save scanner state }
- oldc:=c;
- oldpattern:=pattern;
- oldorgpattern:=orgpattern;
- oldtoken:=token;
- oldidtoken:=idtoken;
- old_block_type:=block_type;
- oldtokenpos:=current_tokenpos;
- {
- consuming the semicolon after a uses clause can add to the
- pending state if the first directives change warning state.
- So we must flush before context switch. See for example:
- ppcgen/cgppc.pas
- line 144 has a WARN 6018 OFF...
- }
- flushpendingswitchesstate;
- old_switchesstatestack:=switchesstatestack;
- old_switchesstatestackpos:=switchesstatestackpos;
- { save cg }
- oldparse_only:=parse_only;
- { save akt... state }
- { handle the postponed case first }
- oldcurrent_filepos:=current_filepos;
- old_settings:=current_settings;
- old_verbosity:=status.verbosity;
- if full then
- begin
- old_asmdata:=current_asmdata;
- old_debuginfo:=current_debuginfo;
- old_parser_file:=parser_current_file;
- old_scanner:=current_scanner;
- end;
- end;
- procedure tglobalstate.restore(full: boolean);
- begin
- { restore scanner }
- c:=oldc;
- pattern:=oldpattern;
- orgpattern:=oldorgpattern;
- token:=oldtoken;
- idtoken:=oldidtoken;
- current_tokenpos:=oldtokenpos;
- block_type:=old_block_type;
- switchesstatestack:=old_switchesstatestack;
- switchesstatestackpos:=old_switchesstatestackpos;
- { restore cg }
- parse_only:=oldparse_only;
- { restore symtable state }
- symtablestack:=oldsymtablestack;
- macrosymtablestack:=oldmacrosymtablestack;
- current_procinfo:=oldcurrent_procinfo;
- current_filepos:=oldcurrent_filepos;
- current_settings:=old_settings;
- status.verbosity:=old_verbosity;
- { restore message settings which were recorded prior to unit switch }
- RestoreLocalVerbosity(current_settings.pmessage);
- if full then
- begin
- set_current_module(old_current_module);
- // These can be different
- current_asmdata:=old_asmdata;
- current_debuginfo:=old_debuginfo;
- end;
- end;
- constructor tglobalstate.create(savefull: boolean);
- begin
- addstate(self);
- save(savefull);
- end;
- destructor tglobalstate.destroy;
- begin
- removestate(self);
- inherited destroy;
- end;
- procedure tglobalstate.clearscanner;
- begin
- old_scanner:=nil;
- oldidtoken:=NOTOKEN;
- oldtoken:=NOTOKEN;
- oldtokenpos:=Default(tfileposinfo);
- oldc:=#0;
- oldpattern:='';
- oldorgpattern:='';
- old_block_type:=bt_none;
- end;
- initialization
- onfreescanner:[email protected]_scanner_from_states;
- finalization
- onfreescanner:=Nil;
- end.
|