mirror of
https://github.com/reactos/reactos.git
synced 2024-12-31 19:42:51 +00:00
[WIN32SS:USER] Some minimal work and fixes concerning message queues timeouts.
CORE-15147 - Rename CLIENTTHREADINFO::tickLastMsgChecked into timeLastRead as documented in https://reactos.org/wiki/Techwiki:Win32k/CLIENTTHREADINFO . This is the last time the message queue was read. - This is the structure member one must compare against the current tick count timestamp in order to heuristically determine whether a message queue thread is hung!! Fix MsqIsHung() in accordance, add extra debug logging in order to help us determining which of our code present regular GUI hangs, and add as well an extra "TimeOut" parameter so as not to hardcode a fixed value within that function but instead allowing its caller to specify possible different values. - THREADINFO::timeLast is on the contrary the last message time stamp, and will definitively differ from CLIENTTHREADINFO::timeLastRead . It should only be used for information purposes! - Accordingly, in NtUserGetThreadState()::THREADSTATE_UPTIMELASTREAD and in InitThreadCallback(), only (re-)initialize the timeLastRead member of the CLIENTTHREADINFO structure of the THREADINFO of interest. - In co_IntPeekMessage(), update more often the timeLastRead timestamp whenever the current message queue has been read (but NOT timeLast!! That one will be updated ONLY WHEN a message is found!). - In co_IntSendMessageTimeoutSingle() first check whether the window to which we send the message is being destroyed, before checking for queue hangs etc. Collapse the logic checks for queue hang and increase the hang timeout check to 4 times MSQ_HUNG (== 4 * 5 seconds) and display a debug trace.
This commit is contained in:
parent
0d26bbf4b5
commit
a8e7defb01
12 changed files with 57 additions and 37 deletions
|
@ -167,7 +167,7 @@ typedef struct _CLIENTTHREADINFO
|
||||||
WORD fsWakeBits;
|
WORD fsWakeBits;
|
||||||
WORD fsWakeBitsJournal;
|
WORD fsWakeBitsJournal;
|
||||||
WORD fsWakeMask;
|
WORD fsWakeMask;
|
||||||
ULONG tickLastMsgChecked;
|
ULONG timeLastRead; // Last time the message queue was read.
|
||||||
DWORD dwcPumpHook;
|
DWORD dwcPumpHook;
|
||||||
} CLIENTTHREADINFO, *PCLIENTTHREADINFO;
|
} CLIENTTHREADINFO, *PCLIENTTHREADINFO;
|
||||||
|
|
||||||
|
|
|
@ -162,7 +162,7 @@ BOOL FASTCALL IntMakeHungWindowGhosted(HWND hwndHung)
|
||||||
return FALSE; // not a window
|
return FALSE; // not a window
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!MsqIsHung(pHungWnd->head.pti))
|
if (!MsqIsHung(pHungWnd->head.pti, MSQ_HUNG))
|
||||||
{
|
{
|
||||||
DPRINT1("Not hung window\n");
|
DPRINT1("Not hung window\n");
|
||||||
return FALSE; // not hung window
|
return FALSE; // not hung window
|
||||||
|
|
|
@ -467,6 +467,7 @@ InitThreadCallback(PETHREAD Thread)
|
||||||
IntReferenceProcessInfo(ptiCurrent->ppi);
|
IntReferenceProcessInfo(ptiCurrent->ppi);
|
||||||
pTeb->Win32ThreadInfo = ptiCurrent;
|
pTeb->Win32ThreadInfo = ptiCurrent;
|
||||||
ptiCurrent->pClientInfo = (PCLIENTINFO)pTeb->Win32ClientInfo;
|
ptiCurrent->pClientInfo = (PCLIENTINFO)pTeb->Win32ClientInfo;
|
||||||
|
ptiCurrent->pcti = &ptiCurrent->cti;
|
||||||
|
|
||||||
/* Mark the process as having threads */
|
/* Mark the process as having threads */
|
||||||
ptiCurrent->ppi->W32PF_flags |= W32PF_THREADCONNECTED;
|
ptiCurrent->ppi->W32PF_flags |= W32PF_THREADCONNECTED;
|
||||||
|
@ -503,7 +504,8 @@ InitThreadCallback(PETHREAD Thread)
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
ptiCurrent->timeLast = EngGetTickCount32();
|
ptiCurrent->pcti->timeLastRead = EngGetTickCount32();
|
||||||
|
|
||||||
ptiCurrent->MessageQueue = MsqCreateMessageQueue(ptiCurrent);
|
ptiCurrent->MessageQueue = MsqCreateMessageQueue(ptiCurrent);
|
||||||
if (ptiCurrent->MessageQueue == NULL)
|
if (ptiCurrent->MessageQueue == NULL)
|
||||||
{
|
{
|
||||||
|
@ -524,8 +526,6 @@ InitThreadCallback(PETHREAD Thread)
|
||||||
if (Process == gpepCSRSS || !gpepCSRSS)
|
if (Process == gpepCSRSS || !gpepCSRSS)
|
||||||
ptiCurrent->TIF_flags = TIF_CSRSSTHREAD | TIF_DONTATTACHQUEUE;
|
ptiCurrent->TIF_flags = TIF_CSRSSTHREAD | TIF_DONTATTACHQUEUE;
|
||||||
|
|
||||||
ptiCurrent->pcti = &ptiCurrent->cti;
|
|
||||||
|
|
||||||
/* Initialize the CLIENTINFO */
|
/* Initialize the CLIENTINFO */
|
||||||
pci = (PCLIENTINFO)pTeb->Win32ClientInfo;
|
pci = (PCLIENTINFO)pTeb->Win32ClientInfo;
|
||||||
RtlZeroMemory(pci, sizeof(*pci));
|
RtlZeroMemory(pci, sizeof(*pci));
|
||||||
|
|
|
@ -829,8 +829,8 @@ co_IntPeekMessage( PMSG Msg,
|
||||||
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
pti->timeLast = EngGetTickCount32();
|
/* Update the last message-queue access time */
|
||||||
pti->pcti->tickLastMsgChecked = pti->timeLast;
|
pti->pcti->timeLastRead = EngGetTickCount32();
|
||||||
|
|
||||||
// Post mouse moves while looping through peek messages.
|
// Post mouse moves while looping through peek messages.
|
||||||
if (pti->MessageQueue->QF_flags & QF_MOUSEMOVED)
|
if (pti->MessageQueue->QF_flags & QF_MOUSEMOVED)
|
||||||
|
@ -874,7 +874,7 @@ co_IntPeekMessage( PMSG Msg,
|
||||||
0,
|
0,
|
||||||
Msg ))
|
Msg ))
|
||||||
{
|
{
|
||||||
return TRUE;
|
goto GotMessage;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Only check for quit messages if not posted messages pending. */
|
/* Only check for quit messages if not posted messages pending. */
|
||||||
|
@ -893,7 +893,7 @@ co_IntPeekMessage( PMSG Msg,
|
||||||
pti->pcti->fsWakeBits &= ~QS_ALLPOSTMESSAGE;
|
pti->pcti->fsWakeBits &= ~QS_ALLPOSTMESSAGE;
|
||||||
pti->pcti->fsChangeBits &= ~QS_ALLPOSTMESSAGE;
|
pti->pcti->fsChangeBits &= ~QS_ALLPOSTMESSAGE;
|
||||||
}
|
}
|
||||||
return TRUE;
|
goto GotMessage;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check for hardware events. */
|
/* Check for hardware events. */
|
||||||
|
@ -906,7 +906,7 @@ co_IntPeekMessage( PMSG Msg,
|
||||||
ProcessMask,
|
ProcessMask,
|
||||||
Msg))
|
Msg))
|
||||||
{
|
{
|
||||||
return TRUE;
|
goto GotMessage;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Now check for System Event messages. */
|
/* Now check for System Event messages. */
|
||||||
|
@ -946,7 +946,7 @@ co_IntPeekMessage( PMSG Msg,
|
||||||
Msg,
|
Msg,
|
||||||
RemoveMessages))
|
RemoveMessages))
|
||||||
{
|
{
|
||||||
return TRUE;
|
goto GotMessage;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This is correct, check for the current threads timers waiting to be
|
/* This is correct, check for the current threads timers waiting to be
|
||||||
|
@ -962,6 +962,9 @@ co_IntPeekMessage( PMSG Msg,
|
||||||
}
|
}
|
||||||
while (TRUE);
|
while (TRUE);
|
||||||
|
|
||||||
|
GotMessage:
|
||||||
|
/* Update the last message-queue access time */
|
||||||
|
pti->pcti->timeLastRead = EngGetTickCount32();
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1423,16 +1426,6 @@ co_IntSendMessageTimeoutSingle( HWND hWnd,
|
||||||
RETURN( TRUE);
|
RETURN( TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (MsqIsHung(ptiSendTo))
|
|
||||||
{
|
|
||||||
if (uFlags & SMTO_ABORTIFHUNG)
|
|
||||||
{
|
|
||||||
// FIXME: Set window hung and add to a list.
|
|
||||||
/* FIXME: Set a LastError? */
|
|
||||||
RETURN( FALSE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Window->state & WNDS_DESTROYED)
|
if (Window->state & WNDS_DESTROYED)
|
||||||
{
|
{
|
||||||
/* FIXME: Last error? */
|
/* FIXME: Last error? */
|
||||||
|
@ -1440,6 +1433,14 @@ co_IntSendMessageTimeoutSingle( HWND hWnd,
|
||||||
RETURN( FALSE);
|
RETURN( FALSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((uFlags & SMTO_ABORTIFHUNG) && MsqIsHung(ptiSendTo, 4 * MSQ_HUNG))
|
||||||
|
{
|
||||||
|
// FIXME: Set window hung and add to a list.
|
||||||
|
/* FIXME: Set a LastError? */
|
||||||
|
ERR("Window %p (%p) (pti %p) is hung!\n", hWnd, Window, ptiSendTo);
|
||||||
|
RETURN( FALSE);
|
||||||
|
}
|
||||||
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
Status = co_MsqSendMessage( ptiSendTo,
|
Status = co_MsqSendMessage( ptiSendTo,
|
||||||
|
@ -1452,13 +1453,13 @@ co_IntSendMessageTimeoutSingle( HWND hWnd,
|
||||||
MSQ_NORMAL,
|
MSQ_NORMAL,
|
||||||
uResult );
|
uResult );
|
||||||
}
|
}
|
||||||
while ((STATUS_TIMEOUT == Status) &&
|
while ((Status == STATUS_TIMEOUT) &&
|
||||||
(uFlags & SMTO_NOTIMEOUTIFNOTHUNG) &&
|
(uFlags & SMTO_NOTIMEOUTIFNOTHUNG) &&
|
||||||
!MsqIsHung(ptiSendTo)); // FIXME: Set window hung and add to a list.
|
!MsqIsHung(ptiSendTo, MSQ_HUNG)); // FIXME: Set window hung and add to a list.
|
||||||
|
|
||||||
if (Status == STATUS_TIMEOUT)
|
if (Status == STATUS_TIMEOUT)
|
||||||
{
|
{
|
||||||
if (0 && MsqIsHung(ptiSendTo))
|
if (0 && MsqIsHung(ptiSendTo, MSQ_HUNG))
|
||||||
{
|
{
|
||||||
TRACE("Let's go Ghost!\n");
|
TRACE("Let's go Ghost!\n");
|
||||||
IntMakeHungWindowGhosted(hWnd);
|
IntMakeHungWindowGhosted(hWnd);
|
||||||
|
|
|
@ -296,10 +296,9 @@ NtUserGetThreadState(
|
||||||
{
|
{
|
||||||
PTHREADINFO pti;
|
PTHREADINFO pti;
|
||||||
pti = PsGetCurrentThreadWin32Thread();
|
pti = PsGetCurrentThreadWin32Thread();
|
||||||
pti->timeLast = EngGetTickCount32();
|
pti->pcti->timeLastRead = EngGetTickCount32();
|
||||||
pti->pcti->tickLastMsgChecked = pti->timeLast;
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case THREADSTATE_GETINPUTSTATE:
|
case THREADSTATE_GETINPUTSTATE:
|
||||||
ret = LOWORD(IntGetQueueStatus(QS_POSTMESSAGE|QS_TIMER|QS_PAINT|QS_SENDMESSAGE|QS_INPUT)) & (QS_KEY | QS_MOUSEBUTTON);
|
ret = LOWORD(IntGetQueueStatus(QS_POSTMESSAGE|QS_TIMER|QS_PAINT|QS_SENDMESSAGE|QS_INPUT)) & (QS_KEY | QS_MOUSEBUTTON);
|
||||||
|
|
|
@ -2089,8 +2089,8 @@ co_MsqPeekHardwareMessage(IN PTHREADINFO pti,
|
||||||
{
|
{
|
||||||
pti->TIF_flags |= TIF_MSGPOSCHANGED;
|
pti->TIF_flags |= TIF_MSGPOSCHANGED;
|
||||||
}
|
}
|
||||||
pti->ptLast = msg.pt;
|
|
||||||
pti->timeLast = msg.time;
|
pti->timeLast = msg.time;
|
||||||
|
pti->ptLast = msg.pt;
|
||||||
MessageQueue->ExtraInfo = ExtraInfo;
|
MessageQueue->ExtraInfo = ExtraInfo;
|
||||||
Ret = TRUE;
|
Ret = TRUE;
|
||||||
break;
|
break;
|
||||||
|
@ -2191,13 +2191,32 @@ co_MsqWaitForNewMessages(PTHREADINFO pti, PWND WndFilter,
|
||||||
}
|
}
|
||||||
|
|
||||||
BOOL FASTCALL
|
BOOL FASTCALL
|
||||||
MsqIsHung(PTHREADINFO pti)
|
MsqIsHung(PTHREADINFO pti, DWORD TimeOut)
|
||||||
{
|
{
|
||||||
if (EngGetTickCount32() - pti->timeLast > MSQ_HUNG &&
|
DWORD dwTimeStamp = EngGetTickCount32();
|
||||||
|
if (dwTimeStamp - pti->pcti->timeLastRead > TimeOut &&
|
||||||
!(pti->pcti->fsWakeMask & QS_INPUT) &&
|
!(pti->pcti->fsWakeMask & QS_INPUT) &&
|
||||||
!PsGetThreadFreezeCount(pti->pEThread) &&
|
!PsGetThreadFreezeCount(pti->pEThread) &&
|
||||||
!(pti->ppi->W32PF_flags & W32PF_APPSTARTING))
|
!(pti->ppi->W32PF_flags & W32PF_APPSTARTING))
|
||||||
|
{
|
||||||
|
ERR("\nMsqIsHung(pti %p, TimeOut %lu)\n"
|
||||||
|
"pEThread %p, ThreadsProcess %p, ImageFileName '%s'\n"
|
||||||
|
"dwTimeStamp = %lu\n"
|
||||||
|
"pti->pcti->timeLastRead = %lu\n"
|
||||||
|
"pti->timeLast = %lu\n"
|
||||||
|
"PsGetThreadFreezeCount(pti->pEThread) = %lu\n",
|
||||||
|
pti, TimeOut,
|
||||||
|
pti->pEThread,
|
||||||
|
pti->pEThread ? pti->pEThread->ThreadsProcess : NULL,
|
||||||
|
(pti->pEThread && pti->pEThread->ThreadsProcess)
|
||||||
|
? pti->pEThread->ThreadsProcess->ImageFileName : "(None)",
|
||||||
|
dwTimeStamp,
|
||||||
|
pti->pcti->timeLastRead,
|
||||||
|
pti->timeLast,
|
||||||
|
PsGetThreadFreezeCount(pti->pEThread));
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
|
@ -127,7 +127,7 @@ enum internal_event_message
|
||||||
|
|
||||||
extern LIST_ENTRY usmList;
|
extern LIST_ENTRY usmList;
|
||||||
|
|
||||||
BOOL FASTCALL MsqIsHung(PTHREADINFO pti);
|
BOOL FASTCALL MsqIsHung(PTHREADINFO pti, DWORD TimeOut);
|
||||||
VOID CALLBACK HungAppSysTimerProc(HWND,UINT,UINT_PTR,DWORD);
|
VOID CALLBACK HungAppSysTimerProc(HWND,UINT,UINT_PTR,DWORD);
|
||||||
NTSTATUS FASTCALL co_MsqSendMessage(PTHREADINFO ptirec,
|
NTSTATUS FASTCALL co_MsqSendMessage(PTHREADINFO ptirec,
|
||||||
HWND Wnd, UINT Msg, WPARAM wParam, LPARAM lParam,
|
HWND Wnd, UINT Msg, WPARAM wParam, LPARAM lParam,
|
||||||
|
|
|
@ -1092,7 +1092,7 @@ UpdateThreadWindows(PWND pWnd, PTHREADINFO pti, HRGN hRgn)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (IsThreadSuspended(pwndTemp->head.pti) || MsqIsHung(pwndTemp->head.pti))
|
if (IsThreadSuspended(pwndTemp->head.pti) || MsqIsHung(pwndTemp->head.pti, MSQ_HUNG))
|
||||||
{
|
{
|
||||||
UpdateTheadChildren(pwndTemp, hRgn);
|
UpdateTheadChildren(pwndTemp, hRgn);
|
||||||
}
|
}
|
||||||
|
|
|
@ -515,7 +515,7 @@ NtUserCallTwoParam(
|
||||||
{
|
{
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (MsqIsHung(Window->head.pti))
|
if (MsqIsHung(Window->head.pti, MSQ_HUNG))
|
||||||
{
|
{
|
||||||
// TODO: Make the window ghosted and activate.
|
// TODO: Make the window ghosted and activate.
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -97,7 +97,7 @@ typedef struct _THREADINFO
|
||||||
struct _USER_SENT_MESSAGE *pusmCurrent;
|
struct _USER_SENT_MESSAGE *pusmCurrent;
|
||||||
/* Queue of messages sent to the queue. */
|
/* Queue of messages sent to the queue. */
|
||||||
LIST_ENTRY SentMessagesListHead; // psmsReceiveList
|
LIST_ENTRY SentMessagesListHead; // psmsReceiveList
|
||||||
/* Last time PeekMessage() was called. */
|
/* Last message time and ID */
|
||||||
LONG timeLast;
|
LONG timeLast;
|
||||||
ULONG_PTR idLast;
|
ULONG_PTR idLast;
|
||||||
/* True if a WM_QUIT message is pending. */
|
/* True if a WM_QUIT message is pending. */
|
||||||
|
@ -124,6 +124,7 @@ typedef struct _THREADINFO
|
||||||
PKEVENT pEventQueueServer;
|
PKEVENT pEventQueueServer;
|
||||||
LIST_ENTRY PtiLink;
|
LIST_ENTRY PtiLink;
|
||||||
INT iCursorLevel;
|
INT iCursorLevel;
|
||||||
|
/* Last message cursor position */
|
||||||
POINT ptLast;
|
POINT ptLast;
|
||||||
|
|
||||||
INT cEnterCount;
|
INT cEnterCount;
|
||||||
|
|
|
@ -4057,7 +4057,7 @@ NtUserQueryWindow(HWND hWnd, DWORD Index)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case QUERY_WINDOW_ISHUNG:
|
case QUERY_WINDOW_ISHUNG:
|
||||||
Result = (DWORD_PTR)MsqIsHung(pWnd->head.pti);
|
Result = (DWORD_PTR)MsqIsHung(pWnd->head.pti, MSQ_HUNG);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case QUERY_WINDOW_REAL_ID:
|
case QUERY_WINDOW_REAL_ID:
|
||||||
|
|
|
@ -2162,7 +2162,7 @@ PeekMessageWorker( PMSG pMsg,
|
||||||
{ // Not waiting for idle event.
|
{ // Not waiting for idle event.
|
||||||
if (!pcti->fsChangeBits && !pcti->fsWakeBits)
|
if (!pcti->fsChangeBits && !pcti->fsWakeBits)
|
||||||
{ // No messages are available.
|
{ // No messages are available.
|
||||||
if ((GetTickCount() - pcti->tickLastMsgChecked) > 1000)
|
if ((GetTickCount() - pcti->timeLastRead) > 1000)
|
||||||
{ // Up the msg read count if over 1 sec.
|
{ // Up the msg read count if over 1 sec.
|
||||||
NtUserGetThreadState(THREADSTATE_UPTIMELASTREAD);
|
NtUserGetThreadState(THREADSTATE_UPTIMELASTREAD);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue