Почему DeviceIoControl генерирует ошибку 21 (Device Not Ready) из C #, когда эквивалент в C работает нормально? - PullRequest
4 голосов
/ 24 декабря 2010

Я пытаюсь отправить команду IOCTL_SERVICE_REFRESH в службу промежуточных драйверов GPS, используя C #, например:

handle = CreateFile("GPD0:",
    GENERIC_READ,
    FILE_SHARE_READ | FILE_SHARE_WRITE,
    IntPtr.Zero,
    OPEN_EXISTING, 
    FILE_ATTRIBUTE_NORMAL, 
    IntPtr.Zero);

if (handle.ToInt32() == INVALID_HANDLE_VALUE)
{
    rc = Marshal.GetLastWin32Error();
    return rc;
}

int numBytesReturned = 0;

rc = DeviceIoControl(
    handle,
    IOCTL_SERVICE_REFRESH,
    null,
    0,
    null,
    0,
    ref numBytesReturned,
    IntPtr.Zero);

int error = Marshal.GetLastWin32Error();

GetLastWin32Error всегда выдает ошибку 21 (устройство не готово). Однако эквивалентный вызов, когда он сделан из C ++, работает нормально:

HANDLE hGPS = CreateFile(L"GPD0:", GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
if (hGPS != INVALID_HANDLE_VALUE) {
  BOOL ret = DeviceIoControl(hGPS,IOCTL_SERVICE_REFRESH,0,0,0,0,0,0);
  DWORD err = GetLastError();       
}

Я подозревал проблему с подписями PInvoke, но я не могу найти здесь что-то неправильное:

[DllImport("coredll.dll", EntryPoint = "DeviceIoControl", SetLastError = true)]
public static extern int DeviceIoControl(
    IntPtr hDevice,
    uint dwIoControlCode,
    byte[] lpInBuffer,
    int nInBufferSize,
    byte[] lpOutBuffer,
    int nOutBufferSize,
    ref int lpBytesReturned,
    IntPtr lpOverlapped);

[DllImport("coredll.dll", SetLastError = true)]
public static extern IntPtr CreateFile(
    String lpFileName,
    uint dwDesiredAccess,
    uint dwShareMode,
    IntPtr attr,
    uint dwCreationDisposition,
    uint dwFlagsAndAttributes,
    IntPtr hTemplateFile);

Что мне здесь не хватает?

...