mirror of
https://github.com/reactos/reactos.git
synced 2025-01-07 06:45:24 +00:00
321 lines
7.7 KiB
C
321 lines
7.7 KiB
C
/*
|
|
* PROJECT: ReactOS Win32 Base API
|
|
* LICENSE: See COPYING in the top level directory
|
|
* FILE: dll/win32/kernel32/client/timerqueue.c
|
|
* PURPOSE: Timer Queue Functions
|
|
* PROGRAMMERS: Thomas Weidenmueller <w3seek@reactos.com>
|
|
*/
|
|
|
|
/* INCLUDES *******************************************************************/
|
|
|
|
#include <k32.h>
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
/* GLOBALS ********************************************************************/
|
|
|
|
HANDLE BasepDefaultTimerQueue = NULL;
|
|
|
|
/* PRIVATE FUNCTIONS **********************************************************/
|
|
|
|
/* FIXME - make this thread safe */
|
|
BOOL
|
|
WINAPI
|
|
BasepCreateDefaultTimerQueue(VOID)
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
/* Create the timer queue */
|
|
Status = RtlCreateTimerQueue(&BasepDefaultTimerQueue);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
BaseSetLastNTError(Status);
|
|
DPRINT1("Unable to create the default timer queue!\n");
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* PUBLIC FUNCTIONS ***********************************************************/
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
BOOL
|
|
WINAPI
|
|
CancelTimerQueueTimer(IN HANDLE TimerQueue,
|
|
IN HANDLE Timer)
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
if (!TimerQueue)
|
|
{
|
|
/* Use the default timer queue */
|
|
TimerQueue = BasepDefaultTimerQueue;
|
|
if (!TimerQueue)
|
|
{
|
|
/* A timer is being cancelled before one was created... fail */
|
|
SetLastError(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* Delete the timer */
|
|
Status = RtlDeleteTimer(TimerQueue, Timer, NULL);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
BaseSetLastNTError(Status);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
BOOL
|
|
WINAPI
|
|
ChangeTimerQueueTimer(IN HANDLE TimerQueue,
|
|
IN HANDLE Timer,
|
|
IN ULONG DueTime,
|
|
IN ULONG Period)
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
if (!TimerQueue)
|
|
{
|
|
/* Use the default timer queue */
|
|
TimerQueue = BasepDefaultTimerQueue;
|
|
if (!TimerQueue)
|
|
{
|
|
/* A timer is being cancelled before one was created... fail */
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* Delete the timer */
|
|
Status = RtlUpdateTimer(TimerQueue, Timer, DueTime, Period);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
BaseSetLastNTError(Status);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
HANDLE
|
|
WINAPI
|
|
CreateTimerQueue(VOID)
|
|
{
|
|
HANDLE Handle;
|
|
NTSTATUS Status;
|
|
|
|
/* Create the timer queue */
|
|
Status = RtlCreateTimerQueue(&Handle);
|
|
if(!NT_SUCCESS(Status))
|
|
{
|
|
BaseSetLastNTError(Status);
|
|
return NULL;
|
|
}
|
|
|
|
return Handle;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
BOOL
|
|
WINAPI
|
|
CreateTimerQueueTimer(OUT PHANDLE phNewTimer,
|
|
IN HANDLE TimerQueue,
|
|
IN WAITORTIMERCALLBACK Callback,
|
|
IN PVOID Parameter,
|
|
IN DWORD DueTime,
|
|
IN DWORD Period,
|
|
IN ULONG Flags)
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
/* Parameter is not optional -- clear it now */
|
|
*phNewTimer = NULL;
|
|
|
|
/* Check if no queue was given */
|
|
if (!TimerQueue)
|
|
{
|
|
/* Create the queue if it didn't already exist */
|
|
if (!(BasepDefaultTimerQueue) && !(BasepCreateDefaultTimerQueue()))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* Use the default queue */
|
|
TimerQueue = BasepDefaultTimerQueue;
|
|
}
|
|
|
|
/* Create the timer. Note that no parameter checking is done here */
|
|
Status = RtlCreateTimer(TimerQueue,
|
|
phNewTimer,
|
|
Callback,
|
|
Parameter,
|
|
DueTime,
|
|
Period,
|
|
Flags);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
BaseSetLastNTError(Status);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
BOOL
|
|
WINAPI
|
|
DeleteTimerQueue(IN HANDLE TimerQueue)
|
|
{
|
|
/* We don't allow the user to delete the default timer queue */
|
|
if (!TimerQueue)
|
|
{
|
|
SetLastError(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Delete the timer queue */
|
|
RtlDeleteTimerQueueEx(TimerQueue, 0);
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
BOOL
|
|
WINAPI
|
|
DeleteTimerQueueEx(IN HANDLE TimerQueue,
|
|
IN HANDLE CompletionEvent)
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
/* We don't allow the user to delete the default timer queue */
|
|
if (!TimerQueue)
|
|
{
|
|
SetLastError(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Delete the timer queue */
|
|
Status = RtlDeleteTimerQueueEx(TimerQueue, CompletionEvent);
|
|
if (((CompletionEvent != INVALID_HANDLE_VALUE) && (Status == STATUS_PENDING)) ||
|
|
!(NT_SUCCESS(Status)))
|
|
{
|
|
/* In case CompletionEvent == NULL, RtlDeleteTimerQueueEx() returns before
|
|
all callback routines returned. We set the last error code to STATUS_PENDING
|
|
and return FALSE. In case CompletionEvent == INVALID_HANDLE_VALUE we only
|
|
can get here if another error occured. In case CompletionEvent is something
|
|
else, we get here and fail, even though it isn't really an error (if Status == STATUS_PENDING).
|
|
We also handle all other failures the same way. */
|
|
BaseSetLastNTError(Status);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
BOOL
|
|
WINAPI
|
|
DeleteTimerQueueTimer(IN HANDLE TimerQueue,
|
|
IN HANDLE Timer,
|
|
IN HANDLE CompletionEvent)
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
if (!TimerQueue)
|
|
{
|
|
/* Use the default timer queue */
|
|
TimerQueue = BasepDefaultTimerQueue;
|
|
if (!TimerQueue)
|
|
{
|
|
/* A timer is being cancelled before one was created... fail */
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* Delete the timer */
|
|
Status = RtlDeleteTimer(TimerQueue, Timer, CompletionEvent);
|
|
if (((CompletionEvent != INVALID_HANDLE_VALUE) && (Status == STATUS_PENDING)) ||
|
|
!(NT_SUCCESS(Status)))
|
|
{
|
|
/* In case CompletionEvent == NULL, RtlDeleteTimerQueueEx() returns before
|
|
all callback routines returned. We set the last error code to STATUS_PENDING
|
|
and return FALSE. In case CompletionEvent == INVALID_HANDLE_VALUE we only
|
|
can get here if another error occured. In case CompletionEvent is something
|
|
else, we get here and fail, even though it isn't really an error (if Status == STATUS_PENDING).
|
|
We also handle all other failures the same way. */
|
|
BaseSetLastNTError(Status);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
HANDLE
|
|
WINAPI
|
|
SetTimerQueueTimer(IN HANDLE TimerQueue,
|
|
IN WAITORTIMERCALLBACK Callback,
|
|
IN PVOID Parameter,
|
|
IN DWORD DueTime,
|
|
IN DWORD Period,
|
|
IN BOOL PreferIo)
|
|
{
|
|
HANDLE Timer;
|
|
NTSTATUS Status;
|
|
|
|
/* Check if no queue was given */
|
|
if (!TimerQueue)
|
|
{
|
|
/* Create the queue if it didn't already exist */
|
|
if (!(BasepDefaultTimerQueue) && !(BasepCreateDefaultTimerQueue()))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* Use the default queue */
|
|
TimerQueue = BasepDefaultTimerQueue;
|
|
}
|
|
|
|
/* Create the timer */
|
|
Status = RtlCreateTimer(TimerQueue,
|
|
&Timer,
|
|
Callback,
|
|
Parameter,
|
|
DueTime,
|
|
Period,
|
|
PreferIo ? WT_EXECUTEINIOTHREAD : WT_EXECUTEDEFAULT);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
BaseSetLastNTError(Status);
|
|
return NULL;
|
|
}
|
|
|
|
return Timer;
|
|
}
|
|
|
|
/* EOF */
|