mirror of
https://github.com/go-delve/delve.git
synced 2025-11-03 05:47:34 +08:00
Initial Dwarf .debug_frame parser
Still needs to handle the following: * Correctly adapt to augmentation strings * Correctly parse instructions
This commit is contained in:
276
dwarf/frame.go
Normal file
276
dwarf/frame.go
Normal file
@ -0,0 +1,276 @@
|
|||||||
|
// Package frame contains data structures and
|
||||||
|
// related functions for parsing and searching
|
||||||
|
// through Dwarf .debug_frame data.
|
||||||
|
package frame
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"unicode/utf8"
|
||||||
|
)
|
||||||
|
|
||||||
|
type parsefunc func(*bytes.Buffer, CommonEntries, uint32) (parsefunc, CommonEntries, uint32)
|
||||||
|
|
||||||
|
type CommonEntries []*CommonInformationEntry
|
||||||
|
|
||||||
|
// Represents a Common Information Entry in
|
||||||
|
// the Dwarf .debug_frame section.
|
||||||
|
type CommonInformationEntry struct {
|
||||||
|
Length uint32
|
||||||
|
CIE_id uint32
|
||||||
|
Version uint8
|
||||||
|
Augmentation string
|
||||||
|
CodeAlignmentFactor uint64
|
||||||
|
DataAlignmentFactor uint64
|
||||||
|
ReturnAddressRegister byte
|
||||||
|
InitialInstructions []byte
|
||||||
|
FrameDescriptorEntries []*FrameDescriptorEntry
|
||||||
|
}
|
||||||
|
|
||||||
|
// Represents a Frame Descriptor Entry in the
|
||||||
|
// Dwarf .debug_frame section.
|
||||||
|
type FrameDescriptorEntry struct {
|
||||||
|
Length uint32
|
||||||
|
CIE_pointer *CommonInformationEntry
|
||||||
|
InitialLocation uint64
|
||||||
|
AddressRange uint64
|
||||||
|
Instructions []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
DW_CFA_advance_loc = (0x1 << 6) // High 2 bits: 0x1, low 6: delta
|
||||||
|
DW_CFA_offset = (0x2 << 6) // High 2 bits: 0x2, low 6: register
|
||||||
|
DW_CFA_restore = (0x3 << 6) // High 2 bits: 0x3, low 6: register
|
||||||
|
DW_CFA_nop = 0x0 // No ops
|
||||||
|
DW_CFA_set_loc = 0x1 // op1: address
|
||||||
|
DW_CFA_advance_loc1 = iota // op1: 1-bytes delta
|
||||||
|
DW_CFA_advance_loc2 // op1: 2-byte delta
|
||||||
|
DW_CFA_advance_loc4 // op1: 4-byte delta
|
||||||
|
DW_CFA_offset_extended // op1: ULEB128 register, op2: ULEB128 offset
|
||||||
|
DW_CFA_restore_extended // op1: ULEB128 register
|
||||||
|
DW_CFA_undefined // op1: ULEB128 register
|
||||||
|
DW_CFA_same_value // op1: ULEB128 register
|
||||||
|
DW_CFA_register // op1: ULEB128 register, op2: ULEB128 register
|
||||||
|
DW_CFA_remember_state // No ops
|
||||||
|
DW_CFA_restore_state // No ops
|
||||||
|
DW_CFA_def_cfa // op1: ULEB128 register, op2: ULEB128 offset
|
||||||
|
DW_CFA_def_cfa_register // op1: ULEB128 register
|
||||||
|
DW_CFA_def_cfa_offset // op1: ULEB128 offset
|
||||||
|
DW_CFA_def_cfa_expression // op1: BLOCK
|
||||||
|
DW_CFA_expression // op1: ULEB128 register, op2: BLOCK
|
||||||
|
DW_CFA_offset_extended_sf // op1: ULEB128 register, op2: SLEB128 offset
|
||||||
|
DW_CFA_def_cfa_sf // op1: ULEB128 register, op2: SLEB128 offset
|
||||||
|
DW_CFA_def_cfa_offset_sf // op1: SLEB128 offset
|
||||||
|
DW_CFA_val_offset // op1: ULEB128, op2: ULEB128
|
||||||
|
DW_CFA_val_offset_sf // op1: ULEB128, op2: SLEB128
|
||||||
|
DW_CFA_val_expression // op1: ULEB128, op2: BLOCK
|
||||||
|
DW_CFA_lo_user = 0x1c // op1: BLOCK
|
||||||
|
DW_CFA_hi_user = 0x3f // op1: ULEB128 register, op2: BLOCK
|
||||||
|
)
|
||||||
|
|
||||||
|
// Parse take in data (a byte slice) and returns a slice of
|
||||||
|
// CommonInformationEntry structures. Each CommonInformationEntry
|
||||||
|
// has a slice of FrameDescriptorEntry structures.
|
||||||
|
func Parse(data []byte) CommonEntries {
|
||||||
|
var (
|
||||||
|
length uint32
|
||||||
|
entries CommonEntries
|
||||||
|
reader = bytes.NewBuffer(data)
|
||||||
|
)
|
||||||
|
|
||||||
|
for fn := parseLength; reader.Len() != 0; {
|
||||||
|
fn, entries, length = fn(reader, entries, length)
|
||||||
|
}
|
||||||
|
|
||||||
|
return entries
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeLEB128 decodes a Little Endian Base 128
|
||||||
|
// represented number.
|
||||||
|
func DecodeLEB128(reader *bytes.Buffer) (uint64, uint32) {
|
||||||
|
var (
|
||||||
|
result uint64
|
||||||
|
shift uint64
|
||||||
|
length uint32
|
||||||
|
)
|
||||||
|
|
||||||
|
for {
|
||||||
|
b, err := reader.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic("Could not parse LEB128 value")
|
||||||
|
}
|
||||||
|
length++
|
||||||
|
|
||||||
|
result |= uint64((uint(b) & 0x7f) << shift)
|
||||||
|
|
||||||
|
// If high order bit is 1.
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
shift += 7
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, length
|
||||||
|
}
|
||||||
|
|
||||||
|
func cieEntry(data []byte) bool {
|
||||||
|
return bytes.Equal(data, []byte{0xff, 0xff, 0xff, 0xff})
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseLength(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
binary.Read(reader, binary.LittleEndian, &length)
|
||||||
|
|
||||||
|
if cieEntry(reader.Bytes()[0:4]) {
|
||||||
|
return parseCIEID, append(entries, &CommonInformationEntry{Length: length}), length
|
||||||
|
}
|
||||||
|
|
||||||
|
entry := entries[len(entries)-1]
|
||||||
|
entry.FrameDescriptorEntries = append(entry.FrameDescriptorEntries, &FrameDescriptorEntry{Length: length, CIE_pointer: entry})
|
||||||
|
|
||||||
|
return parseInitialLocation, entries, length
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseInitialLocation(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
var (
|
||||||
|
frameEntries = entries[len(entries)-1].FrameDescriptorEntries
|
||||||
|
frame = frameEntries[len(frameEntries)-1]
|
||||||
|
)
|
||||||
|
|
||||||
|
binary.Read(reader, binary.LittleEndian, &frame.InitialLocation)
|
||||||
|
|
||||||
|
return parseAddressRange, entries, length - 4
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseAddressRange(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
var (
|
||||||
|
frameEntries = entries[len(entries)-1].FrameDescriptorEntries
|
||||||
|
frame = frameEntries[len(frameEntries)-1]
|
||||||
|
)
|
||||||
|
|
||||||
|
binary.Read(reader, binary.LittleEndian, &frame.AddressRange)
|
||||||
|
|
||||||
|
return parseFrameInstructions, entries, length - 4
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseFrameInstructions(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
var (
|
||||||
|
// The rest of this entry consists of the instructions
|
||||||
|
// so we can just grab all of the data from the buffer
|
||||||
|
// cursor to length.
|
||||||
|
buf = make([]byte, length)
|
||||||
|
frameEntries = entries[len(entries)-1].FrameDescriptorEntries
|
||||||
|
frame = frameEntries[len(frameEntries)-1]
|
||||||
|
)
|
||||||
|
|
||||||
|
binary.Read(reader, binary.LittleEndian, &buf)
|
||||||
|
frame.Instructions = buf
|
||||||
|
|
||||||
|
return parseLength, entries, 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseCIEID(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
var entry = entries[len(entries)-1]
|
||||||
|
|
||||||
|
binary.Read(reader, binary.LittleEndian, &entry.CIE_id)
|
||||||
|
|
||||||
|
return parseVersion, entries, length - 4
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseVersion(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
entry := entries[len(entries)-1]
|
||||||
|
|
||||||
|
binary.Read(reader, binary.LittleEndian, &entry.Version)
|
||||||
|
|
||||||
|
return parseAugmentation, entries, length - 1
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseAugmentation(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
var (
|
||||||
|
entry = entries[len(entries)-1]
|
||||||
|
str, c = parseString(reader)
|
||||||
|
)
|
||||||
|
|
||||||
|
entry.Augmentation = str
|
||||||
|
return parseCodeAlignmentFactor, entries, length - c
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseCodeAlignmentFactor(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
var (
|
||||||
|
entry = entries[len(entries)-1]
|
||||||
|
caf, c = DecodeLEB128(reader)
|
||||||
|
)
|
||||||
|
|
||||||
|
entry.CodeAlignmentFactor = caf
|
||||||
|
|
||||||
|
return parseDataAlignmentFactor, entries, length - c
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseDataAlignmentFactor(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
var (
|
||||||
|
entry = entries[len(entries)-1]
|
||||||
|
daf, c = DecodeLEB128(reader)
|
||||||
|
)
|
||||||
|
|
||||||
|
entry.DataAlignmentFactor = daf
|
||||||
|
|
||||||
|
return parseReturnAddressRegister, entries, length - c
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseReturnAddressRegister(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
entry := entries[len(entries)-1]
|
||||||
|
|
||||||
|
binary.Read(reader, binary.LittleEndian, &entry.ReturnAddressRegister)
|
||||||
|
|
||||||
|
return parseInitialInstructions, entries, length - 1
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseInitialInstructions(reader *bytes.Buffer, entries CommonEntries, length uint32) (parsefunc, CommonEntries, uint32) {
|
||||||
|
var (
|
||||||
|
// The rest of this entry consists of the instructions
|
||||||
|
// so we can just grab all of the data from the buffer
|
||||||
|
// cursor to length.
|
||||||
|
buf = make([]byte, length)
|
||||||
|
entry = entries[len(entries)-1]
|
||||||
|
)
|
||||||
|
|
||||||
|
binary.Read(reader, binary.LittleEndian, &buf)
|
||||||
|
entry.InitialInstructions = buf
|
||||||
|
|
||||||
|
return parseLength, entries, 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseString(data *bytes.Buffer) (string, uint32) {
|
||||||
|
var (
|
||||||
|
size uint32
|
||||||
|
str []rune
|
||||||
|
strb []byte
|
||||||
|
)
|
||||||
|
|
||||||
|
for {
|
||||||
|
b, err := data.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic("parseString(): Could not read byte")
|
||||||
|
}
|
||||||
|
size++
|
||||||
|
|
||||||
|
if b == 0x0 {
|
||||||
|
if size == 1 {
|
||||||
|
return "", size
|
||||||
|
}
|
||||||
|
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
strb = append(strb, b)
|
||||||
|
|
||||||
|
if utf8.FullRune(strb) {
|
||||||
|
r, _ := utf8.DecodeRune(strb)
|
||||||
|
str = append(str, r)
|
||||||
|
size++
|
||||||
|
strb = strb[0:0]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return string(str), size
|
||||||
|
}
|
||||||
98
dwarf/frame_test.go
Normal file
98
dwarf/frame_test.go
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
package frame
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"debug/elf"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func grabDebugFrameSection(fp string, t *testing.T) []byte {
|
||||||
|
p, err := filepath.Abs("../fixtures/testprog")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := os.Open(p)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ef, err := elf.NewFile(f)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := ef.Section(".debug_frame").Data()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return data
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDecodeLEB128(t *testing.T) {
|
||||||
|
var leb128 = bytes.NewBuffer([]byte{0xE5, 0x8E, 0x26})
|
||||||
|
|
||||||
|
n, c := DecodeLEB128(leb128)
|
||||||
|
if n != 624485 {
|
||||||
|
t.Fatal("Number was not decoded properly, got: ", n, c)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseString(t *testing.T) {
|
||||||
|
bstr := bytes.NewBuffer([]byte{'h', 'i', 0x0, 0xFF, 0xCC})
|
||||||
|
str, _ := parseString(bstr)
|
||||||
|
|
||||||
|
if str != "hi" {
|
||||||
|
t.Fatal("String was not parsed correctly")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParse(t *testing.T) {
|
||||||
|
data := grabDebugFrameSection("../fixtures/testprog", t)
|
||||||
|
ce := Parse(data)[0]
|
||||||
|
|
||||||
|
if ce.Length != 16 {
|
||||||
|
t.Fatal("Length was not parsed correctly")
|
||||||
|
}
|
||||||
|
|
||||||
|
if ce.CIE_id != 0xffffffff {
|
||||||
|
t.Fatal("CIE id was not parsed correctly")
|
||||||
|
}
|
||||||
|
|
||||||
|
if ce.Version != 0x3 {
|
||||||
|
t.Fatalf("Version was not parsed correctly expected %#v got %#v, data was %#v", 0x3, ce.Version, data[0:40])
|
||||||
|
}
|
||||||
|
|
||||||
|
if ce.Augmentation != "" {
|
||||||
|
t.Fatal("Augmentation was not parsed correctly")
|
||||||
|
}
|
||||||
|
|
||||||
|
if ce.CodeAlignmentFactor != 0x1 {
|
||||||
|
t.Fatal("Code Alignment Factor was not parsed correctly")
|
||||||
|
}
|
||||||
|
|
||||||
|
if ce.DataAlignmentFactor != 0x7c {
|
||||||
|
t.Fatal("Data Alignment Factor was not parsed correctly")
|
||||||
|
}
|
||||||
|
|
||||||
|
fe := ce.FrameDescriptorEntries[0]
|
||||||
|
|
||||||
|
if fe.Length != 32 {
|
||||||
|
t.Fatal("Length was not parsed correctly")
|
||||||
|
}
|
||||||
|
|
||||||
|
if fe.CIE_pointer != ce {
|
||||||
|
t.Fatal("Frame entry does not point to parent CIE")
|
||||||
|
}
|
||||||
|
|
||||||
|
if fe.InitialLocation != 0x400c0000000000 {
|
||||||
|
t.Fatal("Initial location not parsed correctly")
|
||||||
|
}
|
||||||
|
|
||||||
|
if fe.AddressRange != 0xbc00000000 {
|
||||||
|
t.Fatal("Address range not parsed correctly")
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user