proctl_test.go 4.0 KB
Newer Older
D
Derek Parker 已提交
1 2 3
package proctl

import (
4
	"bytes"
D
Derek Parker 已提交
5
	"os/exec"
6
	"syscall"
D
Derek Parker 已提交
7 8 9
	"testing"
)

10 11 12 13 14 15 16 17 18 19
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
}

20 21 22 23 24 25 26 27 28
func getRegisters(p *DebuggedProcess, t *testing.T) *syscall.PtraceRegs {
	regs, err := p.Registers()
	if err != nil {
		t.Fatal("Registers():", err)
	}

	return regs
}

29 30
func StartTestProcess(name string) (*exec.Cmd, error) {
	cmd := exec.Command("../fixtures/" + name)
D
Derek Parker 已提交
31 32 33

	err := cmd.Start()
	if err != nil {
34
		return nil, err
D
Derek Parker 已提交
35 36
	}

37
	return cmd, nil
D
Derek Parker 已提交
38 39
}

40
func TestAttachProcess(t *testing.T) {
41
	cmd, err := StartTestProcess("testprog")
42 43 44 45 46 47 48 49 50 51 52 53 54
	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")
	}
55 56

	cmd.Process.Kill()
57 58
}

D
Derek Parker 已提交
59
func TestStep(t *testing.T) {
60
	cmd, err := StartTestProcess("testprog")
D
Derek Parker 已提交
61 62 63 64
	if err != nil {
		t.Fatal("Starting test process:", err)
	}

65
	pid := cmd.Process.Pid
D
Derek Parker 已提交
66 67 68 69 70
	p, err := NewDebugProcess(pid)
	if err != nil {
		t.Fatal("NewDebugProcess():", err)
	}

71
	regs := getRegisters(p, t)
D
Derek Parker 已提交
72 73 74 75 76 77 78
	rip := regs.PC()

	err = p.Step()
	if err != nil {
		t.Fatal("Step():", err)
	}

79
	regs = getRegisters(p, t)
D
Derek Parker 已提交
80 81 82 83

	if rip >= regs.PC() {
		t.Errorf("Expected %#v to be greater than %#v", regs.PC(), rip)
	}
84 85

	cmd.Process.Kill()
D
Derek Parker 已提交
86
}
87 88

func TestContinue(t *testing.T) {
89
	cmd, err := StartTestProcess("continuetestprog")
90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108
	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)
	}

109 110
	if !p.ProcessState.Success() {
		t.Fatal("Process did not exit successfully")
111 112
	}
}
113 114 115 116 117 118 119 120 121 122 123 124 125

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)
	}

D
Derek Parker 已提交
126 127 128
	sleepytimefunc := p.GoSymTable.LookupFunc("main.sleepytime")
	sleepyaddr := sleepytimefunc.Entry

129
	bp, err := p.Break(uintptr(sleepyaddr))
130 131 132 133
	if err != nil {
		t.Fatal("Break():", err)
	}

134
	breakpc := bp.Addr + 1
135 136 137 138 139
	err = p.Continue()
	if err != nil {
		t.Fatal("Continue():", err)
	}

140
	regs := getRegisters(p, t)
141 142

	pc := regs.PC()
143 144
	if pc != breakpc {
		t.Fatalf("Break not respected:\nPC:%d\nFN:%d\n", pc, breakpc)
145 146
	}

147
	err = p.Step()
148
	if err != nil {
149
		t.Fatal(err)
150 151
	}

152
	regs = getRegisters(p, t)
153

154
	pc = regs.PC()
155 156
	if pc == breakpc {
		t.Fatalf("Step not respected:\nPC:%d\nFN:%d\n", pc, breakpc)
157 158
	}

159
	cmd.Process.Kill()
160
}
161 162 163 164 165 166 167 168 169 170 171 172 173

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)
	}

D
Derek Parker 已提交
174
	_, err = p.Break(uintptr(0))
175 176 177
	if err == nil {
		t.Fatal("Should not be able to break at non existant function")
	}
178 179

	cmd.Process.Kill()
180
}
181 182 183 184 185 186 187 188 189 190 191 192 193

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)
	}

D
Derek Parker 已提交
194 195
	fn := p.GoSymTable.LookupFunc("main.sleepytime")
	bp, err := p.Break(uintptr(fn.Entry))
196 197 198 199 200 201 202 203 204
	if err != nil {
		t.Fatal("Break():", err)
	}

	int3, err := dataAtAddr(pid, bp.Addr)
	if err != nil {
		t.Fatal(err)
	}

D
Derek Parker 已提交
205
	bp, err = p.Clear(fn.Entry)
206 207 208 209 210 211 212 213 214 215 216 217
	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)
	}
218 219 220 221

	if len(p.BreakPoints) != 0 {
		t.Fatal("Breakpoint not removed internally")
	}
222 223

	cmd.Process.Kill()
224
}