fpc/compiler/dbgbase.pas
paul 915c412ae3 compiler: generate debug info for record methods:
- extended write_symtable_procdefs to handle record definitions
  - fix stabs info generator to use prefixed symbol names for record methods, also handle records static symbols the same way as for object types
  - fix dwarf info record generation: add visibility info and write methods, also prefix record members the same way as object members

git-svn-id: branches/paul/extended_records@16570 -
2010-12-16 06:12:05 +00:00

624 lines
18 KiB
ObjectPascal

{
Copyright (c) 2003-2006 by Peter Vreman and Florian Klaempfl
This units contains the base class for debug info generation
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 dbgbase;
{$i fpcdefs.inc}
interface
uses
cclasses,
systems,
parabase,
symconst,symbase,symdef,symtype,symsym,symtable,
fmodule,
aasmtai,aasmdata;
type
TDebugInfo=class
protected
{ definitions }
{ collect all defs in one list so we can reset them easily }
defnumberlist : TFPObjectList;
deftowritelist : TFPObjectList;
procedure appenddef(list:TAsmList;def:tdef);
procedure beforeappenddef(list:TAsmList;def:tdef);virtual;
procedure afterappenddef(list:TAsmList;def:tdef);virtual;
procedure appenddef_ord(list:TAsmList;def:torddef);virtual;
procedure appenddef_float(list:TAsmList;def:tfloatdef);virtual;
procedure appenddef_file(list:TAsmList;def:tfiledef);virtual;
procedure appenddef_enum(list:TAsmList;def:tenumdef);virtual;
procedure appenddef_array(list:TAsmList;def:tarraydef);virtual;
procedure appenddef_record(list:TAsmList;def:trecorddef);virtual;
procedure appenddef_object(list:TAsmList;def:tobjectdef);virtual;
procedure appenddef_classref(list:TAsmList;def: tclassrefdef);virtual;
procedure appenddef_pointer(list:TAsmList;def:tpointerdef);virtual;
procedure appenddef_string(list:TAsmList;def:tstringdef);virtual;
procedure appenddef_procvar(list:TAsmList;def:tprocvardef);virtual;
procedure appenddef_variant(list:TAsmList;def:tvariantdef);virtual;
procedure appenddef_set(list:TAsmList;def:tsetdef);virtual;
procedure appenddef_formal(list:TAsmList;def:tformaldef);virtual;
procedure appenddef_undefined(list:TAsmList;def: tundefineddef);virtual;
procedure appendprocdef(list:TAsmList;def:tprocdef);virtual;
procedure write_remaining_defs_to_write(list:TAsmList);
{ symbols }
procedure appendsym(list:TAsmList;sym:tsym);
procedure beforeappendsym(list:TAsmList;sym:tsym);virtual;
procedure afterappendsym(list:TAsmList;sym:tsym);virtual;
procedure appendsym_staticvar(list:TAsmList;sym:tstaticvarsym);virtual;
procedure appendsym_paravar(list:TAsmList;sym:tparavarsym);virtual;
procedure appendsym_localvar(list:TAsmList;sym:tlocalvarsym);virtual;
procedure appendsym_fieldvar(list:TAsmList;sym:tfieldvarsym);virtual;
procedure appendsym_unit(list:TAsmList;sym:tunitsym);virtual;
procedure appendsym_const(list:TAsmList;sym:tconstsym);virtual;
procedure appendsym_type(list:TAsmList;sym:ttypesym);virtual;
procedure appendsym_label(list:TAsmList;sym:tlabelsym);virtual;
procedure appendsym_absolute(list:TAsmList;sym:tabsolutevarsym);virtual;
procedure appendsym_property(list:TAsmList;sym:tpropertysym);virtual;
{ symtable }
procedure write_symtable_parasyms(list:TAsmList;paras: tparalist);
procedure write_symtable_syms(list:TAsmList;st:TSymtable);
procedure write_symtable_defs(list:TAsmList;st:TSymtable);
procedure write_symtable_procdefs(list:TAsmList;st:TSymtable);
procedure reset_unit_type_info;
procedure write_used_unit_type_info(list:TAsmList;hp:tmodule);
public
constructor Create;virtual;
procedure inserttypeinfo;virtual;
procedure insertmoduleinfo;virtual;
procedure insertlineinfo(list:TAsmList);virtual;
procedure referencesections(list:TAsmList);virtual;
end;
TDebugInfoClass=class of TDebugInfo;
var
CDebugInfo : array[tdbg] of TDebugInfoClass;
current_debuginfo : tdebuginfo;
procedure InitDebugInfo(hp:tmodule);
procedure DoneDebugInfo(hp:tmodule);
procedure RegisterDebugInfo(const r:tdbginfo;c:TDebugInfoClass);
implementation
uses
cutils,
verbose;
constructor TDebugInfo.Create;
begin
end;
procedure TDebugInfo.insertmoduleinfo;
begin
end;
procedure TDebugInfo.inserttypeinfo;
begin
end;
procedure TDebugInfo.insertlineinfo(list:TAsmList);
begin
end;
procedure TDebugInfo.referencesections(list:TAsmList);
begin
end;
{**************************************
Definition
**************************************}
procedure TDebugInfo.appendprocdef(list:TAsmList;def:tprocdef);
begin
end;
procedure TDebugInfo.beforeappenddef(list:TAsmList;def:tdef);
begin
end;
procedure TDebugInfo.afterappenddef(list:TAsmList;def:tdef);
begin
end;
procedure TDebugInfo.appenddef_ord(list:TAsmList;def:torddef);
begin
end;
procedure TDebugInfo.appenddef_float(list:TAsmList;def:tfloatdef);
begin
end;
procedure TDebugInfo.appenddef_formal(list:TAsmList;def: tformaldef);
begin
end;
procedure TDebugInfo.appenddef_undefined(list:TAsmList;def: tundefineddef);
begin
end;
procedure TDebugInfo.appenddef_set(list:TAsmList;def: tsetdef);
begin
end;
procedure TDebugInfo.appenddef_object(list:TAsmList;def: tobjectdef);
begin
end;
procedure TDebugInfo.appenddef_classref(list:TAsmList;def: tclassrefdef);
begin
appenddef_pointer(list,tpointerdef(pvmttype));
end;
procedure TDebugInfo.appenddef_variant(list:TAsmList;def: tvariantdef);
begin
end;
procedure TDebugInfo.appenddef_enum(list:TAsmList;def:tenumdef);
begin
end;
procedure TDebugInfo.appenddef_file(list:TAsmList;def: tfiledef);
begin
end;
procedure TDebugInfo.appenddef_array(list:TAsmList;def:tarraydef);
begin
end;
procedure TDebugInfo.appenddef_record(list:TAsmList;def:trecorddef);
begin
end;
procedure TDebugInfo.appenddef_pointer(list:TAsmList;def:tpointerdef);
begin
end;
procedure TDebugInfo.appenddef_string(list:TAsmList;def:tstringdef);
begin
end;
procedure TDebugInfo.appenddef_procvar(list:TAsmList;def:tprocvardef);
begin
end;
procedure TDebugInfo.appenddef(list:TAsmList;def:tdef);
begin
if (def.dbg_state in [dbg_state_writing,dbg_state_written]) then
exit;
{ never write generic template defs }
if df_generic in def.defoptions then
begin
def.dbg_state:=dbg_state_written;
exit;
end;
{ to avoid infinite loops }
def.dbg_state := dbg_state_writing;
beforeappenddef(list,def);
{ queued defs have to be written later }
if (def.dbg_state=dbg_state_queued) then
exit;
case def.typ of
stringdef :
appenddef_string(list,tstringdef(def));
enumdef :
appenddef_enum(list,tenumdef(def));
orddef :
appenddef_ord(list,torddef(def));
pointerdef :
appenddef_pointer(list,tpointerdef(def));
floatdef :
appenddef_float(list,tfloatdef(def));
filedef :
appenddef_file(list,tfiledef(def));
recorddef :
appenddef_record(list,trecorddef(def));
variantdef :
appenddef_variant(list,tvariantdef(def));
classrefdef :
appenddef_classref(list,tclassrefdef(def));
setdef :
appenddef_set(list,tsetdef(def));
formaldef :
appenddef_formal(list,tformaldef(def));
arraydef :
appenddef_array(list,tarraydef(def));
procvardef :
appenddef_procvar(list,tprocvardef(def));
objectdef :
appenddef_object(list,tobjectdef(def));
undefineddef :
appenddef_undefined(list,tundefineddef(def));
procdef :
begin
{ procdefs are already written in a separate step. procdef
support in appenddef is only needed for beforeappenddef to
write all local type defs }
end;
else
internalerror(200601281);
end;
afterappenddef(list,def);
def.dbg_state := dbg_state_written;
end;
procedure TDebugInfo.write_remaining_defs_to_write(list:TAsmList);
var
n : integer;
looplist,
templist: TFPObjectList;
def : tdef;
begin
templist := TFPObjectList.Create(False);
looplist := deftowritelist;
while looplist.count > 0 do
begin
deftowritelist := templist;
for n := 0 to looplist.count - 1 do
begin
def := tdef(looplist[n]);
case def.dbg_state of
dbg_state_written:
continue;
dbg_state_writing:
internalerror(200610052);
dbg_state_unused:
internalerror(200610053);
dbg_state_used:
appenddef(list,def);
else
internalerror(200610054);
end;
end;
looplist.clear;
templist := looplist;
looplist := deftowritelist;
end;
templist.free;
end;
{**************************************
Symbols
**************************************}
procedure TDebugInfo.beforeappendsym(list:TAsmList;sym:tsym);
begin
end;
procedure TDebugInfo.afterappendsym(list:TAsmList;sym:tsym);
begin
end;
procedure TDebugInfo.appendsym_staticvar(list:TAsmList;sym:tstaticvarsym);
begin
end;
procedure TDebugInfo.appendsym_paravar(list:TAsmList;sym: tparavarsym);
begin
end;
procedure TDebugInfo.appendsym_localvar(list:TAsmList;sym: tlocalvarsym);
begin
end;
procedure TDebugInfo.appendsym_fieldvar(list:TAsmList;sym: tfieldvarsym);
begin
end;
procedure TDebugInfo.appendsym_const(list:TAsmList;sym:tconstsym);
begin
end;
procedure TDebugInfo.appendsym_label(list:TAsmList;sym: tlabelsym);
begin
end;
procedure TDebugInfo.appendsym_property(list:TAsmList;sym: tpropertysym);
begin
end;
procedure TDebugInfo.appendsym_type(list:TAsmList;sym: ttypesym);
begin
end;
procedure TDebugInfo.appendsym_unit(list:TAsmList;sym: tunitsym);
begin
end;
procedure TDebugInfo.appendsym_absolute(list:TAsmList;sym:tabsolutevarsym);
begin
end;
procedure TDebugInfo.appendsym(list:TAsmList;sym:tsym);
begin
if sym.isdbgwritten then
exit;
beforeappendsym(list,sym);
case sym.typ of
staticvarsym :
appendsym_staticvar(list,tstaticvarsym(sym));
unitsym:
appendsym_unit(list,tunitsym(sym));
labelsym :
appendsym_label(list,tlabelsym(sym));
localvarsym :
appendsym_localvar(list,tlocalvarsym(sym));
paravarsym :
appendsym_paravar(list,tparavarsym(sym));
constsym :
appendsym_const(list,tconstsym(sym));
typesym :
appendsym_type(list,ttypesym(sym));
enumsym :
{ ignore enum syms, they are written by the owner }
;
syssym :
{ ignore sys syms, they are only of internal use }
;
procsym :
{ ignore proc syms, they are written by procdefs }
;
absolutevarsym :
appendsym_absolute(list,tabsolutevarsym(sym));
propertysym :
appendsym_property(list,tpropertysym(sym));
else
internalerror(200601242);
end;
afterappendsym(list,sym);
sym.isdbgwritten:=true;
end;
{**************************************
Symtables
**************************************}
procedure TDebugInfo.write_symtable_defs(list:TAsmList;st:TSymtable);
var
def : tdef;
i : longint;
nonewadded : boolean;
begin
case st.symtabletype of
staticsymtable :
list.concat(tai_comment.Create(strpnew('Defs - Begin Staticsymtable')));
globalsymtable :
list.concat(tai_comment.Create(strpnew('Defs - Begin unit '+st.name^+' has index '+tostr(st.moduleid))));
end;
repeat
nonewadded:=true;
for i:=0 to st.DefList.Count-1 do
begin
def:=tdef(st.DefList[i]);
if (def.dbg_state in [dbg_state_used,dbg_state_queued]) then
begin
appenddef(list,def);
nonewadded:=false;
end;
end;
until nonewadded;
case st.symtabletype of
staticsymtable :
list.concat(tai_comment.Create(strpnew('Defs - End Staticsymtable')));
globalsymtable :
list.concat(tai_comment.Create(strpnew('Defs - End unit '+st.name^+' has index '+tostr(st.moduleid))));
end;
end;
procedure TDebugInfo.write_symtable_parasyms(list:TAsmList;paras: tparalist);
var
i : longint;
sym : tsym;
begin
for i:=0 to paras.Count-1 do
begin
sym:=tsym(paras[i]);
if (sym.visibility<>vis_hidden) then
begin
appendsym(list,sym);
{ if we ever write this procdef again for some reason (this
can happen with DWARF), then we want to write all the
parasyms again as well. }
sym.isdbgwritten:=false;
end;
end;
end;
procedure TDebugInfo.write_symtable_syms(list:TAsmList;st:TSymtable);
var
i : longint;
sym : tsym;
begin
case st.symtabletype of
staticsymtable :
list.concat(tai_comment.Create(strpnew('Syms - Begin Staticsymtable')));
globalsymtable :
list.concat(tai_comment.Create(strpnew('Syms - Begin unit '+st.name^+' has index '+tostr(st.moduleid))));
end;
for i:=0 to st.SymList.Count-1 do
begin
sym:=tsym(st.SymList[i]);
if (sym.visibility<>vis_hidden) and
(not sym.isdbgwritten) then
appendsym(list,sym);
end;
case st.symtabletype of
staticsymtable :
list.concat(tai_comment.Create(strpnew('Syms - End Staticsymtable')));
globalsymtable :
list.concat(tai_comment.Create(strpnew('Syms - End unit '+st.name^+' has index '+tostr(st.moduleid))));
end;
end;
procedure TDebugInfo.write_symtable_procdefs(list:TAsmList;st:TSymtable);
var
i : longint;
def : tdef;
begin
for i:=0 to st.DefList.Count-1 do
begin
def:=tdef(st.DefList[i]);
case def.typ of
procdef :
begin
appendprocdef(list,tprocdef(def));
if assigned(tprocdef(def).localst) then
write_symtable_procdefs(list,tprocdef(def).localst);
end;
objectdef,recorddef :
begin
write_symtable_procdefs(list,tabstractrecorddef(def).symtable);
end;
end;
end;
end;
procedure TDebugInfo.reset_unit_type_info;
var
hp : tmodule;
begin
hp:=tmodule(loaded_units.first);
while assigned(hp) do
begin
hp.is_dbginfo_written:=false;
hp:=tmodule(hp.next);
end;
end;
procedure TDebugInfo.write_used_unit_type_info(list:TAsmList;hp:tmodule);
var
pu : tused_unit;
begin
pu:=tused_unit(hp.used_units.first);
while assigned(pu) do
begin
if not pu.u.is_dbginfo_written then
begin
{ prevent infinte loop for circular dependencies }
pu.u.is_dbginfo_written:=true;
{ write type info from used units, use a depth first
strategy to reduce the recursion in writing all
dependent stabs }
write_used_unit_type_info(list,pu.u);
if assigned(pu.u.globalsymtable) then
write_symtable_defs(list,pu.u.globalsymtable);
end;
pu:=tused_unit(pu.next);
end;
end;
{****************************************************************************
Init / Done
****************************************************************************}
procedure InitDebugInfo(hp:tmodule);
begin
if not assigned(CDebugInfo[target_dbg.id]) then
begin
Comment(V_Fatal,'cg_f_debuginfo_output_not_supported');
exit;
end;
hp.DebugInfo:=CDebugInfo[target_dbg.id].Create;
end;
procedure DoneDebugInfo(hp:tmodule);
begin
if assigned(hp.DebugInfo) then
begin
hp.DebugInfo.Free;
hp.DebugInfo:=nil;
end;
end;
procedure RegisterDebugInfo(const r:tdbginfo;c:TDebugInfoClass);
var
t : tdbg;
begin
t:=r.id;
if assigned(dbginfos[t]) then
writeln('Warning: DebugInfo is already registered!')
else
Getmem(dbginfos[t],sizeof(tdbginfo));
dbginfos[t]^:=r;
CDebugInfo[t]:=c;
end;
const
dbg_none_info : tdbginfo =
(
id : dbg_none;
idtxt : 'NONE';
);
initialization
RegisterDebugInfo(dbg_none_info,TDebugInfo);
end.