2017-04-06 18:14:01 +00:00
|
|
|
package proc
|
|
|
|
|
|
|
|
import (
|
2017-11-21 08:50:14 +00:00
|
|
|
"bytes"
|
2017-05-29 13:20:01 +00:00
|
|
|
"debug/dwarf"
|
|
|
|
"debug/elf"
|
|
|
|
"debug/macho"
|
2017-04-06 18:14:01 +00:00
|
|
|
"debug/pe"
|
2017-08-24 07:46:47 +00:00
|
|
|
"encoding/binary"
|
2018-06-20 10:20:35 +00:00
|
|
|
"encoding/hex"
|
2017-04-06 18:14:01 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
2018-11-07 22:21:35 +00:00
|
|
|
"path/filepath"
|
2017-09-01 13:30:45 +00:00
|
|
|
"sort"
|
|
|
|
"strings"
|
2017-04-06 18:14:01 +00:00
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
2019-01-04 18:39:25 +00:00
|
|
|
"github.com/go-delve/delve/pkg/dwarf/frame"
|
|
|
|
"github.com/go-delve/delve/pkg/dwarf/godwarf"
|
|
|
|
"github.com/go-delve/delve/pkg/dwarf/line"
|
|
|
|
"github.com/go-delve/delve/pkg/dwarf/op"
|
|
|
|
"github.com/go-delve/delve/pkg/dwarf/reader"
|
|
|
|
"github.com/go-delve/delve/pkg/goversion"
|
2017-04-06 18:14:01 +00:00
|
|
|
)
|
|
|
|
|
2019-03-20 17:32:51 +00:00
|
|
|
// BinaryInfo holds information on the binaries being executed (this
|
|
|
|
// includes both the executable and also any loaded libraries).
|
2017-04-06 18:14:01 +00:00
|
|
|
type BinaryInfo struct {
|
2018-11-07 22:21:35 +00:00
|
|
|
// Path on disk of the binary being executed.
|
|
|
|
Path string
|
2018-11-06 22:32:14 +00:00
|
|
|
// Architecture of this binary.
|
|
|
|
Arch Arch
|
|
|
|
|
|
|
|
// GOOS operating system this binary is executing on.
|
|
|
|
GOOS string
|
|
|
|
|
|
|
|
// Functions is a list of all DW_TAG_subprogram entries in debug_info, sorted by entry point
|
|
|
|
Functions []Function
|
|
|
|
// Sources is a list of all source files found in debug_line.
|
|
|
|
Sources []string
|
|
|
|
// LookupFunc maps function names to a description of the function.
|
|
|
|
LookupFunc map[string]*Function
|
|
|
|
|
2019-03-20 17:32:51 +00:00
|
|
|
// Images is a list of loaded shared libraries (also known as
|
|
|
|
// shared objects on linux or DLLs on windws).
|
|
|
|
Images []*Image
|
|
|
|
|
|
|
|
ElfDynamicSection ElfDynamicSection
|
|
|
|
|
2017-04-06 18:14:01 +00:00
|
|
|
lastModified time.Time // Time the executable of this process was last modified
|
|
|
|
|
2018-06-20 10:20:35 +00:00
|
|
|
closer io.Closer
|
|
|
|
sepDebugCloser io.Closer
|
2017-04-06 18:14:01 +00:00
|
|
|
|
2018-05-29 15:01:51 +00:00
|
|
|
staticBase uint64
|
|
|
|
|
2017-04-06 18:14:01 +00:00
|
|
|
// Maps package names to package paths, needed to lookup types inside DWARF info
|
|
|
|
packageMap map[string]string
|
|
|
|
|
2018-11-06 22:32:14 +00:00
|
|
|
dwarf *dwarf.Data
|
|
|
|
dwarfReader *dwarf.Reader
|
|
|
|
frameEntries frame.FrameDescriptionEntries
|
|
|
|
loclist loclistReader
|
|
|
|
compileUnits []*compileUnit
|
|
|
|
types map[string]dwarf.Offset
|
|
|
|
packageVars []packageVar // packageVars is a list of all global/package variables in debug_info, sorted by address
|
|
|
|
typeCache map[dwarf.Offset]godwarf.Type
|
2017-04-06 18:14:01 +00:00
|
|
|
|
2018-11-06 22:32:14 +00:00
|
|
|
gStructOffset uint64
|
2017-05-29 13:20:01 +00:00
|
|
|
|
2017-04-06 18:14:01 +00:00
|
|
|
loadModuleDataOnce sync.Once
|
|
|
|
moduleData []moduleData
|
|
|
|
nameOfRuntimeType map[uintptr]nameOfRuntimeTypeEntry
|
2017-08-15 06:21:24 +00:00
|
|
|
|
2018-04-18 18:09:33 +00:00
|
|
|
// runtimeTypeToDIE maps between the offset of a runtime._type in
|
|
|
|
// runtime.moduledata.types and the offset of the DIE in debug_info. This
|
|
|
|
// map is filled by using the extended attribute godwarf.AttrGoRuntimeType
|
|
|
|
// which was added in go 1.11.
|
|
|
|
runtimeTypeToDIE map[uint64]runtimeTypeDIE
|
|
|
|
|
2017-09-08 10:31:20 +00:00
|
|
|
// consts[off] lists all the constants with the type defined at offset off.
|
|
|
|
consts constantsMap
|
|
|
|
|
2017-08-15 06:21:24 +00:00
|
|
|
loadErrMu sync.Mutex
|
|
|
|
loadErr error
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// ErrUnsupportedLinuxArch is returned when attempting to debug a binary compiled for an unsupported architecture.
|
|
|
|
var ErrUnsupportedLinuxArch = errors.New("unsupported architecture - only linux/amd64 is supported")
|
|
|
|
|
|
|
|
// ErrUnsupportedWindowsArch is returned when attempting to debug a binary compiled for an unsupported architecture.
|
|
|
|
var ErrUnsupportedWindowsArch = errors.New("unsupported architecture of windows/386 - only windows/amd64 is supported")
|
|
|
|
|
|
|
|
// ErrUnsupportedDarwinArch is returned when attempting to debug a binary compiled for an unsupported architecture.
|
|
|
|
var ErrUnsupportedDarwinArch = errors.New("unsupported architecture - only darwin/amd64 is supported")
|
2017-04-06 18:14:01 +00:00
|
|
|
|
2018-11-06 22:34:26 +00:00
|
|
|
// ErrCouldNotDetermineRelocation is an error returned when Delve could not determine the base address of a
|
|
|
|
// position independant executable.
|
2018-05-29 15:01:51 +00:00
|
|
|
var ErrCouldNotDetermineRelocation = errors.New("could not determine the base address of a PIE")
|
|
|
|
|
2019-02-21 17:11:54 +00:00
|
|
|
// ErrNoDebugInfoFound is returned when Delve cannot open the debug_info
|
|
|
|
// section or find an external debug info file.
|
|
|
|
var ErrNoDebugInfoFound = errors.New("could not open debug info")
|
2018-11-07 22:21:35 +00:00
|
|
|
|
2017-09-01 13:30:45 +00:00
|
|
|
const dwarfGoLanguage = 22 // DW_LANG_Go (from DWARF v5, section 7.12, page 231)
|
|
|
|
|
|
|
|
type compileUnit struct {
|
2018-11-06 22:34:26 +00:00
|
|
|
name string // univocal name for non-go compile units
|
|
|
|
lowPC uint64
|
|
|
|
ranges [][2]uint64
|
2018-08-07 01:08:25 +00:00
|
|
|
|
|
|
|
entry *dwarf.Entry // debug_info entry describing this compile unit
|
|
|
|
isgo bool // true if this is the go compile unit
|
|
|
|
lineInfo *line.DebugLineInfo // debug_line segment associated with this compile unit
|
|
|
|
concreteInlinedFns []inlinedFn // list of concrete inlined functions within this compile unit
|
|
|
|
optimized bool // this compile unit is optimized
|
|
|
|
producer string // producer attribute
|
2018-08-17 06:17:22 +00:00
|
|
|
|
|
|
|
startOffset, endOffset dwarf.Offset // interval of offsets contained in this compile unit
|
2017-09-01 13:30:45 +00:00
|
|
|
}
|
|
|
|
|
proc: add support for dwz compressed DWARF
'dwz' is a tool that reduces the size of DWARF sections by
deduplicating symbols. The deduplicated symbols are moved from their
original 'compile unit' to a 'partial unit', which is then referenced
from its original location with an 'imported unit' tag.
In the case of Go binaries, all symbols are located in a single
'compile unit', and the name of each symbol contains a reference to its
package, so 'dwz' is not able to deduplicate them. But still, some C
symbols included in the binary are deduplicated, which also alters the
structure of the DWARF sections, making delve unable to parse them
(crashing in the attempt).
While it would've been possible to simply ignore the C symbols, or
blindly loading all then into BinaryInfo members (packageVars,
Functions...), for correctness sake this change tries to do the right
thing, staging symbols into temporary partialUnit objects, moving them
to BinaryInfo when they are actually requested by a 'imported unit'
tag.
2018-05-18 14:04:37 +00:00
|
|
|
type partialUnitConstant struct {
|
|
|
|
name string
|
|
|
|
typ dwarf.Offset
|
|
|
|
value int64
|
|
|
|
}
|
|
|
|
|
|
|
|
type partialUnit struct {
|
2018-06-18 08:05:21 +00:00
|
|
|
entry *dwarf.Entry
|
|
|
|
types map[string]dwarf.Offset
|
|
|
|
variables []packageVar
|
|
|
|
constants []partialUnitConstant
|
|
|
|
functions []Function
|
proc: add support for dwz compressed DWARF
'dwz' is a tool that reduces the size of DWARF sections by
deduplicating symbols. The deduplicated symbols are moved from their
original 'compile unit' to a 'partial unit', which is then referenced
from its original location with an 'imported unit' tag.
In the case of Go binaries, all symbols are located in a single
'compile unit', and the name of each symbol contains a reference to its
package, so 'dwz' is not able to deduplicate them. But still, some C
symbols included in the binary are deduplicated, which also alters the
structure of the DWARF sections, making delve unable to parse them
(crashing in the attempt).
While it would've been possible to simply ignore the C symbols, or
blindly loading all then into BinaryInfo members (packageVars,
Functions...), for correctness sake this change tries to do the right
thing, staging symbols into temporary partialUnit objects, moving them
to BinaryInfo when they are actually requested by a 'imported unit'
tag.
2018-05-18 14:04:37 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 01:08:25 +00:00
|
|
|
// inlinedFn represents a concrete inlined function, e.g.
|
|
|
|
// an entry for the generated code of an inlined function.
|
|
|
|
type inlinedFn struct {
|
|
|
|
Name string // Name of the function that was inlined
|
|
|
|
LowPC, HighPC uint64 // Address range of the generated inlined instructions
|
|
|
|
CallFile string // File of the call site of the inlined function
|
|
|
|
CallLine int64 // Line of the call site of the inlined function
|
|
|
|
Parent *Function // The function that contains this inlined function
|
|
|
|
}
|
|
|
|
|
2017-09-01 13:30:45 +00:00
|
|
|
// Function describes a function in the target program.
|
|
|
|
type Function struct {
|
|
|
|
Name string
|
|
|
|
Entry, End uint64 // same as DW_AT_lowpc and DW_AT_highpc
|
|
|
|
offset dwarf.Offset
|
|
|
|
cu *compileUnit
|
|
|
|
}
|
|
|
|
|
|
|
|
// PackageName returns the package part of the symbol name,
|
|
|
|
// or the empty string if there is none.
|
|
|
|
// Borrowed from $GOROOT/debug/gosym/symtab.go
|
|
|
|
func (fn *Function) PackageName() string {
|
2017-09-08 10:31:20 +00:00
|
|
|
return packageName(fn.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func packageName(name string) string {
|
|
|
|
pathend := strings.LastIndex(name, "/")
|
2017-09-01 13:30:45 +00:00
|
|
|
if pathend < 0 {
|
|
|
|
pathend = 0
|
|
|
|
}
|
|
|
|
|
2017-09-08 10:31:20 +00:00
|
|
|
if i := strings.Index(name[pathend:], "."); i != -1 {
|
|
|
|
return name[:pathend+i]
|
2017-09-01 13:30:45 +00:00
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReceiverName returns the receiver type name of this symbol,
|
|
|
|
// or the empty string if there is none.
|
|
|
|
// Borrowed from $GOROOT/debug/gosym/symtab.go
|
|
|
|
func (fn *Function) ReceiverName() string {
|
|
|
|
pathend := strings.LastIndex(fn.Name, "/")
|
|
|
|
if pathend < 0 {
|
|
|
|
pathend = 0
|
|
|
|
}
|
|
|
|
l := strings.Index(fn.Name[pathend:], ".")
|
|
|
|
r := strings.LastIndex(fn.Name[pathend:], ".")
|
|
|
|
if l == -1 || r == -1 || l == r {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return fn.Name[pathend+l+1 : pathend+r]
|
|
|
|
}
|
|
|
|
|
|
|
|
// BaseName returns the symbol name without the package or receiver name.
|
|
|
|
// Borrowed from $GOROOT/debug/gosym/symtab.go
|
|
|
|
func (fn *Function) BaseName() string {
|
|
|
|
if i := strings.LastIndex(fn.Name, "."); i != -1 {
|
|
|
|
return fn.Name[i+1:]
|
|
|
|
}
|
|
|
|
return fn.Name
|
|
|
|
}
|
|
|
|
|
2017-10-27 15:00:41 +00:00
|
|
|
// Optimized returns true if the function was optimized by the compiler.
|
|
|
|
func (fn *Function) Optimized() bool {
|
|
|
|
return fn.cu.optimized
|
|
|
|
}
|
|
|
|
|
2017-09-08 10:31:20 +00:00
|
|
|
type constantsMap map[dwarf.Offset]*constantType
|
|
|
|
|
|
|
|
type constantType struct {
|
|
|
|
initialized bool
|
|
|
|
values []constantValue
|
|
|
|
}
|
|
|
|
|
|
|
|
type constantValue struct {
|
|
|
|
name string
|
|
|
|
fullName string
|
|
|
|
value int64
|
|
|
|
singleBit bool
|
|
|
|
}
|
|
|
|
|
2017-10-26 11:37:19 +00:00
|
|
|
// packageVar represents a package-level variable (or a C global variable).
|
|
|
|
// If a global variable does not have an address (for example it's stored in
|
|
|
|
// a register, or non-contiguously) addr will be 0.
|
|
|
|
type packageVar struct {
|
|
|
|
name string
|
|
|
|
offset dwarf.Offset
|
|
|
|
addr uint64
|
|
|
|
}
|
|
|
|
|
2017-08-24 07:46:47 +00:00
|
|
|
type loclistReader struct {
|
|
|
|
data []byte
|
|
|
|
cur int
|
|
|
|
ptrSz int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rdr *loclistReader) Seek(off int) {
|
|
|
|
rdr.cur = off
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rdr *loclistReader) read(sz int) []byte {
|
|
|
|
r := rdr.data[rdr.cur : rdr.cur+sz]
|
|
|
|
rdr.cur += sz
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rdr *loclistReader) oneAddr() uint64 {
|
|
|
|
switch rdr.ptrSz {
|
|
|
|
case 4:
|
|
|
|
addr := binary.LittleEndian.Uint32(rdr.read(rdr.ptrSz))
|
|
|
|
if addr == ^uint32(0) {
|
|
|
|
return ^uint64(0)
|
|
|
|
}
|
|
|
|
return uint64(addr)
|
|
|
|
case 8:
|
|
|
|
addr := uint64(binary.LittleEndian.Uint64(rdr.read(rdr.ptrSz)))
|
|
|
|
return addr
|
|
|
|
default:
|
|
|
|
panic("bad address size")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rdr *loclistReader) Next(e *loclistEntry) bool {
|
|
|
|
e.lowpc = rdr.oneAddr()
|
|
|
|
e.highpc = rdr.oneAddr()
|
|
|
|
|
|
|
|
if e.lowpc == 0 && e.highpc == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
if e.BaseAddressSelection() {
|
|
|
|
e.instr = nil
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
instrlen := binary.LittleEndian.Uint16(rdr.read(2))
|
|
|
|
e.instr = rdr.read(int(instrlen))
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
type loclistEntry struct {
|
|
|
|
lowpc, highpc uint64
|
|
|
|
instr []byte
|
|
|
|
}
|
|
|
|
|
2018-04-18 18:09:33 +00:00
|
|
|
type runtimeTypeDIE struct {
|
|
|
|
offset dwarf.Offset
|
|
|
|
kind int64
|
|
|
|
}
|
|
|
|
|
2017-08-24 07:46:47 +00:00
|
|
|
func (e *loclistEntry) BaseAddressSelection() bool {
|
|
|
|
return e.lowpc == ^uint64(0)
|
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
type buildIDHeader struct {
|
2018-06-20 10:20:35 +00:00
|
|
|
Namesz uint32
|
|
|
|
Descsz uint32
|
|
|
|
Type uint32
|
|
|
|
}
|
|
|
|
|
2019-03-20 17:32:51 +00:00
|
|
|
// ElfDynamicSection describes the .dynamic section of an ELF executable.
|
|
|
|
type ElfDynamicSection struct {
|
|
|
|
Addr uint64 // relocated address of where the .dynamic section is mapped in memory
|
|
|
|
Size uint64 // size of the .dynamic section of the executable
|
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// NewBinaryInfo returns an initialized but unloaded BinaryInfo struct.
|
|
|
|
func NewBinaryInfo(goos, goarch string) *BinaryInfo {
|
|
|
|
r := &BinaryInfo{GOOS: goos, nameOfRuntimeType: make(map[uintptr]nameOfRuntimeTypeEntry), typeCache: make(map[dwarf.Offset]godwarf.Type)}
|
2017-04-06 18:14:01 +00:00
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// TODO: find better way to determine proc arch (perhaps use executable file info).
|
2017-04-06 18:14:01 +00:00
|
|
|
switch goarch {
|
|
|
|
case "amd64":
|
2017-04-21 06:55:53 +00:00
|
|
|
r.Arch = AMD64Arch(goos)
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// LoadBinaryInfo will load and store the information from the binary at 'path'.
|
|
|
|
// It is expected this will be called in parallel with other initialization steps
|
|
|
|
// so a sync.WaitGroup must be provided.
|
2018-11-12 22:52:13 +00:00
|
|
|
func (bi *BinaryInfo) LoadBinaryInfo(path string, entryPoint uint64, debugInfoDirs []string) error {
|
2017-04-06 18:14:01 +00:00
|
|
|
fi, err := os.Stat(path)
|
|
|
|
if err == nil {
|
2018-08-31 18:08:18 +00:00
|
|
|
bi.lastModified = fi.ModTime()
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
|
2018-11-12 22:52:13 +00:00
|
|
|
var wg sync.WaitGroup
|
|
|
|
defer wg.Wait()
|
2018-11-07 22:21:35 +00:00
|
|
|
bi.Path = path
|
2018-08-31 18:08:18 +00:00
|
|
|
switch bi.GOOS {
|
2017-04-06 18:14:01 +00:00
|
|
|
case "linux":
|
2018-11-12 22:52:13 +00:00
|
|
|
return bi.LoadBinaryInfoElf(path, entryPoint, debugInfoDirs, &wg)
|
2017-04-06 18:14:01 +00:00
|
|
|
case "windows":
|
2018-11-12 22:52:13 +00:00
|
|
|
return bi.LoadBinaryInfoPE(path, entryPoint, &wg)
|
2017-04-06 18:14:01 +00:00
|
|
|
case "darwin":
|
2018-11-12 22:52:13 +00:00
|
|
|
return bi.LoadBinaryInfoMacho(path, entryPoint, &wg)
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
return errors.New("unsupported operating system")
|
|
|
|
}
|
|
|
|
|
2017-06-21 22:40:42 +00:00
|
|
|
// GStructOffset returns the offset of the G
|
|
|
|
// struct in thread local storage.
|
|
|
|
func (bi *BinaryInfo) GStructOffset() uint64 {
|
|
|
|
return bi.gStructOffset
|
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// LastModified returns the last modified time of the binary.
|
2017-04-06 18:14:01 +00:00
|
|
|
func (bi *BinaryInfo) LastModified() time.Time {
|
|
|
|
return bi.lastModified
|
|
|
|
}
|
|
|
|
|
|
|
|
// DwarfReader returns a reader for the dwarf data
|
|
|
|
func (bi *BinaryInfo) DwarfReader() *reader.Reader {
|
|
|
|
return reader.New(bi.dwarf)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Types returns list of types present in the debugged program.
|
|
|
|
func (bi *BinaryInfo) Types() ([]string, error) {
|
|
|
|
types := make([]string, 0, len(bi.types))
|
|
|
|
for k := range bi.types {
|
|
|
|
types = append(types, k)
|
|
|
|
}
|
|
|
|
return types, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// PCToLine converts an instruction address to a file/line/function.
|
2017-09-01 13:30:45 +00:00
|
|
|
func (bi *BinaryInfo) PCToLine(pc uint64) (string, int, *Function) {
|
|
|
|
fn := bi.PCToFunc(pc)
|
|
|
|
if fn == nil {
|
|
|
|
return "", 0, nil
|
|
|
|
}
|
2017-08-18 17:49:29 +00:00
|
|
|
f, ln := fn.cu.lineInfo.PCToLine(fn.Entry, pc)
|
2017-09-01 13:30:45 +00:00
|
|
|
return f, ln, fn
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
|
2017-02-10 14:11:40 +00:00
|
|
|
// LineToPC converts a file:line into a memory address.
|
2017-09-01 13:30:45 +00:00
|
|
|
func (bi *BinaryInfo) LineToPC(filename string, lineno int) (pc uint64, fn *Function, err error) {
|
|
|
|
for _, cu := range bi.compileUnits {
|
|
|
|
if cu.lineInfo.Lookup[filename] != nil {
|
|
|
|
pc = cu.lineInfo.LineToPC(filename, lineno)
|
2018-08-07 01:08:25 +00:00
|
|
|
if pc == 0 {
|
|
|
|
// Check to see if this file:line belongs to the call site
|
|
|
|
// of an inlined function.
|
|
|
|
for _, ifn := range cu.concreteInlinedFns {
|
|
|
|
if strings.Contains(ifn.CallFile, filename) && ifn.CallLine == int64(lineno) {
|
|
|
|
pc = ifn.LowPC
|
|
|
|
fn = ifn.Parent
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-09-01 13:30:45 +00:00
|
|
|
fn = bi.PCToFunc(pc)
|
2018-05-12 00:42:58 +00:00
|
|
|
if fn != nil {
|
|
|
|
return
|
2017-09-01 13:30:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
err = fmt.Errorf("could not find %s:%d", filename, lineno)
|
|
|
|
return
|
2017-02-10 14:11:40 +00:00
|
|
|
}
|
|
|
|
|
proc: support inlining
Go 1.10 added inlined calls to debug_info, this commit adds support
for DW_TAG_inlined_call to delve, both for stack traces (where
inlined calls will appear as normal stack frames) and to correct
the behavior of next, step and stepout.
The calls to Next and Frame of stackIterator continue to work
unchanged and only return real stack frames, after reading each line
appendInlinedCalls is called to unpacked all the inlined calls that
involve the current PC.
The fake stack frames produced by appendInlinedCalls are
distinguished from real stack frames by having the Inlined attribute
set to true. Also their Current and Call locations are treated
differently. The Call location will be changed to represent the
position inside the inlined call, while the Current location will
always reference the real stack frame. This is done because:
* next, step and stepout need to access the debug_info entry of
the real function they are stepping through
* we are already manipulating Call in different ways while Current
is just what we read from the call stack
The strategy remains mostly the same, we disassemble the function
and we set a breakpoint on each instruction corresponding to a
different file:line. The function in question will be the one
corresponding to the first real (i.e. non-inlined) stack frame.
* If the current function contains inlined calls, 'next' will not
set any breakpoints on instructions that belong to inlined calls. We
do not do this for 'step'.
* If we are inside an inlined call that makes other inlined
functions, 'next' will not set any breakpoints that belong to
inlined calls that are children of the current inlined call.
* If the current function is inlined the breakpoint on the return
address won't be set, because inlined frames don't have a return
address.
* The code we use for stepout doesn't work at all if we are inside
an inlined call, instead we call 'next' but instruct it to remove
all PCs belonging to the current inlined call.
2017-11-13 15:54:08 +00:00
|
|
|
// AllPCsForFileLine returns all PC addresses for the given filename:lineno.
|
|
|
|
func (bi *BinaryInfo) AllPCsForFileLine(filename string, lineno int) []uint64 {
|
|
|
|
r := make([]uint64, 0, 1)
|
|
|
|
for _, cu := range bi.compileUnits {
|
|
|
|
if cu.lineInfo.Lookup[filename] != nil {
|
|
|
|
r = append(r, cu.lineInfo.AllPCsForFileLine(filename, lineno)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2017-04-21 06:55:53 +00:00
|
|
|
// PCToFunc returns the function containing the given PC address
|
2017-09-01 13:30:45 +00:00
|
|
|
func (bi *BinaryInfo) PCToFunc(pc uint64) *Function {
|
|
|
|
i := sort.Search(len(bi.Functions), func(i int) bool {
|
|
|
|
fn := bi.Functions[i]
|
|
|
|
return pc <= fn.Entry || (fn.Entry <= pc && pc < fn.End)
|
|
|
|
})
|
|
|
|
if i != len(bi.Functions) {
|
|
|
|
fn := &bi.Functions[i]
|
|
|
|
if fn.Entry <= pc && pc < fn.End {
|
|
|
|
return fn
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2017-04-21 06:55:53 +00:00
|
|
|
}
|
|
|
|
|
2019-03-20 17:32:51 +00:00
|
|
|
// Image represents a loaded library file (shared object on linux, DLL on windows).
|
|
|
|
type Image struct {
|
|
|
|
Path string
|
|
|
|
addr uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddImage adds the specified image to bi.
|
|
|
|
func (bi *BinaryInfo) AddImage(path string, addr uint64) {
|
|
|
|
if !strings.HasPrefix(path, "/") {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, image := range bi.Images {
|
|
|
|
if image.Path == path && image.addr == addr {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//TODO(aarzilli): actually load informations about the image here
|
|
|
|
bi.Images = append(bi.Images, &Image{Path: path, addr: addr})
|
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// Close closes all internal readers.
|
2017-04-06 18:14:01 +00:00
|
|
|
func (bi *BinaryInfo) Close() error {
|
2018-06-20 10:20:35 +00:00
|
|
|
if bi.sepDebugCloser != nil {
|
|
|
|
bi.sepDebugCloser.Close()
|
|
|
|
}
|
2019-01-28 13:23:10 +00:00
|
|
|
if bi.closer != nil {
|
|
|
|
return bi.closer.Close()
|
|
|
|
}
|
|
|
|
return nil
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
|
2017-08-15 06:21:24 +00:00
|
|
|
func (bi *BinaryInfo) setLoadError(fmtstr string, args ...interface{}) {
|
|
|
|
bi.loadErrMu.Lock()
|
|
|
|
bi.loadErr = fmt.Errorf(fmtstr, args...)
|
|
|
|
bi.loadErrMu.Unlock()
|
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// LoadError returns any internal load error.
|
2017-08-15 06:21:24 +00:00
|
|
|
func (bi *BinaryInfo) LoadError() error {
|
|
|
|
return bi.loadErr
|
|
|
|
}
|
|
|
|
|
2017-08-22 14:40:01 +00:00
|
|
|
type nilCloser struct{}
|
|
|
|
|
|
|
|
func (c *nilCloser) Close() error { return nil }
|
|
|
|
|
2017-08-24 07:46:47 +00:00
|
|
|
// LoadFromData creates a new BinaryInfo object using the specified data.
|
|
|
|
// This is used for debugging BinaryInfo, you should use LoadBinary instead.
|
|
|
|
func (bi *BinaryInfo) LoadFromData(dwdata *dwarf.Data, debugFrameBytes, debugLineBytes, debugLocBytes []byte) {
|
2017-08-22 14:40:01 +00:00
|
|
|
bi.closer = (*nilCloser)(nil)
|
2018-06-20 10:20:35 +00:00
|
|
|
bi.sepDebugCloser = (*nilCloser)(nil)
|
2017-08-22 14:40:01 +00:00
|
|
|
bi.dwarf = dwdata
|
|
|
|
|
|
|
|
if debugFrameBytes != nil {
|
2018-05-29 15:01:51 +00:00
|
|
|
bi.frameEntries = frame.Parse(debugFrameBytes, frame.DwarfEndian(debugFrameBytes), bi.staticBase)
|
2017-08-22 14:40:01 +00:00
|
|
|
}
|
|
|
|
|
2017-08-24 07:46:47 +00:00
|
|
|
bi.loclistInit(debugLocBytes)
|
|
|
|
|
2018-04-14 09:04:14 +00:00
|
|
|
bi.loadDebugInfoMaps(debugLineBytes, nil, nil)
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (bi *BinaryInfo) loclistInit(data []byte) {
|
|
|
|
bi.loclist.data = data
|
|
|
|
bi.loclist.ptrSz = bi.Arch.PtrSize()
|
|
|
|
}
|
|
|
|
|
2018-07-28 19:12:07 +00:00
|
|
|
func (bi *BinaryInfo) locationExpr(entry reader.Entry, attr dwarf.Attr, pc uint64) ([]byte, string, error) {
|
2017-08-24 07:46:47 +00:00
|
|
|
a := entry.Val(attr)
|
|
|
|
if a == nil {
|
2018-07-28 19:12:07 +00:00
|
|
|
return nil, "", fmt.Errorf("no location attribute %s", attr)
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
if instr, ok := a.([]byte); ok {
|
2017-11-21 08:50:14 +00:00
|
|
|
var descr bytes.Buffer
|
|
|
|
fmt.Fprintf(&descr, "[block] ")
|
|
|
|
op.PrettyPrint(&descr, instr)
|
2018-07-28 19:12:07 +00:00
|
|
|
return instr, descr.String(), nil
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
off, ok := a.(int64)
|
|
|
|
if !ok {
|
2018-07-28 19:12:07 +00:00
|
|
|
return nil, "", fmt.Errorf("could not interpret location attribute %s", attr)
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
if bi.loclist.data == nil {
|
2018-07-28 19:12:07 +00:00
|
|
|
return nil, "", fmt.Errorf("could not find loclist entry at %#x for address %#x (no debug_loc section found)", off, pc)
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
instr := bi.loclistEntry(off, pc)
|
|
|
|
if instr == nil {
|
2018-07-28 19:12:07 +00:00
|
|
|
return nil, "", fmt.Errorf("could not find loclist entry at %#x for address %#x", off, pc)
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
2017-11-21 08:50:14 +00:00
|
|
|
var descr bytes.Buffer
|
|
|
|
fmt.Fprintf(&descr, "[%#x:%#x] ", off, pc)
|
|
|
|
op.PrettyPrint(&descr, instr)
|
2018-07-28 19:12:07 +00:00
|
|
|
return instr, descr.String(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Location returns the location described by attribute attr of entry.
|
|
|
|
// This will either be an int64 address or a slice of Pieces for locations
|
|
|
|
// that don't correspond to a single memory address (registers, composite
|
|
|
|
// locations).
|
|
|
|
func (bi *BinaryInfo) Location(entry reader.Entry, attr dwarf.Attr, pc uint64, regs op.DwarfRegisters) (int64, []op.Piece, string, error) {
|
|
|
|
instr, descr, err := bi.locationExpr(entry, attr, pc)
|
|
|
|
if err != nil {
|
|
|
|
return 0, nil, "", err
|
|
|
|
}
|
2017-11-21 08:50:14 +00:00
|
|
|
addr, pieces, err := op.ExecuteStackProgram(regs, instr)
|
2018-07-28 19:12:07 +00:00
|
|
|
return addr, pieces, descr, err
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// loclistEntry returns the loclist entry in the loclist starting at off,
|
|
|
|
// for address pc.
|
|
|
|
func (bi *BinaryInfo) loclistEntry(off int64, pc uint64) []byte {
|
|
|
|
var base uint64
|
|
|
|
if cu := bi.findCompileUnit(pc); cu != nil {
|
2018-11-06 22:34:26 +00:00
|
|
|
base = cu.lowPC
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bi.loclist.Seek(int(off))
|
|
|
|
var e loclistEntry
|
|
|
|
for bi.loclist.Next(&e) {
|
|
|
|
if e.BaseAddressSelection() {
|
|
|
|
base = e.highpc
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if pc >= e.lowpc+base && pc < e.highpc+base {
|
|
|
|
return e.instr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// findCompileUnit returns the compile unit containing address pc.
|
|
|
|
func (bi *BinaryInfo) findCompileUnit(pc uint64) *compileUnit {
|
|
|
|
for _, cu := range bi.compileUnits {
|
2018-11-06 22:34:26 +00:00
|
|
|
for _, rng := range cu.ranges {
|
2018-05-29 15:01:51 +00:00
|
|
|
if pc >= rng[0] && pc < rng[1] {
|
|
|
|
return cu
|
|
|
|
}
|
2018-08-17 06:17:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bi *BinaryInfo) findCompileUnitForOffset(off dwarf.Offset) *compileUnit {
|
|
|
|
for _, cu := range bi.compileUnits {
|
|
|
|
if off >= cu.startOffset && off < cu.endOffset {
|
|
|
|
return cu
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2017-08-22 14:40:01 +00:00
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// Producer returns the value of DW_AT_producer.
|
2018-04-14 09:04:14 +00:00
|
|
|
func (bi *BinaryInfo) Producer() string {
|
|
|
|
for _, cu := range bi.compileUnits {
|
|
|
|
if cu.isgo && cu.producer != "" {
|
|
|
|
return cu.producer
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2018-05-04 17:31:45 +00:00
|
|
|
// Type returns the Dwarf type entry at `offset`.
|
|
|
|
func (bi *BinaryInfo) Type(offset dwarf.Offset) (godwarf.Type, error) {
|
|
|
|
return godwarf.ReadType(bi.dwarf, offset, bi.typeCache)
|
|
|
|
}
|
|
|
|
|
2017-04-06 18:14:01 +00:00
|
|
|
// ELF ///////////////////////////////////////////////////////////////
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// ErrNoBuildIDNote is used in openSeparateDebugInfo to signal there's no
|
2018-06-20 10:20:35 +00:00
|
|
|
// build-id note on the binary, so LoadBinaryInfoElf will return
|
|
|
|
// the error message coming from elfFile.DWARF() instead.
|
2018-08-31 18:08:18 +00:00
|
|
|
type ErrNoBuildIDNote struct{}
|
2018-06-20 10:20:35 +00:00
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
func (e *ErrNoBuildIDNote) Error() string {
|
2018-06-20 10:20:35 +00:00
|
|
|
return "can't find build-id note on binary"
|
|
|
|
}
|
|
|
|
|
|
|
|
// openSeparateDebugInfo searches for a file containing the separate
|
|
|
|
// debug info for the binary using the "build ID" method as described
|
|
|
|
// in GDB's documentation [1], and if found returns two handles, one
|
|
|
|
// for the bare file, and another for its corresponding elf.File.
|
|
|
|
// [1] https://sourceware.org/gdb/onlinedocs/gdb/Separate-Debug-Files.html
|
2018-11-07 22:21:35 +00:00
|
|
|
//
|
|
|
|
// Alternatively, if the debug file cannot be found be the build-id, Delve
|
|
|
|
// will look in directories specified by the debug-info-directories config value.
|
|
|
|
func (bi *BinaryInfo) openSeparateDebugInfo(exe *elf.File, debugInfoDirectories []string) (*os.File, *elf.File, error) {
|
|
|
|
var debugFilePath string
|
|
|
|
for _, dir := range debugInfoDirectories {
|
|
|
|
var potentialDebugFilePath string
|
|
|
|
if strings.Contains(dir, "build-id") {
|
|
|
|
desc1, desc2, err := parseBuildID(exe)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
potentialDebugFilePath = fmt.Sprintf("%s/%s/%s.debug", dir, desc1, desc2)
|
|
|
|
} else {
|
|
|
|
potentialDebugFilePath = fmt.Sprintf("%s/%s.debug", dir, filepath.Base(bi.Path))
|
|
|
|
}
|
|
|
|
_, err := os.Stat(potentialDebugFilePath)
|
|
|
|
if err == nil {
|
|
|
|
debugFilePath = potentialDebugFilePath
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if debugFilePath == "" {
|
|
|
|
return nil, nil, ErrNoDebugInfoFound
|
|
|
|
}
|
|
|
|
sepFile, err := os.OpenFile(debugFilePath, 0, os.ModePerm)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.New("can't open separate debug file: " + err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
elfFile, err := elf.NewFile(sepFile)
|
|
|
|
if err != nil {
|
|
|
|
sepFile.Close()
|
|
|
|
return nil, nil, fmt.Errorf("can't open separate debug file %q: %v", debugFilePath, err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
if elfFile.Machine != elf.EM_X86_64 {
|
|
|
|
sepFile.Close()
|
|
|
|
return nil, nil, fmt.Errorf("can't open separate debug file %q: %v", debugFilePath, ErrUnsupportedLinuxArch.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return sepFile, elfFile, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseBuildID(exe *elf.File) (string, string, error) {
|
2018-06-20 10:20:35 +00:00
|
|
|
buildid := exe.Section(".note.gnu.build-id")
|
|
|
|
if buildid == nil {
|
2018-11-07 22:21:35 +00:00
|
|
|
return "", "", &ErrNoBuildIDNote{}
|
2018-06-20 10:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
br := buildid.Open()
|
2018-08-31 18:08:18 +00:00
|
|
|
bh := new(buildIDHeader)
|
2018-06-20 10:20:35 +00:00
|
|
|
if err := binary.Read(br, binary.LittleEndian, bh); err != nil {
|
2018-11-07 22:21:35 +00:00
|
|
|
return "", "", errors.New("can't read build-id header: " + err.Error())
|
2018-06-20 10:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
name := make([]byte, bh.Namesz)
|
|
|
|
if err := binary.Read(br, binary.LittleEndian, name); err != nil {
|
2018-11-07 22:21:35 +00:00
|
|
|
return "", "", errors.New("can't read build-id name: " + err.Error())
|
2018-06-20 10:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if strings.TrimSpace(string(name)) != "GNU\x00" {
|
2018-11-07 22:21:35 +00:00
|
|
|
return "", "", errors.New("invalid build-id signature")
|
2018-06-20 10:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
descBinary := make([]byte, bh.Descsz)
|
|
|
|
if err := binary.Read(br, binary.LittleEndian, descBinary); err != nil {
|
2018-11-07 22:21:35 +00:00
|
|
|
return "", "", errors.New("can't read build-id desc: " + err.Error())
|
2018-06-20 10:20:35 +00:00
|
|
|
}
|
|
|
|
desc := hex.EncodeToString(descBinary)
|
2018-11-07 22:21:35 +00:00
|
|
|
return desc[:2], desc[2:], nil
|
2018-06-20 10:20:35 +00:00
|
|
|
}
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// LoadBinaryInfoElf specifically loads information from an ELF binary.
|
2018-11-07 22:21:35 +00:00
|
|
|
func (bi *BinaryInfo) LoadBinaryInfoElf(path string, entryPoint uint64, debugInfoDirectories []string, wg *sync.WaitGroup) error {
|
2017-04-06 18:14:01 +00:00
|
|
|
exe, err := os.OpenFile(path, 0, os.ModePerm)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
bi.closer = exe
|
|
|
|
elfFile, err := elf.NewFile(exe)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if elfFile.Machine != elf.EM_X86_64 {
|
2018-08-31 18:08:18 +00:00
|
|
|
return ErrUnsupportedLinuxArch
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
2018-05-29 15:01:51 +00:00
|
|
|
|
|
|
|
if entryPoint != 0 {
|
|
|
|
bi.staticBase = entryPoint - elfFile.Entry
|
|
|
|
} else {
|
|
|
|
if elfFile.Type == elf.ET_DYN {
|
|
|
|
return ErrCouldNotDetermineRelocation
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-20 17:32:51 +00:00
|
|
|
if dynsec := elfFile.Section(".dynamic"); dynsec != nil {
|
|
|
|
bi.ElfDynamicSection.Addr = dynsec.Addr + bi.staticBase
|
|
|
|
bi.ElfDynamicSection.Size = dynsec.Size
|
|
|
|
}
|
|
|
|
|
2018-06-20 10:20:35 +00:00
|
|
|
dwarfFile := elfFile
|
2018-05-29 15:01:51 +00:00
|
|
|
|
2017-04-06 18:14:01 +00:00
|
|
|
bi.dwarf, err = elfFile.DWARF()
|
|
|
|
if err != nil {
|
2018-06-20 10:20:35 +00:00
|
|
|
var sepFile *os.File
|
|
|
|
var serr error
|
2018-11-07 22:21:35 +00:00
|
|
|
sepFile, dwarfFile, serr = bi.openSeparateDebugInfo(elfFile, debugInfoDirectories)
|
2018-06-20 10:20:35 +00:00
|
|
|
if serr != nil {
|
|
|
|
return serr
|
|
|
|
}
|
|
|
|
bi.sepDebugCloser = sepFile
|
|
|
|
bi.dwarf, err = dwarfFile.DWARF()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
|
2018-02-13 17:20:45 +00:00
|
|
|
bi.dwarfReader = bi.dwarf.Reader()
|
|
|
|
|
2018-06-18 08:05:21 +00:00
|
|
|
debugLineBytes, err := godwarf.GetDebugSectionElf(dwarfFile, "line")
|
2017-09-01 13:30:45 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-06-18 08:05:21 +00:00
|
|
|
debugLocBytes, _ := godwarf.GetDebugSectionElf(dwarfFile, "loc")
|
|
|
|
bi.loclistInit(debugLocBytes)
|
2017-09-01 13:30:45 +00:00
|
|
|
|
|
|
|
wg.Add(3)
|
2018-06-20 10:20:35 +00:00
|
|
|
go bi.parseDebugFrameElf(dwarfFile, wg)
|
2018-04-14 09:04:14 +00:00
|
|
|
go bi.loadDebugInfoMaps(debugLineBytes, wg, nil)
|
2018-06-20 10:20:35 +00:00
|
|
|
go bi.setGStructOffsetElf(dwarfFile, wg)
|
2017-04-06 18:14:01 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bi *BinaryInfo) parseDebugFrameElf(exe *elf.File, wg *sync.WaitGroup) {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2018-06-18 08:05:21 +00:00
|
|
|
debugFrameData, err := godwarf.GetDebugSectionElf(exe, "frame")
|
|
|
|
if err != nil {
|
|
|
|
bi.setLoadError("could not get .debug_frame section: %v", err)
|
2017-08-15 06:21:24 +00:00
|
|
|
return
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
2018-06-18 08:05:21 +00:00
|
|
|
debugInfoData, err := godwarf.GetDebugSectionElf(exe, "info")
|
|
|
|
if err != nil {
|
|
|
|
bi.setLoadError("could not get .debug_info section: %v", err)
|
|
|
|
return
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
|
2018-05-29 15:01:51 +00:00
|
|
|
bi.frameEntries = frame.Parse(debugFrameData, frame.DwarfEndian(debugInfoData), bi.staticBase)
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|
|
|
|
|
2017-06-21 22:40:42 +00:00
|
|
|
func (bi *BinaryInfo) setGStructOffsetElf(exe *elf.File, wg *sync.WaitGroup) {
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
// This is a bit arcane. Essentially:
|
|
|
|
// - If the program is pure Go, it can do whatever it wants, and puts the G
|
|
|
|
// pointer at %fs-8.
|
|
|
|
// - Otherwise, Go asks the external linker to place the G pointer by
|
|
|
|
// emitting runtime.tlsg, a TLS symbol, which is relocated to the chosen
|
|
|
|
// offset in libc's TLS block.
|
|
|
|
symbols, err := exe.Symbols()
|
|
|
|
if err != nil {
|
2017-08-15 06:21:24 +00:00
|
|
|
bi.setLoadError("could not parse ELF symbols: %v", err)
|
|
|
|
return
|
2017-06-21 22:40:42 +00:00
|
|
|
}
|
|
|
|
var tlsg *elf.Symbol
|
|
|
|
for _, symbol := range symbols {
|
|
|
|
if symbol.Name == "runtime.tlsg" {
|
|
|
|
s := symbol
|
|
|
|
tlsg = &s
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if tlsg == nil {
|
|
|
|
bi.gStructOffset = ^uint64(8) + 1 // -8
|
|
|
|
return
|
|
|
|
}
|
|
|
|
var tls *elf.Prog
|
|
|
|
for _, prog := range exe.Progs {
|
|
|
|
if prog.Type == elf.PT_TLS {
|
|
|
|
tls = prog
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2019-02-26 17:17:05 +00:00
|
|
|
if tls == nil {
|
|
|
|
bi.gStructOffset = ^uint64(8) + 1 // -8
|
|
|
|
return
|
|
|
|
}
|
2018-11-28 17:26:01 +00:00
|
|
|
memsz := tls.Memsz
|
|
|
|
|
|
|
|
memsz = (memsz + uint64(bi.Arch.PtrSize()) - 1) & ^uint64(bi.Arch.PtrSize()-1) // align to pointer-sized-boundary
|
2017-06-21 22:40:42 +00:00
|
|
|
// The TLS register points to the end of the TLS block, which is
|
|
|
|
// tls.Memsz long. runtime.tlsg is an offset from the beginning of that block.
|
2018-11-28 17:26:01 +00:00
|
|
|
bi.gStructOffset = ^(memsz) + 1 + tlsg.Value // -tls.Memsz + tlsg.Value
|
2017-06-21 22:40:42 +00:00
|
|
|
}
|
|
|
|
|
2017-04-06 18:14:01 +00:00
|
|
|
// PE ////////////////////////////////////////////////////////////////
|
|
|
|
|
2018-05-29 15:01:51 +00:00
|
|
|
const _IMAGE_DLLCHARACTERISTICS_DYNAMIC_BASE = 0x0040
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// LoadBinaryInfoPE specifically loads information from a PE binary.
|
2018-05-29 15:01:51 +00:00
|
|
|
func (bi *BinaryInfo) LoadBinaryInfoPE(path string, entryPoint uint64, wg *sync.WaitGroup) error {
|
2017-04-06 18:14:01 +00:00
|
|
|
peFile, closer, err := openExecutablePathPE(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
bi.closer = closer
|
|
|
|
if peFile.Machine != pe.IMAGE_FILE_MACHINE_AMD64 {
|
2018-08-31 18:08:18 +00:00
|
|
|
return ErrUnsupportedWindowsArch
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
2017-05-29 13:20:01 +00:00
|
|
|
bi.dwarf, err = peFile.DWARF()
|
2017-04-06 18:14:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-05-29 15:01:51 +00:00
|
|
|
//TODO(aarzilli): actually test this when Go supports PIE buildmode on Windows.
|
|
|
|
opth := peFile.OptionalHeader.(*pe.OptionalHeader64)
|
|
|
|
if entryPoint != 0 {
|
|
|
|
bi.staticBase = entryPoint - opth.ImageBase
|
|
|
|
} else {
|
|
|
|
if opth.DllCharacteristics&_IMAGE_DLLCHARACTERISTICS_DYNAMIC_BASE != 0 {
|
|
|
|
return ErrCouldNotDetermineRelocation
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-13 17:20:45 +00:00
|
|
|
bi.dwarfReader = bi.dwarf.Reader()
|
|
|
|
|
2018-06-18 08:05:21 +00:00
|
|
|
debugLineBytes, err := godwarf.GetDebugSectionPE(peFile, "line")
|
2017-09-01 13:30:45 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-06-18 08:05:21 +00:00
|
|
|
debugLocBytes, _ := godwarf.GetDebugSectionPE(peFile, "loc")
|
|
|
|
bi.loclistInit(debugLocBytes)
|
2017-09-01 13:30:45 +00:00
|
|
|
|
|
|
|
wg.Add(2)
|
2017-04-06 18:14:01 +00:00
|
|
|
go bi.parseDebugFramePE(peFile, wg)
|
2018-04-14 09:04:14 +00:00
|
|
|
go bi.loadDebugInfoMaps(debugLineBytes, wg, nil)
|
2017-06-21 22:40:42 +00:00
|
|
|
|
|
|
|
// Use ArbitraryUserPointer (0x28) as pointer to pointer
|
|
|
|
// to G struct per:
|
|
|
|
// https://golang.org/src/runtime/cgo/gcc_windows_amd64.c
|
|
|
|
|
|
|
|
bi.gStructOffset = 0x28
|
2017-04-06 18:14:01 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func openExecutablePathPE(path string) (*pe.File, io.Closer, error) {
|
|
|
|
f, err := os.OpenFile(path, 0, os.ModePerm)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
peFile, err := pe.NewFile(f)
|
|
|
|
if err != nil {
|
|
|
|
f.Close()
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
return peFile, f, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bi *BinaryInfo) parseDebugFramePE(exe *pe.File, wg *sync.WaitGroup) {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2018-06-18 08:05:21 +00:00
|
|
|
debugFrameBytes, err := godwarf.GetDebugSectionPE(exe, "frame")
|
|
|
|
if err != nil {
|
|
|
|
bi.setLoadError("could not get .debug_frame section: %v", err)
|
2017-08-15 06:21:24 +00:00
|
|
|
return
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
2018-06-18 08:05:21 +00:00
|
|
|
debugInfoBytes, err := godwarf.GetDebugSectionPE(exe, "info")
|
|
|
|
if err != nil {
|
|
|
|
bi.setLoadError("could not get .debug_info section: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-05-29 15:01:51 +00:00
|
|
|
bi.frameEntries = frame.Parse(debugFrameBytes, frame.DwarfEndian(debugInfoBytes), bi.staticBase)
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Borrowed from https://golang.org/src/cmd/internal/objfile/pe.go
|
|
|
|
func findPESymbol(f *pe.File, name string) (*pe.Symbol, error) {
|
|
|
|
for _, s := range f.Symbols {
|
|
|
|
if s.Name != name {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if s.SectionNumber <= 0 {
|
|
|
|
return nil, fmt.Errorf("symbol %s: invalid section number %d", name, s.SectionNumber)
|
|
|
|
}
|
|
|
|
if len(f.Sections) < int(s.SectionNumber) {
|
|
|
|
return nil, fmt.Errorf("symbol %s: section number %d is larger than max %d", name, s.SectionNumber, len(f.Sections))
|
|
|
|
}
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("no %s symbol found", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
// MACH-O ////////////////////////////////////////////////////////////
|
|
|
|
|
2018-08-31 18:08:18 +00:00
|
|
|
// LoadBinaryInfoMacho specifically loads information from a Mach-O binary.
|
2018-05-29 15:01:51 +00:00
|
|
|
func (bi *BinaryInfo) LoadBinaryInfoMacho(path string, entryPoint uint64, wg *sync.WaitGroup) error {
|
2017-04-06 18:14:01 +00:00
|
|
|
exe, err := macho.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
bi.closer = exe
|
|
|
|
if exe.Cpu != macho.CpuAmd64 {
|
2018-08-31 18:08:18 +00:00
|
|
|
return ErrUnsupportedDarwinArch
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
|
|
|
bi.dwarf, err = exe.DWARF()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-02-13 17:20:45 +00:00
|
|
|
bi.dwarfReader = bi.dwarf.Reader()
|
|
|
|
|
2018-06-18 08:05:21 +00:00
|
|
|
debugLineBytes, err := godwarf.GetDebugSectionMacho(exe, "line")
|
2017-09-01 13:30:45 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-06-18 08:05:21 +00:00
|
|
|
debugLocBytes, _ := godwarf.GetDebugSectionMacho(exe, "loc")
|
|
|
|
bi.loclistInit(debugLocBytes)
|
2017-09-01 13:30:45 +00:00
|
|
|
|
|
|
|
wg.Add(2)
|
2017-04-06 18:14:01 +00:00
|
|
|
go bi.parseDebugFrameMacho(exe, wg)
|
2018-04-14 09:04:14 +00:00
|
|
|
go bi.loadDebugInfoMaps(debugLineBytes, wg, bi.setGStructOffsetMacho)
|
2017-04-06 18:14:01 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-04-14 09:04:14 +00:00
|
|
|
func (bi *BinaryInfo) setGStructOffsetMacho() {
|
|
|
|
// In go1.11 it's 0x30, before 0x8a0, see:
|
|
|
|
// https://github.com/golang/go/issues/23617
|
|
|
|
// and go commit b3a854c733257c5249c3435ffcee194f8439676a
|
|
|
|
producer := bi.Producer()
|
|
|
|
if producer != "" && goversion.ProducerAfterOrEqual(producer, 1, 11) {
|
|
|
|
bi.gStructOffset = 0x30
|
|
|
|
return
|
|
|
|
}
|
|
|
|
bi.gStructOffset = 0x8a0
|
|
|
|
}
|
|
|
|
|
2017-04-06 18:14:01 +00:00
|
|
|
func (bi *BinaryInfo) parseDebugFrameMacho(exe *macho.File, wg *sync.WaitGroup) {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2018-06-18 08:05:21 +00:00
|
|
|
debugFrameBytes, err := godwarf.GetDebugSectionMacho(exe, "frame")
|
|
|
|
if err != nil {
|
|
|
|
bi.setLoadError("could not get __debug_frame section: %v", err)
|
2017-08-15 06:21:24 +00:00
|
|
|
return
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
2018-06-18 08:05:21 +00:00
|
|
|
debugInfoBytes, err := godwarf.GetDebugSectionMacho(exe, "info")
|
|
|
|
if err != nil {
|
|
|
|
bi.setLoadError("could not get .debug_info section: %v", err)
|
|
|
|
return
|
2017-04-06 18:14:01 +00:00
|
|
|
}
|
2017-08-24 07:46:47 +00:00
|
|
|
|
2018-05-29 15:01:51 +00:00
|
|
|
bi.frameEntries = frame.Parse(debugFrameBytes, frame.DwarfEndian(debugInfoBytes), bi.staticBase)
|
2017-08-24 07:46:47 +00:00
|
|
|
}
|