fpc/compiler/symtype.pas
nickysn 3ecb41cd2d * improvements to the prevention of dangling pointers in tmodule.deflist,
introduced in r49417. Now it no longer depends on current_module, so it
  handles the case when current_module changes between the time the tdef was
  registered and the time it was freed. It also supports freeing the tmodule
  before the defs, so the freeing order of the object is once again flexible.

git-svn-id: trunk@49426 -
2021-06-01 15:02:26 +00:00

1321 lines
37 KiB
ObjectPascal

{
Copyright (c) 1998-2002 by Florian Klaempfl, Pierre Muller
This unit handles the symbol tables
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 symtype;
{$i fpcdefs.inc}
interface
uses
{ common }
cutils,
cclasses,
{ global }
globtype,globals,constexp,
{ symtable }
symconst,symbase,
{ aasm }
aasmbase,ppu,
finput
;
type
{************************************************
Required Forwards
************************************************}
tsym = class;
Tcompilerppufile=class;
{************************************************
TDef
************************************************}
tgetsymtable = (gs_none,gs_record,gs_local,gs_para);
tdef = class(TDefEntry)
protected
{ whether this def is already registered in the unit's def list }
function registered : boolean;
{ initialize the defid field; only call from a constructor as it threats
0 as an invalid value! }
procedure init_defid;
{$ifdef DEBUG_NODE_XML}
procedure XMLPrintDefTree(var T: Text; Sym: TSym); virtual;
procedure XMLPrintDefInfo(var T: Text; Sym: TSym); dynamic;
procedure XMLPrintDefData(var T: Text; Sym: TSym); virtual;
function XMLPrintType: ansistring; virtual;
{$endif DEBUG_NODE_XML}
public
registered_in_module : tmodulebase;
typesym : tsym; { which type the definition was generated this def }
{ stabs debugging }
stab_number : word;
dbg_state : tdefdbgstatus;
defoptions : tdefoptions;
defstates : tdefstates;
constructor create(dt:tdeftyp);
destructor destroy; override;
procedure buildderef;virtual;abstract;
procedure buildderefimpl;virtual;abstract;
procedure deref;virtual;abstract;
procedure derefimpl;virtual;abstract;
function typename:string;
function fulltypename:string;
function GetTypeName:string;virtual;
function typesymbolprettyname:string;virtual;
function mangledparaname:string;
function getmangledparaname:TSymStr;virtual;
function rtti_mangledname(rt:trttitype):TSymStr;virtual;abstract;
function OwnerHierarchyName: string; virtual; abstract;
function fullownerhierarchyname(skipprocparams:boolean):TSymStr;virtual;abstract;
function unique_id_str: string;
function size:asizeint;virtual;abstract;
function packedbitsize:asizeint;virtual;
function alignment:shortint;virtual;abstract;
{ alignment when this type appears in a record/class/... }
function structalignment:shortint;virtual;
function aggregatealignment:shortint;virtual;
function getvardef:longint;virtual;abstract;
function getparentdef:tdef;virtual;
function getsymtable(t:tgetsymtable):TSymtable;virtual;
function is_publishable:boolean;virtual;abstract;
function needs_inittable:boolean;virtual;abstract;
{ contains a (managed) child that is not initialized to 0/Nil }
function has_non_trivial_init_child(check_parent:boolean):boolean;virtual;abstract;
function needs_separate_initrtti:boolean;virtual;abstract;
procedure ChangeOwner(st:TSymtable);
function getreusablesymtab: tsymtable;
procedure register_created_object_type;virtual;
function get_top_level_symtable(skipprocdefs: boolean): tsymtable;
{ only valid for registered defs and defs for which a unique id string
has been requested; otherwise, first call register_def }
function deflist_index: longint;
procedure register_def; virtual; abstract;
{$ifdef DEBUG_NODE_XML}
procedure XMLPrintDef(Sym: TSym);
{$endif DEBUG_NODE_XML}
property is_registered: boolean read registered;
end;
{************************************************
TSym
************************************************}
{ this object is the base for all symbol objects }
{ tsym }
tsym = class(TSymEntry)
protected
function registered : boolean;
public
fileinfo : tfileposinfo;
{ size of fileinfo is 10 bytes, so if a >word aligned type would follow,
two bytes of memory would be wasted, so we put two one byte fields over here }
visibility : tvisibility;
isdbgwritten : boolean;
symoptions : tsymoptions;
refs : longint;
reflist : TLinkedList;
{ deprecated optionally can have a message }
deprecatedmsg: pshortstring;
constructor create(st:tsymtyp;const aname:string);
destructor destroy;override;
function mangledname:TSymStr; virtual;
function prettyname:string; virtual;
procedure buildderef;virtual;
procedure deref;virtual;
procedure ChangeOwner(st:TSymtable);
procedure IncRefCount;
procedure IncRefCountBy(AValue : longint);
procedure MaybeCreateRefList;
procedure AddRef;
procedure register_sym; virtual; abstract;
property is_registered:boolean read registered;
end;
tsymarr = array[0..maxlongint div sizeof(pointer)-1] of tsym;
psymarr = ^tsymarr;
{************************************************
TDeref
************************************************}
tderef = object
dataidx : longint;
procedure reset;
procedure build(s:TObject);
function resolve:TObject;
end;
pderef = ^tderef;
{************************************************
tpropaccesslist
************************************************}
ppropaccesslistitem = ^tpropaccesslistitem;
tpropaccesslistitem = record
sltype : tsltype;
next : ppropaccesslistitem;
case byte of
0 : (sym : tsym; symderef : tderef);
1 : (value : TConstExprInt; valuedef: tdef; valuedefderef:tderef);
2 : (def: tdef; defderef:tderef);
end;
tpropaccesslist = class
procdef : tdef;
procdefderef : tderef;
firstsym,
lastsym : ppropaccesslistitem;
constructor create;
destructor destroy;override;
function empty:boolean;
function getcopy: tpropaccesslist;
procedure addsym(slt:tsltype;p:tsym);
procedure addconst(slt:tsltype;const v:TConstExprInt;d:tdef);
procedure addtype(slt:tsltype;d:tdef);
procedure addsymderef(slt:tsltype;d:tderef);
procedure addconstderef(slt:tsltype;const v:TConstExprInt;d:tderef);
procedure addtypederef(slt:tsltype;d:tderef);
procedure clear;
procedure resolve;
procedure buildderef;
end;
{************************************************
Tcompilerppufile
************************************************}
tcompilerppufile=class(tppufile)
public
procedure checkerror;
procedure getguid(var g: tguid);
function getexprint:Tconstexprint;
procedure getposinfo(var p:tfileposinfo);
procedure getderef(var d:tderef);
function getpropaccesslist:tpropaccesslist;
function getasmsymbol:tasmsymbol;
procedure putguid(const g: tguid);
procedure putexprint(const v:tconstexprint);
procedure putposinfo(const p:tfileposinfo);
procedure putderef(const d:tderef);
procedure putpropaccesslist(p:tpropaccesslist);
procedure putasmsymbol(s:tasmsymbol);
protected
procedure RaiseAssertion(Code: Longint); override;
end;
{$ifdef MEMDEBUG}
var
memmanglednames,
memprocpara,
memprocparast,
memproclocalst,
memprocnodetree : tmemdebug;
{$endif MEMDEBUG}
function FindUnitSymtable(st:TSymtable):TSymtable;
implementation
uses
crefs,
verbose,
fmodule
;
{****************************************************************************
Utils
****************************************************************************}
function FindUnitSymtable(st:TSymtable):TSymtable;
begin
result:=nil;
repeat
if not assigned(st) then
internalerror(200602034);
case st.symtabletype of
localmacrosymtable,
exportedmacrosymtable,
staticsymtable,
globalsymtable :
begin
result:=st;
exit;
end;
recordsymtable,
enumsymtable,
arraysymtable,
localsymtable,
parasymtable,
ObjectSymtable :
st:=st.defowner.owner;
else
internalerror(200602035);
end;
until false;
end;
{****************************************************************************
Tdef
****************************************************************************}
function tdef.registered: boolean;
begin
result:=defid>defid_not_registered;
end;
procedure tdef.init_defid;
begin
if defid=0 then
defid:=defid_not_registered;
end;
{$ifdef DEBUG_NODE_XML}
procedure tdef.XMLPrintDefTree(var T: Text; Sym: TSym);
begin
Write(T, PrintNodeIndention, '<definition');
XMLPrintDefInfo(T, Sym);
WriteLn(T, '>');
PrintNodeIndent;
{ Printing the type here instead of in XMLPrintDefData ensures it
always appears first no matter how XMLPrintDefData is overridden }
WriteLn(T, PrintNodeIndention, '<type>', XMLPrintType, '</type>');
XMLPrintDefData(T, Sym);
PrintNodeUnindent;
WriteLn(T, PrintNodeIndention, '</definition>');
WriteLn(T, PrintNodeIndention);
end;
procedure tdef.XMLPrintDefInfo(var T: Text; Sym: TSym);
var
i: TSymOption;
first: Boolean;
begin
{ Note that if we've declared something like "INT = Integer", the
INT name gets lost in the system and 'typename' just returns
Integer, so the correct details can be found via Sym }
Write(T, ' name="', SanitiseXMLString(Sym.RealName),
'" pos="', Sym.fileinfo.line, ',', Sym.fileinfo.column);
First := True;
for i := Low(TSymOption) to High(TSymOption) do
if i in Sym.symoptions then
begin
if First then
begin
Write(T, '" symoptions="', i);
First := False;
end
else
Write(T, ',', i)
end;
Write(T, '"');
end;
procedure tdef.XMLPrintDefData(var T: Text; Sym: TSym);
begin
WriteLn(T, PrintNodeIndention, '<size>', size, '</size>');
if (alignment = structalignment) and (alignment = aggregatealignment) then
begin
{ Straightforward and simple }
WriteLn(T, PrintNodeIndention, '<alignment>', alignment, '</alignment>');
end
else
begin
WriteLn(T, PrintNodeIndention, '<alignment>');
printnodeindent;
WriteLn(T, PrintNodeIndention, '<basic>', alignment, '</basic>');
if (structalignment <> alignment) then
WriteLn(T, PrintNodeIndention, '<struct>', structalignment, '</struct>');
if (aggregatealignment <> alignment) and (aggregatealignment <> structalignment) then
WriteLn(T, PrintNodeIndention, '<aggregate>', aggregatealignment, '</aggregate>');
printnodeunindent;
WriteLn(T, PrintNodeIndention, '</alignment>');
end;
end;
function tdef.XMLPrintType: ansistring;
begin
Result := SanitiseXMLString(GetTypeName);
end;
{$endif DEBUG_NODE_XML}
constructor tdef.create(dt:tdeftyp);
begin
inherited create;
typ:=dt;
owner := nil;
typesym := nil;
defoptions:=[];
dbg_state:=dbg_state_unused;
stab_number:=0;
init_defid;
end;
destructor tdef.destroy;
begin
{ set self to nil in registered_in_module's deflist, if the def has been
registered, in order to avoid dangling pointers in registered_in_module.deflist }
if assigned(registered_in_module) then
begin
if defid>=0 then
tmodule(registered_in_module).deflist[defid]:=nil
else if defid<defid_not_registered then
tmodule(registered_in_module).deflist[-(defid-defid_not_registered+1)]:=nil
else
internalerror(2021060101);
end;
inherited;
end;
function tdef.typename:string;
begin
result:=OwnerHierarchyName;
if assigned(typesym) and
not(typ in [procvardef,procdef]) and
(typesym.realname[1]<>'$') then
result:=result+typesym.realname
else
result:=result+GetTypeName;
end;
function tdef.fulltypename:string;
begin
result:=fullownerhierarchyname(false);
if assigned(typesym) and
not(typ in [procvardef,procdef]) and
(typesym.realname[1]<>'$') then
result:=result+typesym.realname
else
result:=result+GetTypeName;
end;
function tdef.GetTypeName : string;
begin
GetTypeName:='<unknown type>'
end;
function tdef.typesymbolprettyname:string;
begin
result:=OwnerHierarchyName;
if assigned(typesym) then
result:=result+typesym.prettyname
else
result:=result+'<no type symbol>'
end;
function tdef.mangledparaname:string;
begin
result:=OwnerHierarchyName;
if assigned(typesym) then
mangledparaname:=result+typesym.name
else
mangledparaname:=result+getmangledparaname;
end;
function tdef.getmangledparaname:TSymStr;
begin
result:='<unknown type>';
end;
function tdef.unique_id_str: string;
begin
if (defid=defid_not_registered) or
(defid=defid_registered_nost) then
begin
if not assigned(current_module) then
internalerror(2015102505);
current_module.deflist.Add(self);
registered_in_module:=current_module;
{ invert the defid to indicate that it was only set because we
needed a unique number -- then add defid_not_registered so we
don't get the values between defid_registered and 0 }
defid:=-(current_module.deflist.Count-1)+defid_not_registered-1;
end;
{ use deflist_index so that it will remain the same if def first gets a
defid just for the unique id (as above) and later it gets registered
because it must be saved to the ppu }
result:=hexstr(deflist_index,sizeof(defid)*2);
end;
function tdef.getparentdef:tdef;
begin
result:=nil;
end;
function tdef.getsymtable(t:tgetsymtable):TSymtable;
begin
result:=nil;
end;
function tdef.packedbitsize:asizeint;
begin
result:=size * 8;
end;
function tdef.structalignment: shortint;
begin
result:=alignment;
end;
function tdef.aggregatealignment: shortint;
begin
if Assigned(Owner) and Assigned(Owner.defowner) and (Owner.defowner is TDef) and (Owner.defowner <> Self) then
Result := max(structalignment, TDef(Owner.defowner).aggregatealignment)
else
Result := structalignment;
end;
procedure tdef.ChangeOwner(st:TSymtable);
begin
// if assigned(Owner) then
// Owner.DefList.List[i]:=nil;
Owner:=st;
Owner.DefList.Add(self);
end;
function tdef.getreusablesymtab: tsymtable;
var
origowner: TSymtable;
begin
{ if the original def was in a localsymtable, don't create a
reusable copy in the unit's staticsymtable since the localsymtable
won't be saved to the ppu and as a result we can get unreachable
defs when reloading the derived ones from the ppu }
origowner:=owner;
while not(origowner.symtabletype in [localsymtable,staticsymtable,globalsymtable,stt_excepTSymtable]) do
origowner:=origowner.defowner.owner;
{ if the def is in an exceptionsymtable, we can't create a reusable
def because the original one will be freed when the (always
temprary) exceptionsymtable is freed }
if origowner.symtabletype=stt_excepTSymtable then
internalerror(2015111701)
else if origowner.symtabletype=localsymtable then
result:=origowner
else if assigned(current_module.localsymtable) then
result:=current_module.localsymtable
else
result:=current_module.globalsymtable;
end;
procedure tdef.register_created_object_type;
begin
end;
function tdef.get_top_level_symtable(skipprocdefs: boolean): tsymtable;
begin
result:=owner;
while assigned(result) and
assigned(result.defowner) and
(skipprocdefs or (result.symtabletype in [ObjectSymtable,recordsymtable])) do
result:=tdef(result.defowner).owner;
end;
function tdef.deflist_index: longint;
begin
if defid<defid_not_registered then
result:=-(defid-defid_not_registered+1)
else if defid>=0 then
result:=defid
else
internalerror(2015102502)
end;
{$ifdef DEBUG_NODE_XML}
procedure TDef.XMLPrintDef(Sym: TSym);
var
T: Text;
begin
if current_module.ppxfilefail then
Exit;
Assign(T, current_module.ppxfilename);
{$push} {$I-}
Append(T);
if IOResult <> 0 then
begin
Message1(exec_e_cant_create_archivefile,current_module.ppxfilename);
current_module.ppxfilefail := True;
Exit;
end;
{$pop}
XMLPrintDefTree(T, Sym);
Close(T);
end;
{$endif DEBUG_NODE_XML}
{****************************************************************************
TSYM (base for all symtypes)
****************************************************************************}
function tsym.registered: boolean;
begin
result:=symid>symid_not_registered;
end;
constructor tsym.create(st:tsymtyp;const aname:string);
begin
inherited CreateNotOwned;
realname:=aname;
typ:=st;
RefList:=nil;
symoptions:=[];
fileinfo:=current_tokenpos;
isdbgwritten := false;
visibility:=vis_public;
deprecatedmsg:=nil;
symid:=symid_not_registered;
end;
destructor Tsym.destroy;
begin
stringdispose(deprecatedmsg);
if assigned(RefList) then
RefList.Free;
inherited Destroy;
end;
procedure Tsym.IncRefCount;
begin
inc(refs);
if cs_browser in current_settings.moduleswitches then
begin
MaybeCreateRefList;
AddRef;
end;
end;
procedure Tsym.IncRefCountBy(AValue : longint);
begin
inc(refs,AValue);
end;
procedure Tsym.MaybeCreateRefList;
begin
if not assigned(reflist) then
reflist:=TRefLinkedList.create;
end;
procedure Tsym.AddRef;
var
RefItem: TRefItem;
begin
RefItem:=TRefItem.Create(current_tokenpos);
RefList.Concat(RefItem);
end;
procedure Tsym.buildderef;
begin
end;
procedure Tsym.deref;
begin
end;
function tsym.mangledname : TSymStr;
begin
internalerror(200204171);
result:='';
end;
function tsym.prettyname : string;
begin
result:=realname;
end;
procedure tsym.ChangeOwner(st:TSymtable);
begin
Owner:=st;
inherited ChangeOwner(Owner.SymList);
end;
{****************************************************************************
tpropaccesslist
****************************************************************************}
constructor tpropaccesslist.create;
begin
procdef:=nil; { needed for procedures }
firstsym:=nil;
lastsym:=nil;
end;
destructor tpropaccesslist.destroy;
begin
clear;
end;
function tpropaccesslist.empty:boolean;
begin
empty:=(firstsym=nil);
end;
function tpropaccesslist.getcopy: tpropaccesslist;
var
hp, dest : ppropaccesslistitem;
begin
result:=tpropaccesslist.create;
result.procdef:=procdef;
hp:=firstsym;
while assigned(hp) do
begin
new(dest);
dest^:=hp^;
dest^.next:=nil;
if not assigned(result.firstsym) then
result.firstsym:=dest;
if assigned(result.lastsym) then
result.lastsym^.next:=dest;
result.lastsym:=dest;
hp:=hp^.next;
end;
end;
procedure tpropaccesslist.clear;
var
hp : ppropaccesslistitem;
begin
while assigned(firstsym) do
begin
hp:=firstsym;
firstsym:=firstsym^.next;
dispose(hp);
end;
firstsym:=nil;
lastsym:=nil;
procdef:=nil;
end;
procedure tpropaccesslist.addsym(slt:tsltype;p:tsym);
var
hp : ppropaccesslistitem;
begin
new(hp);
fillchar(hp^,sizeof(tpropaccesslistitem),0);
hp^.sltype:=slt;
hp^.sym:=p;
hp^.symderef.reset;
if assigned(lastsym) then
lastsym^.next:=hp
else
firstsym:=hp;
lastsym:=hp;
end;
procedure tpropaccesslist.addconst(slt:tsltype;const v:TConstExprInt;d:tdef);
var
hp : ppropaccesslistitem;
begin
new(hp);
fillchar(hp^,sizeof(tpropaccesslistitem),0);
hp^.sltype:=slt;
hp^.value:=v;
hp^.valuedef:=d;
hp^.valuedefderef.reset;
if assigned(lastsym) then
lastsym^.next:=hp
else
firstsym:=hp;
lastsym:=hp;
end;
procedure tpropaccesslist.addtype(slt:tsltype;d:tdef);
var
hp : ppropaccesslistitem;
begin
new(hp);
fillchar(hp^,sizeof(tpropaccesslistitem),0);
hp^.sltype:=slt;
hp^.def:=d;
hp^.defderef.reset;
if assigned(lastsym) then
lastsym^.next:=hp
else
firstsym:=hp;
lastsym:=hp;
end;
procedure tpropaccesslist.addsymderef(slt:tsltype;d:tderef);
begin
addsym(slt,nil);
lastsym^.symderef:=d;
end;
procedure tpropaccesslist.addconstderef(slt:tsltype;const v:TConstExprInt;d:tderef);
begin
addconst(slt,v,nil);
lastsym^.valuedefderef:=d;
end;
procedure tpropaccesslist.addtypederef(slt:tsltype;d:tderef);
begin
addtype(slt,nil);
lastsym^.defderef:=d;
end;
procedure tpropaccesslist.resolve;
var
hp : ppropaccesslistitem;
begin
procdef:=tdef(procdefderef.resolve);
hp:=firstsym;
while assigned(hp) do
begin
case hp^.sltype of
sl_call,
sl_load,
sl_subscript :
hp^.sym:=tsym(hp^.symderef.resolve);
sl_absolutetype,
sl_typeconv :
hp^.def:=tdef(hp^.defderef.resolve);
sl_vec:
hp^.valuedef:=tdef(hp^.valuedefderef.resolve);
else
internalerror(2001102001);
end;
hp:=hp^.next;
end;
end;
procedure tpropaccesslist.buildderef;
var
hp : ppropaccesslistitem;
begin
procdefderef.build(procdef);
hp:=firstsym;
while assigned(hp) do
begin
case hp^.sltype of
sl_call,
sl_load,
sl_subscript :
hp^.symderef.build(hp^.sym);
sl_absolutetype,
sl_typeconv :
hp^.defderef.build(hp^.def);
sl_vec:
hp^.valuedefderef.build(hp^.valuedef);
else
internalerror(2001102002);
end;
hp:=hp^.next;
end;
end;
{****************************************************************************
Tderef
****************************************************************************}
procedure tderef.reset;
begin
dataidx:=-1;
end;
procedure tderef.build(s:TObject);
var
len : byte;
st : TSymtable;
data : array[0..255] of byte;
idx : word;
begin
{ skip length byte }
len:=1;
if assigned(s) then
begin
{ TODO: ugly hack}
if s is tsym then
begin
{ if it has been registered but it wasn't put in a symbol table,
this symbol shouldn't be written to a ppu }
if tsym(s).SymId=symid_registered_nost then
Internalerror(2015102504);
if not tsym(s).registered then
tsym(s).register_sym;
st:=FindUnitSymtable(tsym(s).owner)
end
else if s is tdef then
begin
{ same as above }
if tdef(s).defid=defid_registered_nost then
Internalerror(2015102501);
if not tdef(s).registered then
tdef(s).register_def;
st:=FindUnitSymtable(tdef(s).owner);
end
else
internalerror(2016090204);
if not st.iscurrentunit then
begin
{ register that the unit is needed for resolving }
data[len]:=ord(deref_unit);
idx:=current_module.derefidx_unit(st.moduleid);
data[len+1]:=idx shr 8 and $ff;
data[len+2]:=idx and $ff;
inc(len,3);
end;
if s is tsym then
begin
data[len]:=ord(deref_symid);
data[len+1]:=tsym(s).symid shr 24 and $ff;
data[len+2]:=tsym(s).symid shr 16 and $ff;
data[len+3]:=tsym(s).symid shr 8 and $ff;
data[len+4]:=tsym(s).symid and $ff;
inc(len,5);
end
else
begin
data[len]:=ord(deref_defid);
data[len+1]:=tdef(s).defid shr 24 and $ff;
data[len+2]:=tdef(s).defid shr 16 and $ff;
data[len+3]:=tdef(s).defid shr 8 and $ff;
data[len+4]:=tdef(s).defid and $ff;
inc(len,5);
end;
end
else
begin
{ nil pointer }
data[len]:=ord(deref_nil);
inc(len);
end;
{ store data length in first byte }
data[0]:=len-1;
{ store index and write to derefdata }
dataidx:=current_module.derefdata.size;
current_module.derefdata.write(data,len);
end;
function tderef.resolve:TObject;
var
pm : tmodule;
typ : tdereftype;
idx : longint;
i : aint;
len : byte;
data : array[0..255] of byte;
begin
result:=nil;
{ not initialized or error }
if dataidx<0 then
internalerror(200306067);
{ read data }
current_module.derefdata.seek(dataidx);
if current_module.derefdata.read(len,1)<>1 then
internalerror(200310221);
if len>0 then
begin
if current_module.derefdata.read(data,len)<>len then
internalerror(200310222);
end;
{ process data }
pm:=current_module;
i:=0;
while (i<len) do
begin
typ:=tdereftype(data[i]);
inc(i);
case typ of
deref_unit :
begin
idx:=(data[i] shl 8) or data[i+1];
inc(i,2);
pm:=current_module.resolve_unit(idx);
end;
deref_defid :
begin
idx:=longint((data[i] shl 24) or (data[i+1] shl 16) or (data[i+2] shl 8) or data[i+3]);
inc(i,4);
result:=tdef(pm.deflist[idx]);
end;
deref_symid :
begin
idx:=longint((data[i] shl 24) or (data[i+1] shl 16) or (data[i+2] shl 8) or data[i+3]);
inc(i,4);
result:=tsym(pm.symlist[idx]);
end;
deref_nil :
begin
result:=nil;
{ Only allowed when no other deref is available }
if len<>1 then
internalerror(200306232);
end;
end;
end;
end;
{*****************************************************************************
TCompilerPPUFile
*****************************************************************************}
procedure tcompilerppufile.checkerror;
begin
if error then
Message(unit_f_ppu_read_error);
end;
procedure tcompilerppufile.RaiseAssertion(Code: Longint);
begin
InternalError(Code);
end;
procedure tcompilerppufile.getguid(var g: tguid);
begin
longint(g.d1):=getlongint;
g.d2:=getword;
g.d3:=getword;
getdata(g.d4,sizeof(g.d4));
end;
function tcompilerppufile.getexprint:Tconstexprint;
begin
getexprint.overflow:=false;
getexprint.signed:=getboolean;
getexprint.svalue:=getint64;
end;
procedure tcompilerppufile.getposinfo(var p:tfileposinfo);
var
info : byte;
begin
{
info byte layout in bits:
0-1 - amount of bytes for fileindex
2-3 - amount of bytes for line
4-5 - amount of bytes for column
}
info:=getbyte;
case (info and $03) of
0 : p.fileindex:=getbyte;
1 : p.fileindex:=getword;
2 : p.fileindex:=(getbyte shl 16) or getword;
3 : p.fileindex:=getlongint;
end;
case ((info shr 2) and $03) of
0 : p.line:=getbyte;
1 : p.line:=getword;
2 : p.line:=(getbyte shl 16) or getword;
3 : p.line:=getlongint;
end;
case ((info shr 4) and $03) of
0 : p.column:=getbyte;
1 : p.column:=getword;
2 : p.column:=(getbyte shl 16) or getword;
3 : p.column:=getlongint;
end;
p.moduleindex:=current_module.unit_index;
end;
procedure tcompilerppufile.getderef(var d:tderef);
begin
d.dataidx:=getlongint;
end;
function tcompilerppufile.getpropaccesslist:tpropaccesslist;
var
hderef : tderef;
slt : tsltype;
idx : longint;
p : tpropaccesslist;
begin
p:=tpropaccesslist.create;
getderef(p.procdefderef);
repeat
slt:=tsltype(getbyte);
case slt of
sl_none :
break;
sl_call,
sl_load,
sl_subscript :
begin
getderef(hderef);
p.addsymderef(slt,hderef);
end;
sl_absolutetype,
sl_typeconv :
begin
getderef(hderef);
p.addtypederef(slt,hderef);
end;
sl_vec :
begin
idx:=getlongint;
getderef(hderef);
p.addconstderef(slt,idx,hderef);
end;
end;
until false;
getpropaccesslist:=tpropaccesslist(p);
end;
function tcompilerppufile.getasmsymbol:tasmsymbol;
begin
getlongint;
getasmsymbol:=nil;
end;
procedure tcompilerppufile.putposinfo(const p:tfileposinfo);
var
oldcrc : boolean;
info : byte;
begin
{ posinfo is not relevant for changes in PPU }
oldcrc:=do_crc;
do_crc:=false;
{
info byte layout in bits:
0-1 - amount of bytes for fileindex
2-3 - amount of bytes for line
4-5 - amount of bytes for column
}
info:=0;
{ calculate info byte }
if (p.fileindex>$ff) then
begin
info:=info or $1;
{ uncomment this code if tfileposinfo.fileindex type was changed
if (p.fileindex<=$ffff) then
info:=info or $1
else
if (p.fileindex<=$ffffff) then
info:=info or $2
else
info:=info or $3;
}
end;
if (p.line>$ff) then
begin
if (p.line<=$ffff) then
info:=info or $4
else
if (p.line<=$ffffff) then
info:=info or $8
else
info:=info or $c;
end;
if (p.column>$ff) then
begin
info:=info or $10;
{ uncomment this code if tfileposinfo.column type was changed
if (p.column<=$ffff) then
info:=info or $10
else
if (p.column<=$ffffff) then
info:=info or $20
else
info:=info or $30;
}
end;
{ write data }
putbyte(info);
case (info and $03) of
0 : putbyte(p.fileindex);
1 : putword(p.fileindex);
2 : begin
putbyte(p.fileindex shr 16);
putword(p.fileindex and $ffff);
end;
3 : putlongint(p.fileindex);
end;
case ((info shr 2) and $03) of
0 : putbyte(p.line);
1 : putword(p.line);
2 : begin
putbyte(p.line shr 16);
putword(p.line and $ffff);
end;
3 : putlongint(p.line);
end;
case ((info shr 4) and $03) of
0 : putbyte(p.column);
1 : putword(p.column);
2 : begin
putbyte(p.column shr 16);
putword(p.column and $ffff);
end;
3 : putlongint(p.column);
end;
do_crc:=oldcrc;
end;
procedure tcompilerppufile.putguid(const g: tguid);
begin
putlongint(longint(g.d1));
putword(g.d2);
putword(g.d3);
putdata(g.d4,sizeof(g.d4));
end;
procedure Tcompilerppufile.putexprint(const v:Tconstexprint);
begin
if v.overflow then
internalerror(200706102);
putboolean(v.signed);
putint64(v.svalue);
end;
procedure tcompilerppufile.putderef(const d:tderef);
var
oldcrc : boolean;
begin
oldcrc:=do_crc;
do_crc:=false;
if d.dataidx=-1 then
internalerror(2019022201)
else
putlongint(d.dataidx);
do_crc:=oldcrc;
end;
procedure tcompilerppufile.putpropaccesslist(p:tpropaccesslist);
var
hp : ppropaccesslistitem;
begin
putderef(p.procdefderef);
hp:=p.firstsym;
while assigned(hp) do
begin
putbyte(byte(hp^.sltype));
case hp^.sltype of
sl_call,
sl_load,
sl_subscript :
putderef(hp^.symderef);
sl_absolutetype,
sl_typeconv :
putderef(hp^.defderef);
sl_vec :
begin
putlongint(int64(hp^.value));
putderef(hp^.valuedefderef);
end;
else
internalerror(2001102003);
end;
hp:=hp^.next;
end;
putbyte(byte(sl_none));
end;
procedure tcompilerppufile.putasmsymbol(s:tasmsymbol);
begin
putlongint(0);
end;
{$ifdef MEMDEBUG}
initialization
memmanglednames:=TMemDebug.create('Manglednames');
memmanglednames.stop;
memprocpara:=TMemDebug.create('ProcPara');
memprocpara.stop;
memprocparast:=TMemDebug.create('ProcParaSt');
memprocparast.stop;
memproclocalst:=TMemDebug.create('ProcLocalSt');
memproclocalst.stop;
memprocnodetree:=TMemDebug.create('ProcNodeTree');
memprocnodetree.stop;
finalization
memmanglednames.free;
memprocpara.free;
memprocparast.free;
memproclocalst.free;
memprocnodetree.free;
{$endif MEMDEBUG}
end.