delve/dwarf/frame/frame_parser.go

148 lines
3.4 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"
"io"
"github.com/derekparker/dbg/dwarf/util"
)
type parsefunc func(*parseContext) parsefunc
type parseContext struct {
Buf *bytes.Buffer
Entries FrameDescriptionEntries
2014-06-07 00:04:15 +00:00
Common *CommonInformationEntry
Frame *FrameDescriptionEntry
Length uint32
}
2014-06-06 20:28:47 +00:00
// Parse takes in data (a byte slice) and returns a slice of
// CommonInformationEntry structures. Each CommonInformationEntry
// has a slice of FrameDescriptionEntry structures.
func Parse(data []byte) FrameDescriptionEntries {
var (
buf = bytes.NewBuffer(data)
pctx = &parseContext{Buf: buf}
)
for fn := parseLength; buf.Len() != 0; {
fn = fn(pctx)
}
return pctx.Entries
}
func cieEntry(data []byte) bool {
return bytes.Equal(data, []byte{0xff, 0xff, 0xff, 0xff})
}
func parseLength(ctx *parseContext) parsefunc {
var fn parsefunc
binary.Read(ctx.Buf, binary.LittleEndian, &ctx.Length)
cieid := ctx.Buf.Next(4)
if cieEntry(cieid) {
2014-06-07 00:04:15 +00:00
ctx.Common = &CommonInformationEntry{Length: ctx.Length}
fn = parseVersion
} else {
ctx.Frame = &FrameDescriptionEntry{Length: ctx.Length, CIE: ctx.Common, AddressRange: &addrange{}}
ctx.Entries = append(ctx.Entries, ctx.Frame)
fn = parseInitialLocation
}
// Take off the length of the CIE id / CIE pointer.
ctx.Length -= 4
return fn
}
func parseInitialLocation(ctx *parseContext) parsefunc {
binary.Read(ctx.Buf, binary.LittleEndian, &ctx.Frame.AddressRange.begin)
ctx.Length -= 8
return parseAddressRange
}
func parseAddressRange(ctx *parseContext) parsefunc {
binary.Read(ctx.Buf, binary.LittleEndian, &ctx.Frame.AddressRange.end)
ctx.Length -= 8
return parseFrameInstructions
}
func parseFrameInstructions(ctx *parseContext) parsefunc {
2014-06-07 00:04:15 +00:00
// The rest of this entry consists of the instructions
// so we can just grab all of the data from the buffer
// cursor to length.
var buf = make([]byte, ctx.Length)
io.ReadFull(ctx.Buf, buf)
2014-06-07 00:04:15 +00:00
ctx.Frame.Instructions = buf
ctx.Length = 0
return parseLength
}
func parseVersion(ctx *parseContext) parsefunc {
2014-06-07 00:04:15 +00:00
binary.Read(ctx.Buf, binary.LittleEndian, &ctx.Common.Version)
ctx.Length -= 1
return parseAugmentation
}
func parseAugmentation(ctx *parseContext) parsefunc {
var str, c = util.ParseString(ctx.Buf)
2014-06-07 00:04:15 +00:00
ctx.Common.Augmentation = str
ctx.Length -= c
return parseCodeAlignmentFactor
}
func parseCodeAlignmentFactor(ctx *parseContext) parsefunc {
var caf, c = util.DecodeULEB128(ctx.Buf)
2014-06-07 00:04:15 +00:00
ctx.Common.CodeAlignmentFactor = caf
ctx.Length -= c
return parseDataAlignmentFactor
}
func parseDataAlignmentFactor(ctx *parseContext) parsefunc {
var daf, c = util.DecodeSLEB128(ctx.Buf)
2014-06-07 00:04:15 +00:00
ctx.Common.DataAlignmentFactor = daf
ctx.Length -= c
return parseReturnAddressRegister
}
func parseReturnAddressRegister(ctx *parseContext) parsefunc {
reg, c := util.DecodeULEB128(ctx.Buf)
ctx.Common.ReturnAddressRegister = uint8(reg)
ctx.Length -= c
return parseInitialInstructions
}
func parseInitialInstructions(ctx *parseContext) parsefunc {
2014-06-07 00:04:15 +00:00
// The rest of this entry consists of the instructions
// so we can just grab all of the data from the buffer
// cursor to length.
var buf = make([]byte, ctx.Length)
binary.Read(ctx.Buf, binary.LittleEndian, &buf)
2014-06-07 00:04:15 +00:00
ctx.Common.InitialInstructions = buf
ctx.Length = 0
return parseLength
}