如本文题目所示,这是因为只有在 Windows 8.1(Windows Server 2012 RC)及以上 Windows 操作版本才提供了运行时,修改/删除完成端口关联的ABI接口。
boost::asio 在 release 函数底层实现之中是调用了 FileReplaceCompletionInformation 函数来删除完成端口关联的。
微软官方的MSDN文档,上面有明确说明。
FILE_INFORMATION_CLASS (wdm.h) - Windows drivers | Microsoft Learn
FileReplaceCompletionInformation
Value: 61
A FILE_COMPLETION_INFORMATION structure to change or remove the completion port associated with a file handle. This value is available starting with Windows 8.1.
所以如果想要切 io_context 驱动线程,可能就以下几个思路。
1、共享同一个 io_context,在上层自己控制 strand 来切
2、多个 io_context 的情况,想要把 socket 交给另一个 io_context,可以考虑二次转发来解决。
3、做成多模式,Windows 8.1 以下的版本,走原本流程(不转移 io_context),Windows 8.1及以上可以转移 io_context 驱动。
4、如果能够提前知道某一个 sockfd,最终会放在那个 io_context 上是最好的。
如果修改 asio 的源代码,如果调用失败就忽略呢?也是不行的哟,因为 iocp 没有解除关联,所以就会出现致命问题,因为没法重新 assign 附加到 iocp 上。