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)
    {
        // 
    }
}

я получил opencmStatus51 что соответствует 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, не знаю, имеет ли это значение.

Другие вопросы по тегам