mirror of
				https://github.com/go-delve/delve.git
				synced 2025-10-31 02:36:18 +08:00 
			
		
		
		
	 58762685e3
			
		
	
	58762685e3
	
	
	
		
			
			Adds the low-level support for watchpoints (aka data breakpoints) to the native linux/amd64 backend. Does not add user interface or functioning support for watchpoints on stack variables. Updates #279
		
			
				
	
	
		
			122 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package native
 | |
| 
 | |
| import (
 | |
| 	"syscall"
 | |
| 	"unsafe"
 | |
| 
 | |
| 	sys "golang.org/x/sys/unix"
 | |
| 
 | |
| 	"github.com/go-delve/delve/pkg/proc"
 | |
| 	"github.com/go-delve/delve/pkg/proc/amd64util"
 | |
| 	"github.com/go-delve/delve/pkg/proc/linutil"
 | |
| )
 | |
| 
 | |
| func (t *nativeThread) restoreRegisters(savedRegs proc.Registers) error {
 | |
| 	sr := savedRegs.(*linutil.AMD64Registers)
 | |
| 
 | |
| 	var restoreRegistersErr error
 | |
| 	t.dbp.execPtraceFunc(func() {
 | |
| 		oldRegs := (*sys.PtraceRegs)(sr.Regs)
 | |
| 
 | |
| 		var currentRegs sys.PtraceRegs
 | |
| 		restoreRegistersErr = sys.PtraceGetRegs(t.ID, ¤tRegs)
 | |
| 		if restoreRegistersErr != nil {
 | |
| 			return
 | |
| 		}
 | |
| 		// restoreRegisters is only supposed to restore CPU registers, not FS_BASE and GS_BASE
 | |
| 		oldRegs.Fs_base = currentRegs.Fs_base
 | |
| 		oldRegs.Gs_base = currentRegs.Gs_base
 | |
| 
 | |
| 		restoreRegistersErr = sys.PtraceSetRegs(t.ID, oldRegs)
 | |
| 
 | |
| 		if restoreRegistersErr != nil {
 | |
| 			return
 | |
| 		}
 | |
| 		if sr.Fpregset.Xsave != nil {
 | |
| 			iov := sys.Iovec{Base: &sr.Fpregset.Xsave[0], Len: uint64(len(sr.Fpregset.Xsave))}
 | |
| 			_, _, restoreRegistersErr = syscall.Syscall6(syscall.SYS_PTRACE, sys.PTRACE_SETREGSET, uintptr(t.ID), _NT_X86_XSTATE, uintptr(unsafe.Pointer(&iov)), 0, 0)
 | |
| 			return
 | |
| 		}
 | |
| 
 | |
| 		_, _, restoreRegistersErr = syscall.Syscall6(syscall.SYS_PTRACE, sys.PTRACE_SETFPREGS, uintptr(t.ID), uintptr(0), uintptr(unsafe.Pointer(&sr.Fpregset.AMD64PtraceFpRegs)), 0, 0)
 | |
| 		return
 | |
| 	})
 | |
| 	if restoreRegistersErr == syscall.Errno(0) {
 | |
| 		restoreRegistersErr = nil
 | |
| 	}
 | |
| 	return restoreRegistersErr
 | |
| }
 | |
| 
 | |
| const debugRegUserOffset = 848 // offset of debug registers in the user struct, see source/arch/x86/kernel/ptrace.c
 | |
| 
 | |
| func (t *nativeThread) withDebugRegisters(f func(*amd64util.DebugRegisters) error) error {
 | |
| 	var err error
 | |
| 	t.dbp.execPtraceFunc(func() {
 | |
| 		debugregs := make([]uint64, 8)
 | |
| 
 | |
| 		for i := range debugregs {
 | |
| 			if i == 4 || i == 5 {
 | |
| 				continue
 | |
| 			}
 | |
| 			_, _, err = sys.Syscall6(sys.SYS_PTRACE, sys.PTRACE_PEEKUSR, uintptr(t.ID), uintptr(debugRegUserOffset+uintptr(i)*unsafe.Sizeof(debugregs[0])), uintptr(unsafe.Pointer(&debugregs[i])), 0, 0)
 | |
| 			if err != nil && err != syscall.Errno(0) {
 | |
| 				return
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		drs := amd64util.NewDebugRegisters(&debugregs[0], &debugregs[1], &debugregs[2], &debugregs[3], &debugregs[6], &debugregs[7])
 | |
| 
 | |
| 		err = f(drs)
 | |
| 
 | |
| 		if drs.Dirty {
 | |
| 			for i := range debugregs {
 | |
| 				if i == 4 || i == 5 {
 | |
| 					// Linux will return EIO for DR4 and DR5
 | |
| 					continue
 | |
| 				}
 | |
| 				_, _, err = sys.Syscall6(sys.SYS_PTRACE, sys.PTRACE_POKEUSR, uintptr(t.ID), uintptr(debugRegUserOffset+uintptr(i)*unsafe.Sizeof(debugregs[0])), uintptr(debugregs[i]), 0, 0)
 | |
| 				if err != nil && err != syscall.Errno(0) {
 | |
| 					return
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	})
 | |
| 	if err == syscall.Errno(0) || err == sys.ESRCH {
 | |
| 		err = nil
 | |
| 	}
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| func (t *nativeThread) writeHardwareBreakpoint(addr uint64, wtype proc.WatchType, idx uint8) error {
 | |
| 	return t.withDebugRegisters(func(drs *amd64util.DebugRegisters) error {
 | |
| 		return drs.SetBreakpoint(idx, addr, wtype.Read(), wtype.Write(), wtype.Size())
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func (t *nativeThread) clearHardwareBreakpoint(addr uint64, wtype proc.WatchType, idx uint8) error {
 | |
| 	return t.withDebugRegisters(func(drs *amd64util.DebugRegisters) error {
 | |
| 		drs.ClearBreakpoint(idx)
 | |
| 		return nil
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func (t *nativeThread) findHardwareBreakpoint() (*proc.Breakpoint, error) {
 | |
| 	var retbp *proc.Breakpoint
 | |
| 	err := t.withDebugRegisters(func(drs *amd64util.DebugRegisters) error {
 | |
| 		ok, idx := drs.GetActiveBreakpoint()
 | |
| 		if ok {
 | |
| 			for _, bp := range t.dbp.Breakpoints().M {
 | |
| 				if bp.WatchType != 0 && bp.HWBreakIndex == idx {
 | |
| 					retbp = bp
 | |
| 					break
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return nil
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return retbp, nil
 | |
| }
 |