0

IOCTL コマンドを使用して USB スキャナにアクセスしようとしています。これは Windows 7 上にあります。以前は IOCTL コーディングを扱ったことがなかったので、最初に、クイック検索で見つけたものに基づいて次のスニペットを試しました。

#include "stdafx.h"
#include <stddef.h>
#include <Windows.h>
#include <ntddscsi.h>
#include <usbscan.h>

typedef struct
{
    SCSI_PASS_THROUGH spt;
    BYTE sense[18];
    BYTE data[36];
} SPTSD;

LPTSTR ErrorMessage(DWORD error)
{
    LPTSTR errorText = NULL;

    FormatMessage(
       FORMAT_MESSAGE_FROM_SYSTEM
       |FORMAT_MESSAGE_ALLOCATE_BUFFER
       |FORMAT_MESSAGE_IGNORE_INSERTS,  
       NULL,
       error,
       MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
       (LPTSTR)&errorText,
       0, 
       NULL);

    return errorText;
}

int _tmain(int argc, _TCHAR* argv[])
{
    HANDLE h = CreateFile(L"\\\\.\\Usbscan0", GENERIC_READ|GENERIC_WRITE, FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);

    if (INVALID_HANDLE_VALUE != h)
    {
        SPTSD sptsd={0};
        sptsd.spt.Length = sizeof (sptsd.spt);
        sptsd.spt.SenseInfoLength = sizeof(sptsd.sense);
        sptsd.spt.DataTransferLength = sizeof(sptsd.data);
        sptsd.spt.SenseInfoOffset = offsetof (SPTSD, sense);
        sptsd.spt.DataBufferOffset = offsetof (SPTSD, data);
        sptsd.spt.TimeOutValue = 30;
        sptsd.spt.DataIn = SCSI_IOCTL_DATA_IN;
        sptsd.spt.CdbLength = 6;
        sptsd.spt.Cdb[0] = 0x12; // SCSI INQUIRY command
        sptsd.spt.Cdb[1] = 0;
        sptsd.spt.Cdb[2] = 0;
        sptsd.spt.Cdb[3] = 0;
        sptsd.spt.Cdb[4] = sizeof(sptsd.data);
        sptsd.spt.Cdb[5] = 0;

        DWORD dwReturnedBytes;
        BOOL b;

        b = DeviceIoControl(h, IOCTL_SCSI_PASS_THROUGH, &sptsd, sizeof(sptsd), &sptsd, sizeof(sptsd), &dwReturnedBytes, NULL);

        if (b == 0)
        {
            LPTSTR errortext = ErrorMessage(GetLastError());
            wprintf(L"DeviceIoControl(IOCTL_SCSI_PASS_THROUGH-INQUIRY) failed with error %d : %s\r\n", GetLastError(), errortext);
            LocalFree(errortext);
        }
        else
        {
            wprintf(L"DeviceIoControl(IOCTL_SCSI_PASS_THROUGH-INQUIRY) succeeded\r\n");
            for (int i=0; i<dwReturnedBytes; i++)
            {
                wprintf(L"%02x ", sptsd.data[i]);
            }

            wprintf(L"\r\nEnd of returned data\r\n");
        }

        DEVICE_DESCRIPTOR dd;
        b = DeviceIoControl(h, IOCTL_GET_DEVICE_DESCRIPTOR, &dd, sizeof(dd), &dd, sizeof(dd), &dwReturnedBytes, NULL);

        if (b == 0)
        {
            LPTSTR errortext = ErrorMessage(GetLastError());
            wprintf(L"DeviceIoControl(IOCTL_GET_DEVICE_DESCRIPTOR) failed with error %d : %s\r\n", GetLastError(), errortext);
            LocalFree(errortext);
        }
        else
        {
            wprintf(L"DeviceIoControl(IOCTL_GET_DEVICE_DESCRIPTOR) succeeded\r\n");
            wprintf(L"VendorId = %x, ProductId = %x, Version = %x\r\n", dd.usVendorId, dd.usProductId, dd.usBcdDevice);
            wprintf(L"End of returned data\r\n");
        }

        CloseHandle(h);
    }

    return 0;
}

Windows 7 の 32 ビット バージョンと 64 ビット バージョンの両方を試しましたが、結果はどちらも同じです (エラー 50 : 要求はサポートされていません)。興味深いことに、2 番目の DeviceIoControl 呼び出しが機能し、デバイスの VID/PID とファームウェア バージョンが返されます。

エラー メッセージに基づいて、この IOCTL はサポートされていないと思います。しかし、調べてみると、この IOCTL コードはすべてのデバイスで必須であることがわかったので、何か間違っているに違いありません。INQUIRY コマンドが成功するようにするには、このコードをどのように変更する必要がありますか?

4

1 に答える 1

0

http://msdn.microsoft.com/en-us/library/ff548569%28v=vs.85%29.aspxによると、これらの IOCTL コードは、USB バスのカーネルモード静止画像ドライバーによって認識されます。

  • IOCTL_CANCEL_IO
  • IOCTL_GET_CHANNEL_ALIGN_RQST
  • IOCTL_GET_DEVICE_DESCRIPTOR
  • IOCTL_GET_PIPE_CONFIGURATION
  • IOCTL_GET_USB_記述子
  • IOCTL_GET_VERSION
  • IOCTL_READ_REGISTERS
  • IOCTL_RESET_PIPE
  • IOCTL_SEND_USB_REQUEST
  • IOCTL_SET_TIMEOUT
  • IOCTL_WAIT_ON_DEVICE_EVENT IOCTL_WRITE_REGISTERS

私の理解では、他の IOCTL コードはIOCTL_SEND_USB_REQUEST 制御コードを介して送信する必要があります。これは、上記のコードを使用して INQUERY コマンドを送信しようとしても機能しない理由を説明しています。

編集: WriteFileを使用して INQUIRY コマンドを送信し、ReadFileを使用して応答を読み取るだけでした。ただし、理解できない追加の問題があるようです。デバイスは、応答を送信するために、INQUIRY コマンドの 6 バイトの後に余分なバイトが必要です。それ以外の場合、ReadFile は 1 バイト (0x3) のみを返します。ここで何が起こっているのかがわかったら、この返信をもう一度更新します。

于 2014-05-31T18:41:47.790 に答える