summaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorJason A. Donenfeld <Jason@zx2c4.com>2018-04-18 07:54:39 +0200
committerJason A. Donenfeld <Jason@zx2c4.com>2018-04-18 10:17:04 +0200
commit0b940a756838efcb90363ad99cb085d077b78e96 (patch)
tree7438647654e830a76da66758606d03c7826ab5c8
parenteb6728400ba656990aa4518dc4f5f2a6f0f19f8e (diff)
Use socketcall on x86
-rw-r--r--conn_linux.go50
-rw-r--r--syscall_linux.go30
-rw-r--r--syscall_linux_386.go53
3 files changed, 89 insertions, 44 deletions
diff --git a/conn_linux.go b/conn_linux.go
index cdba74f..8b60d65 100644
--- a/conn_linux.go
+++ b/conn_linux.go
@@ -391,14 +391,7 @@ func send6(sock int, end *NativeEndpoint, buff []byte) error {
msghdr.SetControllen(int(unsafe.Sizeof(cmsg)))
- // sendmsg(sock, &msghdr, 0)
-
- _, _, errno := unix.Syscall(
- unix.SYS_SENDMSG,
- uintptr(sock),
- uintptr(unsafe.Pointer(&msghdr)),
- 0,
- )
+ _, _, errno := sendmsg(sock, &msghdr, 0)
if errno == 0 {
return nil
@@ -409,12 +402,7 @@ func send6(sock int, end *NativeEndpoint, buff []byte) error {
if errno == unix.EINVAL {
end.ClearSrc()
cmsg.pktinfo = unix.Inet6Pktinfo{}
- _, _, errno = unix.Syscall(
- unix.SYS_SENDMSG,
- uintptr(sock),
- uintptr(unsafe.Pointer(&msghdr)),
- 0,
- )
+ _, _, errno = sendmsg(sock, &msghdr, 0)
}
return errno
@@ -455,26 +443,14 @@ func send4(sock int, end *NativeEndpoint, buff []byte) error {
}
msghdr.SetControllen(int(unsafe.Sizeof(cmsg)))
- // sendmsg(sock, &msghdr, 0)
-
- _, _, errno := unix.Syscall(
- unix.SYS_SENDMSG,
- uintptr(sock),
- uintptr(unsafe.Pointer(&msghdr)),
- 0,
- )
+ _, _, errno := sendmsg(sock, &msghdr, 0)
// clear source and try again
if errno == unix.EINVAL {
end.ClearSrc()
cmsg.pktinfo = unix.Inet4Pktinfo{}
- _, _, errno = unix.Syscall(
- unix.SYS_SENDMSG,
- uintptr(sock),
- uintptr(unsafe.Pointer(&msghdr)),
- 0,
- )
+ _, _, errno = sendmsg(sock, &msghdr, 0)
}
// errno = 0 is still an error instance
@@ -507,14 +483,7 @@ func receive4(sock int, buff []byte, end *NativeEndpoint) (int, error) {
msghdr.Control = (*byte)(unsafe.Pointer(&cmsg))
msghdr.SetControllen(int(unsafe.Sizeof(cmsg)))
- // recvmsg(sock, &mskhdr, 0)
-
- size, _, errno := unix.Syscall(
- unix.SYS_RECVMSG,
- uintptr(sock),
- uintptr(unsafe.Pointer(&msghdr)),
- 0,
- )
+ size, _, errno := recvmsg(sock, &msghdr, 0)
if errno != 0 {
return 0, errno
@@ -555,14 +524,7 @@ func receive6(sock int, buff []byte, end *NativeEndpoint) (int, error) {
msg.Control = (*byte)(unsafe.Pointer(&cmsg))
msg.SetControllen(int(unsafe.Sizeof(cmsg)))
- // recvmsg(sock, &mskhdr, 0)
-
- size, _, errno := unix.Syscall(
- unix.SYS_RECVMSG,
- uintptr(sock),
- uintptr(unsafe.Pointer(&msg)),
- 0,
- )
+ size, _, errno := recvmsg(sock, &msg, 0)
if errno != 0 {
return 0, errno
diff --git a/syscall_linux.go b/syscall_linux.go
new file mode 100644
index 0000000..3403544
--- /dev/null
+++ b/syscall_linux.go
@@ -0,0 +1,30 @@
+// +build linux,!386
+
+/* Copyright 2018 Jason A. Donenfeld <Jason@zx2c4.com>. All Rights Reserved.
+ */
+
+package main
+
+import (
+ "golang.org/x/sys/unix"
+ "syscall"
+ "unsafe"
+)
+
+func sendmsg(fd int, msghdr *unix.Msghdr, flags int) (uintptr, uintptr, syscall.Errno) {
+ return unix.Syscall(
+ unix.SYS_SENDMSG,
+ uintptr(fd),
+ uintptr(unsafe.Pointer(msghdr)),
+ uintptr(flags),
+ )
+}
+
+func recvmsg(fd int, msghdr *unix.Msghdr, flags int) (uintptr, uintptr, syscall.Errno) {
+ return unix.Syscall(
+ unix.SYS_RECVMSG,
+ uintptr(fd),
+ uintptr(unsafe.Pointer(msghdr)),
+ uintptr(flags),
+ )
+}
diff --git a/syscall_linux_386.go b/syscall_linux_386.go
new file mode 100644
index 0000000..76d7c7e
--- /dev/null
+++ b/syscall_linux_386.go
@@ -0,0 +1,53 @@
+// +build linux,386
+
+/* Copyright 2018 Jason A. Donenfeld <Jason@zx2c4.com>. All Rights Reserved.
+ */
+
+package main
+
+import (
+ "golang.org/x/sys/unix"
+ "syscall"
+ "unsafe"
+)
+
+const (
+ _SENDMSG = 16
+ _RECVMSG = 17
+)
+
+func sendmsg(fd int, msghdr *unix.Msghdr, flags int) (uintptr, uintptr, syscall.Errno) {
+ args := struct {
+ fd uintptr
+ msghdr uintptr
+ flags uintptr
+ }{
+ uintptr(fd),
+ uintptr(unsafe.Pointer(msghdr)),
+ uintptr(flags),
+ }
+ return unix.Syscall(
+ unix.SYS_SOCKETCALL,
+ _SENDMSG,
+ uintptr(unsafe.Pointer(&args)),
+ 0,
+ )
+}
+
+func recvmsg(fd int, msghdr *unix.Msghdr, flags int) (uintptr, uintptr, syscall.Errno) {
+ args := struct {
+ fd uintptr
+ msghdr uintptr
+ flags uintptr
+ }{
+ uintptr(fd),
+ uintptr(unsafe.Pointer(msghdr)),
+ uintptr(flags),
+ }
+ return unix.Syscall(
+ unix.SYS_SOCKETCALL,
+ _RECVMSG,
+ uintptr(unsafe.Pointer(&args)),
+ 0,
+ )
+}