* get rid of the hack that tlabelnode inherits from tunarynode and stores a statement in left, resolves #40964

This commit is contained in:
florian 2024-10-26 16:10:24 +02:00
parent 220971289a
commit 511beac49c
6 changed files with 31 additions and 43 deletions

View File

@ -509,8 +509,6 @@ implementation
if assigned(labsym) and if assigned(labsym) and
assigned(labsym.asmblocklabel) then assigned(labsym.asmblocklabel) then
hlcg.a_label(current_asmdata.CurrAsmList,labsym.asmblocklabel); hlcg.a_label(current_asmdata.CurrAsmList,labsym.asmblocklabel);
secondpass(left);
end; end;

View File

@ -199,7 +199,7 @@ interface
end; end;
tgotonodeclass = class of tgotonode; tgotonodeclass = class of tgotonode;
tlabelnode = class(tunarynode) tlabelnode = class(tnode)
exceptionblock : integer; exceptionblock : integer;
{ when copying trees, this points to the newly created copy of a label } { when copying trees, this points to the newly created copy of a label }
copiedto : tlabelnode; copiedto : tlabelnode;
@ -2439,7 +2439,7 @@ implementation
constructor tlabelnode.create(l:tnode;alabsym:tlabelsym); constructor tlabelnode.create(l:tnode;alabsym:tlabelsym);
begin begin
inherited create(labeln,l); inherited create(labeln);
exceptionblock:=current_exceptblock; exceptionblock:=current_exceptblock;
labsym:=alabsym; labsym:=alabsym;
{ Register labelnode in labelsym } { Register labelnode in labelsym }
@ -2494,9 +2494,6 @@ implementation
function tlabelnode.pass_typecheck:tnode; function tlabelnode.pass_typecheck:tnode;
begin begin
result:=nil; result:=nil;
{ left could still be unassigned }
if assigned(left) then
typecheckpass(left);
resultdef:=voidtype; resultdef:=voidtype;
end; end;
@ -2509,8 +2506,6 @@ implementation
if not (nf_internal in flags) then if not (nf_internal in flags) then
include(current_procinfo.flags,pi_has_label); include(current_procinfo.flags,pi_has_label);
if assigned(left) then
firstpass(left);
if (m_non_local_goto in current_settings.modeswitches) and if (m_non_local_goto in current_settings.modeswitches) and
{ the owner can be Nil for internal labels } { the owner can be Nil for internal labels }
assigned(labsym.owner) and assigned(labsym.owner) and

View File

@ -574,17 +574,7 @@ unit optdfa;
calclife(node); calclife(node);
end; end;
labeln: labeln,
begin
calclife(node);
if assigned(tlabelnode(node).left) then
begin
l:=node.optinfo^.life;
DFASetIncludeSet(l,tlabelnode(node).optinfo^.life);
UpdateLifeInfo(node,l);
end;
end;
tempcreaten, tempcreaten,
tempdeleten, tempdeleten,
nothingn, nothingn,
@ -946,9 +936,6 @@ unit optdfa;
MaybeDoCheck(tcasenode(node).elseblock); MaybeDoCheck(tcasenode(node).elseblock);
end; end;
labeln:
MaybeDoCheck(tlabelnode(node).left);
{ we are aware of the following nodes so if new node types are added to the compiler { we are aware of the following nodes so if new node types are added to the compiler
and pop up in the search, the ie below kicks in as a reminder } and pop up in the search, the ie below kicks in as a reminder }
exitn: exitn:
@ -979,6 +966,7 @@ unit optdfa;
{ all other platforms except jvm translate raise nodes into call nodes during pass_1 } { all other platforms except jvm translate raise nodes into call nodes during pass_1 }
raisen, raisen,
{$endif JVM} {$endif JVM}
labeln,
loadn, loadn,
assignn, assignn,
calln, calln,

View File

@ -324,13 +324,7 @@ unit optutils;
labeln: labeln:
begin begin
result:=p; result:=p;
if assigned(tlabelnode(p).left) then p.successor:=succ;
begin
DoSet(tlabelnode(p).left,succ);
p.successor:=tlabelnode(p).left;
end
else
p.successor:=succ;
end; end;
assignn: assignn:
begin begin

View File

@ -1319,6 +1319,7 @@ implementation
function statement : tnode; function statement : tnode;
var var
p, p,
astatement,
code : tnode; code : tnode;
filepos : tfileposinfo; filepos : tfileposinfo;
srsym : tsym; srsym : tsym;
@ -1487,21 +1488,19 @@ implementation
if p.nodetype=labeln then if p.nodetype=labeln then
begin begin
{ the pointer to the following instruction } if not(token in endtokens) then
{ isn't a very clean way } begin
if token in endtokens then astatement:=statement();
tlabelnode(p).left:=cnothingnode.create typecheckpass(astatement);
else p:=cblocknode.create(cstatementnode.create(p,cstatementnode.create(astatement,nil)));
tlabelnode(p).left:=statement(); Include(TBlockNode(p).blocknodeflags, bnf_strippable);
{ be sure to have left also typecheckpass } end;
typecheckpass(tlabelnode(p).left);
end end
else else
{ change a load of a procvar to a call. this is also
{ change a load of a procvar to a call. this is also supported in fpc mode }
supported in fpc mode } if p.nodetype in [vecn,derefn,typeconvn,subscriptn,loadn] then
if p.nodetype in [vecn,derefn,typeconvn,subscriptn,loadn] then maybe_call_procvar(p,false);
maybe_call_procvar(p,false);
{ blockn support because a read/write is changed into a blocknode { blockn support because a read/write is changed into a blocknode
with a separate statement for each read/write operation (JM) with a separate statement for each read/write operation (JM)

14
tests/webtbs/tw40964.pp Normal file
View File

@ -0,0 +1,14 @@
{$goto on}
program ie200211262;
function func(v: pointer): string; inline;
begin
func:='';
end;
label lab;
begin
lab:
func(@lab); // app.lpr(11,3) Error: Internal error 200211262
end.