/* * Test winmm timer * * Copyright (c) 2005 Robert Reif * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include #include "wine/test.h" #include "windef.h" #include "winbase.h" #include "winnls.h" #include "mmsystem.h" #define NOBITMAP #include "mmreg.h" #include "winmm_test.h" static TIMECAPS tc; static void test_timeGetDevCaps(void) { MMRESULT rc; rc = timeGetDevCaps(&tc, 0); ok(rc == TIMERR_NOCANDO || rc == MMSYSERR_INVALPARAM, "timeGetDevCaps() returned %s, should have returned TIMERR_NOCANDO " "or MMSYSERR_INVALPARAM\n", mmsys_error(rc)); rc = timeGetDevCaps(0, sizeof(tc)); ok(rc == TIMERR_NOCANDO || rc == TIMERR_STRUCT, "timeGetDevCaps() returned %s, should have returned TIMERR_NOCANDO " "or TIMERR_STRUCT\n", mmsys_error(rc)); rc = timeGetDevCaps(0, 0); ok(rc == TIMERR_NOCANDO || rc == MMSYSERR_INVALPARAM, "timeGetDevCaps() returned %s, should have returned TIMERR_NOCANDO " "or MMSYSERR_INVALPARAM\n", mmsys_error(rc)); rc = timeGetDevCaps(&tc, sizeof(tc)); ok(rc == TIMERR_NOERROR, "timeGetDevCaps() returned %s, " "should have returned TIMERR_NOERROR\n", mmsys_error(rc)); if (rc == TIMERR_NOERROR) trace("wPeriodMin = %u, wPeriodMax = %u\n", tc.wPeriodMin, tc.wPeriodMax); } #define NUM_SAMPLES 100 static DWORD count = 0; static DWORD times[NUM_SAMPLES]; static void CALLBACK testTimeProc(UINT uID, UINT uMsg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2) { if (count < NUM_SAMPLES) times[count++] = timeGetTime(); } static void test_timer(UINT period, UINT resolution) { MMRESULT rc; UINT i, id, delta; DWORD dwMin = 0xffffffff, dwMax = 0; double sum = 0.0; double deviation = 0.0; count = 0; for (i = 0; i < NUM_SAMPLES; i++) times[i] = 0; rc = timeBeginPeriod(period); ok(rc == TIMERR_NOERROR, "timeBeginPeriod(%u) returned %s, " "should have returned TIMERR_NOERROR\n", period, mmsys_error(rc)); if (rc != TIMERR_NOERROR) return; id = timeSetEvent(period, resolution, testTimeProc, 0, TIME_PERIODIC); ok(id != 0, "timeSetEvent(%u, %u, %p, 0, TIME_PERIODIC) returned %d, " "should have returned id > 0\n", period, resolution, testTimeProc, id); if (id == 0) return; Sleep((NUM_SAMPLES * period) + (2 * period)); rc = timeEndPeriod(period); ok(rc == TIMERR_NOERROR, "timeEndPeriod(%u) returned %s, " "should have returned TIMERR_NOERROR\n", period, mmsys_error(rc)); if (rc != TIMERR_NOERROR) return; rc = timeKillEvent(id); ok(rc == TIMERR_NOERROR, "timeKillEvent(%u) returned %s, " "should have returned TIMERR_NOERROR\n", id, mmsys_error(rc)); trace("period = %u, resolution = %u\n", period, resolution); for (i = 0; i < count; i++) { if (i == 0) { if (winetest_debug > 1) trace("time[%d] = %u\n", i, times[i]); } else { delta = times[i] - times[i - 1]; if (winetest_debug > 1) trace("time[%d] = %u delta = %d\n", i, times[i], delta); sum += delta; deviation += ((delta - period) * (delta - period)); if (delta < dwMin) dwMin = delta; if (delta > dwMax) dwMax = delta; } } trace("min = %u, max = %u, average = %f, standard deviation = %f\n", dwMin, dwMax, sum / (count - 1), sqrt(deviation / (count - 2))); } static const char * get_priority(int priority) { static char tmp[32]; #define STR(x) case x: return #x switch(priority) { STR(THREAD_PRIORITY_LOWEST); STR(THREAD_PRIORITY_BELOW_NORMAL); STR(THREAD_PRIORITY_NORMAL); STR(THREAD_PRIORITY_HIGHEST); STR(THREAD_PRIORITY_ABOVE_NORMAL); STR(THREAD_PRIORITY_TIME_CRITICAL); STR(THREAD_PRIORITY_IDLE); } sprintf(tmp, "UNKNOWN(%d)", priority); return tmp; } static int priority = 0; static BOOL fired = FALSE; static void CALLBACK priorityTimeProc(UINT uID, UINT uMsg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2) { priority = GetThreadPriority(GetCurrentThread()); ok(priority!=THREAD_PRIORITY_ERROR_RETURN, "GetThreadPriority() failed, GetLastError() = %u\n", GetLastError()); fired = TRUE; } static void test_priority(void) { UINT id; id = timeSetEvent(100, 100, priorityTimeProc, 0, TIME_ONESHOT); ok(id != 0, "timeSetEvent(100, 100, %p, 0, TIME_ONESHOT) returned %d, " "should have returned id > 0\n", priorityTimeProc, id); if (id == 0) return; Sleep(200); ok(fired == TRUE, "Callback not called\n"); if (fired) { ok(priority == THREAD_PRIORITY_TIME_CRITICAL, "thread priority is %s, should be THREAD_PRIORITY_TIME_CRITICAL\n", get_priority(priority)); } timeKillEvent(id); } static void CALLBACK kill_timer_callback(UINT id, UINT msg, DWORD_PTR arg, DWORD_PTR dw1, DWORD_PTR dw2) { HANDLE event = (HANDLE)arg; MMRESULT res; res = timeKillEvent(id); ok(res == TIMERR_NOERROR, "timeKillEvent failed in callback\n"); SetEvent(event); } static void test_timer_lifetime(void) { UINT timers[20], i; HANDLE event; MMRESULT res; event = CreateEventW(NULL, FALSE, FALSE, NULL); ok(event != NULL, "CreateEvent failed\n"); for (i = 0; i < 20; i++) { timers[i] = timeSetEvent(10000, 0, event, 0, TIME_CALLBACK_EVENT_SET); if (i < 16) ok(timers[i] != 0, "%d) timeSetEvent failed\n", i); else ok(!timers[i], "%d) timeSetEvent succeeded\n", i); } for (i = 0; i < 16; i++) { res = timeKillEvent(timers[i]); ok(res == TIMERR_NOERROR, "%d) timeKillEvent failed\n", i); } timers[0] = timeSetEvent(10, 0, event, 0, TIME_CALLBACK_EVENT_SET); ok(timers[0], "timeSetEvent failed\n"); WaitForSingleObject(event, 1000); timers[1] = timeSetEvent(1000, 0, event, 0, TIME_CALLBACK_EVENT_SET); ok(timers[1], "timeSetEvent failed\n"); ok(timers[0] != timers[1], "timer got the same id as just destroyed timer\n"); res = timeKillEvent(timers[0]); ok(res == TIMERR_NOCANDO, "timeKillEvent returned %d\n", res); res = timeKillEvent(timers[1]); ok(res == TIMERR_NOERROR, "timeKillEvent returned %d\n", res); timers[0] = timeSetEvent(10, 0, kill_timer_callback, (DWORD_PTR)event, TIME_ONESHOT); WaitForSingleObject(event, 1000); timers[0] = timeSetEvent(10, 0, kill_timer_callback, (DWORD_PTR)event, TIME_KILL_SYNCHRONOUS); WaitForSingleObject(event, 1000); CloseHandle(event); } START_TEST(timer) { test_timeGetDevCaps(); if (tc.wPeriodMin <= 1) { test_timer(1, 0); test_timer(1, 1); } if (tc.wPeriodMin <= 10) { test_timer(10, 0); test_timer(10, 1); test_timer(10, 10); } if (tc.wPeriodMin <= 20) { test_timer(20, 0); test_timer(20, 1); test_timer(20, 10); test_timer(20, 20); } test_priority(); test_timer_lifetime(); }