2014-05-20 18:23:35 +00:00
|
|
|
package proctl
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os/exec"
|
2014-05-23 19:07:10 +00:00
|
|
|
"syscall"
|
2014-05-20 18:23:35 +00:00
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
2014-05-24 00:44:54 +00:00
|
|
|
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) {
|
2014-05-24 00:44:54 +00:00
|
|
|
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")
|
|
|
|
}
|
2014-05-24 00:44:54 +00:00
|
|
|
|
|
|
|
cmd.Process.Kill()
|
2014-05-23 19:07:10 +00:00
|
|
|
}
|
|
|
|
|
2014-05-20 18:23:35 +00:00
|
|
|
func TestStep(t *testing.T) {
|
2014-05-24 00:44:54 +00:00
|
|
|
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) {
|
2014-05-24 00:44:54 +00:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
2014-05-24 00:44:54 +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-24 16:22:06 +00:00
|
|
|
_, err = p.Break("main.sleepytime")
|
2014-05-24 00:44:54 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal("Break():", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
sleepytimefunc := p.GoSymTable.LookupFunc("main.sleepytime")
|
|
|
|
sleepyaddr := sleepytimefunc.LineTable.PC
|
|
|
|
|
|
|
|
err = p.Continue()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("Continue():", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
regs, err := p.Registers()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("Registers():", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
pc := regs.PC()
|
|
|
|
if pc != sleepyaddr {
|
|
|
|
t.Fatal("Break not respected:\nPC:%d\nFN:%d\n", pc, sleepyaddr)
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd.Process.Kill()
|
|
|
|
}
|
2014-05-24 16:22:06 +00:00
|
|
|
|
|
|
|
func TestBreakPointIsSetOnlyOnce(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)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = p.Break("main.sleepytime")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("Break():", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = p.Break("main.sleepytime")
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("Should not be able to add breakpoint twice")
|
|
|
|
}
|
|
|
|
}
|