delve/dwarf/frame/frame_table_test.go
2014-08-01 16:34:27 -05:00

93 lines
1.7 KiB
Go

package frame_test
import (
"debug/elf"
"encoding/binary"
"os"
"path/filepath"
"syscall"
"testing"
"github.com/derekparker/dbg/_helper"
"github.com/derekparker/dbg/dwarf/_helper"
"github.com/derekparker/dbg/dwarf/frame"
"github.com/derekparker/dbg/proctl"
)
func grabDebugFrameSection(fp string, t *testing.T) []byte {
p, err := filepath.Abs(fp)
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 TestFindReturnAddress(t *testing.T) {
var (
testfile, _ = filepath.Abs("../../_fixtures/testnextprog")
dbframe = grabDebugFrameSection(testfile, t)
fdes = frame.Parse(dbframe)
gsd = dwarfhelper.GosymData(testfile, t)
)
helper.WithTestProcess(testfile, t, func(p *proctl.DebuggedProcess) {
testsourcefile := testfile + ".go"
start, _, err := gsd.LineToPC(testsourcefile, 22)
if err != nil {
t.Fatal(err)
}
_, err = p.Break(uintptr(start))
if err != nil {
t.Fatal(err)
}
err = p.Continue()
if err != nil {
t.Fatal(err)
}
regs, err := p.Registers()
if err != nil {
t.Fatal(err)
}
fde, err := fdes.FDEForPC(start)
if err != nil {
t.Fatal(err)
}
ret := fde.ReturnAddressOffset(start)
if err != nil {
t.Fatal(err)
}
addr := uint64(int64(regs.Rsp) + ret)
data := make([]byte, 8)
syscall.PtracePeekText(p.Pid, uintptr(addr), data)
addr = binary.LittleEndian.Uint64(data)
end := uint64(0x400d9f)
if addr != end {
t.Fatalf("return address not found correctly, expected %#v got %#v", end, addr)
}
})
}