mirror of
https://github.com/cwinfo/yggdrasil-go.git
synced 2025-04-03 00:33:48 +00:00
141 lines
3.9 KiB
Go
141 lines
3.9 KiB
Go
//go:build windows
|
||
// +build windows
|
||
|
||
package tun
|
||
|
||
import (
|
||
"errors"
|
||
"fmt"
|
||
"log"
|
||
"net/netip"
|
||
|
||
"github.com/yggdrasil-network/yggdrasil-go/src/config"
|
||
"golang.org/x/sys/windows"
|
||
|
||
wgtun "golang.zx2c4.com/wireguard/tun"
|
||
"golang.zx2c4.com/wireguard/windows/elevate"
|
||
"golang.zx2c4.com/wireguard/windows/tunnel/winipcfg"
|
||
)
|
||
|
||
// This is to catch Windows platforms
|
||
|
||
// Configures the TUN adapter with the correct IPv6 address and MTU.
|
||
func (tun *TunAdapter) setup(ifname string, addr string, mtu uint64) error {
|
||
if ifname == "auto" {
|
||
ifname = config.GetDefaults().DefaultIfName
|
||
}
|
||
return elevate.DoAsSystem(func() error {
|
||
var err error
|
||
var iface wgtun.Device
|
||
var guid windows.GUID
|
||
if guid, err = windows.GUIDFromString("{8f59971a-7872-4aa6-b2eb-061fc4e9d0a7}"); err != nil {
|
||
return err
|
||
}
|
||
if iface, err = wgtun.CreateTUNWithRequestedGUID(ifname, &guid, int(mtu)); err != nil {
|
||
return err
|
||
}
|
||
tun.iface = iface
|
||
if err = tun.setupAddress(addr); err != nil {
|
||
tun.log.Errorln("Failed to set up TUN address:", err)
|
||
return err
|
||
}
|
||
if err = tun.setupMTU(getSupportedMTU(mtu)); err != nil {
|
||
tun.log.Errorln("Failed to set up TUN MTU:", err)
|
||
return err
|
||
}
|
||
if mtu, err := iface.MTU(); err == nil {
|
||
tun.mtu = uint64(mtu)
|
||
}
|
||
return nil
|
||
})
|
||
}
|
||
|
||
// Configures the "utun" adapter from an existing file descriptor.
|
||
func (tun *TunAdapter) setupFD(fd int32, addr string, mtu uint64) error {
|
||
return fmt.Errorf("setup via FD not supported on this platform")
|
||
}
|
||
|
||
// Sets the MTU of the TUN adapter.
|
||
func (tun *TunAdapter) setupMTU(mtu uint64) error {
|
||
if tun.iface == nil || tun.Name() == "" {
|
||
return errors.New("Can't configure MTU as TUN adapter is not present")
|
||
}
|
||
if intf, ok := tun.iface.(*wgtun.NativeTun); ok {
|
||
luid := winipcfg.LUID(intf.LUID())
|
||
ipfamily, err := luid.IPInterface(windows.AF_INET6)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
ipfamily.NLMTU = uint32(mtu)
|
||
intf.ForceMTU(int(ipfamily.NLMTU))
|
||
ipfamily.UseAutomaticMetric = false
|
||
ipfamily.Metric = 0
|
||
ipfamily.DadTransmits = 0
|
||
ipfamily.RouterDiscoveryBehavior = winipcfg.RouterDiscoveryDisabled
|
||
|
||
if err := ipfamily.Set(); err != nil {
|
||
return err
|
||
}
|
||
}
|
||
|
||
return nil
|
||
}
|
||
|
||
// Sets the IPv6 address of the TUN adapter.
|
||
func (tun *TunAdapter) setupAddress(addr string) error {
|
||
if tun.iface == nil || tun.Name() == "" {
|
||
return errors.New("Can't configure IPv6 address as TUN adapter is not present")
|
||
}
|
||
if intf, ok := tun.iface.(*wgtun.NativeTun); ok {
|
||
if ipnet, err := netip.ParsePrefix(addr); err == nil {
|
||
luid := winipcfg.LUID(intf.LUID())
|
||
addresses := []netip.Prefix{ipnet}
|
||
err := luid.SetIPAddressesForFamily(windows.AF_INET6, addresses)
|
||
if err == windows.ERROR_OBJECT_ALREADY_EXISTS {
|
||
cleanupAddressesOnDisconnectedInterfaces(windows.AF_INET6, addresses)
|
||
err = luid.SetIPAddressesForFamily(windows.AF_INET6, addresses)
|
||
}
|
||
if err != nil {
|
||
return err
|
||
}
|
||
} else {
|
||
return err
|
||
}
|
||
} else {
|
||
return errors.New("unable to get NativeTUN")
|
||
}
|
||
return nil
|
||
}
|
||
|
||
/*
|
||
* cleanupAddressesOnDisconnectedInterfaces
|
||
* SPDX-License-Identifier: MIT
|
||
* Copyright (C) 2019 WireGuard LLC. All Rights Reserved.
|
||
*/
|
||
func cleanupAddressesOnDisconnectedInterfaces(family winipcfg.AddressFamily, addresses []netip.Prefix) {
|
||
if len(addresses) == 0 {
|
||
return
|
||
}
|
||
addrHash := make(map[netip.Addr]bool, len(addresses))
|
||
for i := range addresses {
|
||
addrHash[addresses[i].Addr()] = true
|
||
}
|
||
interfaces, err := winipcfg.GetAdaptersAddresses(family, winipcfg.GAAFlagDefault)
|
||
if err != nil {
|
||
return
|
||
}
|
||
for _, iface := range interfaces {
|
||
if iface.OperStatus == winipcfg.IfOperStatusUp {
|
||
continue
|
||
}
|
||
for address := iface.FirstUnicastAddress; address != nil; address = address.Next {
|
||
if ip, _ := netip.AddrFromSlice(address.Address.IP()); addrHash[ip] {
|
||
prefix := netip.PrefixFrom(ip, int(address.OnLinkPrefixLength))
|
||
log.Printf("Cleaning up stale address %s from interface ‘%s’", prefix.String(), iface.FriendlyName())
|
||
iface.LUID.DeleteIPAddress(prefix)
|
||
}
|
||
}
|
||
}
|
||
}
|