
Replaces sys.Iovec with a similar struct that uses uintptr instead of *byte for the base field when referring to addresses of the target process, so that we do not generate invalid pointers. Fixes #2919
40 lines
1.2 KiB
Go
40 lines
1.2 KiB
Go
//go:build (linux && amd64) || (linux && arm64)
|
|
// +build linux,amd64 linux,arm64
|
|
|
|
package native
|
|
|
|
import (
|
|
"syscall"
|
|
"unsafe"
|
|
|
|
sys "golang.org/x/sys/unix"
|
|
)
|
|
|
|
// processVmRead calls process_vm_readv
|
|
func processVmRead(tid int, addr uintptr, data []byte) (int, error) {
|
|
len_iov := uint64(len(data))
|
|
local_iov := sys.Iovec{Base: &data[0], Len: len_iov}
|
|
remote_iov := remoteIovec{base: addr, len: uintptr(len_iov)}
|
|
p_local := uintptr(unsafe.Pointer(&local_iov))
|
|
p_remote := uintptr(unsafe.Pointer(&remote_iov))
|
|
n, _, err := syscall.Syscall6(sys.SYS_PROCESS_VM_READV, uintptr(tid), p_local, 1, p_remote, 1, 0)
|
|
if err != syscall.Errno(0) {
|
|
return 0, err
|
|
}
|
|
return int(n), nil
|
|
}
|
|
|
|
// processVmWrite calls process_vm_writev
|
|
func processVmWrite(tid int, addr uintptr, data []byte) (int, error) {
|
|
len_iov := uint64(len(data))
|
|
local_iov := sys.Iovec{Base: &data[0], Len: len_iov}
|
|
remote_iov := remoteIovec{base: addr, len: uintptr(len_iov)}
|
|
p_local := uintptr(unsafe.Pointer(&local_iov))
|
|
p_remote := uintptr(unsafe.Pointer(&remote_iov))
|
|
n, _, err := syscall.Syscall6(sys.SYS_PROCESS_VM_WRITEV, uintptr(tid), p_local, 1, p_remote, 1, 0)
|
|
if err != syscall.Errno(0) {
|
|
return 0, err
|
|
}
|
|
return int(n), nil
|
|
}
|