| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691 | {    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,      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; restore_current_debuginfo : boolean);    procedure DoneDebugInfo(hp:tmodule;var current_debuginfo_reset : boolean);    procedure RegisterDebugInfo(const r:tdbginfo;c:TDebugInfoClass);implementation    uses      cutils,      globals,      verbose,      cgbase;    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);                  dbg_state_queued:                    begin                      { can happen in case an objectdef was used from another                        unit that was compiled without debug info, and we are                        using Stabs (which means that parent types have to be                        written before child types). In this case, the child                        objectdef will be queued and never written, because its                        definition is not inside the current unit and hence will                        not be encountered }                      if def.typ<>objectdef then                        internalerror(2012072401);                      if not assigned(tobjectdef(def).childof) or                         (tobjectdef(def).childof.dbg_state=dbg_state_written) then                        appenddef(list,def)                      else if tobjectdef(def).childof.dbg_state=dbg_state_queued then                        begin                          { ensure that the parent is indeed queued }                          deftowritelist.add(tobjectdef(def).childof);                          deftowritelist.add(def);                        end                      else if tobjectdef(def).childof.dbg_state=dbg_state_used then                        { comes somewhere after the current def in the looplist                          and will be written at that point, so we will have to                          wait until the next iteration }                        deftowritelist.add(def)                      else                        internalerror(2012072402);                    end;                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 :            if not assigned(tstaticvarsym(sym).fieldvarsym) or               not(df_generic in tdef(tstaticvarsym(sym).fieldvarsym.owner.defowner).defoptions) then              appendsym_staticvar(list,tstaticvarsym(sym));          unitsym:            appendsym_unit(list,tunitsym(sym));          labelsym :            appendsym_label(list,tlabelsym(sym));          localvarsym :            appendsym_localvar(list,tlocalvarsym(sym));          paravarsym :            if tparavarsym(sym).localloc.loc<>LOC_VOID then              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));          namespacesym :            { ignore namespace syms, they are only of internal use }            ;          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))));          else            ;        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))));          else            ;        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))));          else            ;        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) and               { avoid all generic symbols }               not (sp_generic_dummy in sym.symoptions) and               not ((sym.typ=typesym) and assigned(ttypesym(sym).typedef) and                    (df_generic in ttypesym(sym).typedef.defoptions)) 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))));          else            ;        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;              else                ;            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 and not assigned(pu.u.package) 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; restore_current_debuginfo : boolean);      begin        if not assigned(CDebugInfo[target_dbg.id]) then          begin            Comment(V_Fatal,'cg_f_debuginfo_output_not_supported');            exit;          end;{$ifndef llvm}        hp.DebugInfo:=CDebugInfo[target_dbg.id].Create;{$else}        { we can't override the assignment of target_dbg with the LLVM class,          because we still need to know whether to tell LLVM to generate          DWARFv2/3/4/5/... }        hp.DebugInfo:=CDebugInfo[dbg_llvm].Create;{$endif}        if restore_current_debuginfo then          begin            if current_debuginfo=nil then              current_debuginfo:=tdebuginfo(hp.DebugInfo)            else              internalerror(2012032101);          end;      end;    procedure DoneDebugInfo(hp:tmodule;var current_debuginfo_reset : boolean);      begin        current_debuginfo_reset:=false;        if assigned(hp.DebugInfo) then          begin            if hp.DebugInfo=current_debuginfo then              begin                current_debuginfo:=nil;                current_debuginfo_reset:=true;              end;            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.
 |