mirror of
https://github.com/reactos/reactos.git
synced 2025-01-01 03:54:02 +00:00
cf955094b4
Based on the Doug Lyons' test in #7087, I found that my previous fix stopped working partially. Or rather, it would only work until the 32767 indexes were exhausted. It seems to me that the behavior of the bitfield has changed, because when I published the previous patch, it passed my tests.
- Bit array generates free ID cyclically, in the previous code after 32767 indexes expired the same index was returned, because of this the previous fix would stop working after expiration, so change the logic of calculating the next index.
- Change the index range to 256-32767 to match Windows, indexes 0-255 can theoretically be used as reserved for system purposes.
Addendum to fd327db20f
. CORE-9141
720 lines
17 KiB
C
720 lines
17 KiB
C
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS kernel
|
|
* PURPOSE: Window timers messages
|
|
* FILE: win32ss/user/ntuser/timer.c
|
|
* PROGRAMER: Gunnar
|
|
* Thomas Weidenmueller (w3seek@users.sourceforge.net)
|
|
* Michael Martin (michael.martin@reactos.org)
|
|
*/
|
|
|
|
#include <win32k.h>
|
|
DBG_DEFAULT_CHANNEL(UserTimer);
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
static LIST_ENTRY TimersListHead;
|
|
static LONG TimeLast = 0;
|
|
|
|
/* Windows 2000 has room for 32768 window-less timers */
|
|
/* These values give timer IDs [256,32767], same as on Windows */
|
|
#define MAX_WINDOW_LESS_TIMER_ID (32768 - 1)
|
|
#define NUM_WINDOW_LESS_TIMERS (32768 - 256)
|
|
|
|
#define HINTINDEX_BEGIN_VALUE 0
|
|
|
|
static PFAST_MUTEX Mutex;
|
|
static RTL_BITMAP WindowLessTimersBitMap;
|
|
static PVOID WindowLessTimersBitMapBuffer;
|
|
static ULONG HintIndex = HINTINDEX_BEGIN_VALUE;
|
|
|
|
ERESOURCE TimerLock;
|
|
|
|
#define IntLockWindowlessTimerBitmap() \
|
|
ExEnterCriticalRegionAndAcquireFastMutexUnsafe(Mutex)
|
|
|
|
#define IntUnlockWindowlessTimerBitmap() \
|
|
ExReleaseFastMutexUnsafeAndLeaveCriticalRegion(Mutex)
|
|
|
|
#define TimerEnterExclusive() \
|
|
{ \
|
|
KeEnterCriticalRegion(); \
|
|
ExAcquireResourceExclusiveLite(&TimerLock, TRUE); \
|
|
}
|
|
|
|
#define TimerLeave() \
|
|
{ \
|
|
ExReleaseResourceLite(&TimerLock); \
|
|
KeLeaveCriticalRegion(); \
|
|
}
|
|
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
static
|
|
PTIMER
|
|
FASTCALL
|
|
CreateTimer(VOID)
|
|
{
|
|
HANDLE Handle;
|
|
PTIMER Ret = NULL;
|
|
|
|
Ret = UserCreateObject(gHandleTable, NULL, NULL, &Handle, TYPE_TIMER, sizeof(TIMER));
|
|
if (Ret)
|
|
{
|
|
UserHMSetHandle(Ret, Handle);
|
|
InsertTailList(&TimersListHead, &Ret->ptmrList);
|
|
}
|
|
|
|
return Ret;
|
|
}
|
|
|
|
static
|
|
BOOL
|
|
FASTCALL
|
|
RemoveTimer(PTIMER pTmr)
|
|
{
|
|
BOOL Ret = FALSE;
|
|
if (pTmr)
|
|
{
|
|
/* Set the flag, it will be removed when ready */
|
|
RemoveEntryList(&pTmr->ptmrList);
|
|
if ((pTmr->pWnd == NULL) && (!(pTmr->flags & TMRF_SYSTEM))) // System timers are reusable.
|
|
{
|
|
ULONG ulBitmapIndex;
|
|
|
|
ASSERT(pTmr->nID <= MAX_WINDOW_LESS_TIMER_ID);
|
|
ulBitmapIndex = (ULONG)(MAX_WINDOW_LESS_TIMER_ID - pTmr->nID);
|
|
IntLockWindowlessTimerBitmap();
|
|
RtlClearBit(&WindowLessTimersBitMap, ulBitmapIndex);
|
|
IntUnlockWindowlessTimerBitmap();
|
|
}
|
|
UserDereferenceObject(pTmr);
|
|
Ret = UserDeleteObject( UserHMGetHandle(pTmr), TYPE_TIMER);
|
|
}
|
|
if (!Ret) ERR("Warning: Unable to delete timer\n");
|
|
|
|
return Ret;
|
|
}
|
|
|
|
PTIMER
|
|
FASTCALL
|
|
FindTimer(PWND Window,
|
|
UINT_PTR nID,
|
|
UINT flags)
|
|
{
|
|
PLIST_ENTRY pLE;
|
|
PTIMER pTmr, RetTmr = NULL;
|
|
|
|
TimerEnterExclusive();
|
|
pLE = TimersListHead.Flink;
|
|
while (pLE != &TimersListHead)
|
|
{
|
|
pTmr = CONTAINING_RECORD(pLE, TIMER, ptmrList);
|
|
|
|
if ( pTmr->nID == nID &&
|
|
pTmr->pWnd == Window &&
|
|
(pTmr->flags & (TMRF_SYSTEM|TMRF_RIT)) == (flags & (TMRF_SYSTEM|TMRF_RIT)))
|
|
{
|
|
RetTmr = pTmr;
|
|
break;
|
|
}
|
|
|
|
pLE = pLE->Flink;
|
|
}
|
|
TimerLeave();
|
|
|
|
return RetTmr;
|
|
}
|
|
|
|
PTIMER
|
|
FASTCALL
|
|
FindSystemTimer(PMSG pMsg)
|
|
{
|
|
PLIST_ENTRY pLE;
|
|
PTIMER pTmr = NULL;
|
|
|
|
TimerEnterExclusive();
|
|
pLE = TimersListHead.Flink;
|
|
while (pLE != &TimersListHead)
|
|
{
|
|
pTmr = CONTAINING_RECORD(pLE, TIMER, ptmrList);
|
|
|
|
if ( pMsg->lParam == (LPARAM)pTmr->pfn &&
|
|
(pTmr->flags & TMRF_SYSTEM) )
|
|
break;
|
|
|
|
pLE = pLE->Flink;
|
|
}
|
|
TimerLeave();
|
|
|
|
return pTmr;
|
|
}
|
|
|
|
BOOL
|
|
FASTCALL
|
|
ValidateTimerCallback(PTHREADINFO pti,
|
|
LPARAM lParam)
|
|
{
|
|
PLIST_ENTRY pLE;
|
|
BOOL Ret = FALSE;
|
|
PTIMER pTmr;
|
|
|
|
TimerEnterExclusive();
|
|
pLE = TimersListHead.Flink;
|
|
while (pLE != &TimersListHead)
|
|
{
|
|
pTmr = CONTAINING_RECORD(pLE, TIMER, ptmrList);
|
|
if ( (lParam == (LPARAM)pTmr->pfn) &&
|
|
!(pTmr->flags & (TMRF_SYSTEM|TMRF_RIT)) &&
|
|
(pTmr->pti->ppi == pti->ppi) )
|
|
{
|
|
Ret = TRUE;
|
|
break;
|
|
}
|
|
pLE = pLE->Flink;
|
|
}
|
|
TimerLeave();
|
|
|
|
return Ret;
|
|
}
|
|
|
|
UINT_PTR FASTCALL
|
|
IntSetTimer( PWND Window,
|
|
UINT_PTR IDEvent,
|
|
UINT Elapse,
|
|
TIMERPROC TimerFunc,
|
|
INT Type)
|
|
{
|
|
PTIMER pTmr;
|
|
UINT_PTR Ret = IDEvent;
|
|
ULONG ulBitmapIndex;
|
|
LARGE_INTEGER DueTime;
|
|
DueTime.QuadPart = (LONGLONG)(-97656); // 1024hz .9765625 ms set to 10.0 ms
|
|
|
|
#if 0
|
|
/* Windows NT/2k/XP behaviour */
|
|
if (Elapse > USER_TIMER_MAXIMUM)
|
|
{
|
|
TRACE("Adjusting uElapse\n");
|
|
Elapse = 1;
|
|
}
|
|
#else
|
|
/* Windows XP SP2 and Windows Server 2003 behaviour */
|
|
if (Elapse > USER_TIMER_MAXIMUM)
|
|
{
|
|
TRACE("Adjusting uElapse\n");
|
|
Elapse = USER_TIMER_MAXIMUM;
|
|
}
|
|
#endif
|
|
|
|
/* Windows 2k/XP and Windows Server 2003 SP1 behaviour */
|
|
if (Elapse < USER_TIMER_MINIMUM)
|
|
{
|
|
TRACE("Adjusting uElapse\n");
|
|
Elapse = USER_TIMER_MINIMUM; // 1024hz .9765625 ms, set to 10.0 ms (+/-)1 ms
|
|
}
|
|
|
|
/* Passing an IDEvent of 0 and the SetTimer returns 1.
|
|
It will create the timer with an ID of 0 */
|
|
if ((Window) && (IDEvent == 0))
|
|
Ret = 1;
|
|
|
|
pTmr = FindTimer(Window, IDEvent, Type);
|
|
|
|
if ((!pTmr) && (Window == NULL) && (!(Type & TMRF_SYSTEM)))
|
|
{
|
|
IntLockWindowlessTimerBitmap();
|
|
|
|
ulBitmapIndex = RtlFindClearBitsAndSet(&WindowLessTimersBitMap, 1, HintIndex);
|
|
HintIndex = (ulBitmapIndex + 1) % NUM_WINDOW_LESS_TIMERS;
|
|
if (ulBitmapIndex == ULONG_MAX)
|
|
{
|
|
IntUnlockWindowlessTimerBitmap();
|
|
ERR("Unable to find a free window-less timer id\n");
|
|
EngSetLastError(ERROR_NO_SYSTEM_RESOURCES);
|
|
return 0;
|
|
}
|
|
|
|
ASSERT(ulBitmapIndex < NUM_WINDOW_LESS_TIMERS);
|
|
IDEvent = MAX_WINDOW_LESS_TIMER_ID - ulBitmapIndex;
|
|
Ret = IDEvent;
|
|
|
|
IntUnlockWindowlessTimerBitmap();
|
|
}
|
|
|
|
if (!pTmr)
|
|
{
|
|
pTmr = CreateTimer();
|
|
if (!pTmr) return 0;
|
|
|
|
if (Window && (Type & TMRF_TIFROMWND))
|
|
pTmr->pti = Window->head.pti->pEThread->Tcb.Win32Thread;
|
|
else
|
|
{
|
|
if (Type & TMRF_RIT)
|
|
pTmr->pti = ptiRawInput;
|
|
else
|
|
pTmr->pti = PsGetCurrentThreadWin32Thread();
|
|
}
|
|
|
|
pTmr->pWnd = Window;
|
|
pTmr->cmsCountdown = Elapse;
|
|
pTmr->cmsRate = Elapse;
|
|
pTmr->pfn = TimerFunc;
|
|
pTmr->nID = IDEvent;
|
|
pTmr->flags = Type|TMRF_INIT;
|
|
}
|
|
else
|
|
{
|
|
pTmr->cmsCountdown = Elapse;
|
|
pTmr->cmsRate = Elapse;
|
|
}
|
|
|
|
ASSERT(MasterTimer != NULL);
|
|
// Start the timer thread!
|
|
if (TimersListHead.Flink == TimersListHead.Blink) // There is only one timer
|
|
KeSetTimer(MasterTimer, DueTime, NULL);
|
|
|
|
return Ret;
|
|
}
|
|
|
|
//
|
|
// Process win32k system timers.
|
|
//
|
|
VOID
|
|
CALLBACK
|
|
SystemTimerProc(HWND hwnd,
|
|
UINT uMsg,
|
|
UINT_PTR idEvent,
|
|
DWORD dwTime)
|
|
{
|
|
PDESKTOP pDesk;
|
|
PWND pWnd = NULL;
|
|
|
|
if (hwnd)
|
|
{
|
|
pWnd = UserGetWindowObject(hwnd);
|
|
if (!pWnd)
|
|
{
|
|
ERR("System Timer Proc has invalid window handle! %p Id: %u\n", hwnd, idEvent);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
TRACE( "Windowless Timer Running!\n" );
|
|
return;
|
|
}
|
|
|
|
switch (idEvent)
|
|
{
|
|
/*
|
|
Used in NtUserTrackMouseEvent.
|
|
*/
|
|
case ID_EVENT_SYSTIMER_MOUSEHOVER:
|
|
{
|
|
POINT Point;
|
|
UINT Msg;
|
|
WPARAM wParam;
|
|
|
|
pDesk = pWnd->head.rpdesk;
|
|
if ( pDesk->dwDTFlags & DF_TME_HOVER &&
|
|
pWnd == pDesk->spwndTrack )
|
|
{
|
|
Point = gpsi->ptCursor;
|
|
if ( RECTL_bPointInRect(&pDesk->rcMouseHover, Point.x, Point.y) )
|
|
{
|
|
if (pDesk->htEx == HTCLIENT) // In a client area.
|
|
{
|
|
wParam = MsqGetDownKeyState(pWnd->head.pti->MessageQueue);
|
|
Msg = WM_MOUSEHOVER;
|
|
|
|
if (pWnd->ExStyle & WS_EX_LAYOUTRTL)
|
|
{
|
|
Point.x = pWnd->rcClient.right - Point.x - 1;
|
|
}
|
|
else
|
|
Point.x -= pWnd->rcClient.left;
|
|
Point.y -= pWnd->rcClient.top;
|
|
}
|
|
else
|
|
{
|
|
wParam = pDesk->htEx; // Need to support all HTXYZ hits.
|
|
Msg = WM_NCMOUSEHOVER;
|
|
}
|
|
TRACE("Generating WM_NCMOUSEHOVER\n");
|
|
UserPostMessage(hwnd, Msg, wParam, MAKELPARAM(Point.x, Point.y));
|
|
pDesk->dwDTFlags &= ~DF_TME_HOVER;
|
|
break; // Kill this timer.
|
|
}
|
|
}
|
|
}
|
|
return; // Not this window so just return.
|
|
|
|
case ID_EVENT_SYSTIMER_FLASHWIN:
|
|
{
|
|
FLASHWINFO fwi =
|
|
{sizeof(FLASHWINFO),
|
|
UserHMGetHandle(pWnd),
|
|
FLASHW_SYSTIMER,0,0};
|
|
|
|
IntFlashWindowEx(pWnd, &fwi);
|
|
}
|
|
return;
|
|
|
|
default:
|
|
ERR("System Timer Proc invalid id %u!\n", idEvent);
|
|
break;
|
|
}
|
|
IntKillTimer(pWnd, idEvent, TRUE);
|
|
}
|
|
|
|
VOID
|
|
FASTCALL
|
|
StartTheTimers(VOID)
|
|
{
|
|
// Need to start gdi syncro timers then start timer with Hang App proc
|
|
// that calles Idle process so the screen savers will know to run......
|
|
IntSetTimer(NULL, 0, 1000, HungAppSysTimerProc, TMRF_RIT);
|
|
// Test Timers
|
|
// IntSetTimer(NULL, 0, 1000, SystemTimerProc, TMRF_RIT);
|
|
}
|
|
|
|
UINT_PTR
|
|
FASTCALL
|
|
SystemTimerSet( PWND Window,
|
|
UINT_PTR nIDEvent,
|
|
UINT uElapse,
|
|
TIMERPROC lpTimerFunc)
|
|
{
|
|
if (Window && Window->head.pti->pEThread->ThreadsProcess != PsGetCurrentProcess())
|
|
{
|
|
EngSetLastError(ERROR_ACCESS_DENIED);
|
|
TRACE("SysemTimerSet: Access Denied!\n");
|
|
return 0;
|
|
}
|
|
return IntSetTimer( Window, nIDEvent, uElapse, lpTimerFunc, TMRF_SYSTEM);
|
|
}
|
|
|
|
BOOL
|
|
FASTCALL
|
|
PostTimerMessages(PWND Window)
|
|
{
|
|
PLIST_ENTRY pLE;
|
|
MSG Msg;
|
|
PTHREADINFO pti;
|
|
BOOL Hit = FALSE;
|
|
PTIMER pTmr;
|
|
|
|
pti = PsGetCurrentThreadWin32Thread();
|
|
|
|
TimerEnterExclusive();
|
|
pLE = TimersListHead.Flink;
|
|
while(pLE != &TimersListHead)
|
|
{
|
|
pTmr = CONTAINING_RECORD(pLE, TIMER, ptmrList);
|
|
if ( (pTmr->flags & TMRF_READY) &&
|
|
(pTmr->pti == pti) &&
|
|
((pTmr->pWnd == Window) || (Window == NULL)) )
|
|
{
|
|
Msg.hwnd = (pTmr->pWnd ? UserHMGetHandle(pTmr->pWnd) : NULL);
|
|
Msg.message = (pTmr->flags & TMRF_SYSTEM) ? WM_SYSTIMER : WM_TIMER;
|
|
Msg.wParam = (WPARAM) pTmr->nID;
|
|
Msg.lParam = (LPARAM) pTmr->pfn;
|
|
Msg.time = EngGetTickCount32();
|
|
// Fix all wine win:test_GetMessagePos WM_TIMER tests. See CORE-10867.
|
|
Msg.pt = gpsi->ptCursor;
|
|
|
|
MsqPostMessage(pti, &Msg, FALSE, (QS_POSTMESSAGE|QS_ALLPOSTMESSAGE), 0, 0);
|
|
pTmr->flags &= ~TMRF_READY;
|
|
ClearMsgBitsMask(pti, QS_TIMER);
|
|
Hit = TRUE;
|
|
// Now move this entry to the end of the list so it will not be
|
|
// called again in the next msg loop.
|
|
if (pLE != &TimersListHead)
|
|
{
|
|
RemoveEntryList(&pTmr->ptmrList);
|
|
InsertTailList(&TimersListHead, &pTmr->ptmrList);
|
|
}
|
|
break;
|
|
}
|
|
|
|
pLE = pLE->Flink;
|
|
}
|
|
|
|
TimerLeave();
|
|
|
|
return Hit;
|
|
}
|
|
|
|
VOID
|
|
FASTCALL
|
|
ProcessTimers(VOID)
|
|
{
|
|
LARGE_INTEGER DueTime;
|
|
LONG Time;
|
|
PLIST_ENTRY pLE;
|
|
PTIMER pTmr;
|
|
LONG TimerCount = 0;
|
|
|
|
TimerEnterExclusive();
|
|
pLE = TimersListHead.Flink;
|
|
Time = EngGetTickCount32();
|
|
|
|
DueTime.QuadPart = (LONGLONG)(-97656); // 1024hz .9765625 ms set to 10.0 ms
|
|
|
|
while(pLE != &TimersListHead)
|
|
{
|
|
pTmr = CONTAINING_RECORD(pLE, TIMER, ptmrList);
|
|
TimerCount++;
|
|
if (pTmr->flags & TMRF_WAITING)
|
|
{
|
|
pLE = pTmr->ptmrList.Flink;
|
|
pTmr = CONTAINING_RECORD(pLE, TIMER, ptmrList);
|
|
continue;
|
|
}
|
|
|
|
if (pTmr->flags & TMRF_INIT)
|
|
{
|
|
pTmr->flags &= ~TMRF_INIT; // Skip this run.
|
|
}
|
|
else
|
|
{
|
|
if (pTmr->cmsCountdown < 0)
|
|
{
|
|
ASSERT(pTmr->pti);
|
|
if ((!(pTmr->flags & TMRF_READY)) && (!(pTmr->pti->TIF_flags & TIF_INCLEANUP)))
|
|
{
|
|
if (pTmr->flags & TMRF_ONESHOT)
|
|
pTmr->flags |= TMRF_WAITING;
|
|
|
|
if (pTmr->flags & TMRF_RIT)
|
|
{
|
|
// Hard coded call here, inside raw input thread.
|
|
pTmr->pfn(NULL, WM_SYSTIMER, pTmr->nID, (LPARAM)pTmr);
|
|
}
|
|
else
|
|
{
|
|
pTmr->flags |= TMRF_READY; // Set timer ready to be ran.
|
|
// Set thread message queue for this timer.
|
|
if (pTmr->pti)
|
|
{ // Wakeup thread
|
|
pTmr->pti->cTimersReady++;
|
|
ASSERT(pTmr->pti->pEventQueueServer != NULL);
|
|
MsqWakeQueue(pTmr->pti, QS_TIMER, TRUE);
|
|
}
|
|
}
|
|
}
|
|
pTmr->cmsCountdown = pTmr->cmsRate;
|
|
}
|
|
else
|
|
pTmr->cmsCountdown -= Time - TimeLast;
|
|
}
|
|
|
|
pLE = pLE->Flink;
|
|
}
|
|
|
|
// Restart the timer thread!
|
|
ASSERT(MasterTimer != NULL);
|
|
KeSetTimer(MasterTimer, DueTime, NULL);
|
|
|
|
TimeLast = Time;
|
|
|
|
TimerLeave();
|
|
TRACE("TimerCount = %d\n", TimerCount);
|
|
}
|
|
|
|
BOOL FASTCALL
|
|
DestroyTimersForWindow(PTHREADINFO pti, PWND Window)
|
|
{
|
|
PLIST_ENTRY pLE;
|
|
PTIMER pTmr;
|
|
BOOL TimersRemoved = FALSE;
|
|
|
|
if (Window == NULL)
|
|
return FALSE;
|
|
|
|
TimerEnterExclusive();
|
|
pLE = TimersListHead.Flink;
|
|
while(pLE != &TimersListHead)
|
|
{
|
|
pTmr = CONTAINING_RECORD(pLE, TIMER, ptmrList);
|
|
pLE = pLE->Flink; /* get next timer list entry before current timer is removed */
|
|
if ((pTmr) && (pTmr->pti == pti) && (pTmr->pWnd == Window))
|
|
{
|
|
TimersRemoved = RemoveTimer(pTmr);
|
|
}
|
|
}
|
|
|
|
TimerLeave();
|
|
|
|
return TimersRemoved;
|
|
}
|
|
|
|
BOOL FASTCALL
|
|
DestroyTimersForThread(PTHREADINFO pti)
|
|
{
|
|
PLIST_ENTRY pLE = TimersListHead.Flink;
|
|
PTIMER pTmr;
|
|
BOOL TimersRemoved = FALSE;
|
|
|
|
TimerEnterExclusive();
|
|
|
|
while(pLE != &TimersListHead)
|
|
{
|
|
pTmr = CONTAINING_RECORD(pLE, TIMER, ptmrList);
|
|
pLE = pLE->Flink; /* get next timer list entry before current timer is removed */
|
|
if ((pTmr) && (pTmr->pti == pti))
|
|
{
|
|
TimersRemoved = RemoveTimer(pTmr);
|
|
}
|
|
}
|
|
|
|
TimerLeave();
|
|
|
|
return TimersRemoved;
|
|
}
|
|
|
|
BOOL FASTCALL
|
|
IntKillTimer(PWND Window, UINT_PTR IDEvent, BOOL SystemTimer)
|
|
{
|
|
PTIMER pTmr = NULL;
|
|
TRACE("IntKillTimer Window %p id %uI systemtimer %s\n",
|
|
Window, IDEvent, SystemTimer ? "TRUE" : "FALSE");
|
|
|
|
TimerEnterExclusive();
|
|
pTmr = FindTimer(Window, IDEvent, SystemTimer ? TMRF_SYSTEM : 0);
|
|
|
|
if (pTmr)
|
|
{
|
|
RemoveTimer(pTmr);
|
|
}
|
|
TimerLeave();
|
|
|
|
return pTmr ? TRUE : FALSE;
|
|
}
|
|
|
|
CODE_SEG("INIT")
|
|
NTSTATUS
|
|
NTAPI
|
|
InitTimerImpl(VOID)
|
|
{
|
|
ULONG BitmapBytes;
|
|
|
|
/* Allocate FAST_MUTEX from non paged pool */
|
|
Mutex = ExAllocatePoolWithTag(NonPagedPool, sizeof(FAST_MUTEX), TAG_INTERNAL_SYNC);
|
|
if (!Mutex)
|
|
{
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
ExInitializeFastMutex(Mutex);
|
|
|
|
BitmapBytes = ALIGN_UP_BY(NUM_WINDOW_LESS_TIMERS, sizeof(ULONG) * 8) / 8;
|
|
WindowLessTimersBitMapBuffer = ExAllocatePoolWithTag(NonPagedPool, BitmapBytes, TAG_TIMERBMP);
|
|
if (WindowLessTimersBitMapBuffer == NULL)
|
|
{
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
RtlInitializeBitMap(&WindowLessTimersBitMap,
|
|
WindowLessTimersBitMapBuffer,
|
|
NUM_WINDOW_LESS_TIMERS);
|
|
|
|
/* Yes we need this, since ExAllocatePoolWithTag isn't supposed to zero out allocated memory */
|
|
RtlClearAllBits(&WindowLessTimersBitMap);
|
|
|
|
ExInitializeResourceLite(&TimerLock);
|
|
InitializeListHead(&TimersListHead);
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
UINT_PTR
|
|
APIENTRY
|
|
NtUserSetTimer
|
|
(
|
|
HWND hWnd,
|
|
UINT_PTR nIDEvent,
|
|
UINT uElapse,
|
|
TIMERPROC lpTimerFunc
|
|
)
|
|
{
|
|
PWND Window = NULL;
|
|
UINT_PTR ret;
|
|
|
|
TRACE("Enter NtUserSetTimer\n");
|
|
UserEnterExclusive();
|
|
if (hWnd) Window = UserGetWindowObject(hWnd);
|
|
|
|
ret = IntSetTimer(Window, nIDEvent, uElapse, lpTimerFunc, TMRF_TIFROMWND);
|
|
|
|
UserLeave();
|
|
TRACE("Leave NtUserSetTimer, ret=%u\n", ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
BOOL
|
|
APIENTRY
|
|
NtUserKillTimer
|
|
(
|
|
HWND hWnd,
|
|
UINT_PTR uIDEvent
|
|
)
|
|
{
|
|
PWND Window = NULL;
|
|
BOOL ret;
|
|
|
|
TRACE("Enter NtUserKillTimer\n");
|
|
UserEnterExclusive();
|
|
if (hWnd) Window = UserGetWindowObject(hWnd);
|
|
|
|
ret = IntKillTimer(Window, uIDEvent, FALSE);
|
|
|
|
UserLeave();
|
|
|
|
TRACE("Leave NtUserKillTimer, ret=%i\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
|
|
UINT_PTR
|
|
APIENTRY
|
|
NtUserSetSystemTimer(
|
|
HWND hWnd,
|
|
UINT_PTR nIDEvent,
|
|
UINT uElapse,
|
|
TIMERPROC lpTimerFunc
|
|
)
|
|
{
|
|
UINT_PTR ret;
|
|
|
|
UserEnterExclusive();
|
|
TRACE("Enter NtUserSetSystemTimer\n");
|
|
|
|
ret = IntSetTimer(UserGetWindowObject(hWnd), nIDEvent, uElapse, NULL, TMRF_SYSTEM);
|
|
|
|
UserLeave();
|
|
|
|
TRACE("Leave NtUserSetSystemTimer, ret=%u\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
BOOL
|
|
APIENTRY
|
|
NtUserValidateTimerCallback(
|
|
LPARAM lParam)
|
|
{
|
|
BOOL Ret = FALSE;
|
|
|
|
UserEnterShared();
|
|
|
|
Ret = ValidateTimerCallback(PsGetCurrentThreadWin32Thread(), lParam);
|
|
|
|
UserLeave();
|
|
return Ret;
|
|
}
|
|
|
|
/* EOF */
|