package main
import (
+ "./rwcancel"
"errors"
"golang.org/x/sys/unix"
"net"
}
type NativeBind struct {
- sock4 int
- sock6 int
- netlinkSock int
- lastMark uint32
+ sock4 int
+ sock6 int
+ netlinkSock int
+ netlinkCancel *rwcancel.RWCancel
+ lastMark uint32
}
var _ Endpoint = (*NativeEndpoint)(nil)
if err != nil {
return nil, 0, err
}
+ bind.netlinkCancel, err = rwcancel.NewRWCancel(bind.netlinkSock)
+ if err != nil {
+ unix.Close(bind.netlinkSock)
+ return nil, 0, err
+ }
go bind.routineRouteListener(device)
bind.sock6, port, err = create6(port)
if err != nil {
- unix.Close(bind.netlinkSock)
+ bind.netlinkCancel.Cancel()
return nil, port, err
}
bind.sock4, port, err = create4(port)
if err != nil {
- unix.Close(bind.netlinkSock)
+ bind.netlinkCancel.Cancel()
unix.Close(bind.sock6)
}
return &bind, port, err
func (bind *NativeBind) Close() error {
err1 := closeUnblock(bind.sock6)
err2 := closeUnblock(bind.sock4)
- err3 := closeUnblock(bind.netlinkSock)
+ err3 := bind.netlinkCancel.Cancel()
+
if err1 != nil {
return err1
}
func (bind *NativeBind) routineRouteListener(device *Device) {
var reqPeer map[uint32]*Peer
+ defer unix.Close(bind.netlinkSock)
+
for msg := make([]byte, 1<<16); ; {
- msgn, _, _, _, err := unix.Recvmsg(bind.netlinkSock, msg[:], nil, 0)
+ var err error
+ var msgn int
+ for {
+ msgn, _, _, _, err = unix.Recvmsg(bind.netlinkSock, msg[:], nil, 0)
+ if err == nil || !rwcancel.ErrorIsEAGAIN(err) {
+ break
+ }
+ if !bind.netlinkCancel.ReadyRead() {
+ return
+ }
+ }
if err != nil {
return
}
)
type NativeTun struct {
- fd *os.File
- index int32 // if index
- name string // name of interface
- errors chan error // async error handling
- events chan TUNEvent // device related events
- nopi bool // the device was pased IFF_NO_PI
- rwcancel *rwcancel.RWCancel
- netlinkSock int
+ fd *os.File
+ fdCancel *rwcancel.RWCancel
+ index int32 // if index
+ name string // name of interface
+ errors chan error // async error handling
+ events chan TUNEvent // device related events
+ nopi bool // the device was pased IFF_NO_PI
+ netlinkSock int
+ netlinkCancel *rwcancel.RWCancel
+
statusListenersShutdown chan struct{}
}
}
func (tun *NativeTun) RoutineNetlinkListener() {
+ defer unix.Close(tun.netlinkSock)
+
for msg := make([]byte, 1<<16); ; {
- msgn, _, _, _, err := unix.Recvmsg(tun.netlinkSock, msg[:], nil, 0)
+ var err error
+ var msgn int
+ for {
+ msgn, _, _, _, err = unix.Recvmsg(tun.netlinkSock, msg[:], nil, 0)
+ if err == nil || !rwcancel.ErrorIsEAGAIN(err) {
+ break
+ }
+ if !tun.netlinkCancel.ReadyRead() {
+ tun.errors <- fmt.Errorf("netlink socket closed: %s", err.Error())
+ return
+ }
+ }
if err != nil {
tun.errors <- fmt.Errorf("failed to receive netlink message: %s", err.Error())
return
if err == nil || !rwcancel.ErrorIsEAGAIN(err) {
return n, err
}
- if !tun.rwcancel.ReadyRead() {
+ if !tun.fdCancel.ReadyRead() {
return 0, errors.New("tun device closed")
}
}
}
func (tun *NativeTun) Close() error {
+ var err1 error
close(tun.statusListenersShutdown)
- err1 := closeUnblock(tun.netlinkSock)
+ if tun.netlinkCancel != nil {
+ err1 = tun.netlinkCancel.Cancel()
+ }
err2 := tun.fd.Close()
- err3 := tun.rwcancel.Cancel()
+ err3 := tun.fdCancel.Cancel()
close(tun.events)
if err1 != nil {
}
var err error
- tun.rwcancel, err = rwcancel.NewRWCancel(int(fd.Fd()))
+ tun.fdCancel, err = rwcancel.NewRWCancel(int(fd.Fd()))
if err != nil {
+ tun.fd.Close()
return nil, err
}
_, err = tun.Name()
if err != nil {
+ tun.fd.Close()
return nil, err
}
tun.netlinkSock, err = createNetlinkSocket()
if err != nil {
+ tun.fd.Close()
+ return nil, err
+ }
+ tun.netlinkCancel, err = rwcancel.NewRWCancel(tun.netlinkSock)
+ if err != nil {
+ tun.fd.Close()
return nil, err
}