简体   繁体   中英

C#: Error in getting device instance handle

In my C# code I am trying to use C++ functions: CM_Locate_DevNodeW and CM_Open_DevNode_Key (using pinvoke ). My code seems something like this:

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

After calling to CM_Locate_DevNodeW , the devInst becomes 1 , and the cmStatus is 0 = CR_SUCCESS . But the call to CM_Open_DevNode_Key fails. I don't know if CM_Locate_DevNodeW returns CR_SUCCESS but puts incorrect data within devInst ? ('1' does not seems like real device instance handle...)

Or maybe the call to CM_Open_DevNode_Key is wrong?

I declared the functions like this:

[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);

Any help would be appreciated!

I fiddled around with your code and this is what I've got so far. After reading some docs I found out that phkDevice parameter of CM_Open_DevNode_Key function was likely an out parameter so I updated function signature

[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);

And I tried to run the following code

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

I got opencmStatus 51 which corresponds to CR_ACCESS_DENIED . Then I thought " Hmm, don't I just requesting to much access? Let's try only read access options " So I replaced KEY_ALL_ACCESS with 1 ( KEY_QUERY_VALUE ) and ran the following code

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

It worked as expected. Finally, this version gave me opencmStatus equals to 0 .

I did all tests against my keyboard PCI identifier, dont know if it matters.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM