mirror of
				https://github.com/go-delve/delve.git
				synced 2025-11-01 03:42:59 +08:00 
			
		
		
		
	Ensure we can step/continue past breakpoints
This commit is contained in:
		| @ -25,8 +25,10 @@ type DebuggedProcess struct { | |||||||
|  |  | ||||||
| type BreakPoint struct { | type BreakPoint struct { | ||||||
| 	FunctionName string | 	FunctionName string | ||||||
|  | 	File         string | ||||||
| 	Line         int | 	Line         int | ||||||
| 	Addr         uint64 | 	Addr         uint64 | ||||||
|  | 	OriginalData []byte | ||||||
| } | } | ||||||
|  |  | ||||||
| // Returns a new DebuggedProcess struct with sensible defaults. | // Returns a new DebuggedProcess struct with sensible defaults. | ||||||
| @ -97,21 +99,26 @@ func (dbp *DebuggedProcess) Break(fname string) (*BreakPoint, error) { | |||||||
| 		return nil, fmt.Errorf("No function named %s\n", fname) | 		return nil, fmt.Errorf("No function named %s\n", fname) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	_, ok := dbp.BreakPoints[fname] | 	f, l, _ := dbp.GoSymTable.PCToLine(fn.Entry) | ||||||
| 	if ok { |  | ||||||
| 		return nil, fmt.Errorf("Breakpoint already set") | 	orginalData := make([]byte, 1) | ||||||
|  | 	addr := uintptr(fn.Entry) | ||||||
|  | 	_, err := syscall.PtracePeekData(dbp.Pid, addr, orginalData) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	addr := uintptr(fn.LineTable.PC) | 	_, err = syscall.PtracePokeData(dbp.Pid, addr, int3) | ||||||
| 	_, err := syscall.PtracePokeData(dbp.Pid, addr, int3) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	breakpoint := &BreakPoint{ | 	breakpoint := &BreakPoint{ | ||||||
| 		FunctionName: fn.Name, | 		FunctionName: fn.Name, | ||||||
| 		Line:         fn.LineTable.Line, | 		File:         f, | ||||||
| 		Addr:         fn.LineTable.PC, | 		Line:         l, | ||||||
|  | 		Addr:         fn.Entry, | ||||||
|  | 		OriginalData: orginalData, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	dbp.BreakPoints[fname] = breakpoint | 	dbp.BreakPoints[fname] = breakpoint | ||||||
| @ -121,24 +128,43 @@ func (dbp *DebuggedProcess) Break(fname string) (*BreakPoint, error) { | |||||||
|  |  | ||||||
| // Steps through process. | // Steps through process. | ||||||
| func (dbp *DebuggedProcess) Step() error { | func (dbp *DebuggedProcess) Step() error { | ||||||
| 	err := dbp.handleResult(syscall.PtraceSingleStep(dbp.Pid)) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return fmt.Errorf("step failed: ", err.Error()) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	regs, err := dbp.Registers() | 	regs, err := dbp.Registers() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	bp, ok := dbp.PCtoBP(regs.PC()) | ||||||
|  | 	if ok { | ||||||
|  | 		dbp.restoreInstruction(regs.PC(), bp.OriginalData) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = dbp.handleResult(syscall.PtraceSingleStep(dbp.Pid)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fmt.Errorf("step failed: ", err.Error()) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	f, l, fn := dbp.GoSymTable.PCToLine(regs.PC()) | 	f, l, fn := dbp.GoSymTable.PCToLine(regs.PC()) | ||||||
| 	fmt.Printf("Stopped at: %s %s:%d\n", fn.Name, f, l) | 	fmt.Printf("Stopped at: %s %s:%d\n", fn.Name, f, l) | ||||||
|  |  | ||||||
|  | 	if ok { | ||||||
|  | 		_, err = dbp.Break(bp.FunctionName) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // Continue process until next breakpoint. | // Continue process until next breakpoint. | ||||||
| func (dbp *DebuggedProcess) Continue() error { | func (dbp *DebuggedProcess) Continue() error { | ||||||
|  | 	// Stepping first will ensure we are able to continue | ||||||
|  | 	// past a breakpoint if that's currently where we are stopped. | ||||||
|  | 	err := dbp.Step() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	return dbp.handleResult(syscall.PtraceCont(dbp.Pid, 0)) | 	return dbp.handleResult(syscall.PtraceCont(dbp.Pid, 0)) | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -196,3 +222,14 @@ func (dbp *DebuggedProcess) obtainGoSymbols() error { | |||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (dbp *DebuggedProcess) PCtoBP(pc uint64) (*BreakPoint, bool) { | ||||||
|  | 	_, _, fn := dbp.GoSymTable.PCToLine(pc) | ||||||
|  | 	bp, ok := dbp.BreakPoints[fn.Name] | ||||||
|  | 	return bp, ok | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (dbp *DebuggedProcess) restoreInstruction(pc uint64, data []byte) error { | ||||||
|  | 	_, err := syscall.PtracePokeData(dbp.Pid, uintptr(pc), data) | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  | |||||||
| @ -116,7 +116,7 @@ func TestBreakPoint(t *testing.T) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	sleepytimefunc := p.GoSymTable.LookupFunc("main.sleepytime") | 	sleepytimefunc := p.GoSymTable.LookupFunc("main.sleepytime") | ||||||
| 	sleepyaddr := sleepytimefunc.LineTable.PC | 	sleepyaddr := sleepytimefunc.Entry | ||||||
|  |  | ||||||
| 	err = p.Continue() | 	err = p.Continue() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -130,35 +130,27 @@ func TestBreakPoint(t *testing.T) { | |||||||
|  |  | ||||||
| 	pc := regs.PC() | 	pc := regs.PC() | ||||||
| 	if pc != sleepyaddr { | 	if pc != sleepyaddr { | ||||||
| 		t.Fatal("Break not respected:\nPC:%d\nFN:%d\n", pc, sleepyaddr) | 		t.Fatalf("Break not respected:\nPC:%d\nFN:%d\n", pc, sleepyaddr) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = p.Step() | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatal(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	regs, err = p.Registers() | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatal("Registers():", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	pc = regs.PC() | ||||||
|  | 	if pc == sleepyaddr { | ||||||
|  | 		t.Fatalf("Step not respected:\nPC:%d\nFN:%d\n", pc, sleepyaddr) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	cmd.Process.Kill() | 	cmd.Process.Kill() | ||||||
| } | } | ||||||
|  |  | ||||||
| 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") |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func TestBreakPointWithNonExistantFunction(t *testing.T) { | func TestBreakPointWithNonExistantFunction(t *testing.T) { | ||||||
| 	cmd, err := StartTestProcess("testprog") | 	cmd, err := StartTestProcess("testprog") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user
	 Derek Parker
					Derek Parker