delve/proctl/proctl_test.go

229 lines
4.1 KiB
Go
Raw Normal View History

2014-05-20 18:23:35 +00:00
package proctl
import (
"bytes"
2014-05-20 18:23:35 +00:00
"os/exec"
2014-05-23 19:07:10 +00:00
"syscall"
2014-05-20 18:23:35 +00:00
"testing"
)
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 StartTestProcess(name string) (*exec.Cmd, error) {
cmd := exec.Command("../fixtures/" + name)
2014-05-20 18:23:35 +00:00
err := cmd.Start()
if err != nil {
2014-05-20 18:23:36 +00:00
return nil, err
2014-05-20 18:23:35 +00:00
}
2014-05-20 18:23:36 +00:00
return cmd, nil
2014-05-20 18:23:35 +00:00
}
2014-05-23 19:07:10 +00:00
func TestAttachProcess(t *testing.T) {
cmd, err := StartTestProcess("testprog")
2014-05-23 19:07:10 +00:00
if err != nil {
t.Fatal("Starting test process:", err)
}
pid := cmd.Process.Pid
p, err := NewDebugProcess(pid)
if err != nil {
t.Fatal("NewDebugProcess():", err)
}
if !p.ProcessState.Sys().(syscall.WaitStatus).Stopped() {
t.Errorf("Process was not stopped correctly")
}
cmd.Process.Kill()
2014-05-23 19:07:10 +00:00
}
2014-05-20 18:23:35 +00:00
func TestStep(t *testing.T) {
cmd, err := StartTestProcess("testprog")
2014-05-20 18:23:35 +00:00
if err != nil {
t.Fatal("Starting test process:", err)
}
2014-05-20 18:23:36 +00:00
pid := cmd.Process.Pid
2014-05-20 18:23:35 +00:00
p, err := NewDebugProcess(pid)
if err != nil {
t.Fatal("NewDebugProcess():", err)
}
regs, err := p.Registers()
if err != nil {
2014-05-23 19:07:10 +00:00
t.Fatal("Registers():", err, pid)
2014-05-20 18:23:35 +00:00
}
rip := regs.PC()
err = p.Step()
if err != nil {
t.Fatal("Step():", err)
}
regs, err = p.Registers()
if err != nil {
t.Fatal("Registers():", err)
}
if rip >= regs.PC() {
t.Errorf("Expected %#v to be greater than %#v", regs.PC(), rip)
}
2014-05-23 19:07:10 +00:00
cmd.Process.Kill()
2014-05-20 18:23:35 +00:00
}
2014-05-20 18:23:36 +00:00
func TestContinue(t *testing.T) {
cmd, err := StartTestProcess("continuetestprog")
2014-05-20 18:23:36 +00:00
if err != nil {
t.Fatal("Starting test process:", err)
}
pid := cmd.Process.Pid
p, err := NewDebugProcess(pid)
if err != nil {
t.Fatal("NewDebugProcess():", err)
}
if p.ProcessState.Exited() {
t.Fatal("Process already exited")
}
err = p.Continue()
if err != nil {
t.Fatal("Continue():", err)
}
2014-05-23 19:07:10 +00:00
if !p.ProcessState.Success() {
t.Fatal("Process did not exit successfully")
2014-05-20 18:23:36 +00:00
}
}
func TestBreakPoint(t *testing.T) {
cmd, err := StartTestProcess("testprog")
if err != nil {
t.Fatal("Starting test process:", err)
}
pid := cmd.Process.Pid
p, err := NewDebugProcess(pid)
if err != nil {
t.Fatal("NewDebugProcess():", err)
}
2014-05-27 23:15:18 +00:00
sleepytimefunc := p.GoSymTable.LookupFunc("main.sleepytime")
sleepyaddr := sleepytimefunc.Entry
2014-05-29 14:34:37 +00:00
bp, err := p.Break(uintptr(sleepyaddr))
if err != nil {
t.Fatal("Break():", err)
}
2014-05-29 14:34:37 +00:00
breakpc := bp.Addr + 1
err = p.Continue()
if err != nil {
t.Fatal("Continue():", err)
}
regs, err := p.Registers()
if err != nil {
t.Fatal("Registers():", err)
}
pc := regs.PC()
2014-05-29 14:34:37 +00:00
if pc != breakpc {
t.Fatalf("Break not respected:\nPC:%d\nFN:%d\n", pc, breakpc)
}
err = p.Step()
2014-05-24 16:22:06 +00:00
if err != nil {
t.Fatal(err)
2014-05-24 16:22:06 +00:00
}
regs, err = p.Registers()
2014-05-24 16:22:06 +00:00
if err != nil {
t.Fatal("Registers():", err)
2014-05-24 16:22:06 +00:00
}
pc = regs.PC()
2014-05-29 14:34:37 +00:00
if pc == breakpc {
t.Fatalf("Step not respected:\nPC:%d\nFN:%d\n", pc, breakpc)
2014-05-24 16:22:06 +00:00
}
cmd.Process.Kill()
2014-05-24 16:22:06 +00:00
}
func TestBreakPointWithNonExistantFunction(t *testing.T) {
cmd, err := StartTestProcess("testprog")
if err != nil {
t.Fatal("Starting test process:", err)
}
pid := cmd.Process.Pid
p, err := NewDebugProcess(pid)
if err != nil {
t.Fatal("NewDebugProcess():", err)
}
2014-05-27 23:15:18 +00:00
_, err = p.Break(uintptr(0))
if err == nil {
t.Fatal("Should not be able to break at non existant function")
}
cmd.Process.Kill()
}
func TestClearBreakPoint(t *testing.T) {
cmd, err := StartTestProcess("testprog")
if err != nil {
t.Fatal("Starting test process:", err)
}
pid := cmd.Process.Pid
p, err := NewDebugProcess(pid)
if err != nil {
t.Fatal("NewDebugProcess():", err)
}
2014-05-27 23:15:18 +00:00
fn := p.GoSymTable.LookupFunc("main.sleepytime")
bp, err := p.Break(uintptr(fn.Entry))
if err != nil {
t.Fatal("Break():", err)
}
int3, err := dataAtAddr(pid, bp.Addr)
if err != nil {
t.Fatal(err)
}
2014-05-27 23:15:18 +00:00
bp, err = p.Clear(fn.Entry)
if err != nil {
t.Fatal("Break():", err)
}
data, err := dataAtAddr(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)
}
2014-05-28 22:47:29 +00:00
if len(p.BreakPoints) != 0 {
t.Fatal("Breakpoint not removed internally")
}
cmd.Process.Kill()
}