C#: ошибка в получении дескриптора экземпляра устройства
В моем коде C# я пытаюсь использовать функции C++: CM_Locate_DevNodeW
а также CM_Open_DevNode_Key
(используя pinvoke). Мой код выглядит примерно так:
String deviceId = "PCI\\VEN_8086&DEV_591B&SUBSYS_22128086&REV_01\\3&11583659&0&10";
int devInst = 0;
cmStatus = CM_Locate_DevNodeW(&devInst, deviceId, CM_LOCATE_DEVNODE_NORMAL);
if (cmStatus == CR_SUCCESS)
{
UIntPtr pHKey = new UIntPtr();
cmStatus = CM_Open_DevNode_Key(devInst, KEY_ALL_ACCESS, 0, RegDisposition_OpenExisting, pHKey, CM_REGISTRY_SOFTWARE);
if (cmStatus == CR_SUCCESS)
{
//but here cmStatus=3 (Invalid Pointer)
}
}
После звонка CM_Locate_DevNodeW
, devInst
становится 1
и cmStatus
это 0 = CR_SUCCESS
, Но призыв к CM_Open_DevNode_Key
выходит из строя. Я не знаю, если CM_Locate_DevNodeW
возвращается CR_SUCCESS
но помещает неверные данные в devInst
? ('1' не похоже на дескриптор реального устройства...)
Или, может быть, звонок CM_Open_DevNode_Key
неправильно?
Я объявил функции следующим образом:
[DllImport("cfgmgr32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
public static extern unsafe int CM_Locate_DevNodeW(
int* pdnDevInst,
string pDeviceID,
ulong ulFlags);
[DllImport("cfgmgr32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
public static extern unsafe int CM_Open_DevNode_Key(
int dnDevNode,
int samDesired,
int ulHardwareProfile,
int Disposition,
IntPtr phkDevice,
int ulFlags);
Любая помощь будет оценена!
1 ответ
Я возился с твоим кодом, и это то, что я получил до сих пор. Прочитав несколько документов, я узнал, что phkDevice
параметр CM_Open_DevNode_Key
функция была, вероятно, out
параметр, поэтому я обновил сигнатуру функции
[DllImport("cfgmgr32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
public static extern unsafe int CM_Open_DevNode_Key(
int dnDevNode,
int samDesired,
int ulHardwareProfile,
int Disposition,
out IntPtr phkDevice, //added out keyword
int ulFlags);
И я попытался запустить следующий код
IntPtr pHKey;
string deviceId = @"my keyboard pci id";
int devInst = 0;
int cmStatus = CM_Locate_DevNodeW(&devInst, deviceId, CM_LOCATE_DEVNODE_NORMAL);
if (cmStatus == CR_SUCCESS)
{
int opencmStatus = CM_Open_DevNode_Key(devInst, KEY_ALL_ACCESS, 0, RegDisposition_OpenExisting, out pHKey, CM_REGISTRY_SOFTWARE);
if (opencmStatus == CR_SUCCESS)
{
//
}
}
я получил opencmStatus
51
что соответствует CR_ACCESS_DENIED
, Тогда я подумал: "Хм, разве я не запрашивал много доступа? Давайте попробуем только читать параметры доступа", поэтому я заменил KEY_ALL_ACCESS
с 1
(KEY_QUERY_VALUE
) и запустил следующий код
IntPtr pHKey;
string deviceId = @"my keyboard pci id";
int devInst = 0;
int cmStatus = CM_Locate_DevNodeW(&devInst, deviceId, CM_LOCATE_DEVNODE_NORMAL);
if (cmStatus == CR_SUCCESS)
{
int opencmStatus = CM_Open_DevNode_Key(devInst, 1, 0, RegDisposition_OpenExisting, out pHKey, CM_REGISTRY_SOFTWARE);
if (opencmStatus == CR_SUCCESS)
{
//
}
}
Сработало как положено. Наконец, эта версия дала мне opencmStatus
равно 0
,
Я сделал все тесты с моим клавиатурным идентификатором PCI, не знаю, имеет ли это значение.