wintun: use nci.dll directly instead of buggy netshell
This commit is contained in:
parent
ef23100a4f
commit
68fea631d8
8
tun/wintun/nci/mksyscall.go
Normal file
8
tun/wintun/nci/mksyscall.go
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
/* SPDX-License-Identifier: MIT
|
||||||
|
*
|
||||||
|
* Copyright (C) 2019 WireGuard LLC. All Rights Reserved.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package nci
|
||||||
|
|
||||||
|
//go:generate go run $GOROOT/src/syscall/mksyscall_windows.go -output zsyscall_windows.go nci_windows.go
|
28
tun/wintun/nci/nci_windows.go
Normal file
28
tun/wintun/nci/nci_windows.go
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
/* SPDX-License-Identifier: MIT
|
||||||
|
*
|
||||||
|
* Copyright (C) 2019 WireGuard LLC. All Rights Reserved.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package nci
|
||||||
|
|
||||||
|
import "golang.org/x/sys/windows"
|
||||||
|
|
||||||
|
//sys nciSetConnectionName(guid *windows.GUID, newName *uint16) (ret error) = nci.NciSetConnectionName
|
||||||
|
//sys nciGetConnectionName(guid *windows.GUID, destName *uint16, inDestNameBytes uint32, outDestNameBytes *uint32) (ret error) = nci.NciGetConnectionName
|
||||||
|
|
||||||
|
func SetConnectionName(guid *windows.GUID, newName string) error {
|
||||||
|
newName16, err := windows.UTF16PtrFromString(newName)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nciSetConnectionName(guid, newName16)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ConnectionName(guid *windows.GUID) (string, error) {
|
||||||
|
var name [0x400]uint16
|
||||||
|
err := nciGetConnectionName(guid, &name[0], uint32(len(name)*2), nil)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return windows.UTF16ToString(name[:]), nil
|
||||||
|
}
|
60
tun/wintun/nci/zsyscall_windows.go
Normal file
60
tun/wintun/nci/zsyscall_windows.go
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
// Code generated by 'go generate'; DO NOT EDIT.
|
||||||
|
|
||||||
|
package nci
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"golang.org/x/sys/windows"
|
||||||
|
)
|
||||||
|
|
||||||
|
var _ unsafe.Pointer
|
||||||
|
|
||||||
|
// Do the interface allocations only once for common
|
||||||
|
// Errno values.
|
||||||
|
const (
|
||||||
|
errnoERROR_IO_PENDING = 997
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
errERROR_IO_PENDING error = syscall.Errno(errnoERROR_IO_PENDING)
|
||||||
|
)
|
||||||
|
|
||||||
|
// errnoErr returns common boxed Errno values, to prevent
|
||||||
|
// allocations at runtime.
|
||||||
|
func errnoErr(e syscall.Errno) error {
|
||||||
|
switch e {
|
||||||
|
case 0:
|
||||||
|
return nil
|
||||||
|
case errnoERROR_IO_PENDING:
|
||||||
|
return errERROR_IO_PENDING
|
||||||
|
}
|
||||||
|
// TODO: add more here, after collecting data on the common
|
||||||
|
// error values see on Windows. (perhaps when running
|
||||||
|
// all.bat?)
|
||||||
|
return e
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
modnci = windows.NewLazySystemDLL("nci.dll")
|
||||||
|
|
||||||
|
procNciSetConnectionName = modnci.NewProc("NciSetConnectionName")
|
||||||
|
procNciGetConnectionName = modnci.NewProc("NciGetConnectionName")
|
||||||
|
)
|
||||||
|
|
||||||
|
func nciSetConnectionName(guid *windows.GUID, newName *uint16) (ret error) {
|
||||||
|
r0, _, _ := syscall.Syscall(procNciSetConnectionName.Addr(), 2, uintptr(unsafe.Pointer(guid)), uintptr(unsafe.Pointer(newName)), 0)
|
||||||
|
if r0 != 0 {
|
||||||
|
ret = syscall.Errno(r0)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func nciGetConnectionName(guid *windows.GUID, destName *uint16, inDestNameBytes uint32, outDestNameBytes *uint32) (ret error) {
|
||||||
|
r0, _, _ := syscall.Syscall6(procNciGetConnectionName.Addr(), 4, uintptr(unsafe.Pointer(guid)), uintptr(unsafe.Pointer(destName)), uintptr(inDestNameBytes), uintptr(unsafe.Pointer(outDestNameBytes)), 0, 0)
|
||||||
|
if r0 != 0 {
|
||||||
|
ret = syscall.Errno(r0)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
|
@ -1,32 +0,0 @@
|
||||||
/* SPDX-License-Identifier: MIT
|
|
||||||
*
|
|
||||||
* Copyright (C) 2019 WireGuard LLC. All Rights Reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package netshell
|
|
||||||
|
|
||||||
import (
|
|
||||||
"syscall"
|
|
||||||
"unsafe"
|
|
||||||
|
|
||||||
"golang.org/x/sys/windows"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
modnetshell = windows.NewLazySystemDLL("netshell.dll")
|
|
||||||
procHrRenameConnection = modnetshell.NewProc("HrRenameConnection")
|
|
||||||
)
|
|
||||||
|
|
||||||
func HrRenameConnection(guid *windows.GUID, newName *uint16) (err error) {
|
|
||||||
err = procHrRenameConnection.Find()
|
|
||||||
if err != nil {
|
|
||||||
// Missing from servercore, so we can't presume it's always there.
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
ret, _, _ := syscall.Syscall(procHrRenameConnection.Addr(), 2, uintptr(unsafe.Pointer(guid)), uintptr(unsafe.Pointer(newName)), 0)
|
|
||||||
if ret != 0 {
|
|
||||||
err = syscall.Errno(ret)
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
|
@ -15,7 +15,7 @@ import (
|
||||||
"golang.org/x/sys/windows"
|
"golang.org/x/sys/windows"
|
||||||
"golang.org/x/sys/windows/registry"
|
"golang.org/x/sys/windows/registry"
|
||||||
|
|
||||||
"golang.zx2c4.com/wireguard/tun/wintun/netshell"
|
"golang.zx2c4.com/wireguard/tun/wintun/nci"
|
||||||
registryEx "golang.zx2c4.com/wireguard/tun/wintun/registry"
|
registryEx "golang.zx2c4.com/wireguard/tun/wintun/registry"
|
||||||
"golang.zx2c4.com/wireguard/tun/wintun/setupapi"
|
"golang.zx2c4.com/wireguard/tun/wintun/setupapi"
|
||||||
)
|
)
|
||||||
|
@ -348,28 +348,6 @@ func CreateInterface(description string, requestedGUID *windows.GUID) (wintun *W
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wait for network registry key to emerge and populate.
|
|
||||||
netRegKey, err := registryEx.OpenKeyWait(
|
|
||||||
registry.LOCAL_MACHINE,
|
|
||||||
wintun.netRegKeyName(),
|
|
||||||
registry.QUERY_VALUE|registry.NOTIFY,
|
|
||||||
waitForRegistryTimeout)
|
|
||||||
if err != nil {
|
|
||||||
err = fmt.Errorf("makeWintun failed: %v", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer netRegKey.Close()
|
|
||||||
_, err = registryEx.GetStringValueWait(netRegKey, "Name", waitForRegistryTimeout)
|
|
||||||
if err != nil {
|
|
||||||
err = fmt.Errorf("GetStringValueWait(Name) failed: %v", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
_, err = registryEx.GetStringValueWait(netRegKey, "PnPInstanceId", waitForRegistryTimeout)
|
|
||||||
if err != nil {
|
|
||||||
err = fmt.Errorf("GetStringValueWait(PnPInstanceId) failed: %v", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Wait for TCP/IP adapter registry key to emerge and populate.
|
// Wait for TCP/IP adapter registry key to emerge and populate.
|
||||||
tcpipAdapterRegKey, err := registryEx.OpenKeyWait(
|
tcpipAdapterRegKey, err := registryEx.OpenKeyWait(
|
||||||
registry.LOCAL_MACHINE,
|
registry.LOCAL_MACHINE,
|
||||||
|
@ -539,29 +517,17 @@ func setQuietInstall(deviceInfoSet setupapi.DevInfo, deviceInfoData *setupapi.De
|
||||||
|
|
||||||
// InterfaceName returns the name of the Wintun interface.
|
// InterfaceName returns the name of the Wintun interface.
|
||||||
func (wintun *Wintun) InterfaceName() (string, error) {
|
func (wintun *Wintun) InterfaceName() (string, error) {
|
||||||
key, err := registry.OpenKey(registry.LOCAL_MACHINE, wintun.netRegKeyName(), registry.QUERY_VALUE)
|
return nci.ConnectionName(&wintun.cfgInstanceID)
|
||||||
if err != nil {
|
|
||||||
return "", fmt.Errorf("Network-specific registry key open failed: %v", err)
|
|
||||||
}
|
|
||||||
defer key.Close()
|
|
||||||
|
|
||||||
// Get the interface name.
|
|
||||||
return registryEx.GetStringValue(key, "Name")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetInterfaceName sets name of the Wintun interface.
|
// SetInterfaceName sets name of the Wintun interface.
|
||||||
func (wintun *Wintun) SetInterfaceName(ifname string) error {
|
func (wintun *Wintun) SetInterfaceName(ifname string) error {
|
||||||
netRegKey, err := registry.OpenKey(registry.LOCAL_MACHINE, wintun.netRegKeyName(), registry.SET_VALUE)
|
err := nci.SetConnectionName(&wintun.cfgInstanceID, ifname)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Network-specific registry key open failed: %v", err)
|
return fmt.Errorf("NciSetConnectionName failed: %v", err)
|
||||||
}
|
|
||||||
defer netRegKey.Close()
|
|
||||||
err = netRegKey.SetStringValue("Name", ifname)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: This only sometimes works.
|
// TODO: This should use NetSetup2 so that it doesn't get unset.
|
||||||
deviceRegKey, err := registry.OpenKey(registry.LOCAL_MACHINE, wintun.deviceRegKeyName(), registry.SET_VALUE)
|
deviceRegKey, err := registry.OpenKey(registry.LOCAL_MACHINE, wintun.deviceRegKeyName(), registry.SET_VALUE)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Device-level registry key open failed: %v", err)
|
return fmt.Errorf("Device-level registry key open failed: %v", err)
|
||||||
|
@ -569,23 +535,11 @@ func (wintun *Wintun) SetInterfaceName(ifname string) error {
|
||||||
defer deviceRegKey.Close()
|
defer deviceRegKey.Close()
|
||||||
err = deviceRegKey.SetStringValue("FriendlyName", deviceTypeName)
|
err = deviceRegKey.SetStringValue("FriendlyName", deviceTypeName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return fmt.Errorf("SetStringValue(FriendlyName) failed: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// We have to tell the various runtime COM services about the new name too. We ignore the
|
|
||||||
// error because netshell isn't available on servercore.
|
|
||||||
// TODO: netsh.exe falls back to NciSetConnection in this case. If somebody complains, maybe
|
|
||||||
// we should do the same.
|
|
||||||
// TODO: This only sometimes works.
|
|
||||||
netshell.HrRenameConnection(&wintun.cfgInstanceID, windows.StringToUTF16Ptr(ifname))
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// netRegKeyName returns the interface-specific network registry key name.
|
|
||||||
func (wintun *Wintun) netRegKeyName() string {
|
|
||||||
return fmt.Sprintf("SYSTEM\\CurrentControlSet\\Control\\Network\\%v\\%v\\Connection", deviceClassNetGUID, wintun.cfgInstanceID)
|
|
||||||
}
|
|
||||||
|
|
||||||
// tcpipAdapterRegKeyName returns the adapter-specific TCP/IP network registry key name.
|
// tcpipAdapterRegKeyName returns the adapter-specific TCP/IP network registry key name.
|
||||||
func (wintun *Wintun) tcpipAdapterRegKeyName() string {
|
func (wintun *Wintun) tcpipAdapterRegKeyName() string {
|
||||||
return fmt.Sprintf("SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters\\Adapters\\%v", wintun.cfgInstanceID)
|
return fmt.Sprintf("SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters\\Adapters\\%v", wintun.cfgInstanceID)
|
||||||
|
|
Loading…
Reference in a new issue