2014-12-09 16:51:17 +00:00
|
|
|
package proctl
|
2014-05-20 18:23:35 +00:00
|
|
|
|
|
|
|
import (
|
2014-05-27 18:33:49 +00:00
|
|
|
"bytes"
|
2014-12-09 16:51:17 +00:00
|
|
|
"encoding/binary"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
2014-06-29 16:52:21 +00:00
|
|
|
"path/filepath"
|
2014-12-09 16:51:17 +00:00
|
|
|
"runtime"
|
2014-05-20 18:23:35 +00:00
|
|
|
"testing"
|
2015-04-13 22:17:06 +00:00
|
|
|
"time"
|
2014-06-25 19:06:04 +00:00
|
|
|
)
|
2014-05-30 15:12:18 +00:00
|
|
|
|
2014-12-09 16:51:17 +00:00
|
|
|
func withTestProcess(name string, t *testing.T, fn func(p *DebuggedProcess)) {
|
|
|
|
runtime.LockOSThread()
|
|
|
|
base := filepath.Base(name)
|
|
|
|
if err := exec.Command("go", "build", "-gcflags=-N -l", "-o", base, name+".go").Run(); err != nil {
|
|
|
|
t.Fatalf("Could not compile %s due to %s", name, err)
|
|
|
|
}
|
|
|
|
defer os.Remove("./" + base)
|
|
|
|
|
|
|
|
p, err := Launch([]string{"./" + base})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("Launch():", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
defer p.Process.Kill()
|
|
|
|
|
|
|
|
fn(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getRegisters(p *DebuggedProcess, t *testing.T) Registers {
|
2014-12-09 16:35:55 +00:00
|
|
|
regs, err := p.Registers()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("Registers():", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return regs
|
|
|
|
}
|
|
|
|
|
2015-01-14 02:37:10 +00:00
|
|
|
func dataAtAddr(thread *ThreadContext, addr uint64) ([]byte, error) {
|
2014-05-27 18:33:49 +00:00
|
|
|
data := make([]byte, 1)
|
2015-01-14 02:37:10 +00:00
|
|
|
_, err := readMemory(thread, uintptr(addr), data)
|
2014-05-27 18:33:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return data, nil
|
|
|
|
}
|
|
|
|
|
2014-06-29 16:52:21 +00:00
|
|
|
func assertNoError(err error, t *testing.T, s string) {
|
|
|
|
if err != nil {
|
2015-01-21 21:26:01 +00:00
|
|
|
_, file, line, _ := runtime.Caller(1)
|
|
|
|
fname := filepath.Base(file)
|
|
|
|
t.Fatalf("failed assertion at %s:%d: %s : %s\n", fname, line, s, err)
|
2014-06-29 16:52:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-09 16:51:17 +00:00
|
|
|
func currentPC(p *DebuggedProcess, t *testing.T) uint64 {
|
2015-04-23 15:40:33 +00:00
|
|
|
pc, err := p.PC()
|
2014-06-29 16:52:21 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return pc
|
|
|
|
}
|
|
|
|
|
2014-12-09 16:51:17 +00:00
|
|
|
func currentLineNumber(p *DebuggedProcess, t *testing.T) (string, int) {
|
2014-06-29 16:52:21 +00:00
|
|
|
pc := currentPC(p, t)
|
2015-04-03 16:10:35 +00:00
|
|
|
f, l, _ := p.goSymTable.PCToLine(pc)
|
2014-06-29 16:52:21 +00:00
|
|
|
|
2014-10-07 17:25:33 +00:00
|
|
|
return f, l
|
2014-06-29 16:52:21 +00:00
|
|
|
}
|
|
|
|
|
2015-03-05 22:59:51 +00:00
|
|
|
func TestExit(t *testing.T) {
|
|
|
|
withTestProcess("../_fixtures/continuetestprog", t, func(p *DebuggedProcess) {
|
|
|
|
err := p.Continue()
|
|
|
|
pe, ok := err.(ProcessExitedError)
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("Continue() returned unexpected error type")
|
|
|
|
}
|
|
|
|
if pe.Status != 0 {
|
|
|
|
t.Errorf("Unexpected error status: %d", pe.Status)
|
|
|
|
}
|
|
|
|
if pe.Pid != p.Pid {
|
|
|
|
t.Errorf("Unexpected process id: %d", pe.Pid)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2015-04-13 22:17:06 +00:00
|
|
|
func TestHalt(t *testing.T) {
|
|
|
|
withTestProcess("../_fixtures/testprog", t, func(p *DebuggedProcess) {
|
|
|
|
go func() {
|
|
|
|
time.Sleep(10 * time.Millisecond)
|
|
|
|
err := p.RequestManualStop()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
err := p.Continue()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Loop through threads and make sure they are all
|
|
|
|
// actually stopped, err will not be nil if the process
|
|
|
|
// is still running.
|
|
|
|
for _, th := range p.Threads {
|
|
|
|
_, err := th.Registers()
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2014-05-30 15:12:18 +00:00
|
|
|
func TestStep(t *testing.T) {
|
2014-12-09 16:51:17 +00:00
|
|
|
withTestProcess("../_fixtures/testprog", t, func(p *DebuggedProcess) {
|
2015-04-03 16:10:35 +00:00
|
|
|
helloworldfunc := p.goSymTable.LookupFunc("main.helloworld")
|
2014-11-27 02:35:53 +00:00
|
|
|
helloworldaddr := helloworldfunc.Entry
|
|
|
|
|
2015-01-01 13:23:55 +00:00
|
|
|
_, err := p.Break(helloworldaddr)
|
2014-11-27 02:35:53 +00:00
|
|
|
assertNoError(err, t, "Break()")
|
|
|
|
assertNoError(p.Continue(), t, "Continue()")
|
|
|
|
|
2014-12-09 16:35:55 +00:00
|
|
|
regs := getRegisters(p, t)
|
2014-05-30 15:12:18 +00:00
|
|
|
rip := regs.PC()
|
2014-05-20 18:23:36 +00:00
|
|
|
|
2014-11-27 02:35:53 +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-12-09 16:35:55 +00:00
|
|
|
regs = getRegisters(p, t)
|
2014-05-30 15:12:18 +00:00
|
|
|
if rip >= regs.PC() {
|
|
|
|
t.Errorf("Expected %#v to be greater than %#v", regs.PC(), rip)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2014-05-20 18:23:36 +00:00
|
|
|
|
2014-05-24 00:44:54 +00:00
|
|
|
func TestBreakPoint(t *testing.T) {
|
2014-12-09 16:51:17 +00:00
|
|
|
withTestProcess("../_fixtures/testprog", t, func(p *DebuggedProcess) {
|
2015-04-03 16:10:35 +00:00
|
|
|
helloworldfunc := p.goSymTable.LookupFunc("main.helloworld")
|
2015-01-14 02:37:10 +00:00
|
|
|
helloworldaddr := helloworldfunc.Entry
|
2014-05-30 15:12:18 +00:00
|
|
|
|
2015-01-14 02:37:10 +00:00
|
|
|
bp, err := p.Break(helloworldaddr)
|
2014-09-06 23:53:22 +00:00
|
|
|
assertNoError(err, t, "Break()")
|
2015-01-14 02:37:10 +00:00
|
|
|
assertNoError(p.Continue(), t, "Continue()")
|
2014-05-30 15:12:18 +00:00
|
|
|
|
2015-04-23 15:40:33 +00:00
|
|
|
pc, err := p.PC()
|
2014-10-18 01:34:58 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2014-05-30 15:12:18 +00:00
|
|
|
|
2015-02-28 03:35:26 +00:00
|
|
|
if pc-1 != bp.Addr && pc != bp.Addr {
|
2015-04-03 16:10:35 +00:00
|
|
|
f, l, _ := p.goSymTable.PCToLine(pc)
|
2015-01-14 02:37:10 +00:00
|
|
|
t.Fatalf("Break not respected:\nPC:%#v %s:%d\nFN:%#v \n", pc, f, l, bp.Addr)
|
2014-05-30 15:12:18 +00:00
|
|
|
}
|
|
|
|
})
|
2014-05-24 16:22:06 +00:00
|
|
|
}
|
2014-05-24 16:36:18 +00:00
|
|
|
|
2014-10-18 01:34:58 +00:00
|
|
|
func TestBreakPointInSeperateGoRoutine(t *testing.T) {
|
2014-12-09 16:51:17 +00:00
|
|
|
withTestProcess("../_fixtures/testthreads", t, func(p *DebuggedProcess) {
|
2015-04-03 16:10:35 +00:00
|
|
|
fn := p.goSymTable.LookupFunc("main.anotherthread")
|
2014-10-18 01:34:58 +00:00
|
|
|
if fn == nil {
|
|
|
|
t.Fatal("No fn exists")
|
|
|
|
}
|
|
|
|
|
2015-01-01 13:23:55 +00:00
|
|
|
_, err := p.Break(fn.Entry)
|
2014-10-18 01:34:58 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = p.Continue()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2015-04-23 15:40:33 +00:00
|
|
|
pc, err := p.PC()
|
2014-10-18 01:34:58 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2015-04-03 16:10:35 +00:00
|
|
|
f, l, _ := p.goSymTable.PCToLine(pc)
|
2014-10-18 01:34:58 +00:00
|
|
|
if f != "testthreads.go" && l != 8 {
|
|
|
|
t.Fatal("Program did not hit breakpoint")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2014-05-24 16:36:18 +00:00
|
|
|
func TestBreakPointWithNonExistantFunction(t *testing.T) {
|
2014-12-09 16:51:17 +00:00
|
|
|
withTestProcess("../_fixtures/testprog", t, func(p *DebuggedProcess) {
|
2015-01-01 13:23:55 +00:00
|
|
|
_, err := p.Break(0)
|
2014-05-30 15:12:18 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("Should not be able to break at non existant function")
|
|
|
|
}
|
|
|
|
})
|
2014-05-24 16:36:18 +00:00
|
|
|
}
|
2014-05-27 18:33:49 +00:00
|
|
|
|
|
|
|
func TestClearBreakPoint(t *testing.T) {
|
2014-12-09 16:51:17 +00:00
|
|
|
withTestProcess("../_fixtures/testprog", t, func(p *DebuggedProcess) {
|
2015-04-03 16:10:35 +00:00
|
|
|
fn := p.goSymTable.LookupFunc("main.sleepytime")
|
2015-01-01 13:23:55 +00:00
|
|
|
bp, err := p.Break(fn.Entry)
|
2014-09-06 23:53:22 +00:00
|
|
|
assertNoError(err, t, "Break()")
|
2014-05-30 15:12:18 +00:00
|
|
|
|
|
|
|
bp, err = p.Clear(fn.Entry)
|
2014-09-06 23:53:22 +00:00
|
|
|
assertNoError(err, t, "Clear()")
|
2014-05-30 15:12:18 +00:00
|
|
|
|
2015-01-14 02:37:10 +00:00
|
|
|
data, err := dataAtAddr(p.CurrentThread, bp.Addr)
|
2014-05-30 15:12:18 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2015-01-10 23:33:06 +00:00
|
|
|
int3 := []byte{0xcc}
|
2014-05-30 15:12:18 +00:00
|
|
|
if bytes.Equal(data, int3) {
|
|
|
|
t.Fatalf("Breakpoint was not cleared data: %#v, int3: %#v", data, int3)
|
|
|
|
}
|
|
|
|
|
2014-10-25 14:17:05 +00:00
|
|
|
if len(p.BreakPoints) != 0 {
|
2014-05-30 15:12:18 +00:00
|
|
|
t.Fatal("Breakpoint not removed internally")
|
|
|
|
}
|
|
|
|
})
|
2014-05-27 18:33:49 +00:00
|
|
|
}
|
2014-06-29 16:52:21 +00:00
|
|
|
|
2015-04-19 22:11:33 +00:00
|
|
|
type nextTest struct {
|
|
|
|
begin, end int
|
|
|
|
}
|
2014-06-29 16:52:21 +00:00
|
|
|
|
2015-04-19 22:11:33 +00:00
|
|
|
func testnext(testcases []nextTest, initialLocation string, t *testing.T) {
|
|
|
|
var executablePath = "../_fixtures/testnextprog"
|
2014-12-09 16:51:17 +00:00
|
|
|
withTestProcess(executablePath, t, func(p *DebuggedProcess) {
|
2015-04-19 22:11:33 +00:00
|
|
|
bp, err := p.BreakByLocation(initialLocation)
|
2014-09-06 23:56:25 +00:00
|
|
|
assertNoError(err, t, "Break()")
|
|
|
|
assertNoError(p.Continue(), t, "Continue()")
|
2015-04-19 22:11:33 +00:00
|
|
|
p.Clear(bp.Addr)
|
|
|
|
p.CurrentThread.SetPC(bp.Addr)
|
2014-06-29 16:52:21 +00:00
|
|
|
|
2014-12-29 02:48:58 +00:00
|
|
|
f, ln := currentLineNumber(p, t)
|
2014-06-29 16:52:21 +00:00
|
|
|
for _, tc := range testcases {
|
|
|
|
if ln != tc.begin {
|
2015-03-28 01:12:07 +00:00
|
|
|
t.Fatalf("Program not stopped at correct spot expected %d was %s:%d", tc.begin, filepath.Base(f), ln)
|
2014-06-29 16:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assertNoError(p.Next(), t, "Next() returned an error")
|
|
|
|
|
2014-10-07 17:25:33 +00:00
|
|
|
f, ln = currentLineNumber(p, t)
|
2014-06-29 16:52:21 +00:00
|
|
|
if ln != tc.end {
|
2015-03-28 01:12:07 +00:00
|
|
|
t.Fatalf("Program did not continue to correct next location expected %d was %s:%d", tc.end, filepath.Base(f), ln)
|
2014-06-29 16:52:21 +00:00
|
|
|
}
|
|
|
|
}
|
2014-09-17 03:37:48 +00:00
|
|
|
|
2015-01-10 23:33:06 +00:00
|
|
|
if len(p.BreakPoints) != 0 {
|
2015-04-19 22:11:33 +00:00
|
|
|
t.Fatal("Not all breakpoints were cleaned up", len(p.BreakPoints))
|
2015-01-10 23:33:06 +00:00
|
|
|
}
|
|
|
|
for _, bp := range p.HWBreakPoints {
|
|
|
|
if bp != nil {
|
|
|
|
t.Fatal("Not all breakpoints were cleaned up", bp.Addr)
|
|
|
|
}
|
2014-09-17 03:37:48 +00:00
|
|
|
}
|
2014-06-29 16:52:21 +00:00
|
|
|
})
|
|
|
|
}
|
2014-12-09 16:51:17 +00:00
|
|
|
|
2015-04-19 22:11:33 +00:00
|
|
|
func TestNextGeneral(t *testing.T) {
|
|
|
|
testcases := []nextTest{
|
|
|
|
{17, 19},
|
|
|
|
{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},
|
|
|
|
}
|
|
|
|
testnext(testcases, "main.testnext", t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNextGoroutine(t *testing.T) {
|
|
|
|
testcases := []nextTest{
|
|
|
|
{46, 47},
|
|
|
|
{47, 42},
|
|
|
|
}
|
|
|
|
testnext(testcases, "main.testgoroutine", t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNextFunctionReturn(t *testing.T) {
|
|
|
|
testcases := []nextTest{
|
|
|
|
{13, 14},
|
|
|
|
{14, 35},
|
|
|
|
}
|
|
|
|
testnext(testcases, "main.helloworld", t)
|
|
|
|
}
|
|
|
|
|
2014-12-09 16:51:17 +00:00
|
|
|
func TestFindReturnAddress(t *testing.T) {
|
|
|
|
var testfile, _ = filepath.Abs("../_fixtures/testnextprog")
|
|
|
|
|
|
|
|
withTestProcess(testfile, t, func(p *DebuggedProcess) {
|
|
|
|
var (
|
2015-04-03 16:10:35 +00:00
|
|
|
fdes = p.frameEntries
|
|
|
|
gsd = p.goSymTable
|
2014-12-09 16:51:17 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
testsourcefile := testfile + ".go"
|
|
|
|
start, _, err := gsd.LineToPC(testsourcefile, 24)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2015-01-01 13:23:55 +00:00
|
|
|
_, err = p.Break(start)
|
2014-12-09 16:51:17 +00:00
|
|
|
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.SP()) + ret)
|
|
|
|
data := make([]byte, 8)
|
|
|
|
|
2015-01-14 02:37:10 +00:00
|
|
|
readMemory(p.CurrentThread, uintptr(addr), data)
|
2014-12-09 16:51:17 +00:00
|
|
|
addr = binary.LittleEndian.Uint64(data)
|
|
|
|
|
2015-04-19 22:11:33 +00:00
|
|
|
_, l, _ := p.goSymTable.PCToLine(addr)
|
|
|
|
if l != 40 {
|
|
|
|
t.Fatalf("return address not found correctly, expected line 40")
|
2014-12-09 16:51:17 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2015-03-26 18:15:35 +00:00
|
|
|
|
|
|
|
func TestSwitchThread(t *testing.T) {
|
|
|
|
var testfile, _ = filepath.Abs("../_fixtures/testnextprog")
|
|
|
|
|
|
|
|
withTestProcess(testfile, t, func(p *DebuggedProcess) {
|
|
|
|
// With invalid thread id
|
|
|
|
err := p.SwitchThread(-1)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("Expected error for invalid thread id")
|
|
|
|
}
|
|
|
|
pc, err := p.FindLocation("main.main")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
_, err = p.Break(pc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
err = p.Continue()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var nt int
|
|
|
|
ct := p.CurrentThread.Id
|
|
|
|
for tid, _ := range p.Threads {
|
|
|
|
if tid != ct {
|
|
|
|
nt = tid
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if nt == 0 {
|
|
|
|
t.Fatal("could not find thread to switch to")
|
|
|
|
}
|
|
|
|
// With valid thread id
|
|
|
|
err = p.SwitchThread(nt)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if p.CurrentThread.Id != nt {
|
|
|
|
t.Fatal("Did not switch threads")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2015-03-28 01:12:07 +00:00
|
|
|
|
|
|
|
func TestFunctionCall(t *testing.T) {
|
|
|
|
var testfile, _ = filepath.Abs("../_fixtures/testprog")
|
|
|
|
|
|
|
|
withTestProcess(testfile, t, func(p *DebuggedProcess) {
|
|
|
|
pc, err := p.FindLocation("main.main")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
_, err = p.Break(pc)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
err = p.Continue()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2015-04-23 15:40:33 +00:00
|
|
|
pc, err = p.PC()
|
2015-03-28 01:12:07 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2015-04-03 16:10:35 +00:00
|
|
|
fn := p.goSymTable.PCToFunc(pc)
|
2015-03-28 01:12:07 +00:00
|
|
|
if fn == nil {
|
|
|
|
t.Fatalf("Could not find func for PC: %#v", pc)
|
|
|
|
}
|
|
|
|
if fn.Name != "main.main" {
|
|
|
|
t.Fatal("Program stopped at incorrect place")
|
|
|
|
}
|
2015-04-19 22:11:33 +00:00
|
|
|
if err = p.CallFn("runtime.getg", func() error {
|
|
|
|
th := p.CurrentThread
|
2015-04-23 15:40:33 +00:00
|
|
|
pc, err := th.PC()
|
2015-03-28 01:12:07 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2015-04-03 16:10:35 +00:00
|
|
|
f := th.Process.goSymTable.LookupFunc("runtime.getg")
|
2015-03-28 01:12:07 +00:00
|
|
|
if f == nil {
|
|
|
|
t.Fatalf("could not find function %s", "runtime.getg")
|
|
|
|
}
|
|
|
|
if pc-1 != f.End-2 && pc != f.End-2 {
|
|
|
|
t.Fatalf("wrong pc expected %#v got %#v", f.End-2, pc-1)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2015-04-23 15:40:33 +00:00
|
|
|
pc, err = p.PC()
|
2015-03-28 01:12:07 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2015-04-03 16:10:35 +00:00
|
|
|
fn = p.goSymTable.PCToFunc(pc)
|
2015-03-28 01:12:07 +00:00
|
|
|
if fn == nil {
|
|
|
|
t.Fatalf("Could not find func for PC: %#v", pc)
|
|
|
|
}
|
|
|
|
if fn.Name != "main.main" {
|
|
|
|
t.Fatal("Program stopped at incorrect place")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|