delve/proctl/proctl_test.go

265 lines
5.9 KiB
Go
Raw Normal View History

2014-06-25 19:06:04 +00:00
package proctl_test
2014-05-20 18:23:35 +00:00
import (
"bytes"
"path/filepath"
2014-05-23 19:07:10 +00:00
"syscall"
2014-05-20 18:23:35 +00:00
"testing"
2014-10-14 14:50:23 +00:00
"github.com/derekparker/dbg/helper"
2014-06-25 19:06:04 +00:00
"github.com/derekparker/dbg/proctl"
)
func dataAtAddr(pid int, addr uint64) ([]byte, error) {
data := make([]byte, 1)
_, err := syscall.PtracePeekData(pid, uintptr(addr), data)
if err != nil {
return nil, err
}
return data, nil
}
func assertNoError(err error, t *testing.T, s string) {
if err != nil {
t.Fatal(s, ":", err)
}
}
func currentPC(p *proctl.DebuggedProcess, t *testing.T) uint64 {
pc, err := p.CurrentPC()
if err != nil {
t.Fatal(err)
}
return pc
}
func currentLineNumber(p *proctl.DebuggedProcess, t *testing.T) (string, int) {
pc := currentPC(p, t)
f, l, _ := p.GoSymTable.PCToLine(pc)
return f, l
}
func TestAttachProcess(t *testing.T) {
helper.WithTestProcess("../_fixtures/testprog", t, func(p *proctl.DebuggedProcess) {
if !p.ProcessState.Stopped() {
t.Errorf("Process was not stopped correctly")
}
})
2014-05-20 18:23:35 +00:00
}
2014-05-20 18:23:36 +00:00
func TestStep(t *testing.T) {
helper.WithTestProcess("../_fixtures/testprog", t, func(p *proctl.DebuggedProcess) {
2014-06-09 19:56:10 +00:00
if p.ProcessState.Exited() {
t.Fatal("Process already exited")
}
2014-06-25 19:06:04 +00:00
regs := helper.GetRegisters(p, t)
rip := regs.PC()
2014-05-20 18:23:36 +00:00
err := p.Step()
2014-09-06 23:53:22 +00:00
assertNoError(err, t, "Step()")
2014-05-20 18:23:36 +00:00
2014-06-25 19:06:04 +00:00
regs = helper.GetRegisters(p, t)
if rip >= regs.PC() {
t.Errorf("Expected %#v to be greater than %#v", regs.PC(), rip)
}
})
}
2014-05-20 18:23:36 +00:00
func TestContinue(t *testing.T) {
helper.WithTestProcess("../_fixtures/continuetestprog", t, func(p *proctl.DebuggedProcess) {
if p.ProcessState.Exited() {
t.Fatal("Process already exited")
}
err := p.Continue()
2014-09-06 23:53:22 +00:00
assertNoError(err, t, "Continue()")
if p.ProcessState.ExitStatus() != 0 {
t.Fatal("Process did not exit successfully")
}
})
2014-05-20 18:23:36 +00:00
}
func TestBreakPoint(t *testing.T) {
helper.WithTestProcess("../_fixtures/testprog", t, func(p *proctl.DebuggedProcess) {
sleepytimefunc := p.GoSymTable.LookupFunc("main.sleepytime")
sleepyaddr := sleepytimefunc.Entry
bp, err := p.Break(uintptr(sleepyaddr))
2014-09-06 23:53:22 +00:00
assertNoError(err, t, "Break()")
breakpc := bp.Addr + 1
err = p.Continue()
2014-09-06 23:53:22 +00:00
assertNoError(err, t, "Continue()")
2014-06-25 19:06:04 +00:00
regs := helper.GetRegisters(p, t)
pc := regs.PC()
if pc != breakpc {
t.Fatalf("Break not respected:\nPC:%d\nFN:%d\n", pc, breakpc)
}
err = p.Step()
2014-09-06 23:53:22 +00:00
assertNoError(err, t, "Step()")
2014-06-25 19:06:04 +00:00
regs = helper.GetRegisters(p, t)
pc = regs.PC()
if pc == breakpc {
t.Fatalf("Step not respected:\nPC:%d\nFN:%d\n", pc, breakpc)
}
})
2014-05-24 16:22:06 +00:00
}
func TestBreakPointWithNonExistantFunction(t *testing.T) {
helper.WithTestProcess("../_fixtures/testprog", t, func(p *proctl.DebuggedProcess) {
_, err := p.Break(uintptr(0))
if err == nil {
t.Fatal("Should not be able to break at non existant function")
}
})
}
func TestClearBreakPoint(t *testing.T) {
helper.WithTestProcess("../_fixtures/testprog", t, func(p *proctl.DebuggedProcess) {
fn := p.GoSymTable.LookupFunc("main.sleepytime")
bp, err := p.Break(uintptr(fn.Entry))
2014-09-06 23:53:22 +00:00
assertNoError(err, t, "Break()")
int3, err := dataAtAddr(p.Pid, bp.Addr)
if err != nil {
t.Fatal(err)
}
bp, err = p.Clear(fn.Entry)
2014-09-06 23:53:22 +00:00
assertNoError(err, t, "Clear()")
data, err := dataAtAddr(p.Pid, bp.Addr)
if err != nil {
t.Fatal(err)
}
if bytes.Equal(data, int3) {
t.Fatalf("Breakpoint was not cleared data: %#v, int3: %#v", data, int3)
}
if len(p.BreakPoints) != 0 {
t.Fatal("Breakpoint not removed internally")
}
})
}
func TestNext(t *testing.T) {
var (
2014-07-30 00:00:24 +00:00
err error
executablePath = "../_fixtures/testnextprog"
)
testcases := []struct {
begin, end int
}{
{19, 20},
{20, 23},
{23, 24},
{24, 26},
{26, 31},
{31, 23},
{23, 24},
{24, 26},
{26, 31},
{31, 23},
{23, 24},
{24, 26},
{26, 27},
{27, 34},
{34, 35},
{35, 41},
2014-10-11 02:00:07 +00:00
{41, 40},
{40, 41},
}
fp, err := filepath.Abs("../_fixtures/testnextprog.go")
if err != nil {
t.Fatal(err)
}
2014-07-30 00:00:24 +00:00
helper.WithTestProcess(executablePath, t, func(p *proctl.DebuggedProcess) {
pc, _, _ := p.GoSymTable.LineToPC(fp, testcases[0].begin)
_, err := p.Break(uintptr(pc))
2014-09-06 23:56:25 +00:00
assertNoError(err, t, "Break()")
assertNoError(p.Continue(), t, "Continue()")
for _, tc := range testcases {
f, ln := currentLineNumber(p, t)
if ln != tc.begin {
t.Fatalf("Program not stopped at correct spot expected %d was %s:%d", tc.begin, f, ln)
}
assertNoError(p.Next(), t, "Next() returned an error")
f, ln = currentLineNumber(p, t)
if ln != tc.end {
t.Fatalf("Program did not continue to correct next location expected %d was %s:%d", tc.end, f, ln)
}
}
2014-10-04 23:13:20 +00:00
if len(p.BreakPoints) != 1 {
t.Fatal("Not all breakpoints were cleaned up")
}
})
}
2014-07-30 00:00:24 +00:00
func TestVariableEvaluation(t *testing.T) {
executablePath := "../_fixtures/testvariables"
fp, err := filepath.Abs(executablePath + ".go")
if err != nil {
t.Fatal(err)
}
testcases := []struct {
name string
value string
varType string
}{
2014-08-04 18:53:15 +00:00
{"a1", "foo", "struct string"},
2014-08-04 18:20:20 +00:00
{"a2", "6", "int"},
{"a3", "7.23", "float64"},
{"a4", "[2]int [1 2]", "[97]int"}, // There is a weird bug in the Go dwarf parser that is grabbing the wrong size for an array.
2014-08-04 20:41:09 +00:00
{"a5", "len: 5 cap: 5 [1 2 3 4 5]", "struct []int"},
{"a6", "main.FooBar {Baz: 8, Bur: word}", "main.FooBar"},
{"a7", "*main.FooBar {Baz: 5, Bur: strum}", "*main.FooBar"},
2014-07-30 00:00:24 +00:00
}
helper.WithTestProcess(executablePath, t, func(p *proctl.DebuggedProcess) {
2014-08-04 18:20:20 +00:00
pc, _, _ := p.GoSymTable.LineToPC(fp, 21)
2014-07-30 00:00:24 +00:00
_, err := p.Break(uintptr(pc))
assertNoError(err, t, "Break() returned an error")
err = p.Continue()
2014-08-04 18:20:20 +00:00
assertNoError(err, t, "Continue() returned an error")
2014-07-30 00:00:24 +00:00
for _, tc := range testcases {
variable, err := p.EvalSymbol(tc.name)
assertNoError(err, t, "Variable() returned an error")
if variable.Name != tc.name {
t.Fatalf("Expected %s got %s\n", tc.name, variable.Name)
}
if variable.Type != tc.varType {
t.Fatalf("Expected %s got %s\n", tc.varType, variable.Type)
}
if variable.Value != tc.value {
2014-08-04 18:53:15 +00:00
t.Fatalf("Expected %#v got %#v\n", tc.value, variable.Value)
2014-07-30 00:00:24 +00:00
}
}
})
}