当前位置: 首页 > article >正文

DeviceIoControl超时后如何处理

在使用 DeviceIoControl 函数时,如果发生超时并且你希望手动中断或取消正在进行的 I/O 操作,可以通过以下几种方式来处理:

1. 使用 CancelIo 函数

CancelIo 是 Windows API 提供的一个函数,用于取消指定设备句柄的所有挂起的异步 I/O 操作。它只会影响异步操作(即 OVERLAPPED 模式下的 I/O 请求)。

使用方法:

BOOL result = CancelIo(hDevice);
if (!result) {
    DWORD dwError = GetLastError();
    // 处理错误
}

该函数会取消当前设备句柄上的所有未完成的异步 I/O 操作,包括 DeviceIoControl 请求。要注意:

  • 取消操作后,相关的 I/O 操作将不会完成,GetLastError() 会返回 ERROR_OPERATION_ABORTED
  • CancelIo 只能取消未完成的 I/O 请求,已经完成的请求无法取消。

2. 使用 CancelIoEx 函数

CancelIoExCancelIo 的增强版本,它允许指定取消特定线程或设备句柄的 I/O 请求。它也适用于异步模式。

使用方法:

BOOL result = CancelIoEx(hDevice, NULL);
if (!result) {
    DWORD dwError = GetLastError();
    // 处理错误
}

CancelIoEx 允许通过指定 OVERLAPPED 结构来取消特定的异步 I/O 操作,或者传入 NULL 来取消所有挂起的操作。

3. 取消 OVERLAPPED 操作(异步模式)

如果你使用了 OVERLAPPED 结构,且正在等待某个异步 I/O 操作完成(例如使用 WaitForSingleObject 等等待 I/O 完成),你可以通过取消等待来停止操作。

// 假设使用 OVERLAPPED 和 WaitForSingleObject
DWORD dwWaitResult = WaitForSingleObject(overlapped.hEvent, timeoutMilliseconds);
if (dwWaitResult == WAIT_TIMEOUT) {
    // 超时后取消 I/O 操作
    CancelIo(hDevice);  // 或者使用 CancelIoEx
}

 

在超时发生后,你可以使用 CancelIoCancelIoEx 来取消正在进行的异步操作。

4. 异步 I/O 中的错误处理

如果你使用的是异步 I/O(OVERLAPPED),当 DeviceIoControl 返回超时错误时,可以通过检查 GetLastError() 返回的错误代码来判断是否需要中断。

if (GetLastError() == ERROR_OPERATION_ABORTED) {
    // 操作被中断,处理相关逻辑
}

5. 线程间通信中断

如果你在多线程环境中使用异步 I/O,且希望中断操作,可以通过线程间的信号机制(如 EventMutexCriticalSection 等)来控制。你可以在某个线程中设置一个标志或事件,然后在等待 I/O 完成的线程中定期检查该标志,如果标志被设置,就提前退出或中断操作。

示例代码:取消异步 I/O 操作

#include <windows.h>
#include <iostream>

HANDLE hDevice;
OVERLAPPED overlapped = { 0 };
BOOL bResult;

// 超时后中断 I/O 操作
void CancelIoAfterTimeout()
{
    // 假设你已经设置了一个等待事件
    DWORD dwWaitResult = WaitForSingleObject(overlapped.hEvent, 5000);  // 设置 5 秒超时

    if (dwWaitResult == WAIT_TIMEOUT) {
        std::cout << "I/O operation timed out, canceling...\n";
        CancelIo(hDevice);  // 取消所有异步 I/O 操作
        // 或者使用 CancelIoEx(hDevice, &overlapped); 如果需要取消特定操作
    } else {
        std::cout << "I/O operation completed\n";
    }
}

int main()
{
    // 假设hDevice是已打开的设备句柄
    hDevice = CreateFile(L"\\\\.\\Device\\MyDevice", GENERIC_READ | GENERIC_WRITE,
                         0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
    
    if (hDevice == INVALID_HANDLE_VALUE) {
        std::cerr << "Failed to open device: " << GetLastError() << "\n";
        return 1;
    }

    // 发起异步 I/O 操作
    bResult = DeviceIoControl(hDevice, IOCTL_CODE, NULL, 0, NULL, 0, NULL, &overlapped);
    if (!bResult) {
        if (GetLastError() == ERROR_IO_PENDING) {
            // 操作正在进行,等待超时
            CancelIoAfterTimeout();
        }
    }

    CloseHandle(hDevice);
    return 0;
}

注意事项:

  • 在取消操作后,通常需要检查设备的状态或驱动程序的返回值。取消操作后,可能会导致 DeviceIoControl 调用返回错误,例如 ERROR_OPERATION_ABORTED
  • 如果你不使用异步 I/O(即没有 OVERLAPPED 结构),则无法直接取消操作。在这种情况下,你可能需要使用线程或进程间的控制机制来中断操作。

希望这些方法能帮助你处理中断超时的 I/O 操作!


http://www.kler.cn/a/422150.html

相关文章:

  • Wwise 使用MIDI文件、采样音频
  • 运费微服务和redis存热点数据
  • 力扣第89题 格雷编码
  • 华为HarmonyOS 让应用快速拥有账号能力 -- 2 获取用户头像昵称
  • c++总复习
  • 从被动响应到主动帮助,ProActive Agent开启人机交互新篇章
  • 【Spring】接口版本控制最佳实现
  • Vue3 父子组件传值
  • ESLint 规则入门:如何配置重要性及选项(2)
  • 【数据分析】如何根据数据选择图表类型
  • 【Android】组件化嘻嘻嘻gradle耶耶耶
  • 下载 M3U8 格式的视频
  • c++ mfc调用UpdateData(TRUE)时,发生异常
  • ElasticSearch easy-es 聚合函数 group by 混合写法求Top N 词云 分词
  • k8s,声明式API对象理解
  • 基于Java Springboot广西文化传承微信小程序
  • 洛谷 B2029:大象喝水 ← 圆柱体体积
  • 《Vue零基础教程》(5)计算属性和侦听器好讲解
  • 【Linux|计算机网络】HTTPS工作原理与安全机制详解
  • 说说Elasticsearch查询语句如何提升权重?
  • Leetcode 303 Range Sum Query - Immutable
  • 靶机dpwwn-01
  • vue3项目最新eslint9+prettier+husky+stylelint+vscode配置
  • Qt 面试题复习10~12_2024-12-2
  • Android Folding
  • kafka 配置消息编码格式、解决消费者中文乱码