mirror of https://github.com/golang/go.git
201 lines
4.1 KiB
Go
201 lines
4.1 KiB
Go
// Copyright 2009 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package dwarf
|
|
|
|
import (
|
|
"sort"
|
|
"strconv"
|
|
)
|
|
|
|
// DWARF debug info is split into a sequence of compilation units.
|
|
// Each unit has its own abbreviation table and address size.
|
|
|
|
type unit struct {
|
|
base Offset // byte offset of header within the aggregate info
|
|
off Offset // byte offset of data within the aggregate info
|
|
data []byte
|
|
atable abbrevTable
|
|
*unit5 // info specific to DWARF 5 units
|
|
asize int
|
|
vers int
|
|
is64 bool // True for 64-bit DWARF format
|
|
utype uint8 // DWARF 5 unit type
|
|
}
|
|
|
|
type unit5 struct {
|
|
addrBase uint64
|
|
strOffsetsBase uint64
|
|
rngListsBase uint64
|
|
locListsBase uint64
|
|
}
|
|
|
|
// Implement the dataFormat interface.
|
|
|
|
func (u *unit) version() int {
|
|
return u.vers
|
|
}
|
|
|
|
func (u *unit) dwarf64() (bool, bool) {
|
|
return u.is64, true
|
|
}
|
|
|
|
func (u *unit) addrsize() int {
|
|
return u.asize
|
|
}
|
|
|
|
func (u *unit) addrBase() uint64 {
|
|
if u.unit5 != nil {
|
|
return u.unit5.addrBase
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func (u *unit) strOffsetsBase() uint64 {
|
|
if u.unit5 != nil {
|
|
return u.unit5.strOffsetsBase
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func (u *unit) rngListsBase() uint64 {
|
|
if u.unit5 != nil {
|
|
return u.unit5.rngListsBase
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func (u *unit) locListsBase() uint64 {
|
|
if u.unit5 != nil {
|
|
return u.unit5.locListsBase
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func (d *Data) parseUnits() ([]unit, error) {
|
|
// Count units.
|
|
nunit := 0
|
|
b := makeBuf(d, unknownFormat{}, "info", 0, d.info)
|
|
for len(b.data) > 0 {
|
|
len, _ := b.unitLength()
|
|
if len != Offset(uint32(len)) {
|
|
b.error("unit length overflow")
|
|
break
|
|
}
|
|
b.skip(int(len))
|
|
if len > 0 {
|
|
nunit++
|
|
}
|
|
}
|
|
if b.err != nil {
|
|
return nil, b.err
|
|
}
|
|
|
|
// Again, this time writing them down.
|
|
b = makeBuf(d, unknownFormat{}, "info", 0, d.info)
|
|
units := make([]unit, nunit)
|
|
for i := range units {
|
|
u := &units[i]
|
|
u.base = b.off
|
|
var n Offset
|
|
if b.err != nil {
|
|
return nil, b.err
|
|
}
|
|
for n == 0 {
|
|
n, u.is64 = b.unitLength()
|
|
}
|
|
dataOff := b.off
|
|
vers := b.uint16()
|
|
if vers < 2 || vers > 5 {
|
|
b.error("unsupported DWARF version " + strconv.Itoa(int(vers)))
|
|
break
|
|
}
|
|
u.vers = int(vers)
|
|
if vers >= 5 {
|
|
u.utype = b.uint8()
|
|
u.asize = int(b.uint8())
|
|
}
|
|
var abbrevOff uint64
|
|
if u.is64 {
|
|
abbrevOff = b.uint64()
|
|
} else {
|
|
abbrevOff = uint64(b.uint32())
|
|
}
|
|
atable, err := d.parseAbbrev(abbrevOff, u.vers)
|
|
if err != nil {
|
|
if b.err == nil {
|
|
b.err = err
|
|
}
|
|
break
|
|
}
|
|
u.atable = atable
|
|
if vers < 5 {
|
|
u.asize = int(b.uint8())
|
|
}
|
|
|
|
switch u.utype {
|
|
case utSkeleton, utSplitCompile:
|
|
b.uint64() // unit ID
|
|
case utType, utSplitType:
|
|
b.uint64() // type signature
|
|
if u.is64 { // type offset
|
|
b.uint64()
|
|
} else {
|
|
b.uint32()
|
|
}
|
|
}
|
|
|
|
u.off = b.off
|
|
u.data = b.bytes(int(n - (b.off - dataOff)))
|
|
}
|
|
if b.err != nil {
|
|
return nil, b.err
|
|
}
|
|
return units, nil
|
|
}
|
|
|
|
// offsetToUnit returns the index of the unit containing offset off.
|
|
// It returns -1 if no unit contains this offset.
|
|
func (d *Data) offsetToUnit(off Offset) int {
|
|
// Find the unit after off
|
|
next := sort.Search(len(d.unit), func(i int) bool {
|
|
return d.unit[i].off > off
|
|
})
|
|
if next == 0 {
|
|
return -1
|
|
}
|
|
u := &d.unit[next-1]
|
|
if u.off <= off && off < u.off+Offset(len(u.data)) {
|
|
return next - 1
|
|
}
|
|
return -1
|
|
}
|
|
|
|
func (d *Data) collectDwarf5BaseOffsets(u *unit) error {
|
|
if u.unit5 == nil {
|
|
panic("expected unit5 to be set up already")
|
|
}
|
|
b := makeBuf(d, u, "info", u.off, u.data)
|
|
cu := b.entry(nil, u)
|
|
if cu == nil {
|
|
// Unknown abbreviation table entry or some other fatal
|
|
// problem; bail early on the assumption that this will be
|
|
// detected at some later point.
|
|
return b.err
|
|
}
|
|
if iAddrBase, ok := cu.Val(AttrAddrBase).(int64); ok {
|
|
u.unit5.addrBase = uint64(iAddrBase)
|
|
}
|
|
if iStrOffsetsBase, ok := cu.Val(AttrStrOffsetsBase).(int64); ok {
|
|
u.unit5.strOffsetsBase = uint64(iStrOffsetsBase)
|
|
}
|
|
if iRngListsBase, ok := cu.Val(AttrRnglistsBase).(int64); ok {
|
|
u.unit5.rngListsBase = uint64(iRngListsBase)
|
|
}
|
|
if iLocListsBase, ok := cu.Val(AttrLoclistsBase).(int64); ok {
|
|
u.unit5.locListsBase = uint64(iLocListsBase)
|
|
}
|
|
return nil
|
|
}
|