delve/dwarf/frame.go

277 lines
8.1 KiB
Go
Raw Normal View History

// 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
}