Я работаю над спорадической производственной проблемой, которая возникает в нашем 32-битном приложении MFC VC2010. Приложение работает в 64-разрядной версии Windows Server 2008 R2 Standard SP1.
Проблема вызвана невозможностью создать производный класс CWnd. Когда происходит сбой, метод AfxUnhookWindowCreate возвращает false в CWnd :: CreateEx. Это потому, что переменная pThreadState-> m_pWndInit не равна NULL. Похоже, _AfxCbtFilterHook должен установить это значение в NULL, когда HCBT_CREATEWND подключен, но, похоже, этого не происходит. Я вышел из CREATESTRUCT и сравнил его с тем, когда происходит сбой или не происходит, а параметры по существу одинаковы.
У кого-нибудь есть идеи о том, что может вызвать это или как я мог определить причину? Спасибо!
BOOL CWnd::CreateEx(DWORD dwExStyle, LPCTSTR lpszClassName,
LPCTSTR lpszWindowName, DWORD dwStyle,
int x, int y, int nWidth, int nHeight,
HWND hWndParent, HMENU nIDorHMenu, LPVOID lpParam)
{
...
if (!PreCreateWindow(cs))
{
PostNcDestroy();
return FALSE;
}
AfxHookWindowCreate(this);
HWND hWnd = ::AfxCtxCreateWindowEx(cs.dwExStyle, cs.lpszClass,
cs.lpszName, cs.style, cs.x, cs.y, cs.cx, cs.cy,
cs.hwndParent, cs.hMenu, cs.hInstance, cs.lpCreateParams);
...
if (!AfxUnhookWindowCreate())
PostNcDestroy(); // cleanup if CreateWindowEx fails too soon
...
BOOL AFXAPI AfxUnhookWindowCreate()
{
_AFX_THREAD_STATE* pThreadState = _afxThreadState.GetData();
#ifndef _AFXDLL
if (afxContextIsDLL && pThreadState->m_hHookOldCbtFilter != NULL)
{
::UnhookWindowsHookEx(pThreadState->m_hHookOldCbtFilter);
pThreadState->m_hHookOldCbtFilter = NULL;
}
#endif
if (pThreadState->m_pWndInit != NULL)
{
pThreadState->m_pWndInit = NULL;
return FALSE; // was not successfully hooked
}
return TRUE;
}
LRESULT CALLBACK
_AfxCbtFilterHook(int code, WPARAM wParam, LPARAM lParam)
{
_AFX_THREAD_STATE* pThreadState = _afxThreadState.GetData();
if (code != HCBT_CREATEWND)
{
// wait for HCBT_CREATEWND just pass others on...
return CallNextHookEx(pThreadState->m_hHookOldCbtFilter, code,
wParam, lParam);
}
...
pThreadState->m_pWndInit = NULL;
Я отследил проблему до ловушки оконной процедуры, которая не должна была выполняться в это время.
Других решений пока нет …