38c7acd70f
The SetupDiEnumDeviceInfo() now returns a SP_DEVINFO_DATA rather than taking it on input to fill it on return. Signed-off-by: Simon Rozman <simon@rozman.si>
180 lines
8.4 KiB
Go
180 lines
8.4 KiB
Go
/* SPDX-License-Identifier: MIT
|
|
*
|
|
* Copyright (C) 2019 WireGuard LLC. All Rights Reserved.
|
|
*/
|
|
|
|
package setupapi
|
|
|
|
import (
|
|
"syscall"
|
|
"unsafe"
|
|
|
|
"golang.org/x/sys/windows"
|
|
"golang.org/x/sys/windows/registry"
|
|
)
|
|
|
|
//sys setupDiClassNameFromGuidEx(ClassGUID *windows.GUID, ClassName *uint16, ClassNameSize uint32, RequiredSize *uint32, MachineName *uint16, Reserved uintptr) (err error) = setupapi.SetupDiClassNameFromGuidExW
|
|
//sys setupDiClassGuidsFromNameEx(ClassName *uint16, ClassGuidList *windows.GUID, ClassGuidListSize uint32, RequiredSize *uint32, MachineName *uint16, Reserved uintptr) (err error) = setupapi.SetupDiClassGuidsFromNameExW
|
|
//sys setupDiGetClassDevsEx(ClassGUID *windows.GUID, Enumerator *uint16, hwndParent uintptr, Flags DIGCF, DeviceInfoSet DevInfo, MachineName *uint16, reserved uintptr) (handle DevInfo, err error) [failretval==DevInfo(windows.InvalidHandle)] = setupapi.SetupDiGetClassDevsExW
|
|
//sys SetupDiDestroyDeviceInfoList(DeviceInfoSet DevInfo) (err error) = setupapi.SetupDiDestroyDeviceInfoList
|
|
//sys setupDiGetDeviceInfoListDetail(DeviceInfoSet DevInfo, DeviceInfoSetDetailData *_SP_DEVINFO_LIST_DETAIL_DATA) (err error) = setupapi.SetupDiGetDeviceInfoListDetailW
|
|
//sys setupDiEnumDeviceInfo(DeviceInfoSet DevInfo, MemberIndex uint32, DeviceInfoData *SP_DEVINFO_DATA) (err error) = setupapi.SetupDiEnumDeviceInfo
|
|
//sys setupDiOpenDevRegKey(DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA, Scope DICS_FLAG, HwProfile uint32, KeyType DIREG, samDesired uint32) (key windows.Handle, err error) [failretval==windows.InvalidHandle] = setupapi.SetupDiOpenDevRegKey
|
|
//sys setupDiGetDeviceInstallParams(DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA, DeviceInstallParams *_SP_DEVINSTALL_PARAMS) (err error) = setupapi.SetupDiGetDeviceInstallParamsW
|
|
//sys setupDiSetDeviceInstallParams(DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA, DeviceInstallParams *_SP_DEVINSTALL_PARAMS) (err error) = setupapi.SetupDiSetDeviceInstallParamsW
|
|
//sys SetupDiGetClassInstallParams(DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA, ClassInstallParams *SP_CLASSINSTALL_HEADER, ClassInstallParamsSize uint32, RequiredSize *uint32) (err error) = setupapi.SetupDiGetClassInstallParamsW
|
|
//sys SetupDiSetClassInstallParams(DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA, ClassInstallParams *SP_CLASSINSTALL_HEADER, ClassInstallParamsSize uint32) (err error) = setupapi.SetupDiSetClassInstallParamsW
|
|
//sys SetupDiCallClassInstaller(InstallFunction DI_FUNCTION, DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA) (err error) = setupapi.SetupDiCallClassInstaller
|
|
|
|
// SetupDiClassNameFromGuidEx function retrieves the class name associated with a class GUID. The class can be installed on a local or remote computer.
|
|
func SetupDiClassNameFromGuidEx(ClassGUID *windows.GUID, MachineName string) (ClassName string, err error) {
|
|
var classNameUTF16 [MAX_CLASS_NAME_LEN]uint16
|
|
|
|
var machineNameUTF16 *uint16
|
|
if MachineName != "" {
|
|
machineNameUTF16, err = syscall.UTF16PtrFromString(MachineName)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
|
|
err = setupDiClassNameFromGuidEx(ClassGUID, &classNameUTF16[0], MAX_CLASS_NAME_LEN, nil, machineNameUTF16, 0)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
ClassName = windows.UTF16ToString(classNameUTF16[:])
|
|
return
|
|
}
|
|
|
|
// SetupDiClassGuidsFromNameEx function retrieves the GUIDs associated with the specified class name. This resulting list contains the classes currently installed on a local or remote computer.
|
|
func SetupDiClassGuidsFromNameEx(ClassName string, MachineName string) (ClassGuidList []windows.GUID, err error) {
|
|
classNameUTF16, err := syscall.UTF16PtrFromString(ClassName)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
const bufLen = 4
|
|
var buf [bufLen]windows.GUID
|
|
var bufCount uint32
|
|
|
|
var machineNameUTF16 *uint16
|
|
if MachineName != "" {
|
|
machineNameUTF16, err = syscall.UTF16PtrFromString(MachineName)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
|
|
err = setupDiClassGuidsFromNameEx(classNameUTF16, &buf[0], bufLen, &bufCount, machineNameUTF16, 0)
|
|
if err == nil {
|
|
// The GUID array was sufficiently big. Return its slice.
|
|
return buf[:bufCount], nil
|
|
}
|
|
|
|
if errWin, ok := err.(syscall.Errno); ok && errWin == windows.ERROR_INSUFFICIENT_BUFFER {
|
|
// The GUID array was too small. Now that we got the required size, create another one big enough and retry.
|
|
buf := make([]windows.GUID, bufCount)
|
|
err = setupDiClassGuidsFromNameEx(classNameUTF16, &buf[0], bufCount, &bufCount, machineNameUTF16, 0)
|
|
if err == nil {
|
|
return buf[:bufCount], nil
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// SetupDiGetClassDevsEx function returns a handle to a device information set that contains requested device information elements for a local or a remote computer.
|
|
func SetupDiGetClassDevsEx(ClassGUID *windows.GUID, Enumerator string, hwndParent uintptr, Flags DIGCF, DeviceInfoSet DevInfo, MachineName string) (handle DevInfo, err error) {
|
|
var enumeratorUTF16 *uint16
|
|
if Enumerator != "" {
|
|
enumeratorUTF16, err = syscall.UTF16PtrFromString(Enumerator)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
var machineNameUTF16 *uint16
|
|
if MachineName != "" {
|
|
machineNameUTF16, err = syscall.UTF16PtrFromString(MachineName)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
return setupDiGetClassDevsEx(ClassGUID, enumeratorUTF16, hwndParent, Flags, DeviceInfoSet, machineNameUTF16, 0)
|
|
}
|
|
|
|
// SetupDiGetDeviceInfoListDetail function retrieves information associated with a device information set including the class GUID, remote computer handle, and remote computer name.
|
|
func SetupDiGetDeviceInfoListDetail(DeviceInfoSet DevInfo) (data *DevInfoListDetailData, err error) {
|
|
var _data _SP_DEVINFO_LIST_DETAIL_DATA
|
|
_data.Size = uint32(unsafe.Sizeof(_data))
|
|
|
|
err = setupDiGetDeviceInfoListDetail(DeviceInfoSet, &_data)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
data = &DevInfoListDetailData{
|
|
ClassGUID: _data.ClassGUID,
|
|
RemoteMachineHandle: _data.RemoteMachineHandle,
|
|
RemoteMachineName: windows.UTF16ToString(_data.RemoteMachineName[:]),
|
|
}
|
|
return
|
|
}
|
|
|
|
// SetupDiEnumDeviceInfo function returns a SP_DEVINFO_DATA structure that specifies a device information element in a device information set.
|
|
func SetupDiEnumDeviceInfo(DeviceInfoSet DevInfo, MemberIndex int) (DeviceInfoData *SP_DEVINFO_DATA, err error) {
|
|
data := SP_DEVINFO_DATA{}
|
|
data.Size = uint32(unsafe.Sizeof(data))
|
|
|
|
return &data, setupDiEnumDeviceInfo(DeviceInfoSet, uint32(MemberIndex), &data)
|
|
}
|
|
|
|
// SetupDiOpenDevRegKey function opens a registry key for device-specific configuration information.
|
|
func SetupDiOpenDevRegKey(DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA, Scope DICS_FLAG, HwProfile uint32, KeyType DIREG, samDesired uint32) (key registry.Key, err error) {
|
|
handle, err := setupDiOpenDevRegKey(DeviceInfoSet, DeviceInfoData, Scope, HwProfile, KeyType, samDesired)
|
|
return registry.Key(handle), err
|
|
}
|
|
|
|
// SetupDiGetDeviceInstallParams function retrieves device installation parameters for a device information set or a particular device information element.
|
|
func SetupDiGetDeviceInstallParams(DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA) (data *DevInstallParams, err error) {
|
|
var _data _SP_DEVINSTALL_PARAMS
|
|
_data.Size = uint32(unsafe.Sizeof(_data))
|
|
|
|
err = setupDiGetDeviceInstallParams(DeviceInfoSet, DeviceInfoData, &_data)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
data = &DevInstallParams{
|
|
Flags: _data.Flags,
|
|
FlagsEx: _data.FlagsEx,
|
|
hwndParent: _data.hwndParent,
|
|
InstallMsgHandler: _data.InstallMsgHandler,
|
|
InstallMsgHandlerContext: _data.InstallMsgHandlerContext,
|
|
FileQueue: _data.FileQueue,
|
|
DriverPath: windows.UTF16ToString(_data.DriverPath[:]),
|
|
}
|
|
return
|
|
}
|
|
|
|
// SetupDiSetDeviceInstallParams function sets device installation parameters for a device information set or a particular device information element.
|
|
func SetupDiSetDeviceInstallParams(DeviceInfoSet DevInfo, DeviceInfoData *SP_DEVINFO_DATA, DeviceInstallParams *DevInstallParams) (err error) {
|
|
_data := _SP_DEVINSTALL_PARAMS{
|
|
Flags: DeviceInstallParams.Flags,
|
|
FlagsEx: DeviceInstallParams.FlagsEx,
|
|
hwndParent: DeviceInstallParams.hwndParent,
|
|
InstallMsgHandler: DeviceInstallParams.InstallMsgHandler,
|
|
InstallMsgHandlerContext: DeviceInstallParams.InstallMsgHandlerContext,
|
|
FileQueue: DeviceInstallParams.FileQueue,
|
|
}
|
|
_data.Size = uint32(unsafe.Sizeof(_data))
|
|
|
|
driverPathUTF16, err := syscall.UTF16FromString(DeviceInstallParams.DriverPath)
|
|
if err != nil {
|
|
return
|
|
}
|
|
copy(_data.DriverPath[:], driverPathUTF16)
|
|
|
|
return setupDiSetDeviceInstallParams(DeviceInfoSet, DeviceInfoData, &_data)
|
|
}
|