5
0
mirror of https://github.com/cwinfo/yggdrasil-go.git synced 2024-12-23 23:55:40 +00:00
yggdrasil-go/src/tuntap/tun.go

237 lines
7.0 KiB
Go
Raw Normal View History

package tuntap
2017-12-29 04:16:20 +00:00
// This manages the tun driver to send/recv packets to/from applications
2019-04-22 23:04:22 +00:00
// TODO: Crypto-key routing support
2019-04-22 22:12:13 +00:00
// TODO: Set MTU of session properly
2019-04-22 23:04:22 +00:00
// TODO: Reject packets that exceed session MTU with ICMPv6 for PMTU Discovery
// TODO: Connection timeouts (call Conn.Close() when we want to time out)
// TODO: Don't block in reader on writes that are pending searches
2019-04-22 22:12:13 +00:00
2018-06-12 22:50:08 +00:00
import (
2021-05-08 15:39:07 +00:00
"crypto/ed25519"
"errors"
"fmt"
2019-01-14 14:25:52 +00:00
"net"
2019-08-28 18:31:04 +00:00
2019-08-25 23:53:11 +00:00
//"sync"
"github.com/Arceliar/phony"
"github.com/gologme/log"
"golang.zx2c4.com/wireguard/tun"
2018-12-08 01:56:04 +00:00
"github.com/yggdrasil-network/yggdrasil-go/src/address"
"github.com/yggdrasil-network/yggdrasil-go/src/config"
2021-05-23 19:42:26 +00:00
"github.com/yggdrasil-network/yggdrasil-go/src/core"
2018-12-08 01:56:04 +00:00
"github.com/yggdrasil-network/yggdrasil-go/src/defaults"
2018-06-12 22:50:08 +00:00
)
2017-12-29 04:16:20 +00:00
type MTU uint16
// TunAdapter represents a running TUN interface and extends the
// yggdrasil.Adapter type. In order to use the TUN adapter with Yggdrasil, you
// should pass this object to the yggdrasil.SetRouterAdapter() function before
// calling yggdrasil.Start().
type TunAdapter struct {
2021-05-23 19:42:26 +00:00
core *core.Core
2021-05-08 15:39:07 +00:00
store keyStore
config *config.NodeConfig
2019-08-25 23:53:11 +00:00
log *log.Logger
addr address.Address
subnet address.Subnet
mtu uint64
iface tun.Device
2019-08-25 23:53:11 +00:00
phony.Inbox // Currently only used for _handlePacket from the reader, TODO: all the stuff that currently needs a mutex below
//mutex sync.RWMutex // Protects the below
isOpen bool
2021-05-16 18:52:52 +00:00
isEnabled bool // Used by the writer to drop sessionTraffic if not enabled
gatekeeper func(pubkey ed25519.PublicKey, initiator bool) bool
2021-05-23 16:58:52 +00:00
proto protoHandler
}
func (tun *TunAdapter) SetSessionGatekeeper(gatekeeper func(pubkey ed25519.PublicKey, initiator bool) bool) {
phony.Block(tun, func() {
tun.gatekeeper = gatekeeper
})
2017-12-29 04:16:20 +00:00
}
2018-06-12 21:45:53 +00:00
// Gets the maximum supported MTU for the platform based on the defaults in
// defaults.GetDefaults().
func getSupportedMTU(mtu uint64) uint64 {
if mtu < 1280 {
return 1280
}
if mtu > MaximumMTU() {
return MaximumMTU()
}
return mtu
}
2019-03-29 18:18:31 +00:00
// Name returns the name of the adapter, e.g. "tun0". On Windows, this may
// return a canonical adapter name instead.
func (tun *TunAdapter) Name() string {
if name, err := tun.iface.Name(); err == nil {
return name
}
return ""
}
2019-03-29 18:18:31 +00:00
// MTU gets the adapter's MTU. This can range between 1280 and 65535, although
// the maximum value is determined by your platform. The returned value will
// never exceed that of MaximumMTU().
func (tun *TunAdapter) MTU() uint64 {
return getSupportedMTU(tun.mtu)
}
// DefaultName gets the default TUN interface name for your platform.
func DefaultName() string {
return defaults.GetDefaults().DefaultIfName
}
// DefaultMTU gets the default TUN interface MTU for your platform. This can
2019-03-29 18:18:31 +00:00
// be as high as MaximumMTU(), depending on platform, but is never lower than 1280.
func DefaultMTU() uint64 {
return defaults.GetDefaults().DefaultIfMTU
}
// MaximumMTU returns the maximum supported TUN interface MTU for your
2019-03-29 18:18:31 +00:00
// platform. This can be as high as 65535, depending on platform, but is never
// lower than 1280.
func MaximumMTU() uint64 {
return defaults.GetDefaults().MaximumIfMTU
}
// Init initialises the TUN module. You must have acquired a Listener from
// the Yggdrasil core before this point and it must not be in use elsewhere.
func (tun *TunAdapter) Init(core *core.Core, config *config.NodeConfig, log *log.Logger, options interface{}) error {
tun.core = core
2021-05-08 15:39:07 +00:00
tun.store.init(tun)
tun.config = config
tun.log = log
2021-05-23 16:58:52 +00:00
tun.proto.init(tun)
tun.config.RLock()
if err := tun.proto.nodeinfo.setNodeInfo(tun.config.NodeInfo, tun.config.NodeInfoPrivacy); err != nil {
return fmt.Errorf("tun.proto.nodeinfo.setNodeInfo: %w", err)
}
tun.config.RUnlock()
if err := tun.core.SetOutOfBandHandler(tun.oobHandler); err != nil {
return fmt.Errorf("tun.core.SetOutOfBandHander: %w", err)
}
return nil
2017-12-29 04:16:20 +00:00
}
// Start the setup process for the TUN adapter. If successful, starts the
2019-08-25 23:53:11 +00:00
// reader actor to handle packets on that interface.
func (tun *TunAdapter) Start() error {
2019-08-25 23:53:11 +00:00
var err error
phony.Block(tun, func() {
2019-08-25 23:53:11 +00:00
err = tun._start()
})
return err
}
func (tun *TunAdapter) _start() error {
2019-10-24 22:37:39 +00:00
if tun.isOpen {
return errors.New("TUN module is already started")
2019-10-24 22:37:39 +00:00
}
tun.config.RLock()
defer tun.config.RUnlock()
2021-05-08 15:39:07 +00:00
if tun.config == nil {
return errors.New("no configuration available to TUN")
}
sk := tun.core.PrivateKey()
pk := sk.Public().(ed25519.PublicKey)
tun.addr = *address.AddrForKey(pk)
tun.subnet = *address.SubnetForKey(pk)
addr := fmt.Sprintf("%s/%d", net.IP(tun.addr[:]).String(), 8*len(address.GetPrefix())-1)
if tun.config.IfName == "none" || tun.config.IfName == "dummy" {
tun.log.Debugln("Not starting TUN as ifname is none or dummy")
2021-05-16 18:52:52 +00:00
tun.isEnabled = false
go tun.write()
2019-01-02 18:05:54 +00:00
return nil
2018-05-27 21:35:30 +00:00
}
mtu := tun.config.IfMTU
if tun.maxSessionMTU() < mtu {
mtu = tun.maxSessionMTU()
}
if err := tun.setup(tun.config.IfName, addr, mtu); err != nil {
return err
}
if tun.MTU() != mtu {
tun.log.Warnf("Warning: Interface MTU %d automatically adjusted to %d (supported range is 1280-%d)", tun.config.IfMTU, tun.MTU(), MaximumMTU())
}
tun.isOpen = true
2021-05-16 18:52:52 +00:00
tun.isEnabled = true
2021-05-08 16:35:04 +00:00
go tun.read()
go tun.write()
2018-05-27 21:35:30 +00:00
return nil
}
// IsStarted returns true if the module has been started.
func (tun *TunAdapter) IsStarted() bool {
2019-10-24 22:37:39 +00:00
var isOpen bool
phony.Block(tun, func() {
isOpen = tun.isOpen
})
return isOpen
}
// Start the setup process for the TUN adapter. If successful, starts the
// read/write goroutines to handle packets on that interface.
func (tun *TunAdapter) Stop() error {
2019-08-25 23:53:11 +00:00
var err error
phony.Block(tun, func() {
2019-08-25 23:53:11 +00:00
err = tun._stop()
})
return err
}
func (tun *TunAdapter) _stop() error {
tun.isOpen = false
// by TUN, e.g. readers/writers, sessions
if tun.iface != nil {
// Just in case we failed to start up the iface for some reason, this can apparently happen on Windows
tun.iface.Close()
}
return nil
}
func (tun *TunAdapter) oobHandler(fromKey, toKey ed25519.PublicKey, data []byte) {
if len(data) != 1+ed25519.SignatureSize {
return
}
sig := data[1:]
switch data[0] {
case typeKeyLookup:
snet := *address.SubnetForKey(toKey)
if snet == tun.subnet && ed25519.Verify(fromKey, toKey[:], sig) {
// This is looking for at least our subnet (possibly our address)
// Send a response
tun.sendKeyResponse(fromKey)
}
case typeKeyResponse:
// TODO keep a list of something to match against...
// Ignore the response if it doesn't match anything of interest...
if ed25519.Verify(fromKey, toKey[:], sig) {
tun.store.update(fromKey)
}
}
}
2021-05-08 15:39:07 +00:00
func (tun *TunAdapter) sendKeyLookup(partial ed25519.PublicKey) {
sig := ed25519.Sign(tun.core.PrivateKey(), partial[:])
bs := append([]byte{typeKeyLookup}, sig...)
2021-06-02 13:40:09 +00:00
_ = tun.core.SendOutOfBand(partial, bs)
2021-05-08 15:39:07 +00:00
}
func (tun *TunAdapter) sendKeyResponse(dest ed25519.PublicKey) {
sig := ed25519.Sign(tun.core.PrivateKey(), dest[:])
bs := append([]byte{typeKeyResponse}, sig...)
2021-06-02 13:40:09 +00:00
_ = tun.core.SendOutOfBand(dest, bs)
2021-05-08 15:39:07 +00:00
}
2021-05-16 19:05:22 +00:00
func (tun *TunAdapter) maxSessionMTU() uint64 {
const sessionTypeOverhead = 1
2021-05-16 19:05:22 +00:00
return tun.core.MTU() - sessionTypeOverhead
}