mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-06 00:07:54 +02:00
529 lines
16 KiB
ObjectPascal
529 lines
16 KiB
ObjectPascal
{
|
|
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
|
|
|
|
uses fmodule;
|
|
|
|
{$ifdef PREPROCWRITE}
|
|
procedure preprocess(const filename:string);
|
|
{$endif PREPROCWRITE}
|
|
function compile(const filename:string) : boolean;
|
|
function compile_module(module : tmodule) : boolean;
|
|
procedure parsing_done(module : tmodule);
|
|
procedure initparser;
|
|
procedure doneparser;
|
|
|
|
implementation
|
|
|
|
uses
|
|
{$IFNDEF USE_FAKE_SYSUTILS}
|
|
sysutils,
|
|
{$ELSE}
|
|
fksysutl,
|
|
{$ENDIF}
|
|
cclasses,
|
|
globtype,tokens,systems,globals,verbose,switches,globstat,
|
|
symbase,symtable,symdef,
|
|
finput,fppu,
|
|
aasmdata,
|
|
cscript,gendef,
|
|
comphook,
|
|
scanner,scandir,
|
|
pbase,psystem,pmodules,psub,ncgrtti,
|
|
cpuinfo,procinfo;
|
|
|
|
procedure parsing_done(module: tmodule);
|
|
|
|
var
|
|
hp,hp2 : tmodule;
|
|
|
|
begin
|
|
|
|
module.end_of_parsing;
|
|
|
|
if (module.is_initial) and
|
|
(status.errorcount=0) then
|
|
{ Write Browser Collections }
|
|
do_extractsymbolinfo;
|
|
|
|
// olddata.restore(false);
|
|
|
|
{ Restore all locally modified warning messages }
|
|
RestoreLocalVerbosity(current_settings.pmessage);
|
|
current_exceptblock:=0;
|
|
exceptblockcounter:=0;
|
|
|
|
{ Shut down things when the last file is compiled succesfull }
|
|
if (module.is_initial) and (module.state=ms_compiled) 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;
|
|
end;
|
|
|
|
{ free now what we did not free earlier in
|
|
proc_program PM }
|
|
if (module.is_initial) and (module.state=ms_compiled) and needsymbolinfo then
|
|
begin
|
|
hp:=tmodule(loaded_units.first);
|
|
while assigned(hp) do
|
|
begin
|
|
hp2:=tmodule(hp.next);
|
|
if (hp<>module) then
|
|
begin
|
|
loaded_units.remove(hp);
|
|
hp.free;
|
|
end;
|
|
hp:=hp2;
|
|
end;
|
|
{ free also unneeded units we didn't free before }
|
|
unloaded_units.Clear;
|
|
end;
|
|
|
|
{ If used units are compiled current_module is already the same as
|
|
the stored module. Now if the unit is not finished its scanner is
|
|
not yet freed and thus set_current_module would reopen the scanned
|
|
file which will result in pointing to the wrong position in the
|
|
file. In the normal case current_scanner and current_module.scanner
|
|
would be Nil, thus nothing bad would happen }
|
|
{ if olddata.old_current_module<>current_module then
|
|
set_current_module(olddata.old_current_module);}
|
|
|
|
FreeLocalVerbosity(current_settings.pmessage);
|
|
|
|
end;
|
|
|
|
procedure initparser;
|
|
begin
|
|
{ Current compiled module/proc }
|
|
set_current_module(nil);
|
|
current_asmdata:=nil;
|
|
current_procinfo:=nil;
|
|
current_structdef:=nil;
|
|
current_genericdef:=nil;
|
|
current_specializedef:=nil;
|
|
|
|
loaded_units:=TLinkedList.Create;
|
|
|
|
usedunits:=TLinkedList.Create;
|
|
|
|
unloaded_units:=TLinkedList.Create;
|
|
|
|
{ global switches }
|
|
current_settings.globalswitches:=init_settings.globalswitches;
|
|
|
|
current_settings.sourcecodepage:=init_settings.sourcecodepage;
|
|
|
|
{ initialize scanner }
|
|
InitScanner;
|
|
InitScannerDirectives;
|
|
|
|
{ scanner }
|
|
c:=#0;
|
|
pattern:='';
|
|
orgpattern:='';
|
|
cstringpattern:='';
|
|
set_current_scanner(nil);
|
|
switchesstatestackpos:=0;
|
|
|
|
{ register all nodes and tais }
|
|
registernodes;
|
|
registertais;
|
|
|
|
{ memory sizes }
|
|
if stacksize=0 then
|
|
stacksize:=target_info.stacksize;
|
|
|
|
{ RTTI writer }
|
|
RTTIWriter:=TRTTIWriter.Create;
|
|
|
|
{ open assembler response }
|
|
if cs_link_on_target in current_settings.globalswitches then
|
|
GenerateAsmRes(outputexedir+ChangeFileExt(inputfilename,'_ppas'))
|
|
else
|
|
GenerateAsmRes(outputexedir+'ppas');
|
|
|
|
{ open deffile }
|
|
DefFile:=TDefFile.Create(outputexedir+ChangeFileExt(inputfilename,target_info.defext));
|
|
|
|
{ list of generated .o files, so the linker can remove them }
|
|
SmartLinkOFiles:=TCmdStrList.Create;
|
|
|
|
{ codegen }
|
|
if paraprintnodetree<>0 then
|
|
printnode_reset;
|
|
|
|
{ target specific stuff }
|
|
case target_info.system of
|
|
system_arm_aros,
|
|
system_arm_palmos,
|
|
system_m68k_amiga,
|
|
system_m68k_atari,
|
|
system_m68k_palmos,
|
|
system_i386_aros,
|
|
system_powerpc_amiga,
|
|
system_powerpc_morphos,
|
|
system_x86_64_aros:
|
|
include(supported_calling_conventions,pocall_syscall);
|
|
system_m68k_human68k:
|
|
begin
|
|
include(supported_calling_conventions,pocall_syscall);
|
|
if heapsize=0 then
|
|
heapsize:=65536;
|
|
end;
|
|
system_wasm32_wasip1,
|
|
system_wasm32_wasip1threads,
|
|
system_wasm32_wasip2:
|
|
begin
|
|
if ts_wasm_threads in init_settings.targetswitches then
|
|
maxheapsize:=256*1024*1024
|
|
else
|
|
maxheapsize:=0;
|
|
end;
|
|
{$ifdef i8086}
|
|
system_i8086_embedded:
|
|
begin
|
|
if stacksize=0 then
|
|
begin
|
|
if init_settings.x86memorymodel in x86_far_data_models then
|
|
stacksize:=16384
|
|
else
|
|
stacksize:=2048;
|
|
end;
|
|
end;
|
|
system_i8086_msdos:
|
|
begin
|
|
if stacksize=0 then
|
|
begin
|
|
if init_settings.x86memorymodel in x86_far_data_models then
|
|
stacksize:=16384
|
|
else
|
|
stacksize:=4096;
|
|
end;
|
|
if maxheapsize=0 then
|
|
begin
|
|
if init_settings.x86memorymodel in x86_far_data_models then
|
|
maxheapsize:=655360
|
|
else
|
|
maxheapsize:=65520;
|
|
end;
|
|
end;
|
|
system_i8086_win16:
|
|
begin
|
|
if stacksize=0 then
|
|
begin
|
|
if init_settings.x86memorymodel in x86_far_data_models then
|
|
stacksize:=8192
|
|
else
|
|
stacksize:=5120;
|
|
end;
|
|
if heapsize=0 then
|
|
begin
|
|
if init_settings.x86memorymodel in x86_far_data_models then
|
|
heapsize:=8192
|
|
else
|
|
heapsize:=4096;
|
|
end;
|
|
end;
|
|
{$endif i8086}
|
|
else
|
|
;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure doneparser;
|
|
begin
|
|
{ Reset current compiling info, so destroy routines can't
|
|
reference the data that might already be destroyed }
|
|
set_current_module(nil);
|
|
current_procinfo:=nil;
|
|
current_asmdata:=nil;
|
|
current_structdef:=nil;
|
|
current_genericdef:=nil;
|
|
current_specializedef:=nil;
|
|
|
|
{ unload units }
|
|
if assigned(loaded_units) then
|
|
begin
|
|
loaded_units.free;
|
|
loaded_units:=nil;
|
|
end;
|
|
if assigned(usedunits) then
|
|
begin
|
|
usedunits.free;
|
|
usedunits:=nil;
|
|
end;
|
|
if assigned(unloaded_units) then
|
|
begin
|
|
unloaded_units.free;
|
|
unloaded_units:=nil;
|
|
end;
|
|
{ Set default types to nil. At this point they are not valid class pointers. }
|
|
reset_all_default_types;
|
|
|
|
{ if there was an error in the scanner, the scanner is
|
|
still assinged }
|
|
if assigned(current_scanner) then
|
|
begin
|
|
current_scanner.free;
|
|
set_current_scanner(nil);
|
|
|
|
end;
|
|
|
|
{ close scanner }
|
|
DoneScanner;
|
|
|
|
RTTIWriter.free;
|
|
|
|
{ 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
|
|
preprocfile:=tpreprocfile.create('pre_'+filename);
|
|
{ initialize a module }
|
|
set_current_module(tppumodule.create(nil,'',filename,false));
|
|
macrosymtablestack:=TSymtablestack.create;
|
|
|
|
|
|
set_current_scanner(tscannerfile.Create(filename));
|
|
current_scanner.firstfile;
|
|
current_module.scanner:=current_scanner;
|
|
|
|
{ init macros before anything in the file is parsed.}
|
|
current_module.localmacrosymtable:= tmacrosymtable.create(false);
|
|
macrosymtablestack.push(initialmacrosymtable);
|
|
macrosymtablestack.push(current_module.localmacrosymtable);
|
|
|
|
{ read the first token }
|
|
// current_scanner.readtoken(false);
|
|
|
|
main_module:=current_module;
|
|
repeat
|
|
current_scanner.readtoken(true);
|
|
preprocfile.AddSpace;
|
|
case token of
|
|
_ID :
|
|
begin
|
|
preprocfile.Add(orgpattern);
|
|
end;
|
|
_REALNUMBER,
|
|
_INTCONST :
|
|
preprocfile.Add(pattern);
|
|
_CSTRING :
|
|
begin
|
|
i:=0;
|
|
while (i<length(cstringpattern)) do
|
|
begin
|
|
inc(i);
|
|
if cstringpattern[i]='''' then
|
|
begin
|
|
insert('''',cstringpattern,i);
|
|
inc(i);
|
|
end;
|
|
end;
|
|
preprocfile.Add(''''+cstringpattern+'''');
|
|
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 }
|
|
current_scanner.free;
|
|
set_current_scanner(nil);
|
|
{ close }
|
|
preprocfile.free;
|
|
end;
|
|
{$endif PREPROCWRITE}
|
|
|
|
|
|
{*****************************************************************************
|
|
Compile a source file
|
|
*****************************************************************************}
|
|
|
|
function compile(const filename:string) : boolean;
|
|
|
|
var
|
|
m : TModule;
|
|
|
|
begin
|
|
m:=tppumodule.create(nil,'',filename,false);
|
|
m.state:=ms_compile;
|
|
result:=compile_module(m);
|
|
end;
|
|
|
|
function compile_module(module : tmodule) : boolean;
|
|
|
|
var
|
|
hp,hp2 : tmodule;
|
|
finished : boolean;
|
|
sc : tscannerfile;
|
|
|
|
begin
|
|
Result:=True;
|
|
{ parsing a procedure or declaration should be finished }
|
|
if assigned(current_procinfo) then
|
|
internalerror(200811121);
|
|
if assigned(current_structdef) then
|
|
internalerror(200811122);
|
|
inc(module.compilecount);
|
|
parser_current_file:=module.mainsource;
|
|
{ Uses heap memory instead of placing everything on the
|
|
stack. This is needed because compile() can be called
|
|
recursively }
|
|
{ handle the postponed case first }
|
|
flushpendingswitchesstate;
|
|
|
|
{ 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;
|
|
named_args_allowed:=false;
|
|
got_addrn:=false;
|
|
getprocvardef:=nil;
|
|
getfuncrefdef:=nil;
|
|
|
|
{ show info }
|
|
Message1(parser_i_compiling,module.mainsource);
|
|
|
|
{ reset symtable }
|
|
symtablestack:=tdefawaresymtablestack.create;
|
|
macrosymtablestack:=TSymtablestack.create;
|
|
systemunit:=nil;
|
|
current_settings.defproccall:=init_settings.defproccall;
|
|
current_exceptblock:=0;
|
|
exceptblockcounter:=0;
|
|
current_settings.maxfpuregisters:=-1;
|
|
current_settings.pmessage:=nil;
|
|
|
|
{ Load current state from the init values }
|
|
current_settings:=init_settings;
|
|
|
|
set_current_module(module);
|
|
if not (module.state in [ms_compile]) then
|
|
internalerror(200212281);
|
|
|
|
{ load current asmdata from current_module }
|
|
current_asmdata:=TAsmData(module.asmdata);
|
|
|
|
{ startup scanner and load the first file }
|
|
sc:=tscannerfile.Create(module.mainsource);
|
|
sc.firstfile;
|
|
module.scanner:=sc;
|
|
module.mainscanner:=sc;
|
|
set_current_scanner(sc);
|
|
|
|
{ init macros before anything in the file is parsed.}
|
|
module.localmacrosymtable:= tmacrosymtable.create(false);
|
|
macrosymtablestack.push(initialmacrosymtable);
|
|
macrosymtablestack.push(module.localmacrosymtable);
|
|
|
|
{ read the first token }
|
|
current_scanner.readtoken(false);
|
|
|
|
{ this is set to false if a unit needs to wait for other units }
|
|
finished:=true;
|
|
|
|
{ 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 (not module.is_initial) then
|
|
begin
|
|
module.is_unit:=true;
|
|
finished:=proc_unit(module);
|
|
end
|
|
else if (token=_ID) and (idtoken=_PACKAGE) then
|
|
begin
|
|
module.IsPackage:=true;
|
|
finished:=proc_package(module);
|
|
end
|
|
else
|
|
finished:=proc_program(module,token=_LIBRARY);
|
|
except
|
|
on ECompilerAbort do
|
|
raise;
|
|
on Exception do
|
|
begin
|
|
{ Generate exception_raised message,
|
|
but avoid multiple messages by
|
|
guarding with exception_raised global variable }
|
|
if not exception_raised then
|
|
begin
|
|
exception_raised:=true;
|
|
Message(general_e_exception_raised);
|
|
end;
|
|
raise;
|
|
end;
|
|
end;
|
|
Result:=Finished;
|
|
{ the program or the unit at the command line should not need to wait
|
|
for other units }
|
|
// if (module.is_initial) and not finished then
|
|
// internalerror(2012091901);
|
|
finally
|
|
if finished then
|
|
parsing_done(module);
|
|
end;
|
|
end;
|
|
|
|
end.
|