yggdrasil-go/src/tuntap/tun_linux.go

87 lines
2.2 KiB
Go
Raw Normal View History

2019-01-02 18:05:54 +00:00
// +build !mobile
package tuntap
2017-12-28 22:16:20 -06:00
// The linux platform specific tun parts
2018-06-12 17:50:08 -05:00
import (
"errors"
"fmt"
"net"
2017-12-28 22:16:20 -06:00
2018-06-12 17:50:08 -05:00
"github.com/docker/libcontainer/netlink"
2018-06-12 17:50:08 -05:00
water "github.com/yggdrasil-network/water"
)
2018-06-12 22:45:53 +01:00
// Configures the TAP adapter with the correct IPv6 address and MTU.
func (tun *TunAdapter) setup(ifname string, iftapmode bool, addr string, mtu int) error {
var config water.Config
if iftapmode {
config = water.Config{DeviceType: water.TAP}
} else {
config = water.Config{DeviceType: water.TUN}
}
2018-01-04 22:37:51 +00:00
if ifname != "" && ifname != "auto" {
config.Name = ifname
}
iface, err := water.New(config)
2018-01-04 22:37:51 +00:00
if err != nil {
panic(err)
}
tun.iface = iface
2018-03-03 12:30:54 +00:00
tun.mtu = getSupportedMTU(mtu)
2018-07-19 10:01:12 +01:00
// The following check is specific to Linux, as the TAP driver only supports
// an MTU of 65535-14 to make room for the ethernet headers. This makes sure
// that the MTU gets rounded down to 65521 instead of causing a panic.
if iftapmode {
if tun.mtu > 65535-tun_ETHER_HEADER_LENGTH {
2018-07-20 23:02:25 -05:00
tun.mtu = 65535 - tun_ETHER_HEADER_LENGTH
2018-07-19 10:01:12 +01:00
}
}
// Friendly output
tun.log.Infof("Interface name: %s", tun.iface.Name())
tun.log.Infof("Interface IPv6: %s", addr)
tun.log.Infof("Interface MTU: %d", tun.mtu)
2018-01-04 22:37:51 +00:00
return tun.setupAddress(addr)
}
2018-06-12 22:45:53 +01:00
// Configures the TAP adapter with the correct IPv6 address and MTU. Netlink
// is used to do this, so there is not a hard requirement on "ip" or "ifconfig"
// to exist on the system, but this will fail if Netlink is not present in the
// kernel (it nearly always is).
func (tun *TunAdapter) setupAddress(addr string) error {
2018-01-04 22:37:51 +00:00
// Set address
var netIF *net.Interface
ifces, err := net.Interfaces()
if err != nil {
return err
}
for _, ifce := range ifces {
if ifce.Name == tun.iface.Name() {
var newIF = ifce
netIF = &newIF // Don't point inside ifces, it's apparently unsafe?...
}
}
if netIF == nil {
return errors.New(fmt.Sprintf("Failed to find interface: %s", tun.iface.Name()))
}
ip, ipNet, err := net.ParseCIDR(addr)
if err != nil {
return err
}
err = netlink.NetworkLinkAddIp(netIF, ip, ipNet)
if err != nil {
return err
}
err = netlink.NetworkSetMTU(netIF, tun.mtu)
2018-01-04 22:37:51 +00:00
if err != nil {
return err
}
netlink.NetworkLinkUp(netIF)
2018-01-04 22:37:51 +00:00
if err != nil {
return err
}
return nil
2017-12-28 22:16:20 -06:00
}