* try reload before recompilation

This commit is contained in:
Michaël Van Canneyt 2024-07-15 17:19:07 +02:00
parent e86882580d
commit 095c2c7ac7
2 changed files with 62 additions and 19 deletions

View File

@ -88,6 +88,7 @@ uses verbose, fppu, finput, globtype, sysutils, scanner, parser, pmodules;
procedure InitTaskHandler; procedure InitTaskHandler;
begin begin
task_handler:=ttask_handler.create; task_handler:=ttask_handler.create;
schedule_recompile_proc:=@task_handler.addmodule;
end; end;
procedure DoneTaskHandler; procedure DoneTaskHandler;
@ -219,11 +220,12 @@ begin
ms_compiling_waitfinish : cancontinue:=m.nowaitingforunits(firstwaiting); ms_compiling_waitfinish : cancontinue:=m.nowaitingforunits(firstwaiting);
ms_compiling_waitintf : cancontinue:=m.usedunitsloaded(true,firstwaiting); ms_compiling_waitintf : cancontinue:=m.usedunitsloaded(true,firstwaiting);
ms_compiling_wait : cancontinue:=m.usedunitsloaded(true,firstwaiting); ms_compiling_wait : cancontinue:=m.usedunitsloaded(true,firstwaiting);
ms_load : cancontinue:=m.usedunitsloaded(true,firstwaiting);
ms_compiled : cancontinue:=true; ms_compiled : cancontinue:=true;
ms_processed : cancontinue:=true; ms_processed : cancontinue:=true;
ms_moduleerror : cancontinue:=true; ms_moduleerror : cancontinue:=true;
else { else
InternalError(2024011802); InternalError(2024011802);}
end; end;
if (not cancontinue) and checksub then if (not cancontinue) and checksub then
begin begin
@ -266,6 +268,8 @@ begin
t.RestoreState; t.RestoreState;
case m.state of case m.state of
ms_registered : parser.compile_module(m); ms_registered : parser.compile_module(m);
ms_load : with tppumodule(m) do
loadppu(reload_from);
ms_compile : parser.compile_module(m); ms_compile : parser.compile_module(m);
ms_compiled : if (not m.is_initial) or m.is_unit then ms_compiled : if (not m.is_initial) or m.is_unit then
(m as tppumodule).post_load_or_compile(m,m.compilecount>1); (m as tppumodule).post_load_or_compile(m,m.compilecount>1);

View File

@ -46,12 +46,14 @@ interface
{ tppumodule } { tppumodule }
TAvailableUnitFile = (auPPU,auSrc); TAvailableUnitFile = (auPPU,auSrc);
TAvailableUnitFiles = set of TAvailableUnitFile; TAvailableUnitFiles = set of TAvailableUnitFile;
tschedule_recompile_proc = procedure(amodule : tmodule) of object;
tppumodule = class(tmodule) tppumodule = class(tmodule)
ppufile : tcompilerppufile; { the PPU file } ppufile : tcompilerppufile; { the PPU file }
sourcefn : TPathStr; { Source specified with "uses .. in '..'" } sourcefn : TPathStr; { Source specified with "uses .. in '..'" }
comments : TCmdStrList; comments : TCmdStrList;
nsprefix : TCmdStr; { Namespace prefix the unit was found with } nsprefix : TCmdStr; { Namespace prefix the unit was found with }
reload_from : tppumodule; { from_module in case we need to reload }
{$ifdef Test_Double_checksum} {$ifdef Test_Double_checksum}
interface_read_crc_index, interface_read_crc_index,
interface_write_crc_index, interface_write_crc_index,
@ -97,13 +99,13 @@ interface
function loadfrompackage:boolean; function loadfrompackage:boolean;
procedure load_interface; procedure load_interface;
procedure load_implementation; procedure load_implementation;
procedure load_usedunits; function load_usedunits : boolean;
procedure printcomments; procedure printcomments;
procedure queuecomment(const s:TMsgStr;v,w:longint); procedure queuecomment(const s:TMsgStr;v,w:longint);
procedure buildderefunitimportsyms; procedure buildderefunitimportsyms;
procedure derefunitimportsyms; procedure derefunitimportsyms;
procedure freederefunitimportsyms; procedure freederefunitimportsyms;
procedure try_load_ppufile(from_module: tmodule); function try_load_ppufile(from_module: tmodule) : Boolean;
procedure writesourcefiles; procedure writesourcefiles;
procedure writeusedunit(intf:boolean); procedure writeusedunit(intf:boolean);
procedure writelinkcontainer(var p:tlinkcontainer;id:byte;strippath:boolean); procedure writelinkcontainer(var p:tlinkcontainer;id:byte;strippath:boolean);
@ -136,6 +138,9 @@ interface
function registerunit(callermodule:tmodule;const s : TIDString;const fn:string; out is_new:boolean) : tppumodule; function registerunit(callermodule:tmodule;const s : TIDString;const fn:string; out is_new:boolean) : tppumodule;
var
{ Set by task class. To avoid circular dependencies }
schedule_recompile_proc : tschedule_recompile_proc;
implementation implementation
@ -776,7 +781,7 @@ var
{ now load the unit and all used units } { now load the unit and all used units }
load_interface; load_interface;
setdefgeneration; setdefgeneration;
load_usedunits; result:=Load_usedunits;
Message1(unit_u_finished_loading_unit,modulename^); Message1(unit_u_finished_loading_unit,modulename^);
result:=true; result:=true;
@ -1947,25 +1952,36 @@ var
end; end;
procedure tppumodule.load_usedunits; function tppumodule.load_usedunits : boolean;
var var
pu : tused_unit; pu : tused_unit;
s : string;
begin begin
if current_module<>self then if current_module<>self then
internalerror(200212284); internalerror(200212284);
{ load the used units from interface } { load the used units from interface }
in_interface:=true; in_interface:=true;
result:=false;
pu:=tused_unit(used_units.first); pu:=tused_unit(used_units.first);
while assigned(pu) do while assigned(pu) do
begin begin
if pu.in_interface then if pu.in_interface then
begin begin
s:=pu.u.modulename^;
tppumodule(pu.u).loadppu(self); tppumodule(pu.u).loadppu(self);
{ if this unit is scheduled for compilation or compiled we can stop } { if this unit is scheduled for compilation or compiled we can stop }
if state in [ms_compile,ms_compiled,ms_processed] then if state in [ms_compile,ms_compiled,ms_processed] then
exit; exit;
{ add this unit to the dependencies } { add this unit to the dependencies }
pu.u.adddependency(self,true); pu.u.adddependency(self,true);
// Compiler decided the unit must be recompiled.
if pu.u.state=ms_compile then
begin
state:=ms_load;
exit;
end;
{ need to recompile the current unit, check the interface { need to recompile the current unit, check the interface
crc. And when not compiled with -Ur then check the complete crc. And when not compiled with -Ur then check the complete
crc } crc }
@ -1979,10 +1995,10 @@ var
Message2(unit_u_recompile_crc_change,realmodulename^,pu.u.ppufilename,@queuecomment); Message2(unit_u_recompile_crc_change,realmodulename^,pu.u.ppufilename,@queuecomment);
{$ifdef DEBUG_UNIT_CRC_CHANGES} {$ifdef DEBUG_UNIT_CRC_CHANGES}
if (pu.u.interface_crc<>pu.interface_checksum) then if (pu.u.interface_crc<>pu.interface_checksum) then
Comment(V_Normal,' intfcrc change: '+hexstr(pu.u.interface_crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.interface_checksum,8)+' in unit '+realmodulename^) Comment(V_Normal,' intfcrc change: '+hexstr(pu.u.interface_crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.interface_checksum,8)+' in unit '+realmodulename^);
else if (pu.u.indirect_crc<>pu.indirect_checksum) then if (pu.u.indirect_crc<>pu.indirect_checksum) then
Comment(V_Normal,' indcrc change: '+hexstr(pu.u.indirect_crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.indirect_checksum,8)+' in unit '+realmodulename^) Comment(V_Normal,' indcrc change: '+hexstr(pu.u.indirect_crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.indirect_checksum,8)+' in unit '+realmodulename^);
else if (pu.u.crc<>pu.checksum) then
Comment(V_Normal,' implcrc change: '+hexstr(pu.u.crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.checksum,8)+' in unit '+realmodulename^); Comment(V_Normal,' implcrc change: '+hexstr(pu.u.crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.checksum,8)+' in unit '+realmodulename^);
{$endif DEBUG_UNIT_CRC_CHANGES} {$endif DEBUG_UNIT_CRC_CHANGES}
recompile_reason:=rr_crcchanged; recompile_reason:=rr_crcchanged;
@ -1992,6 +2008,7 @@ var
end; end;
pu:=tused_unit(pu.next); pu:=tused_unit(pu.next);
end; end;
{ ok, now load the interface of this unit } { ok, now load the interface of this unit }
if current_module<>self then if current_module<>self then
internalerror(200208187); internalerror(200208187);
@ -2021,10 +2038,19 @@ var
begin begin
if (not pu.in_interface) then if (not pu.in_interface) then
begin begin
s:=pu.u.modulename^;
tppumodule(pu.u).loadppu(self); tppumodule(pu.u).loadppu(self);
{ if this unit is compiled we can stop } { if this unit is compiled we can stop }
if state=ms_compiled then if state=ms_compiled then
exit; exit;
// compiler decided the unit must be recompiled.
if pu.u.state=ms_compile then
begin
state:=ms_load;
exit;
end;
{ add this unit to the dependencies } { add this unit to the dependencies }
pu.u.adddependency(self,false); pu.u.adddependency(self,false);
{ need to recompile the current unit ? } { need to recompile the current unit ? }
@ -2034,8 +2060,8 @@ var
Message2(unit_u_recompile_crc_change,realmodulename^,pu.u.ppufilename+' {impl}',@queuecomment); Message2(unit_u_recompile_crc_change,realmodulename^,pu.u.ppufilename+' {impl}',@queuecomment);
{$ifdef DEBUG_UNIT_CRC_CHANGES} {$ifdef DEBUG_UNIT_CRC_CHANGES}
if (pu.u.interface_crc<>pu.interface_checksum) then if (pu.u.interface_crc<>pu.interface_checksum) then
Comment(V_Normal,' intfcrc change (2): '+hexstr(pu.u.interface_crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.interface_checksum,8)+' in unit '+realmodulename^) Comment(V_Normal,' intfcrc change (2): '+hexstr(pu.u.interface_crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.interface_checksum,8)+' in unit '+realmodulename^);
else if (pu.u.indirect_crc<>pu.indirect_checksum) then if (pu.u.indirect_crc<>pu.indirect_checksum) then
Comment(V_Normal,' indcrc change (2): '+hexstr(pu.u.indirect_crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.indirect_checksum,8)+' in unit '+realmodulename^); Comment(V_Normal,' indcrc change (2): '+hexstr(pu.u.indirect_crc,8)+' for '+pu.u.ppufilename+' <> '+hexstr(pu.indirect_checksum,8)+' in unit '+realmodulename^);
{$endif DEBUG_UNIT_CRC_CHANGES} {$endif DEBUG_UNIT_CRC_CHANGES}
recompile_reason:=rr_crcchanged; recompile_reason:=rr_crcchanged;
@ -2067,6 +2093,7 @@ var
wpoinfo:=tunitwpoinfo.ppuload(ppufile); wpoinfo:=tunitwpoinfo.ppuload(ppufile);
tunitwpoinfo(wpoinfo).deref; tunitwpoinfo(wpoinfo).deref;
tunitwpoinfo(wpoinfo).derefimpl; tunitwpoinfo(wpoinfo).derefimpl;
Result:=True;
end; end;
@ -2214,9 +2241,14 @@ var
state:=ms_load; state:=ms_load;
end; end;
procedure tppumodule.try_load_ppufile(from_module : tmodule); function tppumodule.try_load_ppufile(from_module : tmodule) : Boolean;
{
Return True if the unit was successfully loaded.
False means the unit must be reloaded or recompiled
}
begin begin
Result:=False;
Message1(unit_u_loading_unit,modulename^); Message1(unit_u_loading_unit,modulename^);
if auPPU in search_unit_files(from_module,false) then if auPPU in search_unit_files(from_module,false) then
state:=ms_load state:=ms_load
@ -2228,9 +2260,13 @@ var
setdefgeneration; setdefgeneration;
if not (state=ms_compile) then if not (state=ms_compile) then
begin begin
load_usedunits; if load_usedunits then
if not (state=ms_compile) then begin
Message1(unit_u_finished_loading_unit,modulename^); Message1(unit_u_finished_loading_unit,modulename^);
Result:=true;
end
else
reload_from:=(from_module as tppumodule);
end; end;
end; end;
{ PPU is not needed anymore } { PPU is not needed anymore }
@ -2282,6 +2318,8 @@ var
if not (state in [ms_compile]) then if not (state in [ms_compile]) then
state:=ms_compile; state:=ms_compile;
setdefgeneration; setdefgeneration;
if assigned(schedule_recompile_proc) then
schedule_recompile_proc(self);
end; end;
procedure tppumodule.post_load_or_compile(from_module : tmodule; second_time : boolean); procedure tppumodule.post_load_or_compile(from_module : tmodule; second_time : boolean);
@ -2311,7 +2349,7 @@ var
const const
ImplIntf : array[boolean] of string[15]=('implementation','interface'); ImplIntf : array[boolean] of string[15]=('implementation','interface');
var var
do_load, do_load,load_ok,
second_time : boolean; second_time : boolean;
begin begin
@ -2371,13 +2409,14 @@ var
{ try to opening ppu, skip this when we already { try to opening ppu, skip this when we already
know that we need to compile the unit } know that we need to compile the unit }
load_ok:=False;
if not (state=ms_compile) then if not (state=ms_compile) then
try_load_ppufile(from_module); load_ok:=try_load_ppufile(from_module);
{ Do we need to recompile the unit } { Do we need to recompile the unit }
if (state=ms_compile) then if (state=ms_compile) then
recompile_from_sources(from_module) recompile_from_sources(from_module)
else else if load_ok then
state:=ms_compiled; state:=ms_compiled;
Result:=(state=ms_compiled); Result:=(state=ms_compiled);