mirror of
https://github.com/go-delve/delve.git
synced 2025-10-30 02:07:58 +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