首页 > 解决方案 > 如何在 C++ (VS2010) 中设置超时读取 USB 端口?

问题描述

我正在打开并从 USB 设备(热敏打印机)读取端口:

HANDLE hUsb = CreateFile(symbolicName,
      GENERIC_READ | GENERIC_WRITE, 
      0, 
      NULL, 
      OPEN_EXISTING, 
      0, 
      NULL);

ReadFile(hUsb, buffer, bytes, &read, NULL);

我需要设置超时才能读取,但它是 USB 端口,而不是 COM 端口,所以我不能使用函数 SetCommTimeouts。

有没有我可以使用并且具有与 SetCommTimeouts 相同效果的功能?

如果有一个简单的方法,我宁愿不使用线程。

我正在使用带有 Windows 10 的 Visual Studio 2010。

感激的。

标签: c++windowsvisual-studiousbthermal-printer

解决方案


首先,这里的任何 Visual Studio 都绝对不相关。

一般解决方案 - 使用异步 io,它永远不会阻塞。你可以自己取消一些超时操作。当然是设置定时器(通过CreateTimerQueueTimer)并在定时器回调中取消 io 的好方法。或者如果 io 将在此之前完成 - 取消计时器。但如果想要最简单的实现,即在适当的位置进行同步 io - 可能接下来做:

    inline ULONG BOOL_TO_ERROR(BOOL f)
    {
        return f ? NOERROR : GetLastError();
    }

    //------------------------------------------------------------------------

    HANDLE hFile = CreateFileW(symbolicName, FILE_GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING,
        FILE_FLAG_OVERLAPPED, 0);

    if (hFile != INVALID_HANDLE_VALUE)
    {
        OVERLAPPED ov = {};
        if (ov.hEvent = CreateEventW(0, 0, 0, 0))
        {
            char buf[16];
            ULONG NumberOfBytesRead = 0;
            ULONG err = BOOL_TO_ERROR(ReadFile(hFile, buf, sizeof(buf), 0, &ov));

            switch (err)
            {
            case ERROR_IO_PENDING:
                if (WaitForSingleObject(ov.hEvent, timeout) != WAIT_OBJECT_0)
                {
                    CancelIo(hFile);
                }
            case NOERROR:
                err = BOOL_TO_ERROR(GetOverlappedResult(hFile, &ov, &NumberOfBytesRead, TRUE));
                break;
            }

            CloseHandle(ov.hEvent);
        }
        CloseHandle(hFile);
    }

推荐阅读