/* * Win32 waitable timers * * Copyright 1999 Alexandre Julliard */ #include #include #include "winerror.h" #include "winnls.h" #include "wine/unicode.h" #include "file.h" /* for FILETIME routines */ #include "server.h" /*********************************************************************** * CreateWaitableTimerA (KERNEL32.861) */ HANDLE WINAPI CreateWaitableTimerA( SECURITY_ATTRIBUTES *sa, BOOL manual, LPCSTR name ) { HANDLE ret; DWORD len = name ? MultiByteToWideChar( CP_ACP, 0, name, strlen(name), NULL, 0 ) : 0; if (len >= MAX_PATH) { SetLastError( ERROR_FILENAME_EXCED_RANGE ); return 0; } SERVER_START_REQ { struct create_timer_request *req = server_alloc_req( sizeof(*req), len * sizeof(WCHAR) ); req->manual = manual; req->inherit = (sa && (sa->nLength>=sizeof(*sa)) && sa->bInheritHandle); if (len) MultiByteToWideChar( CP_ACP, 0, name, strlen(name), server_data_ptr(req), len ); SetLastError(0); server_call( REQ_CREATE_TIMER ); ret = req->handle; } SERVER_END_REQ; if (ret == INVALID_HANDLE_VALUE) ret = 0; /* must return 0 on failure, not -1 */ return ret; } /*********************************************************************** * CreateWaitableTimerW (KERNEL32.862) */ HANDLE WINAPI CreateWaitableTimerW( SECURITY_ATTRIBUTES *sa, BOOL manual, LPCWSTR name ) { HANDLE ret; DWORD len = name ? strlenW(name) : 0; if (len >= MAX_PATH) { SetLastError( ERROR_FILENAME_EXCED_RANGE ); return 0; } SERVER_START_REQ { struct create_timer_request *req = server_alloc_req( sizeof(*req), len * sizeof(WCHAR) ); req->manual = manual; req->inherit = (sa && (sa->nLength>=sizeof(*sa)) && sa->bInheritHandle); memcpy( server_data_ptr(req), name, len * sizeof(WCHAR) ); SetLastError(0); server_call( REQ_CREATE_TIMER ); ret = req->handle; } SERVER_END_REQ; if (ret == INVALID_HANDLE_VALUE) ret = 0; /* must return 0 on failure, not -1 */ return ret; } /*********************************************************************** * OpenWaitableTimerA (KERNEL32.881) */ HANDLE WINAPI OpenWaitableTimerA( DWORD access, BOOL inherit, LPCSTR name ) { HANDLE ret; DWORD len = name ? MultiByteToWideChar( CP_ACP, 0, name, strlen(name), NULL, 0 ) : 0; if (len >= MAX_PATH) { SetLastError( ERROR_FILENAME_EXCED_RANGE ); return 0; } SERVER_START_REQ { struct open_timer_request *req = server_alloc_req( sizeof(*req), len * sizeof(WCHAR) ); req->access = access; req->inherit = inherit; if (len) MultiByteToWideChar( CP_ACP, 0, name, strlen(name), server_data_ptr(req), len ); server_call( REQ_OPEN_TIMER ); ret = req->handle; } SERVER_END_REQ; if (ret == INVALID_HANDLE_VALUE) ret = 0; /* must return 0 on failure, not -1 */ return ret; } /*********************************************************************** * OpenWaitableTimerW (KERNEL32.882) */ HANDLE WINAPI OpenWaitableTimerW( DWORD access, BOOL inherit, LPCWSTR name ) { HANDLE ret; DWORD len = name ? strlenW(name) : 0; if (len >= MAX_PATH) { SetLastError( ERROR_FILENAME_EXCED_RANGE ); return 0; } SERVER_START_REQ { struct open_timer_request *req = server_alloc_req( sizeof(*req), len * sizeof(WCHAR) ); req->access = access; req->inherit = inherit; memcpy( server_data_ptr(req), name, len * sizeof(WCHAR) ); server_call( REQ_OPEN_TIMER ); ret = req->handle; } SERVER_END_REQ; if (ret == INVALID_HANDLE_VALUE) ret = 0; /* must return 0 on failure, not -1 */ return ret; } /*********************************************************************** * SetWaitableTimer (KERNEL32.894) */ BOOL WINAPI SetWaitableTimer( HANDLE handle, const LARGE_INTEGER *when, LONG period, PTIMERAPCROUTINE callback, LPVOID arg, BOOL resume ) { BOOL ret; LARGE_INTEGER exp = *when; if (exp.s.HighPart < 0) /* relative time */ { LARGE_INTEGER now; NtQuerySystemTime( &now ); exp.QuadPart = RtlLargeIntegerSubtract( now.QuadPart, exp.QuadPart ); } SERVER_START_REQ { struct set_timer_request *req = server_alloc_req( sizeof(*req), 0 ); if (!exp.s.LowPart && !exp.s.HighPart) { /* special case to start timeout on now+period without too many calculations */ req->sec = 0; req->usec = 0; } else { DWORD remainder; req->sec = DOSFS_FileTimeToUnixTime( (FILETIME *)&exp, &remainder ); req->usec = remainder / 10; /* convert from 100-ns to us units */ } req->handle = handle; req->period = period; req->callback = callback; req->arg = arg; if (resume) SetLastError( ERROR_NOT_SUPPORTED ); /* set error but can still succeed */ ret = !server_call( REQ_SET_TIMER ); } SERVER_END_REQ; return ret; } /*********************************************************************** * CancelWaitableTimer (KERNEL32.857) */ BOOL WINAPI CancelWaitableTimer( HANDLE handle ) { BOOL ret; SERVER_START_REQ { struct cancel_timer_request *req = server_alloc_req( sizeof(*req), 0 ); req->handle = handle; ret = !server_call( REQ_CANCEL_TIMER ); } SERVER_END_REQ; return ret; }