From 4cd06c0925ceb3dccd07b5ca1bdcdd6c24cfc77b Mon Sep 17 00:00:00 2001 From: "Jason A. Donenfeld" Date: Sat, 7 Sep 2019 00:02:07 -0500 Subject: [PATCH] tun: openbsd: check for interface already being up In some cases, we operate on an already-up interface, or the user brings up the interface before we start monitoring. For those situations, we should first check if the interface is already up. This still technically races between the initial check and the start of the route loop, but fixing that is a bit ugly and probably not worth it at the moment. Reported-by: Theo Buehler --- tun/tun_openbsd.go | 62 ++++++++++++++++++++++++++-------------------- 1 file changed, 35 insertions(+), 27 deletions(-) diff --git a/tun/tun_openbsd.go b/tun/tun_openbsd.go index bbc0432..44cedaa 100644 --- a/tun/tun_openbsd.go +++ b/tun/tun_openbsd.go @@ -42,34 +42,11 @@ func (tun *NativeTun) routineRouteListener(tunIfindex int) { defer close(tun.events) - data := make([]byte, os.Getpagesize()) - for { - retry: - n, err := unix.Read(tun.routeSocket, data) - if err != nil { - if errno, ok := err.(syscall.Errno); ok && errno == syscall.EINTR { - goto retry - } - tun.errors <- err - return - } - - if n < 8 { - continue - } - - if data[3 /* type */] != unix.RTM_IFINFO { - continue - } - ifindex := int(*(*uint16)(unsafe.Pointer(&data[6 /* ifindex */]))) - if ifindex != tunIfindex { - continue - } - - iface, err := net.InterfaceByIndex(ifindex) + check := func() bool { + iface, err := net.InterfaceByIndex(tunIfindex) if err != nil { tun.errors <- err - return + return true } // Up / Down event @@ -87,6 +64,38 @@ func (tun *NativeTun) routineRouteListener(tunIfindex int) { tun.events <- EventMTUUpdate } statusMTU = iface.MTU + return false + } + + if check() { + return + } + + data := make([]byte, os.Getpagesize()) + for { + n, err := unix.Read(tun.routeSocket, data) + if err != nil { + if errno, ok := err.(syscall.Errno); ok && errno == syscall.EINTR { + continue + } + tun.errors <- err + return + } + + if n < 8 { + continue + } + + if data[3 /* type */] != unix.RTM_IFINFO { + continue + } + ifindex := int(*(*uint16)(unsafe.Pointer(&data[6 /* ifindex */]))) + if ifindex != tunIfindex { + continue + } + if check() { + return + } } } @@ -140,7 +149,6 @@ func CreateTUN(name string, mtu int) (Device, error) { } func CreateTUNFromFile(file *os.File, mtu int) (Device, error) { - tun := &NativeTun{ tunFile: file, events: make(chan Event, 10),