mirror of
https://github.com/go-delve/delve.git
synced 2025-10-29 17:56:45 +08:00
Ensure we can step/continue past breakpoints
This commit is contained in:
@ -25,8 +25,10 @@ type DebuggedProcess struct {
|
||||
|
||||
type BreakPoint struct {
|
||||
FunctionName string
|
||||
File string
|
||||
Line int
|
||||
Addr uint64
|
||||
OriginalData []byte
|
||||
}
|
||||
|
||||
// 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)
|
||||
}
|
||||
|
||||
_, ok := dbp.BreakPoints[fname]
|
||||
if ok {
|
||||
return nil, fmt.Errorf("Breakpoint already set")
|
||||
f, l, _ := dbp.GoSymTable.PCToLine(fn.Entry)
|
||||
|
||||
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 {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
breakpoint := &BreakPoint{
|
||||
FunctionName: fn.Name,
|
||||
Line: fn.LineTable.Line,
|
||||
Addr: fn.LineTable.PC,
|
||||
File: f,
|
||||
Line: l,
|
||||
Addr: fn.Entry,
|
||||
OriginalData: orginalData,
|
||||
}
|
||||
|
||||
dbp.BreakPoints[fname] = breakpoint
|
||||
@ -121,24 +128,43 @@ func (dbp *DebuggedProcess) Break(fname string) (*BreakPoint, error) {
|
||||
|
||||
// Steps through process.
|
||||
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()
|
||||
if err != nil {
|
||||
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())
|
||||
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
|
||||
}
|
||||
|
||||
// Continue process until next breakpoint.
|
||||
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))
|
||||
}
|
||||
|
||||
@ -196,3 +222,14 @@ func (dbp *DebuggedProcess) obtainGoSymbols() error {
|
||||
|
||||
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")
|
||||
sleepyaddr := sleepytimefunc.LineTable.PC
|
||||
sleepyaddr := sleepytimefunc.Entry
|
||||
|
||||
err = p.Continue()
|
||||
if err != nil {
|
||||
@ -130,35 +130,27 @@ func TestBreakPoint(t *testing.T) {
|
||||
|
||||
pc := regs.PC()
|
||||
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()
|
||||
}
|
||||
|
||||
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) {
|
||||
cmd, err := StartTestProcess("testprog")
|
||||
if err != nil {
|
||||
|
||||
Reference in New Issue
Block a user