小心使用IOCP完成端口
s = createsocket();??? //假定s返回值是10
CreateIoCompletionPort(s, m_hCompletionPort, (DWORD_PTR)&A, 0);
WSASend(s,...);
WSASend(s,...);
WSASend(s,...);
WSASend(s,...);
WSASend(s,...);
這個時候,完成端口里累計了多條跟s相關的數據,由于完成端口的線程暫時繁忙,未來得及處理s關閉后的失敗數據
closesocket(s);
s = createsocket();??? //s的值仍然是10
CreateIoCompletionPort(s, m_hCompletionPort, (DWORD_PTR)&B, 0);
好了,本線程到此告一個段落,回到完成端口線程上來:
GetQueuedCompletionStatus(m_hCompletionPort,&dwTransferred,(LPDWORD)&pCustomData,(OVERLAPPED *) ...,INFINITE);
問:
pCustomData是指向A還是指向B?
?
在我的設計意識里,pCustomData應該指向A,且我沒有再任何文檔和論壇里看到提及此問題,且無論是官方的還是非官方的例子都是我這樣的用法。
實際上,pCustomData指向了B
?
此時,WSASend(和其他函數)操作的數據和在GetQueuedCompletionStatus后處理的數據不再綁定到一致的數據上,bug由此產生,且無法直接觀察,非常難重現,因為closesocket后也不是立即調用createsocket,且socket句柄被立即復用的概率實在不大,因為不是立即復用,完成端口線程就把老句柄相關的數據處理完畢了。
?
這個問題困擾<亂武>項目組和測試組人員至少三周,終在昨日(2008-8.31)獲知原因并得到解決。老謝說值得慶賀,該出去搓一頓。吾深表贊同,不過就是要我買單有些許遺憾——因為代碼是我寫的。
總結
以上是生活随笔為你收集整理的小心使用IOCP完成端口的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 一碗面,害了一个孩子,却惊醒了中国无数父
- 下一篇: 80后投资臆想