| |
| |
| |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| package ld |
|
|
| import ( |
| "cmd/internal/dwarf" |
| "cmd/internal/obj" |
| "cmd/internal/objabi" |
| "cmd/internal/src" |
| "cmd/internal/sys" |
| "cmd/link/internal/loader" |
| "cmd/link/internal/sym" |
| "cmp" |
| "fmt" |
| "internal/abi" |
| "internal/buildcfg" |
| "log" |
| "path" |
| "runtime" |
| "slices" |
| "strings" |
| "sync" |
| ) |
|
|
| |
| |
| |
| |
| |
| |
| type dwctxt struct { |
| linkctxt *Link |
| ldr *loader.Loader |
| arch *sys.Arch |
|
|
| |
| |
| tmap map[string]loader.Sym |
|
|
| |
| |
| |
| |
| |
| rtmap map[loader.Sym]loader.Sym |
|
|
| |
| |
| tdmap map[loader.Sym]loader.Sym |
|
|
| |
| typeRuntimeEface loader.Sym |
| typeRuntimeIface loader.Sym |
| uintptrInfoSym loader.Sym |
|
|
| |
| |
| dwmu *sync.Mutex |
| } |
|
|
| |
| |
| |
| type dwSym loader.Sym |
|
|
| func (c dwctxt) PtrSize() int { |
| return c.arch.PtrSize |
| } |
|
|
| func (c dwctxt) Size(s dwarf.Sym) int64 { |
| return int64(len(c.ldr.Data(loader.Sym(s.(dwSym))))) |
| } |
|
|
| func (c dwctxt) AddInt(s dwarf.Sym, size int, i int64) { |
| ds := loader.Sym(s.(dwSym)) |
| dsu := c.ldr.MakeSymbolUpdater(ds) |
| dsu.AddUintXX(c.arch, uint64(i), size) |
| } |
|
|
| func (c dwctxt) AddBytes(s dwarf.Sym, b []byte) { |
| ds := loader.Sym(s.(dwSym)) |
| dsu := c.ldr.MakeSymbolUpdater(ds) |
| dsu.AddBytes(b) |
| } |
|
|
| func (c dwctxt) AddString(s dwarf.Sym, v string) { |
| ds := loader.Sym(s.(dwSym)) |
| dsu := c.ldr.MakeSymbolUpdater(ds) |
| dsu.Addstring(v) |
| } |
|
|
| func (c dwctxt) AddAddress(s dwarf.Sym, data any, value int64) { |
| ds := loader.Sym(s.(dwSym)) |
| dsu := c.ldr.MakeSymbolUpdater(ds) |
| if value != 0 { |
| value -= dsu.Value() |
| } |
| tgtds := loader.Sym(data.(dwSym)) |
| dsu.AddAddrPlus(c.arch, tgtds, value) |
| } |
|
|
| func (c dwctxt) AddCURelativeAddress(s dwarf.Sym, data any, value int64) { |
| ds := loader.Sym(s.(dwSym)) |
| dsu := c.ldr.MakeSymbolUpdater(ds) |
| if value != 0 { |
| value -= dsu.Value() |
| } |
| tgtds := loader.Sym(data.(dwSym)) |
| dsu.AddCURelativeAddrPlus(c.arch, tgtds, value) |
| } |
|
|
| func (c dwctxt) AddSectionOffset(s dwarf.Sym, size int, t any, ofs int64) { |
| ds := loader.Sym(s.(dwSym)) |
| dsu := c.ldr.MakeSymbolUpdater(ds) |
| tds := loader.Sym(t.(dwSym)) |
| switch size { |
| default: |
| c.linkctxt.Errorf(ds, "invalid size %d in adddwarfref\n", size) |
| case c.arch.PtrSize, 4: |
| } |
| dsu.AddSymRef(c.arch, tds, ofs, objabi.R_ADDROFF, size) |
| } |
|
|
| func (c dwctxt) AddDWARFAddrSectionOffset(s dwarf.Sym, t any, ofs int64) { |
| size := 4 |
| if isDwarf64(c.linkctxt) { |
| size = 8 |
| } |
| ds := loader.Sym(s.(dwSym)) |
| dsu := c.ldr.MakeSymbolUpdater(ds) |
| tds := loader.Sym(t.(dwSym)) |
| switch size { |
| default: |
| c.linkctxt.Errorf(ds, "invalid size %d in adddwarfref\n", size) |
| case c.arch.PtrSize, 4: |
| } |
| dsu.AddSymRef(c.arch, tds, ofs, objabi.R_DWARFSECREF, size) |
| } |
|
|
| func (c dwctxt) AddIndirectTextRef(s dwarf.Sym, t any) { |
| ds := loader.Sym(s.(dwSym)) |
| dsu := c.ldr.MakeSymbolUpdater(ds) |
| tds := loader.Sym(t.(dwSym)) |
| dsu.AddSymRef(c.arch, tds, 0, objabi.R_DWTXTADDR_U4, 4) |
| } |
|
|
| func (c dwctxt) Logf(format string, args ...any) { |
| c.linkctxt.Logf(format, args...) |
| } |
|
|
| |
|
|
| func (c dwctxt) CurrentOffset(s dwarf.Sym) int64 { |
| panic("should be used only in the compiler") |
| } |
|
|
| func (c dwctxt) RecordDclReference(s dwarf.Sym, t dwarf.Sym, dclIdx int, inlIndex int) { |
| panic("should be used only in the compiler") |
| } |
|
|
| func (c dwctxt) RecordChildDieOffsets(s dwarf.Sym, vars []*dwarf.Var, offsets []int32) { |
| panic("should be used only in the compiler") |
| } |
|
|
| func isDwarf64(ctxt *Link) bool { |
| return ctxt.HeadType == objabi.Haix |
| } |
|
|
| |
| |
| |
| const ( |
| GdbScriptPythonFileId = 1 |
| GdbScriptSchemeFileId = 3 |
| GdbScriptPythonTextId = 4 |
| GdbScriptSchemeTextId = 6 |
| ) |
|
|
| var gdbscript string |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| type dwarfSecInfo struct { |
| syms []loader.Sym |
| } |
|
|
| |
| func (dsi *dwarfSecInfo) secSym() loader.Sym { |
| if len(dsi.syms) == 0 { |
| return 0 |
| } |
| return dsi.syms[0] |
| } |
|
|
| |
| func (dsi *dwarfSecInfo) subSyms() []loader.Sym { |
| if len(dsi.syms) == 0 { |
| return []loader.Sym{} |
| } |
| return dsi.syms[1:] |
| } |
|
|
| |
| |
| var dwarfp []dwarfSecInfo |
|
|
| func (d *dwctxt) writeabbrev() dwarfSecInfo { |
| abrvs := d.ldr.CreateSymForUpdate(".debug_abbrev", 0) |
| abrvs.SetType(sym.SDWARFSECT) |
| abrvs.AddBytes(dwarf.GetAbbrev()) |
| return dwarfSecInfo{syms: []loader.Sym{abrvs.Sym()}} |
| } |
|
|
| var dwtypes dwarf.DWDie |
|
|
| |
| |
| |
| |
| |
| |
| |
| func newattr(die *dwarf.DWDie, attr uint16, cls int, value int64, data any) { |
| a := new(dwarf.DWAttr) |
| a.Link = die.Attr |
| die.Attr = a |
| a.Atr = attr |
| a.Cls = uint8(cls) |
| a.Value = value |
| a.Data = data |
| } |
|
|
| |
| |
| |
| func getattr(die *dwarf.DWDie, attr uint16) *dwarf.DWAttr { |
| if die.Attr.Atr == attr { |
| return die.Attr |
| } |
|
|
| a := die.Attr |
| b := a.Link |
| for b != nil { |
| if b.Atr == attr { |
| a.Link = b.Link |
| b.Link = die.Attr |
| die.Attr = b |
| return b |
| } |
|
|
| a = b |
| b = b.Link |
| } |
|
|
| return nil |
| } |
|
|
| |
| |
| |
| |
| |
| func (d *dwctxt) newdie(parent *dwarf.DWDie, abbrev int, name string) *dwarf.DWDie { |
| die := new(dwarf.DWDie) |
| die.Abbrev = abbrev |
| die.Link = parent.Child |
| parent.Child = die |
|
|
| newattr(die, dwarf.DW_AT_name, dwarf.DW_CLS_STRING, int64(len(name)), name) |
|
|
| |
| if name == "" { |
| panic("nameless DWARF DIE") |
| } |
|
|
| var st sym.SymKind |
| switch abbrev { |
| case dwarf.DW_ABRV_FUNCTYPEPARAM, dwarf.DW_ABRV_FUNCTYPEOUTPARAM, dwarf.DW_ABRV_DOTDOTDOT, dwarf.DW_ABRV_STRUCTFIELD, dwarf.DW_ABRV_ARRAYRANGE: |
| |
| |
| return die |
| case dwarf.DW_ABRV_COMPUNIT, dwarf.DW_ABRV_COMPUNIT_TEXTLESS: |
| |
| name = fmt.Sprintf(".pkg.%s.%d", name, len(d.linkctxt.compUnits)) |
| st = sym.SDWARFCUINFO |
| case dwarf.DW_ABRV_VARIABLE: |
| st = sym.SDWARFVAR |
| default: |
| |
| |
| st = sym.SDWARFTYPE |
| } |
| ds := d.ldr.LookupOrCreateSym(dwarf.InfoPrefix+name, 0) |
| dsu := d.ldr.MakeSymbolUpdater(ds) |
| dsu.SetType(st) |
| d.ldr.SetAttrNotInSymbolTable(ds, true) |
| d.ldr.SetAttrReachable(ds, true) |
| die.Sym = dwSym(ds) |
| if abbrev >= dwarf.DW_ABRV_NULLTYPE && abbrev <= dwarf.DW_ABRV_TYPEDECL { |
| d.tmap[name] = ds |
| } |
|
|
| return die |
| } |
|
|
| func walktypedef(die *dwarf.DWDie) *dwarf.DWDie { |
| if die == nil { |
| return nil |
| } |
| |
| if die.Abbrev == dwarf.DW_ABRV_TYPEDECL { |
| for attr := die.Attr; attr != nil; attr = attr.Link { |
| if attr.Atr == dwarf.DW_AT_type && attr.Cls == dwarf.DW_CLS_REFERENCE && attr.Data != nil { |
| return attr.Data.(*dwarf.DWDie) |
| } |
| } |
| } |
|
|
| return die |
| } |
|
|
| func (d *dwctxt) walksymtypedef(symIdx loader.Sym) loader.Sym { |
|
|
| |
| |
| |
| |
|
|
| if ts, ok := d.rtmap[symIdx]; ok { |
| if def, ok := d.tdmap[ts]; ok { |
| return def |
| } |
| d.linkctxt.Errorf(ts, "internal error: no entry for sym %d in tdmap\n", ts) |
| return 0 |
| } |
| d.linkctxt.Errorf(symIdx, "internal error: no entry for sym %d in rtmap\n", symIdx) |
| return 0 |
| } |
|
|
| |
| |
| func findchild(die *dwarf.DWDie, name string) *dwarf.DWDie { |
| var prev *dwarf.DWDie |
| for ; die != prev; prev, die = die, walktypedef(die) { |
| for a := die.Child; a != nil; a = a.Link { |
| if name == getattr(a, dwarf.DW_AT_name).Data { |
| return a |
| } |
| } |
| continue |
| } |
| return nil |
| } |
|
|
| |
| |
| func (d *dwctxt) find(name string) loader.Sym { |
| return d.tmap[name] |
| } |
|
|
| func (d *dwctxt) mustFind(name string) loader.Sym { |
| r := d.find(name) |
| if r == 0 { |
| Exitf("dwarf find: cannot find %s", name) |
| } |
| return r |
| } |
|
|
| func (d *dwctxt) adddwarfref(sb *loader.SymbolBuilder, t loader.Sym, size int) { |
| switch size { |
| default: |
| d.linkctxt.Errorf(sb.Sym(), "invalid size %d in adddwarfref\n", size) |
| case d.arch.PtrSize, 4: |
| } |
| sb.AddSymRef(d.arch, t, 0, objabi.R_DWARFSECREF, size) |
| } |
|
|
| func (d *dwctxt) newrefattr(die *dwarf.DWDie, attr uint16, ref loader.Sym) { |
| if ref == 0 { |
| return |
| } |
| newattr(die, attr, dwarf.DW_CLS_REFERENCE, 0, dwSym(ref)) |
| } |
|
|
| func (d *dwctxt) dtolsym(s dwarf.Sym) loader.Sym { |
| if s == nil { |
| return 0 |
| } |
| dws := loader.Sym(s.(dwSym)) |
| return dws |
| } |
|
|
| func (d *dwctxt) putdie(syms []loader.Sym, die *dwarf.DWDie) []loader.Sym { |
| s := d.dtolsym(die.Sym) |
| if s == 0 { |
| s = syms[len(syms)-1] |
| } else { |
| syms = append(syms, s) |
| } |
| sDwsym := dwSym(s) |
| dwarf.Uleb128put(d, sDwsym, int64(die.Abbrev)) |
| dwarf.PutAttrs(d, sDwsym, die.Abbrev, die.Attr) |
| if dwarf.HasChildren(die) { |
| for die := die.Child; die != nil; die = die.Link { |
| syms = d.putdie(syms, die) |
| } |
| dsu := d.ldr.MakeSymbolUpdater(syms[len(syms)-1]) |
| dsu.AddUint8(0) |
| } |
| return syms |
| } |
|
|
| func reverselist(list **dwarf.DWDie) { |
| curr := *list |
| var prev *dwarf.DWDie |
| for curr != nil { |
| next := curr.Link |
| curr.Link = prev |
| prev = curr |
| curr = next |
| } |
|
|
| *list = prev |
| } |
|
|
| func reversetree(list **dwarf.DWDie) { |
| reverselist(list) |
| for die := *list; die != nil; die = die.Link { |
| if dwarf.HasChildren(die) { |
| reversetree(&die.Child) |
| } |
| } |
| } |
|
|
| func newmemberoffsetattr(die *dwarf.DWDie, offs int32) { |
| newattr(die, dwarf.DW_AT_data_member_location, dwarf.DW_CLS_CONSTANT, int64(offs), nil) |
| } |
|
|
| func (d *dwctxt) lookupOrDiag(n string) loader.Sym { |
| symIdx := d.ldr.Lookup(n, 0) |
| if symIdx == 0 { |
| Exitf("dwarf: missing type: %s", n) |
| } |
| if len(d.ldr.Data(symIdx)) == 0 { |
| Exitf("dwarf: missing type (no data): %s", n) |
| } |
|
|
| return symIdx |
| } |
|
|
| func (d *dwctxt) dotypedef(parent *dwarf.DWDie, name string, def *dwarf.DWDie) *dwarf.DWDie { |
| |
| if strings.HasPrefix(name, "map[") { |
| return nil |
| } |
| if strings.HasPrefix(name, "struct {") { |
| return nil |
| } |
| |
| |
| if strings.HasPrefix(name, "noalg.struct {") { |
| return nil |
| } |
| if strings.HasPrefix(name, "chan ") { |
| return nil |
| } |
| if name[0] == '[' || name[0] == '*' { |
| return nil |
| } |
| if def == nil { |
| Errorf("dwarf: bad def in dotypedef") |
| } |
|
|
| |
| |
| |
| tds := d.ldr.CreateExtSym("", 0) |
| tdsu := d.ldr.MakeSymbolUpdater(tds) |
| tdsu.SetType(sym.SDWARFTYPE) |
| def.Sym = dwSym(tds) |
| d.ldr.SetAttrNotInSymbolTable(tds, true) |
| d.ldr.SetAttrReachable(tds, true) |
|
|
| |
| |
| |
| |
| die := d.newdie(parent, dwarf.DW_ABRV_TYPEDECL, name) |
|
|
| d.newrefattr(die, dwarf.DW_AT_type, tds) |
|
|
| return die |
| } |
|
|
| |
| func (d *dwctxt) defgotype(gotype loader.Sym) loader.Sym { |
| if gotype == 0 { |
| return d.mustFind("<unspecified>") |
| } |
|
|
| |
| if ds, ok := d.tdmap[gotype]; ok { |
| return ds |
| } |
|
|
| sn := d.ldr.SymName(gotype) |
| if !strings.HasPrefix(sn, "type:") { |
| d.linkctxt.Errorf(gotype, "dwarf: type name doesn't start with \"type:\"") |
| return d.mustFind("<unspecified>") |
| } |
| name := sn[len("type:"):] |
|
|
| sdie := d.find(name) |
| if sdie != 0 { |
| return sdie |
| } |
|
|
| gtdwSym := d.newtype(gotype) |
| d.tdmap[gotype] = loader.Sym(gtdwSym.Sym.(dwSym)) |
| return loader.Sym(gtdwSym.Sym.(dwSym)) |
| } |
|
|
| func (d *dwctxt) newtype(gotype loader.Sym) *dwarf.DWDie { |
| sn := d.ldr.SymName(gotype) |
| name := sn[len("type:"):] |
| tdata := d.ldr.Data(gotype) |
| if len(tdata) == 0 { |
| d.linkctxt.Errorf(gotype, "missing type") |
| } |
| kind := decodetypeKind(d.arch, tdata) |
| bytesize := decodetypeSize(d.arch, tdata) |
|
|
| var die, typedefdie *dwarf.DWDie |
| switch kind { |
| case abi.Bool: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_BASETYPE, name) |
| newattr(die, dwarf.DW_AT_encoding, dwarf.DW_CLS_CONSTANT, dwarf.DW_ATE_boolean, 0) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
|
|
| case abi.Int, |
| abi.Int8, |
| abi.Int16, |
| abi.Int32, |
| abi.Int64: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_BASETYPE, name) |
| newattr(die, dwarf.DW_AT_encoding, dwarf.DW_CLS_CONSTANT, dwarf.DW_ATE_signed, 0) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
|
|
| case abi.Uint, |
| abi.Uint8, |
| abi.Uint16, |
| abi.Uint32, |
| abi.Uint64, |
| abi.Uintptr: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_BASETYPE, name) |
| newattr(die, dwarf.DW_AT_encoding, dwarf.DW_CLS_CONSTANT, dwarf.DW_ATE_unsigned, 0) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
|
|
| case abi.Float32, |
| abi.Float64: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_BASETYPE, name) |
| newattr(die, dwarf.DW_AT_encoding, dwarf.DW_CLS_CONSTANT, dwarf.DW_ATE_float, 0) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
|
|
| case abi.Complex64, |
| abi.Complex128: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_BASETYPE, name) |
| newattr(die, dwarf.DW_AT_encoding, dwarf.DW_CLS_CONSTANT, dwarf.DW_ATE_complex_float, 0) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
|
|
| case abi.Array: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_ARRAYTYPE, name) |
| typedefdie = d.dotypedef(&dwtypes, name, die) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
| s := decodetypeArrayElem(d.linkctxt, d.arch, gotype) |
| d.newrefattr(die, dwarf.DW_AT_type, d.defgotype(s)) |
| fld := d.newdie(die, dwarf.DW_ABRV_ARRAYRANGE, "range") |
|
|
| |
| newattr(fld, dwarf.DW_AT_count, dwarf.DW_CLS_CONSTANT, decodetypeArrayLen(d.ldr, d.arch, gotype), 0) |
|
|
| d.newrefattr(fld, dwarf.DW_AT_type, d.uintptrInfoSym) |
|
|
| case abi.Chan: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_CHANTYPE, name) |
| s := decodetypeChanElem(d.ldr, d.arch, gotype) |
| d.newrefattr(die, dwarf.DW_AT_go_elem, d.defgotype(s)) |
| |
| |
| d.newrefattr(die, dwarf.DW_AT_type, s) |
|
|
| case abi.Func: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_FUNCTYPE, name) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
| typedefdie = d.dotypedef(&dwtypes, name, die) |
| data := d.ldr.Data(gotype) |
| |
| relocs := d.ldr.Relocs(gotype) |
| nfields := decodetypeFuncInCount(d.arch, data) |
| for i := 0; i < nfields; i++ { |
| s := decodetypeFuncInType(d.ldr, d.arch, gotype, &relocs, i) |
| sn := d.ldr.SymName(s) |
| fld := d.newdie(die, dwarf.DW_ABRV_FUNCTYPEPARAM, sn[5:]) |
| d.newrefattr(fld, dwarf.DW_AT_type, d.defgotype(s)) |
| } |
|
|
| if decodetypeFuncDotdotdot(d.arch, data) { |
| d.newdie(die, dwarf.DW_ABRV_DOTDOTDOT, "...") |
| } |
| nfields = decodetypeFuncOutCount(d.arch, data) |
| for i := 0; i < nfields; i++ { |
| s := decodetypeFuncOutType(d.ldr, d.arch, gotype, &relocs, i) |
| sn := d.ldr.SymName(s) |
| fld := d.newdie(die, dwarf.DW_ABRV_FUNCTYPEOUTPARAM, sn[5:]) |
| newattr(fld, dwarf.DW_AT_variable_parameter, dwarf.DW_CLS_FLAG, 1, 0) |
| d.newrefattr(fld, dwarf.DW_AT_type, d.defgotype(s)) |
| } |
|
|
| case abi.Interface: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_IFACETYPE, name) |
| typedefdie = d.dotypedef(&dwtypes, name, die) |
| data := d.ldr.Data(gotype) |
| nfields := int(decodetypeIfaceMethodCount(d.arch, data)) |
| var s loader.Sym |
| if nfields == 0 { |
| s = d.typeRuntimeEface |
| } else { |
| s = d.typeRuntimeIface |
| } |
| d.newrefattr(die, dwarf.DW_AT_type, d.defgotype(s)) |
|
|
| case abi.Map: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_MAPTYPE, name) |
| s := decodetypeMapKey(d.ldr, d.arch, gotype) |
| d.newrefattr(die, dwarf.DW_AT_go_key, d.defgotype(s)) |
| s = decodetypeMapValue(d.ldr, d.arch, gotype) |
| d.newrefattr(die, dwarf.DW_AT_go_elem, d.defgotype(s)) |
| |
| |
| d.newrefattr(die, dwarf.DW_AT_type, gotype) |
|
|
| case abi.Pointer: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_PTRTYPE, name) |
| typedefdie = d.dotypedef(&dwtypes, name, die) |
| s := decodetypePtrElem(d.ldr, d.arch, gotype) |
| d.newrefattr(die, dwarf.DW_AT_type, d.defgotype(s)) |
|
|
| case abi.Slice: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_SLICETYPE, name) |
| typedefdie = d.dotypedef(&dwtypes, name, die) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
| s := decodetypeArrayElem(d.linkctxt, d.arch, gotype) |
| elem := d.defgotype(s) |
| d.newrefattr(die, dwarf.DW_AT_go_elem, elem) |
|
|
| case abi.String: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_STRINGTYPE, name) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
|
|
| case abi.Struct: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_STRUCTTYPE, name) |
| typedefdie = d.dotypedef(&dwtypes, name, die) |
| newattr(die, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, bytesize, 0) |
| nfields := decodetypeStructFieldCount(d.ldr, d.arch, gotype) |
| for i := 0; i < nfields; i++ { |
| f := decodetypeStructFieldName(d.ldr, d.arch, gotype, i) |
| s := decodetypeStructFieldType(d.linkctxt, d.arch, gotype, i) |
| if f == "" { |
| sn := d.ldr.SymName(s) |
| f = sn[5:] |
| } |
| fld := d.newdie(die, dwarf.DW_ABRV_STRUCTFIELD, f) |
| d.newrefattr(fld, dwarf.DW_AT_type, d.defgotype(s)) |
| offset := decodetypeStructFieldOffset(d.ldr, d.arch, gotype, i) |
| newmemberoffsetattr(fld, int32(offset)) |
| if decodetypeStructFieldEmbedded(d.ldr, d.arch, gotype, i) { |
| newattr(fld, dwarf.DW_AT_go_embedded_field, dwarf.DW_CLS_FLAG, 1, 0) |
| } |
| } |
|
|
| case abi.UnsafePointer: |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_BARE_PTRTYPE, name) |
|
|
| default: |
| d.linkctxt.Errorf(gotype, "dwarf: definition of unknown kind %d", kind) |
| die = d.newdie(&dwtypes, dwarf.DW_ABRV_TYPEDECL, name) |
| d.newrefattr(die, dwarf.DW_AT_type, d.mustFind("<unspecified>")) |
| } |
|
|
| newattr(die, dwarf.DW_AT_go_kind, dwarf.DW_CLS_CONSTANT, int64(kind), 0) |
|
|
| if d.ldr.AttrReachable(gotype) { |
| newattr(die, dwarf.DW_AT_go_runtime_type, dwarf.DW_CLS_GO_TYPEREF, 0, dwSym(gotype)) |
| } |
|
|
| |
| if _, ok := d.rtmap[gotype]; ok { |
| log.Fatalf("internal error: rtmap entry already installed\n") |
| } |
|
|
| ds := loader.Sym(die.Sym.(dwSym)) |
| if typedefdie != nil { |
| ds = loader.Sym(typedefdie.Sym.(dwSym)) |
| } |
| d.rtmap[ds] = gotype |
|
|
| if _, ok := prototypedies[sn]; ok { |
| prototypedies[sn] = die |
| } |
|
|
| if typedefdie != nil { |
| return typedefdie |
| } |
| return die |
| } |
|
|
| func (d *dwctxt) nameFromDIESym(dwtypeDIESym loader.Sym) string { |
| sn := d.ldr.SymName(dwtypeDIESym) |
| return sn[len(dwarf.InfoPrefix):] |
| } |
|
|
| func (d *dwctxt) defptrto(dwtype loader.Sym) loader.Sym { |
|
|
| |
| |
| |
|
|
| ptrname := "*" + d.nameFromDIESym(dwtype) |
| if die := d.find(ptrname); die != 0 { |
| return die |
| } |
|
|
| pdie := d.newdie(&dwtypes, dwarf.DW_ABRV_PTRTYPE, ptrname) |
| d.newrefattr(pdie, dwarf.DW_AT_type, dwtype) |
|
|
| |
| |
| |
| gts := d.ldr.Lookup("type:"+ptrname, 0) |
| if gts != 0 && d.ldr.AttrReachable(gts) { |
| newattr(pdie, dwarf.DW_AT_go_kind, dwarf.DW_CLS_CONSTANT, int64(abi.Pointer), 0) |
| newattr(pdie, dwarf.DW_AT_go_runtime_type, dwarf.DW_CLS_GO_TYPEREF, 0, dwSym(gts)) |
| } |
|
|
| if gts != 0 { |
| ds := loader.Sym(pdie.Sym.(dwSym)) |
| d.rtmap[ds] = gts |
| d.tdmap[gts] = ds |
| } |
|
|
| return d.dtolsym(pdie.Sym) |
| } |
|
|
| |
| |
| |
| func (d *dwctxt) copychildrenexcept(ctxt *Link, dst *dwarf.DWDie, src *dwarf.DWDie, except *dwarf.DWDie) { |
| for src = src.Child; src != nil; src = src.Link { |
| if src == except { |
| continue |
| } |
| c := d.newdie(dst, src.Abbrev, getattr(src, dwarf.DW_AT_name).Data.(string)) |
| for a := src.Attr; a != nil; a = a.Link { |
| newattr(c, a.Atr, int(a.Cls), a.Value, a.Data) |
| } |
| d.copychildrenexcept(ctxt, c, src, nil) |
| } |
|
|
| reverselist(&dst.Child) |
| } |
|
|
| func (d *dwctxt) copychildren(ctxt *Link, dst *dwarf.DWDie, src *dwarf.DWDie) { |
| d.copychildrenexcept(ctxt, dst, src, nil) |
| } |
|
|
| |
| |
| func (d *dwctxt) substitutetype(structdie *dwarf.DWDie, field string, dwtype loader.Sym) { |
| child := findchild(structdie, field) |
| if child == nil { |
| Exitf("dwarf substitutetype: %s does not have member %s", |
| getattr(structdie, dwarf.DW_AT_name).Data, field) |
| return |
| } |
|
|
| a := getattr(child, dwarf.DW_AT_type) |
| if a != nil { |
| a.Data = dwSym(dwtype) |
| } else { |
| d.newrefattr(child, dwarf.DW_AT_type, dwtype) |
| } |
| } |
|
|
| func (d *dwctxt) findprotodie(ctxt *Link, name string) *dwarf.DWDie { |
| die, ok := prototypedies[name] |
| if ok && die == nil { |
| d.defgotype(d.lookupOrDiag(name)) |
| die = prototypedies[name] |
| } |
| if die == nil { |
| log.Fatalf("internal error: DIE generation failed for %s\n", name) |
| } |
| return die |
| } |
|
|
| func (d *dwctxt) synthesizestringtypes(ctxt *Link, die *dwarf.DWDie) { |
| prototype := walktypedef(d.findprotodie(ctxt, "type:runtime.stringStructDWARF")) |
| if prototype == nil { |
| return |
| } |
|
|
| for ; die != nil; die = die.Link { |
| if die.Abbrev != dwarf.DW_ABRV_STRINGTYPE { |
| continue |
| } |
| d.copychildren(ctxt, die, prototype) |
| } |
| } |
|
|
| func (d *dwctxt) synthesizeslicetypes(ctxt *Link, die *dwarf.DWDie) { |
| prototype := walktypedef(d.findprotodie(ctxt, "type:runtime.slice")) |
| if prototype == nil { |
| return |
| } |
|
|
| for ; die != nil; die = die.Link { |
| if die.Abbrev != dwarf.DW_ABRV_SLICETYPE { |
| continue |
| } |
| d.copychildren(ctxt, die, prototype) |
| elem := loader.Sym(getattr(die, dwarf.DW_AT_go_elem).Data.(dwSym)) |
| d.substitutetype(die, "array", d.defptrto(elem)) |
| } |
| } |
|
|
| func mkinternaltypename(base string, arg1 string, arg2 string) string { |
| if arg2 == "" { |
| return fmt.Sprintf("%s<%s>", base, arg1) |
| } |
| return fmt.Sprintf("%s<%s,%s>", base, arg1, arg2) |
| } |
|
|
| func (d *dwctxt) mkinternaltype(ctxt *Link, abbrev int, typename, keyname, valname string, f func(*dwarf.DWDie)) loader.Sym { |
| name := mkinternaltypename(typename, keyname, valname) |
| symname := dwarf.InfoPrefix + name |
| s := d.ldr.Lookup(symname, 0) |
| if s != 0 && d.ldr.SymType(s) == sym.SDWARFTYPE { |
| return s |
| } |
| die := d.newdie(&dwtypes, abbrev, name) |
| f(die) |
| return d.dtolsym(die.Sym) |
| } |
|
|
| func (d *dwctxt) synthesizemaptypes(ctxt *Link, die *dwarf.DWDie) { |
| mapType := walktypedef(d.findprotodie(ctxt, "type:internal/runtime/maps.Map")) |
| tableType := walktypedef(d.findprotodie(ctxt, "type:internal/runtime/maps.table")) |
| groupsReferenceType := walktypedef(d.findprotodie(ctxt, "type:internal/runtime/maps.groupsReference")) |
|
|
| for ; die != nil; die = die.Link { |
| if die.Abbrev != dwarf.DW_ABRV_MAPTYPE { |
| continue |
| } |
| gotype := loader.Sym(getattr(die, dwarf.DW_AT_type).Data.(dwSym)) |
|
|
| keyType := decodetypeMapKey(d.ldr, d.arch, gotype) |
| valType := decodetypeMapValue(d.ldr, d.arch, gotype) |
| groupType := decodetypeMapGroup(d.ldr, d.arch, gotype) |
|
|
| keyType = d.walksymtypedef(d.defgotype(keyType)) |
| valType = d.walksymtypedef(d.defgotype(valType)) |
| groupType = d.walksymtypedef(d.defgotype(groupType)) |
|
|
| keyName := d.nameFromDIESym(keyType) |
| valName := d.nameFromDIESym(valType) |
|
|
| |
| dwGroupsReference := d.mkinternaltype(ctxt, dwarf.DW_ABRV_STRUCTTYPE, "groupReference", keyName, valName, func(dwh *dwarf.DWDie) { |
| d.copychildren(ctxt, dwh, groupsReferenceType) |
| |
| |
| |
| |
| |
| d.substitutetype(dwh, "data", d.defptrto(groupType)) |
| newattr(dwh, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, getattr(groupsReferenceType, dwarf.DW_AT_byte_size).Value, nil) |
| newattr(dwh, dwarf.DW_AT_go_kind, dwarf.DW_CLS_CONSTANT, int64(abi.Struct), 0) |
| }) |
|
|
| |
| dwTable := d.mkinternaltype(ctxt, dwarf.DW_ABRV_STRUCTTYPE, "table", keyName, valName, func(dwh *dwarf.DWDie) { |
| d.copychildren(ctxt, dwh, tableType) |
| d.substitutetype(dwh, "groups", dwGroupsReference) |
| newattr(dwh, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, getattr(tableType, dwarf.DW_AT_byte_size).Value, nil) |
| newattr(dwh, dwarf.DW_AT_go_kind, dwarf.DW_CLS_CONSTANT, int64(abi.Struct), 0) |
| }) |
|
|
| |
| dwMap := d.mkinternaltype(ctxt, dwarf.DW_ABRV_STRUCTTYPE, "map", keyName, valName, func(dwh *dwarf.DWDie) { |
| d.copychildren(ctxt, dwh, mapType) |
| |
| |
| |
| |
| |
| |
| d.substitutetype(dwh, "dirPtr", d.defptrto(d.defptrto(dwTable))) |
| newattr(dwh, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, getattr(mapType, dwarf.DW_AT_byte_size).Value, nil) |
| newattr(dwh, dwarf.DW_AT_go_kind, dwarf.DW_CLS_CONSTANT, int64(abi.Struct), 0) |
| }) |
|
|
| |
| d.newrefattr(die, dwarf.DW_AT_type, d.defptrto(dwMap)) |
| } |
| } |
|
|
| func (d *dwctxt) synthesizechantypes(ctxt *Link, die *dwarf.DWDie) { |
| sudog := walktypedef(d.findprotodie(ctxt, "type:runtime.sudog")) |
| waitq := walktypedef(d.findprotodie(ctxt, "type:runtime.waitq")) |
| hchan := walktypedef(d.findprotodie(ctxt, "type:runtime.hchan")) |
| if sudog == nil || waitq == nil || hchan == nil { |
| return |
| } |
|
|
| sudogsize := int(getattr(sudog, dwarf.DW_AT_byte_size).Value) |
|
|
| for ; die != nil; die = die.Link { |
| if die.Abbrev != dwarf.DW_ABRV_CHANTYPE { |
| continue |
| } |
| elemgotype := loader.Sym(getattr(die, dwarf.DW_AT_type).Data.(dwSym)) |
| tname := d.ldr.SymName(elemgotype) |
| elemname := tname[5:] |
| elemtype := d.walksymtypedef(d.defgotype(d.lookupOrDiag(tname))) |
|
|
| |
| dwss := d.mkinternaltype(ctxt, dwarf.DW_ABRV_STRUCTTYPE, "sudog", elemname, "", func(dws *dwarf.DWDie) { |
| d.copychildren(ctxt, dws, sudog) |
| d.substitutetype(dws, "elem", d.defptrto(elemtype)) |
| newattr(dws, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, int64(sudogsize), nil) |
| }) |
|
|
| |
| dwws := d.mkinternaltype(ctxt, dwarf.DW_ABRV_STRUCTTYPE, "waitq", elemname, "", func(dww *dwarf.DWDie) { |
|
|
| d.copychildren(ctxt, dww, waitq) |
| d.substitutetype(dww, "first", d.defptrto(dwss)) |
| d.substitutetype(dww, "last", d.defptrto(dwss)) |
| newattr(dww, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, getattr(waitq, dwarf.DW_AT_byte_size).Value, nil) |
| }) |
|
|
| |
| dwhs := d.mkinternaltype(ctxt, dwarf.DW_ABRV_STRUCTTYPE, "hchan", elemname, "", func(dwh *dwarf.DWDie) { |
| d.copychildren(ctxt, dwh, hchan) |
| d.substitutetype(dwh, "recvq", dwws) |
| d.substitutetype(dwh, "sendq", dwws) |
| newattr(dwh, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, getattr(hchan, dwarf.DW_AT_byte_size).Value, nil) |
| }) |
|
|
| d.newrefattr(die, dwarf.DW_AT_type, d.defptrto(dwhs)) |
| } |
| } |
|
|
| |
| |
| func (d *dwctxt) createUnitLength(su *loader.SymbolBuilder, v uint64) { |
| if isDwarf64(d.linkctxt) { |
| su.AddUint32(d.arch, 0xFFFFFFFF) |
| } |
| d.addDwarfAddrField(su, v) |
| } |
|
|
| |
| func (d *dwctxt) addDwarfAddrField(sb *loader.SymbolBuilder, v uint64) { |
| if isDwarf64(d.linkctxt) { |
| sb.AddUint(d.arch, v) |
| } else { |
| sb.AddUint32(d.arch, uint32(v)) |
| } |
| } |
|
|
| |
| func (d *dwctxt) addDwarfAddrRef(sb *loader.SymbolBuilder, t loader.Sym) { |
| if isDwarf64(d.linkctxt) { |
| d.adddwarfref(sb, t, 8) |
| } else { |
| d.adddwarfref(sb, t, 4) |
| } |
| } |
|
|
| |
| func (d *dwctxt) calcCompUnitRanges() { |
| var prevUnit *sym.CompilationUnit |
| for _, s := range d.linkctxt.Textp { |
| sym := s |
|
|
| fi := d.ldr.FuncInfo(sym) |
| if !fi.Valid() { |
| continue |
| } |
|
|
| |
| |
| unit := d.ldr.SymUnit(sym) |
| if unit == nil { |
| continue |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| sval := d.ldr.SymValue(sym) |
| u0val := d.ldr.SymValue(unit.Textp[0]) |
| if prevUnit != unit { |
| unit.PCs = append(unit.PCs, dwarf.Range{Start: sval - u0val}) |
| prevUnit = unit |
| } |
| unit.PCs[len(unit.PCs)-1].End = sval - u0val + int64(len(d.ldr.Data(sym))) |
| } |
| } |
|
|
| func movetomodule(ctxt *Link, parent *dwarf.DWDie) { |
| die := ctxt.runtimeCU.DWInfo.Child |
| if die == nil { |
| ctxt.runtimeCU.DWInfo.Child = parent.Child |
| return |
| } |
| for die.Link != nil { |
| die = die.Link |
| } |
| die.Link = parent.Child |
| } |
|
|
| |
| |
| |
| |
| const ( |
| LINE_BASE = -4 |
| LINE_RANGE = 10 |
| PC_RANGE = (255 - OPCODE_BASE) / LINE_RANGE |
| OPCODE_BASE = 11 |
| ) |
|
|
| |
| |
| |
|
|
| func getCompilationDir() string { |
| |
| |
| |
| |
| |
| |
| return "." |
| } |
|
|
| func (d *dwctxt) importInfoSymbol(dsym loader.Sym) { |
| d.ldr.SetAttrReachable(dsym, true) |
| d.ldr.SetAttrNotInSymbolTable(dsym, true) |
| dst := d.ldr.SymType(dsym) |
| if dst != sym.SDWARFCONST && dst != sym.SDWARFABSFCN { |
| log.Fatalf("error: DWARF info sym %d/%s with incorrect type %s", dsym, d.ldr.SymName(dsym), d.ldr.SymType(dsym).String()) |
| } |
| relocs := d.ldr.Relocs(dsym) |
| for i := 0; i < relocs.Count(); i++ { |
| r := relocs.At(i) |
| if r.Type() != objabi.R_DWARFSECREF { |
| continue |
| } |
| rsym := r.Sym() |
| |
| |
| if _, ok := d.rtmap[rsym]; ok { |
| |
| continue |
| } |
| |
| |
| sn := d.ldr.SymName(rsym) |
| tn := sn[len(dwarf.InfoPrefix):] |
| ts := d.ldr.Lookup("type:"+tn, 0) |
| d.defgotype(ts) |
| } |
| } |
|
|
| func expandFile(fname string) string { |
| fname = strings.TrimPrefix(fname, src.FileSymPrefix) |
| return expandGoroot(fname) |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| func (d *dwctxt) writeDirFileTables(unit *sym.CompilationUnit, lsu *loader.SymbolBuilder) { |
| type fileDir struct { |
| base string |
| dir int |
| } |
| dirNums := make(map[string]int) |
| dirs := []string{"."} |
| dirNums["."] = 0 |
| files := []fileDir{} |
| if buildcfg.Experiment.Dwarf5 { |
| |
| |
| |
| |
| |
| |
| |
| files = append(files, fileDir{base: "?", dir: 0}) |
| } |
|
|
| |
| |
| for i, name := range unit.FileTable { |
| name := expandFile(name) |
| if len(name) == 0 { |
| |
| |
| name = fmt.Sprintf("<missing>_%d", i) |
| } |
| |
| |
| |
| file := path.Base(name) |
| dir := path.Dir(name) |
| dirIdx, ok := dirNums[dir] |
| if !ok && dir != "." { |
| dirIdx = len(dirNums) |
| dirNums[dir] = dirIdx |
| dirs = append(dirs, dir) |
| } |
| files = append(files, fileDir{base: file, dir: dirIdx}) |
|
|
| |
| |
| |
| if i := strings.Index(name, "runtime/proc.go"); i >= 0 && unit.Lib.Pkg == "runtime" { |
| d.dwmu.Lock() |
| if gdbscript == "" { |
| k := strings.Index(name, "runtime/proc.go") |
| gdbscript = name[:k] + "runtime/runtime-gdb.py" |
| } |
| d.dwmu.Unlock() |
| } |
| } |
|
|
| lsDwsym := dwSym(lsu.Sym()) |
| if buildcfg.Experiment.Dwarf5 { |
| |
| |
| |
| |
|
|
| |
| lsu.AddUint8(1) |
| |
| dwarf.Uleb128put(d, lsDwsym, dwarf.DW_LNCT_path) |
| dwarf.Uleb128put(d, lsDwsym, dwarf.DW_FORM_string) |
| |
| dwarf.Uleb128put(d, lsDwsym, int64(len(dirs))) |
| for k := 0; k < len(dirs); k++ { |
| d.AddString(lsDwsym, dirs[k]) |
| } |
|
|
| |
| |
| |
| |
| lsu.AddUint8(2) |
| |
| dwarf.Uleb128put(d, lsDwsym, dwarf.DW_LNCT_path) |
| dwarf.Uleb128put(d, lsDwsym, dwarf.DW_FORM_string) |
| dwarf.Uleb128put(d, lsDwsym, dwarf.DW_LNCT_directory_index) |
| dwarf.Uleb128put(d, lsDwsym, dwarf.DW_FORM_udata) |
| |
| dwarf.Uleb128put(d, lsDwsym, int64(len(files))) |
| for k := 0; k < len(files); k++ { |
| d.AddString(lsDwsym, files[k].base) |
| dwarf.Uleb128put(d, lsDwsym, int64(files[k].dir)) |
| } |
| } else { |
| |
|
|
| |
| |
| for k := 1; k < len(dirs); k++ { |
| d.AddString(lsDwsym, dirs[k]) |
| } |
| lsu.AddUint8(0) |
|
|
| |
| |
| |
| for k := 0; k < len(files); k++ { |
| d.AddString(lsDwsym, files[k].base) |
| dwarf.Uleb128put(d, lsDwsym, int64(files[k].dir)) |
| lsu.AddUint8(0) |
| lsu.AddUint8(0) |
| } |
| lsu.AddUint8(0) |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| func (d *dwctxt) writelines(unit *sym.CompilationUnit, lineProlog loader.Sym) []loader.Sym { |
| is_stmt := uint8(1) |
|
|
| unitstart := int64(-1) |
| headerstart := int64(-1) |
| headerend := int64(-1) |
|
|
| syms := make([]loader.Sym, 0, len(unit.Textp)+2) |
| syms = append(syms, lineProlog) |
| lsu := d.ldr.MakeSymbolUpdater(lineProlog) |
| lsDwsym := dwSym(lineProlog) |
| newattr(unit.DWInfo, dwarf.DW_AT_stmt_list, dwarf.DW_CLS_PTR, 0, lsDwsym) |
|
|
| |
| |
| unitLengthOffset := lsu.Size() |
| d.createUnitLength(lsu, 0) |
| unitstart = lsu.Size() |
| if buildcfg.Experiment.Dwarf5 { |
| lsu.AddUint16(d.arch, 5) |
| |
| |
| lsu.AddUint8(uint8(d.arch.PtrSize)) |
| lsu.AddUint8(0) |
| } else { |
| lsu.AddUint16(d.arch, 2) |
| } |
| headerLengthOffset := lsu.Size() |
| d.addDwarfAddrField(lsu, 0) |
| headerstart = lsu.Size() |
|
|
| lsu.AddUint8(1) |
| if buildcfg.Experiment.Dwarf5 { |
| lsu.AddUint8(1) |
| } |
| lsu.AddUint8(is_stmt) |
| lsu.AddUint8(LINE_BASE & 0xFF) |
| lsu.AddUint8(LINE_RANGE) |
| lsu.AddUint8(OPCODE_BASE) |
| lsu.AddUint8(0) |
| lsu.AddUint8(1) |
| lsu.AddUint8(1) |
| lsu.AddUint8(1) |
| lsu.AddUint8(1) |
| lsu.AddUint8(0) |
| lsu.AddUint8(0) |
| lsu.AddUint8(0) |
| lsu.AddUint8(1) |
| lsu.AddUint8(0) |
|
|
| |
| d.writeDirFileTables(unit, lsu) |
|
|
| |
| headerend = lsu.Size() |
| unitlen := lsu.Size() - unitstart |
|
|
| |
| for _, s := range unit.Textp { |
| fnSym := s |
| _, _, _, lines := d.ldr.GetFuncDwarfAuxSyms(fnSym) |
|
|
| |
| if lines != 0 { |
| syms = append(syms, lines) |
| unitlen += int64(len(d.ldr.Data(lines))) |
| } |
| } |
|
|
| if d.linkctxt.HeadType == objabi.Haix { |
| addDwsectCUSize(".debug_line", unit.Lib.Pkg, uint64(unitlen)) |
| } |
|
|
| if isDwarf64(d.linkctxt) { |
| lsu.SetUint(d.arch, unitLengthOffset+4, uint64(unitlen)) |
| lsu.SetUint(d.arch, headerLengthOffset, uint64(headerend-headerstart)) |
| } else { |
| lsu.SetUint32(d.arch, unitLengthOffset, uint32(unitlen)) |
| lsu.SetUint32(d.arch, headerLengthOffset, uint32(headerend-headerstart)) |
| } |
|
|
| return syms |
| } |
|
|
| |
| |
| |
| func (d *dwctxt) writepcranges(unit *sym.CompilationUnit, base loader.Sym, pcs []dwarf.Range, rangeProlog loader.Sym, debugaddrsym loader.Sym) []loader.Sym { |
|
|
| syms := make([]loader.Sym, 0, len(unit.RangeSyms)+1) |
| syms = append(syms, rangeProlog) |
| rsu := d.ldr.MakeSymbolUpdater(rangeProlog) |
| rDwSym := dwSym(rangeProlog) |
|
|
| |
| newattr(unit.DWInfo, dwarf.DW_AT_ranges, dwarf.DW_CLS_PTR, rsu.Size(), rDwSym) |
| newattr(unit.DWInfo, dwarf.DW_AT_low_pc, dwarf.DW_CLS_ADDRESS, 0, dwSym(base)) |
| if buildcfg.Experiment.Dwarf5 && debugaddrsym != 0 { |
| debugaddr := d.ldr.MakeSymbolUpdater(debugaddrsym) |
| |
| |
| |
| dwarf.PutRngListRanges(d, rDwSym, dwSym(base), pcs) |
| drelocs := d.ldr.Relocs(rangeProlog) |
| for ri := 0; ri < drelocs.Count(); ri++ { |
| r := drelocs.At(ri) |
| if !r.Type().IsDwTxtAddr() { |
| continue |
| } |
| cusym := d.dtolsym(unit.DWInfo.Sym) |
| d.assignDebugAddrSlot(unit, cusym, r, debugaddr) |
| } |
| } else { |
| dwarf.PutBasedRanges(d, rDwSym, pcs) |
| } |
|
|
| |
| rsize := uint64(rsu.Size()) |
| for _, ls := range unit.RangeSyms { |
| s := ls |
| syms = append(syms, s) |
| rsize += uint64(d.ldr.SymSize(s)) |
| } |
|
|
| if d.linkctxt.HeadType == objabi.Haix { |
| addDwsectCUSize(".debug_ranges", unit.Lib.Pkg, rsize) |
| } |
|
|
| return syms |
| } |
|
|
| |
| |
| |
| const ( |
| dataAlignmentFactor = -4 |
| ) |
|
|
| |
| func appendPCDeltaCFA(arch *sys.Arch, b []byte, deltapc, cfa int64) []byte { |
| b = append(b, dwarf.DW_CFA_def_cfa_offset_sf) |
| b = dwarf.AppendSleb128(b, cfa/dataAlignmentFactor) |
|
|
| switch { |
| case deltapc < 0x40: |
| b = append(b, uint8(dwarf.DW_CFA_advance_loc+deltapc)) |
| case deltapc < 0x100: |
| b = append(b, dwarf.DW_CFA_advance_loc1) |
| b = append(b, uint8(deltapc)) |
| case deltapc < 0x10000: |
| b = append(b, dwarf.DW_CFA_advance_loc2, 0, 0) |
| arch.ByteOrder.PutUint16(b[len(b)-2:], uint16(deltapc)) |
| default: |
| b = append(b, dwarf.DW_CFA_advance_loc4, 0, 0, 0, 0) |
| arch.ByteOrder.PutUint32(b[len(b)-4:], uint32(deltapc)) |
| } |
| return b |
| } |
|
|
| func (d *dwctxt) writeframes(fs loader.Sym) dwarfSecInfo { |
| fsd := dwSym(fs) |
| fsu := d.ldr.MakeSymbolUpdater(fs) |
| fsu.SetType(sym.SDWARFSECT) |
| isdw64 := isDwarf64(d.linkctxt) |
| haslr := d.linkctxt.Arch.HasLR |
|
|
| |
| lengthFieldSize := int64(4) |
| if isdw64 { |
| lengthFieldSize += 8 |
| } |
|
|
| |
| cieReserve := uint32(16) |
| if haslr { |
| cieReserve = 32 |
| } |
| if isdw64 { |
| cieReserve += 4 |
| } |
| d.createUnitLength(fsu, uint64(cieReserve)) |
| d.addDwarfAddrField(fsu, ^uint64(0)) |
| fsu.AddUint8(3) |
| fsu.AddUint8(0) |
| dwarf.Uleb128put(d, fsd, 1) |
| dwarf.Sleb128put(d, fsd, dataAlignmentFactor) |
| dwarf.Uleb128put(d, fsd, int64(thearch.Dwarfreglr)) |
|
|
| fsu.AddUint8(dwarf.DW_CFA_def_cfa) |
| dwarf.Uleb128put(d, fsd, int64(thearch.Dwarfregsp)) |
| if haslr { |
| dwarf.Uleb128put(d, fsd, int64(0)) |
|
|
| fsu.AddUint8(dwarf.DW_CFA_same_value) |
| dwarf.Uleb128put(d, fsd, int64(thearch.Dwarfreglr)) |
|
|
| fsu.AddUint8(dwarf.DW_CFA_val_offset) |
| dwarf.Uleb128put(d, fsd, int64(thearch.Dwarfregsp)) |
| dwarf.Uleb128put(d, fsd, int64(0)) |
| } else { |
| dwarf.Uleb128put(d, fsd, int64(d.arch.PtrSize)) |
|
|
| fsu.AddUint8(dwarf.DW_CFA_offset_extended) |
| dwarf.Uleb128put(d, fsd, int64(thearch.Dwarfreglr)) |
| dwarf.Uleb128put(d, fsd, int64(-d.arch.PtrSize)/dataAlignmentFactor) |
| } |
|
|
| pad := int64(cieReserve) + lengthFieldSize - int64(len(d.ldr.Data(fs))) |
|
|
| if pad < 0 { |
| Exitf("dwarf: cieReserve too small by %d bytes.", -pad) |
| } |
|
|
| internalExec := d.linkctxt.BuildMode == BuildModeExe && d.linkctxt.IsInternal() |
| addAddrPlus := loader.GenAddAddrPlusFunc(internalExec) |
|
|
| fsu.AddBytes(zeros[:pad]) |
|
|
| var deltaBuf []byte |
| pcsp := obj.NewPCIter(uint32(d.arch.MinLC)) |
| for _, s := range d.linkctxt.Textp { |
| fn := s |
| fi := d.ldr.FuncInfo(fn) |
| if !fi.Valid() { |
| continue |
| } |
| fpcsp := d.ldr.Pcsp(s) |
|
|
| |
| |
| deltaBuf = deltaBuf[:0] |
| if haslr && fi.TopFrame() { |
| |
| |
| |
| deltaBuf = append(deltaBuf, dwarf.DW_CFA_undefined) |
| deltaBuf = dwarf.AppendUleb128(deltaBuf, uint64(thearch.Dwarfreglr)) |
| } |
|
|
| for pcsp.Init(d.linkctxt.loader.Data(fpcsp)); !pcsp.Done; pcsp.Next() { |
| nextpc := pcsp.NextPC |
|
|
| |
| |
| if int64(nextpc) == int64(len(d.ldr.Data(fn))) { |
| nextpc-- |
| if nextpc < pcsp.PC { |
| continue |
| } |
| } |
|
|
| spdelta := int64(pcsp.Value) |
| if !haslr { |
| |
| spdelta += int64(d.arch.PtrSize) |
| } |
|
|
| if haslr && !fi.TopFrame() { |
| |
| |
| |
| if pcsp.Value > 0 { |
| |
| |
| deltaBuf = append(deltaBuf, dwarf.DW_CFA_offset_extended_sf) |
| deltaBuf = dwarf.AppendUleb128(deltaBuf, uint64(thearch.Dwarfreglr)) |
| deltaBuf = dwarf.AppendSleb128(deltaBuf, -spdelta/dataAlignmentFactor) |
| } else { |
| |
| |
| deltaBuf = append(deltaBuf, dwarf.DW_CFA_same_value) |
| deltaBuf = dwarf.AppendUleb128(deltaBuf, uint64(thearch.Dwarfreglr)) |
| } |
| } |
|
|
| deltaBuf = appendPCDeltaCFA(d.arch, deltaBuf, int64(nextpc)-int64(pcsp.PC), spdelta) |
| } |
| pad := int(Rnd(int64(len(deltaBuf)), int64(d.arch.PtrSize))) - len(deltaBuf) |
| deltaBuf = append(deltaBuf, zeros[:pad]...) |
|
|
| |
| |
| |
| |
| |
|
|
| fdeLength := uint64(4 + 2*d.arch.PtrSize + len(deltaBuf)) |
| if isdw64 { |
| fdeLength += 4 |
| } |
| d.createUnitLength(fsu, fdeLength) |
|
|
| if d.linkctxt.LinkMode == LinkExternal { |
| d.addDwarfAddrRef(fsu, fs) |
| } else { |
| d.addDwarfAddrField(fsu, 0) |
| } |
| addAddrPlus(fsu, d.arch, s, 0) |
| fsu.AddUintXX(d.arch, uint64(len(d.ldr.Data(fn))), d.arch.PtrSize) |
| fsu.AddBytes(deltaBuf) |
|
|
| if d.linkctxt.HeadType == objabi.Haix { |
| addDwsectCUSize(".debug_frame", d.ldr.SymPkg(fn), fdeLength+uint64(lengthFieldSize)) |
| } |
| } |
|
|
| return dwarfSecInfo{syms: []loader.Sym{fs}} |
| } |
|
|
| |
| |
| |
|
|
| func (d *dwctxt) writeUnitInfo(u *sym.CompilationUnit, abbrevsym loader.Sym, addrsym loader.Sym, infoEpilog loader.Sym) []loader.Sym { |
| syms := []loader.Sym{} |
| if len(u.Textp) == 0 && u.DWInfo.Child == nil && len(u.VarDIEs) == 0 { |
| return syms |
| } |
|
|
| compunit := u.DWInfo |
| s := d.dtolsym(compunit.Sym) |
| su := d.ldr.MakeSymbolUpdater(s) |
|
|
| |
| |
| d.createUnitLength(su, 0) |
| dwver := 4 |
| if buildcfg.Experiment.Dwarf5 { |
| dwver = 5 |
| } |
| su.AddUint16(d.arch, uint16(dwver)) |
|
|
| if buildcfg.Experiment.Dwarf5 { |
| |
| |
| |
| |
| su.AddUint8(uint8(dwarf.DW_UT_compile)) |
| su.AddUint8(uint8(d.arch.PtrSize)) |
| d.addDwarfAddrRef(su, abbrevsym) |
| } else { |
| |
| |
| |
| d.addDwarfAddrRef(su, abbrevsym) |
| su.AddUint8(uint8(d.arch.PtrSize)) |
| } |
|
|
| ds := dwSym(s) |
| dwarf.Uleb128put(d, ds, int64(compunit.Abbrev)) |
| if buildcfg.Experiment.Dwarf5 { |
| |
| |
| abattr := getattr(compunit, dwarf.DW_AT_addr_base) |
| if abattr != nil { |
| abattr.Data = dwSym(addrsym) |
| } |
| } |
| dwarf.PutAttrs(d, ds, compunit.Abbrev, compunit.Attr) |
|
|
| |
| cu := make([]loader.Sym, 0, len(u.AbsFnDIEs)+len(u.FuncDIEs)) |
| cu = append(cu, s) |
| cu = append(cu, u.AbsFnDIEs...) |
| cu = append(cu, u.FuncDIEs...) |
| if u.Consts != 0 { |
| cu = append(cu, u.Consts) |
| } |
| cu = append(cu, u.VarDIEs...) |
| var cusize int64 |
| for _, child := range cu { |
| cusize += int64(len(d.ldr.Data(child))) |
| } |
|
|
| for die := compunit.Child; die != nil; die = die.Link { |
| l := len(cu) |
| lastSymSz := int64(len(d.ldr.Data(cu[l-1]))) |
| cu = d.putdie(cu, die) |
| if lastSymSz != int64(len(d.ldr.Data(cu[l-1]))) { |
| |
| cusize = cusize - lastSymSz + int64(len(d.ldr.Data(cu[l-1]))) |
| } |
| for _, child := range cu[l:] { |
| cusize += int64(len(d.ldr.Data(child))) |
| } |
| } |
|
|
| culu := d.ldr.MakeSymbolUpdater(infoEpilog) |
| culu.AddUint8(0) |
| cu = append(cu, infoEpilog) |
| cusize++ |
|
|
| |
| if d.linkctxt.HeadType == objabi.Haix { |
| addDwsectCUSize(".debug_info", d.getPkgFromCUSym(s), uint64(cusize)) |
| } |
| if isDwarf64(d.linkctxt) { |
| cusize -= 12 |
| su.SetUint(d.arch, 4, uint64(cusize)) |
| } else { |
| cusize -= 4 |
| su.SetUint32(d.arch, 0, uint32(cusize)) |
| } |
| return append(syms, cu...) |
| } |
|
|
| func (d *dwctxt) writegdbscript() dwarfSecInfo { |
| |
| if d.linkctxt.HeadType == objabi.Haix { |
| return dwarfSecInfo{} |
| } |
| if d.linkctxt.LinkMode == LinkExternal && d.linkctxt.HeadType == objabi.Hwindows && d.linkctxt.BuildMode == BuildModeCArchive { |
| |
| |
| |
| |
| |
| |
| return dwarfSecInfo{} |
| } |
| if gdbscript == "" { |
| return dwarfSecInfo{} |
| } |
|
|
| gs := d.ldr.CreateSymForUpdate(".debug_gdb_scripts", 0) |
| gs.SetType(sym.SDWARFSECT) |
|
|
| gs.AddUint8(GdbScriptPythonFileId) |
| gs.Addstring(gdbscript) |
| return dwarfSecInfo{syms: []loader.Sym{gs.Sym()}} |
| } |
|
|
| |
| |
|
|
| var prototypedies map[string]*dwarf.DWDie |
|
|
| func dwarfEnabled(ctxt *Link) bool { |
| if *FlagW { |
| return false |
| } |
| if ctxt.HeadType == objabi.Hplan9 || ctxt.HeadType == objabi.Hjs || ctxt.HeadType == objabi.Hwasip1 { |
| return false |
| } |
|
|
| if ctxt.LinkMode == LinkExternal { |
| switch { |
| case ctxt.IsELF: |
| case ctxt.HeadType == objabi.Hdarwin: |
| case ctxt.HeadType == objabi.Hwindows: |
| case ctxt.HeadType == objabi.Haix: |
| res, err := dwarf.IsDWARFEnabledOnAIXLd(ctxt.extld()) |
| if err != nil { |
| Exitf("%v", err) |
| } |
| return res |
| default: |
| return false |
| } |
| } |
|
|
| return true |
| } |
|
|
| |
| |
| func (d *dwctxt) mkBuiltinType(ctxt *Link, abrv int, tname string) *dwarf.DWDie { |
| |
| die := d.newdie(&dwtypes, abrv, tname) |
|
|
| |
| gotype := d.lookupOrDiag("type:" + tname) |
|
|
| |
| ds := loader.Sym(die.Sym.(dwSym)) |
| d.rtmap[ds] = gotype |
|
|
| |
| d.tdmap[gotype] = ds |
|
|
| return die |
| } |
|
|
| |
| |
| |
| |
| func (d *dwctxt) assignDebugAddrSlot(unit *sym.CompilationUnit, fnsym loader.Sym, r loader.Reloc, sb *loader.SymbolBuilder) { |
| rsym := r.Sym() |
| if unit.Addrs == nil { |
| unit.Addrs = make(map[sym.LoaderSym]uint32) |
| } |
| if _, ok := unit.Addrs[rsym]; ok { |
| |
| } else { |
| sl := len(unit.Addrs) |
| rt := r.Type() |
| lim, _ := rt.DwTxtAddrRelocParams() |
| if sl > lim { |
| log.Fatalf("internal error: %s relocation overflow on infosym for %s", rt.String(), d.ldr.SymName(fnsym)) |
| } |
| unit.Addrs[rsym] = uint32(sl) |
| sb.AddAddrPlus(d.arch, rsym, 0) |
| data := sb.Data() |
| if d.arch.PtrSize == 4 { |
| d.arch.ByteOrder.PutUint32(data[len(data)-4:], uint32(d.ldr.SymValue(rsym))) |
| } else { |
| d.arch.ByteOrder.PutUint64(data[len(data)-8:], uint64(d.ldr.SymValue(rsym))) |
| } |
| } |
| } |
|
|
| |
| |
| |
| func (d *dwctxt) dwarfVisitFunction(fnSym loader.Sym, unit *sym.CompilationUnit) { |
| |
| |
| |
| infosym, _, rangesym, _ := d.ldr.GetFuncDwarfAuxSyms(fnSym) |
| if infosym == 0 { |
| return |
| } |
| d.ldr.SetAttrNotInSymbolTable(infosym, true) |
| d.ldr.SetAttrReachable(infosym, true) |
| unit.FuncDIEs = append(unit.FuncDIEs, infosym) |
| if rangesym != 0 { |
| d.ldr.SetAttrNotInSymbolTable(rangesym, true) |
| d.ldr.SetAttrReachable(rangesym, true) |
| unit.RangeSyms = append(unit.RangeSyms, rangesym) |
| } |
|
|
| |
| |
| |
| |
| drelocs := d.ldr.Relocs(infosym) |
| for ri := 0; ri < drelocs.Count(); ri++ { |
| r := drelocs.At(ri) |
| |
| if r.Type() == objabi.R_USETYPE { |
| d.defgotype(r.Sym()) |
| continue |
| } |
| if r.Type() != objabi.R_DWARFSECREF { |
| continue |
| } |
|
|
| rsym := r.Sym() |
| rst := d.ldr.SymType(rsym) |
|
|
| |
| if rst == sym.SDWARFABSFCN { |
| if !d.ldr.AttrOnList(rsym) { |
| |
| d.ldr.SetAttrOnList(rsym, true) |
| unit.AbsFnDIEs = append(unit.AbsFnDIEs, rsym) |
| d.importInfoSymbol(rsym) |
| } |
| continue |
| } |
|
|
| |
| if rst != sym.SDWARFTYPE && rst != sym.Sxxx { |
| continue |
| } |
| if _, ok := d.rtmap[rsym]; ok { |
| |
| continue |
| } |
|
|
| rsn := d.ldr.SymName(rsym) |
| tn := rsn[len(dwarf.InfoPrefix):] |
| ts := d.ldr.Lookup("type:"+tn, 0) |
| d.defgotype(ts) |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| func dwarfGenerateDebugInfo(ctxt *Link) { |
| if !dwarfEnabled(ctxt) { |
| return |
| } |
|
|
| d := &dwctxt{ |
| linkctxt: ctxt, |
| ldr: ctxt.loader, |
| arch: ctxt.Arch, |
| tmap: make(map[string]loader.Sym), |
| tdmap: make(map[loader.Sym]loader.Sym), |
| rtmap: make(map[loader.Sym]loader.Sym), |
| } |
| d.typeRuntimeEface = d.lookupOrDiag("type:runtime.eface") |
| d.typeRuntimeIface = d.lookupOrDiag("type:runtime.iface") |
|
|
| if ctxt.HeadType == objabi.Haix { |
| |
| dwsectCUSize = make(map[string]uint64) |
| } |
|
|
| |
| newattr(&dwtypes, dwarf.DW_AT_name, dwarf.DW_CLS_STRING, int64(len("dwtypes")), "dwtypes") |
|
|
| |
| d.newdie(&dwtypes, dwarf.DW_ABRV_NULLTYPE, "<unspecified>") |
|
|
| |
| |
| unsafeptrDie := d.mkBuiltinType(ctxt, dwarf.DW_ABRV_BARE_PTRTYPE, "unsafe.Pointer") |
| newattr(unsafeptrDie, dwarf.DW_AT_go_runtime_type, dwarf.DW_CLS_GO_TYPEREF, 0, dwSym(d.lookupOrDiag("type:unsafe.Pointer"))) |
| uintptrDie := d.mkBuiltinType(ctxt, dwarf.DW_ABRV_BASETYPE, "uintptr") |
| newattr(uintptrDie, dwarf.DW_AT_encoding, dwarf.DW_CLS_CONSTANT, dwarf.DW_ATE_unsigned, 0) |
| newattr(uintptrDie, dwarf.DW_AT_byte_size, dwarf.DW_CLS_CONSTANT, int64(d.arch.PtrSize), 0) |
| newattr(uintptrDie, dwarf.DW_AT_go_kind, dwarf.DW_CLS_CONSTANT, int64(abi.Uintptr), 0) |
| newattr(uintptrDie, dwarf.DW_AT_go_runtime_type, dwarf.DW_CLS_GO_TYPEREF, 0, dwSym(d.lookupOrDiag("type:uintptr"))) |
|
|
| d.uintptrInfoSym = d.mustFind("uintptr") |
|
|
| |
| prototypedies = map[string]*dwarf.DWDie{ |
| "type:runtime.stringStructDWARF": nil, |
| "type:runtime.slice": nil, |
| "type:runtime.sudog": nil, |
| "type:runtime.waitq": nil, |
| "type:runtime.hchan": nil, |
| "type:internal/runtime/maps.Map": nil, |
| "type:internal/runtime/maps.table": nil, |
| "type:internal/runtime/maps.groupsReference": nil, |
| } |
|
|
| |
| for _, typ := range []string{ |
| "type:internal/abi.Type", |
| "type:internal/abi.ArrayType", |
| "type:internal/abi.ChanType", |
| "type:internal/abi.FuncType", |
| "type:internal/abi.MapType", |
| "type:internal/abi.PtrType", |
| "type:internal/abi.SliceType", |
| "type:internal/abi.StructType", |
| "type:internal/abi.InterfaceType", |
| "type:internal/abi.ITab", |
| "type:internal/abi.Imethod"} { |
| d.defgotype(d.lookupOrDiag(typ)) |
| } |
|
|
| |
| var dwroot dwarf.DWDie |
| flagVariants := make(map[string]bool) |
|
|
| for _, lib := range ctxt.Library { |
|
|
| consts := d.ldr.Lookup(dwarf.ConstInfoPrefix+lib.Pkg, 0) |
| for _, unit := range lib.Units { |
| |
| if consts != 0 { |
| unit.Consts = consts |
| d.importInfoSymbol(consts) |
| consts = 0 |
| } |
| ctxt.compUnits = append(ctxt.compUnits, unit) |
|
|
| |
| if unit.Lib.Pkg == "runtime" { |
| ctxt.runtimeCU = unit |
| } |
|
|
| cuabrv := dwarf.DW_ABRV_COMPUNIT |
| if len(unit.Textp) == 0 { |
| cuabrv = dwarf.DW_ABRV_COMPUNIT_TEXTLESS |
| } |
| unit.DWInfo = d.newdie(&dwroot, cuabrv, unit.Lib.Pkg) |
| newattr(unit.DWInfo, dwarf.DW_AT_language, dwarf.DW_CLS_CONSTANT, int64(dwarf.DW_LANG_Go), 0) |
| |
| compDir := getCompilationDir() |
| |
| |
| |
| newattr(unit.DWInfo, dwarf.DW_AT_comp_dir, dwarf.DW_CLS_STRING, int64(len(compDir)), compDir) |
|
|
| var peData []byte |
| if producerExtra := d.ldr.Lookup(dwarf.CUInfoPrefix+"producer."+unit.Lib.Pkg, 0); producerExtra != 0 { |
| peData = d.ldr.Data(producerExtra) |
| } |
| producer := "Go cmd/compile " + buildcfg.Version |
| if len(peData) > 0 { |
| |
| |
| |
| |
| |
| |
| producer += "; " + string(peData) |
| flagVariants[string(peData)] = true |
| } else { |
| flagVariants[""] = true |
| } |
|
|
| newattr(unit.DWInfo, dwarf.DW_AT_producer, dwarf.DW_CLS_STRING, int64(len(producer)), producer) |
|
|
| var pkgname string |
| if pnSymIdx := d.ldr.Lookup(dwarf.CUInfoPrefix+"packagename."+unit.Lib.Pkg, 0); pnSymIdx != 0 { |
| pnsData := d.ldr.Data(pnSymIdx) |
| pkgname = string(pnsData) |
| } |
| newattr(unit.DWInfo, dwarf.DW_AT_go_package_name, dwarf.DW_CLS_STRING, int64(len(pkgname)), pkgname) |
|
|
| if buildcfg.Experiment.Dwarf5 && cuabrv == dwarf.DW_ABRV_COMPUNIT { |
| |
| |
| |
| |
| |
| |
| newattr(unit.DWInfo, dwarf.DW_AT_addr_base, dwarf.DW_CLS_REFERENCE, 0, 0) |
| } |
|
|
| |
| |
| |
| |
| for _, s := range unit.Textp { |
| d.dwarfVisitFunction(s, unit) |
| } |
| } |
| } |
|
|
| |
| |
| |
| if checkStrictDups > 1 && len(flagVariants) > 1 { |
| checkStrictDups = 1 |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| for idx := loader.Sym(1); idx < loader.Sym(d.ldr.NDef()); idx++ { |
| if !d.ldr.AttrReachable(idx) || |
| d.ldr.AttrNotInSymbolTable(idx) || |
| d.ldr.SymVersion(idx) >= sym.SymVerStatic { |
| continue |
| } |
| t := d.ldr.SymType(idx) |
| switch { |
| case t.IsRODATA(), t.IsDATA(), t.IsNOPTRDATA(), |
| t == sym.STYPE, t == sym.SBSS, t == sym.SNOPTRBSS, t == sym.STLSBSS, t == sym.SMODULEDATA: |
| |
| default: |
| continue |
| } |
| |
| gt := d.ldr.SymGoType(idx) |
| if gt == 0 { |
| if t == sym.SRODATA { |
| if d.ldr.IsDict(idx) { |
| |
| relocs := d.ldr.Relocs(idx) |
| for i := 0; i < relocs.Count(); i++ { |
| reloc := relocs.At(i) |
| if reloc.Type() == objabi.R_USEIFACE { |
| d.defgotype(reloc.Sym()) |
| } |
| } |
| } |
| } |
| continue |
| } |
| |
| |
| if d.ldr.IsFileLocal(idx) { |
| continue |
| } |
|
|
| |
| |
| |
| |
| |
| varDIE := d.ldr.GetVarDwarfAuxSym(idx) |
| if varDIE != 0 { |
| unit := d.ldr.SymUnit(idx) |
| d.defgotype(gt) |
| unit.VarDIEs = append(unit.VarDIEs, varDIE) |
| } |
| } |
|
|
| d.synthesizestringtypes(ctxt, dwtypes.Child) |
| d.synthesizeslicetypes(ctxt, dwtypes.Child) |
| d.synthesizemaptypes(ctxt, dwtypes.Child) |
| d.synthesizechantypes(ctxt, dwtypes.Child) |
| } |
|
|
| |
| |
| |
| func dwarfGenerateDebugSyms(ctxt *Link) { |
| if !dwarfEnabled(ctxt) { |
| return |
| } |
| d := &dwctxt{ |
| linkctxt: ctxt, |
| ldr: ctxt.loader, |
| arch: ctxt.Arch, |
| dwmu: new(sync.Mutex), |
| } |
| d.dwarfGenerateDebugSyms() |
| } |
|
|
| |
| |
| type dwUnitSyms struct { |
| |
| lineProlog loader.Sym |
| rangeProlog loader.Sym |
| infoEpilog loader.Sym |
|
|
| |
| linesyms []loader.Sym |
| infosyms []loader.Sym |
| locsyms []loader.Sym |
| rangessyms []loader.Sym |
| addrsym loader.Sym |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| func (d *dwctxt) dwUnitPortion(u *sym.CompilationUnit, abbrevsym loader.Sym, us *dwUnitSyms) { |
| if u.DWInfo.Abbrev != dwarf.DW_ABRV_COMPUNIT_TEXTLESS { |
| us.linesyms = d.writelines(u, us.lineProlog) |
| base := u.Textp[0] |
| if buildcfg.Experiment.Dwarf5 { |
| d.writedebugaddr(u, us.addrsym) |
| } |
| us.rangessyms = d.writepcranges(u, base, u.PCs, us.rangeProlog, us.addrsym) |
| us.locsyms = d.collectUnitLocs(u) |
| } |
| us.infosyms = d.writeUnitInfo(u, abbrevsym, us.addrsym, us.infoEpilog) |
| } |
|
|
| |
| |
| |
| |
| |
| func (d *dwctxt) writedebugaddr(unit *sym.CompilationUnit, debugaddr loader.Sym) { |
| dasu := d.ldr.MakeSymbolUpdater(debugaddr) |
|
|
| var dsyms []loader.Sym |
| for _, s := range unit.Textp { |
| fnSym := s |
| |
| |
| infosym, locsym, rangessym, _ := d.ldr.GetFuncDwarfAuxSyms(fnSym) |
|
|
| |
| |
| |
| dsyms = dsyms[:0] |
| dsyms = append(dsyms, infosym) |
| if rangessym != 0 { |
| dsyms = append(dsyms, rangessym) |
| } |
| if locsym != 0 { |
| dsyms = append(dsyms, locsym) |
| } |
| for _, dsym := range dsyms { |
| drelocs := d.ldr.Relocs(dsym) |
| for ri := 0; ri < drelocs.Count(); ri++ { |
| r := drelocs.At(ri) |
| if !r.Type().IsDwTxtAddr() { |
| continue |
| } |
| rsym := r.Sym() |
| rst := d.ldr.SymType(rsym) |
| |
| if !rst.IsText() { |
| |
| |
| |
| log.Fatalf("internal error: R_DWTXTADDR_* relocation on dwinfosym for %s against non-function %s type:%s", d.ldr.SymName(fnSym), d.ldr.SymName(rsym), rst.String()) |
| } |
| if runit := d.ldr.SymUnit(rsym); runit != unit { |
| log.Fatalf("internal error: R_DWTXTADDR_* relocation target text sym unit mismatch (want %q got %q)", unit.Lib.Pkg, runit.Lib.Pkg) |
| } |
| d.assignDebugAddrSlot(unit, fnSym, r, dasu) |
| } |
| } |
| } |
| } |
|
|
| func (d *dwctxt) dwarfGenerateDebugSyms() { |
| abbrevSec := d.writeabbrev() |
| dwarfp = append(dwarfp, abbrevSec) |
| d.calcCompUnitRanges() |
| slices.SortFunc(d.linkctxt.compUnits, compilationUnitByStartPCCmp) |
|
|
| |
| |
| |
| for _, u := range d.linkctxt.compUnits { |
| reversetree(&u.DWInfo.Child) |
| } |
| reversetree(&dwtypes.Child) |
| movetomodule(d.linkctxt, &dwtypes) |
|
|
| mkSecSym := func(name string) loader.Sym { |
| s := d.ldr.CreateSymForUpdate(name, 0) |
| s.SetType(sym.SDWARFSECT) |
| s.SetReachable(true) |
| return s.Sym() |
| } |
|
|
| mkAnonSym := func(kind sym.SymKind) loader.Sym { |
| s := d.ldr.MakeSymbolUpdater(d.ldr.CreateExtSym("", 0)) |
| s.SetType(kind) |
| s.SetReachable(true) |
| return s.Sym() |
| } |
|
|
| |
| frameSym := mkSecSym(".debug_frame") |
| lineSym := mkSecSym(".debug_line") |
| var rangesSym, locSym loader.Sym |
| if buildcfg.Experiment.Dwarf5 { |
| rangesSym = mkSecSym(".debug_rnglists") |
| locSym = mkSecSym(".debug_loclists") |
| } else { |
| rangesSym = mkSecSym(".debug_ranges") |
| locSym = mkSecSym(".debug_loc") |
| } |
| infoSym := mkSecSym(".debug_info") |
| var addrSym loader.Sym |
| if buildcfg.Experiment.Dwarf5 { |
| addrSym = mkSecSym(".debug_addr") |
| } |
|
|
| |
| lineSec := dwarfSecInfo{syms: []loader.Sym{lineSym}} |
| frameSec := dwarfSecInfo{syms: []loader.Sym{frameSym}} |
| infoSec := dwarfSecInfo{syms: []loader.Sym{infoSym}} |
| var addrSec, rangesSec, locSec dwarfSecInfo |
| if buildcfg.Experiment.Dwarf5 { |
| addrHdr := d.writeDebugAddrHdr() |
| addrSec.syms = []loader.Sym{addrSym, addrHdr} |
| rnglistsHdr := d.writeDebugRngListsHdr() |
| rangesSec.syms = []loader.Sym{rangesSym, rnglistsHdr} |
| loclistsHdr := d.writeDebugLocListsHdr() |
| locSec.syms = []loader.Sym{locSym, loclistsHdr} |
| } else { |
| rangesSec = dwarfSecInfo{syms: []loader.Sym{rangesSym}} |
| locSec = dwarfSecInfo{syms: []loader.Sym{locSym}} |
| } |
|
|
| |
| |
| ncu := len(d.linkctxt.compUnits) |
| unitSyms := make([]dwUnitSyms, ncu) |
| for i := 0; i < ncu; i++ { |
| us := &unitSyms[i] |
| us.lineProlog = mkAnonSym(sym.SDWARFLINES) |
| us.rangeProlog = mkAnonSym(sym.SDWARFRANGE) |
| us.infoEpilog = mkAnonSym(sym.SDWARFFCN) |
| us.addrsym = mkAnonSym(sym.SDWARFADDR) |
| } |
|
|
| var wg sync.WaitGroup |
| sema := make(chan struct{}, runtime.GOMAXPROCS(0)) |
|
|
| |
| |
| wg.Add(1) |
| go func() { |
| sema <- struct{}{} |
| defer func() { |
| <-sema |
| wg.Done() |
| }() |
| frameSec = d.writeframes(frameSym) |
| }() |
|
|
| |
| |
| |
| wg.Add(len(d.linkctxt.compUnits)) |
| for i := 0; i < ncu; i++ { |
| go func(u *sym.CompilationUnit, us *dwUnitSyms) { |
| sema <- struct{}{} |
| defer func() { |
| <-sema |
| wg.Done() |
| }() |
| d.dwUnitPortion(u, abbrevSec.secSym(), us) |
| }(d.linkctxt.compUnits[i], &unitSyms[i]) |
| } |
| wg.Wait() |
|
|
| markReachable := func(syms []loader.Sym) []loader.Sym { |
| for _, s := range syms { |
| d.ldr.SetAttrNotInSymbolTable(s, true) |
| d.ldr.SetAttrReachable(s, true) |
| } |
| return syms |
| } |
|
|
| patchHdr := func(sec *dwarfSecInfo, len uint64) { |
| hdrsym := sec.syms[1] |
| len += uint64(d.ldr.SymSize(hdrsym)) |
| su := d.ldr.MakeSymbolUpdater(hdrsym) |
| if isDwarf64(d.linkctxt) { |
| len -= 12 |
| su.SetUint(d.arch, 4, len) |
| } else { |
| len -= 4 |
| su.SetUint32(d.arch, 0, uint32(len)) |
| } |
| } |
|
|
| if buildcfg.Experiment.Dwarf5 { |
| |
| |
| var rltot, addrtot, loctot uint64 |
| for i := 0; i < ncu; i++ { |
| addrtot += uint64(d.ldr.SymSize(unitSyms[i].addrsym)) |
| rs := unitSyms[i].rangessyms |
| for _, s := range rs { |
| rltot += uint64(d.ldr.SymSize(s)) |
| } |
| loc := unitSyms[i].locsyms |
| for _, s := range loc { |
| loctot += uint64(d.ldr.SymSize(s)) |
| } |
| } |
| |
| patchHdr(&addrSec, addrtot) |
| patchHdr(&rangesSec, rltot) |
| patchHdr(&locSec, loctot) |
| } |
|
|
| |
| for i := 0; i < ncu; i++ { |
| r := &unitSyms[i] |
| lineSec.syms = append(lineSec.syms, markReachable(r.linesyms)...) |
| infoSec.syms = append(infoSec.syms, markReachable(r.infosyms)...) |
| locSec.syms = append(locSec.syms, markReachable(r.locsyms)...) |
| rangesSec.syms = append(rangesSec.syms, markReachable(r.rangessyms)...) |
| if buildcfg.Experiment.Dwarf5 && r.addrsym != 0 { |
| addrSec.syms = append(addrSec.syms, r.addrsym) |
| } |
| } |
| dwarfp = append(dwarfp, lineSec) |
| dwarfp = append(dwarfp, frameSec) |
| gdbScriptSec := d.writegdbscript() |
| if gdbScriptSec.secSym() != 0 { |
| dwarfp = append(dwarfp, gdbScriptSec) |
| } |
| dwarfp = append(dwarfp, infoSec) |
| if len(locSec.syms) > 1 { |
| dwarfp = append(dwarfp, locSec) |
| } |
| dwarfp = append(dwarfp, rangesSec) |
| if buildcfg.Experiment.Dwarf5 { |
| dwarfp = append(dwarfp, addrSec) |
| } |
|
|
| |
| |
| |
| |
| seen := loader.MakeBitmap(d.ldr.NSym()) |
| for _, s := range infoSec.syms { |
| if seen.Has(s) { |
| log.Fatalf("dwarf symbol %s listed multiple times", |
| d.ldr.SymName(s)) |
| } |
| seen.Set(s) |
| } |
| } |
|
|
| func (d *dwctxt) collectUnitLocs(u *sym.CompilationUnit) []loader.Sym { |
| syms := []loader.Sym{} |
| for _, fn := range u.FuncDIEs { |
| relocs := d.ldr.Relocs(fn) |
| for i := 0; i < relocs.Count(); i++ { |
| reloc := relocs.At(i) |
| if reloc.Type() != objabi.R_DWARFSECREF { |
| continue |
| } |
| rsym := reloc.Sym() |
| if d.ldr.SymType(rsym) == sym.SDWARFLOC { |
| syms = append(syms, rsym) |
| |
| break |
| } |
| } |
| } |
| return syms |
| } |
|
|
| func dwarfaddelfsectionsyms(ctxt *Link) { |
| if *FlagW { |
| return |
| } |
| if ctxt.LinkMode != LinkExternal { |
| return |
| } |
|
|
| ldr := ctxt.loader |
| for _, si := range dwarfp { |
| s := si.secSym() |
| sect := ldr.SymSect(si.secSym()) |
| putelfsectionsym(ctxt, ctxt.Out, s, elfShdrShnum(sect.Elfsect.(*ElfShdr))) |
| } |
| } |
|
|
| |
| |
| |
| func dwarfcompress(ctxt *Link) { |
| |
| type compressedSect struct { |
| index int |
| compressed []byte |
| syms []loader.Sym |
| } |
|
|
| supported := ctxt.IsELF || ctxt.IsWindows() || ctxt.IsDarwin() |
| if !ctxt.compressDWARF || !supported || ctxt.IsExternal() { |
| return |
| } |
|
|
| var compressedCount int |
| resChannel := make(chan compressedSect) |
| for i := range dwarfp { |
| go func(resIndex int, syms []loader.Sym) { |
| resChannel <- compressedSect{resIndex, compressSyms(ctxt, syms), syms} |
| }(compressedCount, dwarfp[i].syms) |
| compressedCount++ |
| } |
| res := make([]compressedSect, compressedCount) |
| for ; compressedCount > 0; compressedCount-- { |
| r := <-resChannel |
| res[r.index] = r |
| } |
|
|
| ldr := ctxt.loader |
| var newDwarfp []dwarfSecInfo |
| Segdwarf.Sections = Segdwarf.Sections[:0] |
| for _, z := range res { |
| s := z.syms[0] |
| if z.compressed == nil { |
| |
| ds := dwarfSecInfo{syms: z.syms} |
| newDwarfp = append(newDwarfp, ds) |
| Segdwarf.Sections = append(Segdwarf.Sections, ldr.SymSect(s)) |
| } else { |
| var compressedSegName string |
| if ctxt.IsELF { |
| compressedSegName = ldr.SymSect(s).Name |
| } else { |
| compressedSegName = ".zdebug_" + ldr.SymSect(s).Name[len(".debug_"):] |
| } |
| sect := addsection(ctxt.loader, ctxt.Arch, &Segdwarf, compressedSegName, 04) |
| sect.Align = int32(ctxt.Arch.Alignment) |
| sect.Length = uint64(len(z.compressed)) |
| sect.Compressed = true |
| newSym := ldr.MakeSymbolBuilder(compressedSegName) |
| ldr.SetAttrReachable(s, true) |
| newSym.SetData(z.compressed) |
| newSym.SetSize(int64(len(z.compressed))) |
| ldr.SetSymSect(newSym.Sym(), sect) |
| ds := dwarfSecInfo{syms: []loader.Sym{newSym.Sym()}} |
| newDwarfp = append(newDwarfp, ds) |
|
|
| |
| for _, s := range z.syms { |
| ldr.SetAttrReachable(s, false) |
| ldr.FreeSym(s) |
| } |
| } |
| } |
| dwarfp = newDwarfp |
|
|
| |
| |
| |
| pos := Segdwarf.Vaddr |
| var prevSect *sym.Section |
| for _, si := range dwarfp { |
| for _, s := range si.syms { |
| sect := ldr.SymSect(s) |
| if sect != prevSect { |
| if ctxt.IsWindows() { |
| pos = uint64(Rnd(int64(pos), PEFILEALIGN)) |
| } |
| sect.Vaddr = pos |
| prevSect = sect |
| } |
| ldr.SetSymValue(s, int64(pos)) |
| if ldr.SubSym(s) != 0 { |
| log.Fatalf("%s: unexpected sub-symbols", ldr.SymName(s)) |
| } |
| pos += uint64(ldr.SymSize(s)) |
| } |
| } |
| Segdwarf.Length = pos - Segdwarf.Vaddr |
| } |
|
|
| func compilationUnitByStartPCCmp(a, b *sym.CompilationUnit) int { |
| switch { |
| case len(a.Textp) == 0 && len(b.Textp) == 0: |
| return strings.Compare(a.Lib.Pkg, b.Lib.Pkg) |
| case len(a.Textp) != 0 && len(b.Textp) == 0: |
| return -1 |
| case len(a.Textp) == 0 && len(b.Textp) != 0: |
| return +1 |
| default: |
| return cmp.Compare(a.PCs[0].Start, b.PCs[0].Start) |
| } |
| } |
|
|
| |
| |
| |
| |
| func (d *dwctxt) getPkgFromCUSym(s loader.Sym) string { |
| return strings.TrimPrefix(d.ldr.SymName(s), dwarf.InfoPrefix+".pkg.") |
| } |
|
|
| |
| |
| |
| |
| |
| var dwsectCUSizeMu sync.Mutex |
| var dwsectCUSize map[string]uint64 |
|
|
| |
| func getDwsectCUSize(sname string, pkgname string) uint64 { |
| return dwsectCUSize[sname+"."+pkgname] |
| } |
|
|
| func addDwsectCUSize(sname string, pkgname string, size uint64) { |
| dwsectCUSizeMu.Lock() |
| defer dwsectCUSizeMu.Unlock() |
| dwsectCUSize[sname+"."+pkgname] += size |
| } |
|
|
| |
| |
| |
| |
| |
| func (d *dwctxt) writeDebugMiscSecHdr(st sym.SymKind, addOffsetEntryCount bool) loader.Sym { |
| su := d.ldr.MakeSymbolUpdater(d.ldr.CreateExtSym("", 0)) |
| su.SetType(st) |
| su.SetReachable(true) |
| d.createUnitLength(su, 0) |
| su.AddUint16(d.arch, 5) |
| su.AddUint8(uint8(d.arch.PtrSize)) |
| su.AddUint8(0) |
| if addOffsetEntryCount { |
| su.AddUint32(d.arch, 0) |
| } |
| return su.Sym() |
| } |
|
|
| func (d *dwctxt) writeDebugRngListsHdr() loader.Sym { |
| return d.writeDebugMiscSecHdr(sym.SDWARFRANGE, true) |
| } |
|
|
| func (d *dwctxt) writeDebugLocListsHdr() loader.Sym { |
| return d.writeDebugMiscSecHdr(sym.SDWARFLOC, true) |
| } |
|
|
| func (d *dwctxt) writeDebugAddrHdr() loader.Sym { |
| return d.writeDebugMiscSecHdr(sym.SDWARFADDR, false) |
| } |
|
|