wintun: use nci.dll directly instead of buggy netshell
This commit is contained in:
		
							parent
							
								
									ef23100a4f
								
							
						
					
					
						commit
						68fea631d8
					
				
					 5 changed files with 102 additions and 84 deletions
				
			
		
							
								
								
									
										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