2017-06-04 19:48:15 +00:00
|
|
|
package main
|
|
|
|
|
2017-08-22 12:57:32 +00:00
|
|
|
import (
|
2017-11-14 17:26:28 +00:00
|
|
|
"os"
|
2017-08-22 12:57:32 +00:00
|
|
|
"sync/atomic"
|
|
|
|
)
|
2017-07-15 14:27:59 +00:00
|
|
|
|
|
|
|
const DefaultMTU = 1420
|
|
|
|
|
2017-08-07 13:25:04 +00:00
|
|
|
type TUNEvent int
|
|
|
|
|
|
|
|
const (
|
|
|
|
TUNEventUp = 1 << iota
|
|
|
|
TUNEventDown
|
|
|
|
TUNEventMTUUpdate
|
|
|
|
)
|
|
|
|
|
2017-06-28 21:45:45 +00:00
|
|
|
type TUNDevice interface {
|
2017-11-14 17:26:28 +00:00
|
|
|
File() *os.File // returns the file descriptor of the device
|
2017-07-11 20:48:58 +00:00
|
|
|
Read([]byte) (int, error) // read a packet from the device (without any additional headers)
|
|
|
|
Write([]byte) (int, error) // writes a packet to the device (without any additional headers)
|
|
|
|
MTU() (int, error) // returns the MTU of the device
|
|
|
|
Name() string // returns the current name
|
2017-08-07 13:25:04 +00:00
|
|
|
Events() chan TUNEvent // returns a constant channel of events related to the device
|
|
|
|
Close() error // stops the device and closes the event channel
|
2017-06-04 19:48:15 +00:00
|
|
|
}
|
2017-08-22 12:57:32 +00:00
|
|
|
|
|
|
|
func (device *Device) RoutineTUNEventReader() {
|
|
|
|
logInfo := device.log.Info
|
|
|
|
logError := device.log.Error
|
|
|
|
|
|
|
|
for event := range device.tun.device.Events() {
|
|
|
|
if event&TUNEventMTUUpdate != 0 {
|
|
|
|
mtu, err := device.tun.device.MTU()
|
|
|
|
old := atomic.LoadInt32(&device.tun.mtu)
|
|
|
|
if err != nil {
|
|
|
|
logError.Println("Failed to load updated MTU of device:", err)
|
|
|
|
} else if int(old) != mtu {
|
|
|
|
if mtu+MessageTransportSize > MaxMessageSize {
|
|
|
|
logInfo.Println("MTU updated:", mtu, "(too large)")
|
|
|
|
} else {
|
|
|
|
logInfo.Println("MTU updated:", mtu)
|
|
|
|
}
|
2017-08-22 15:22:45 +00:00
|
|
|
atomic.StoreInt32(&device.tun.mtu, int32(mtu))
|
2017-08-22 12:57:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if event&TUNEventUp != 0 {
|
|
|
|
if !device.tun.isUp.Get() {
|
2017-11-19 12:35:17 +00:00
|
|
|
// begin listening for incomming datagrams
|
2017-08-22 15:22:45 +00:00
|
|
|
logInfo.Println("Interface set up")
|
2017-08-22 12:57:32 +00:00
|
|
|
device.tun.isUp.Set(true)
|
2017-11-19 12:35:17 +00:00
|
|
|
updateBind(device)
|
2017-08-22 12:57:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if event&TUNEventDown != 0 {
|
|
|
|
if device.tun.isUp.Get() {
|
2017-11-19 12:35:17 +00:00
|
|
|
// stop listening for incomming datagrams
|
2017-08-22 15:22:45 +00:00
|
|
|
logInfo.Println("Interface set down")
|
2017-08-22 12:57:32 +00:00
|
|
|
device.tun.isUp.Set(false)
|
2017-11-19 12:35:17 +00:00
|
|
|
closeBind(device)
|
2017-08-22 12:57:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|