winmm: Use unix fd's instead of win32 events to sync internal events in timer.

This commit is contained in:
Maarten Lankhorst 2007-06-05 19:32:20 +02:00 committed by Alexandre Julliard
parent 90d94efe10
commit 0383e4e499
1 changed files with 45 additions and 20 deletions

View File

@ -24,6 +24,7 @@
#include "wine/port.h" #include "wine/port.h"
#include <stdarg.h> #include <stdarg.h>
#include <errno.h>
#include <time.h> #include <time.h>
#ifdef HAVE_SYS_TIME_H #ifdef HAVE_SYS_TIME_H
# include <sys/time.h> # include <sys/time.h>
@ -31,6 +32,12 @@
#ifdef HAVE_UNISTD_H #ifdef HAVE_UNISTD_H
# include <unistd.h> # include <unistd.h>
#endif #endif
#ifdef HAVE_POLL_H
#include <poll.h>
#endif
#ifdef HAVE_SYS_POLL_H
#include <sys/poll.h>
#endif
#include "windef.h" #include "windef.h"
#include "winbase.h" #include "winbase.h"
@ -44,9 +51,9 @@ WINE_DEFAULT_DEBUG_CHANNEL(mmtime);
static HANDLE TIME_hMMTimer; static HANDLE TIME_hMMTimer;
static LPWINE_TIMERENTRY TIME_TimersList; static LPWINE_TIMERENTRY TIME_TimersList;
static HANDLE TIME_hWakeEvent;
static CRITICAL_SECTION TIME_cbcrst; static CRITICAL_SECTION TIME_cbcrst;
static BOOL TIME_TimeToDie = TRUE; static BOOL TIME_TimeToDie = TRUE;
static int TIME_fdWake[2] = { -1, -1 };
/* /*
* Some observations on the behavior of winmm on Windows. * Some observations on the behavior of winmm on Windows.
@ -219,7 +226,12 @@ static LPWINE_TIMERENTRY lpTimers;
static DWORD CALLBACK TIME_MMSysTimeThread(LPVOID arg) static DWORD CALLBACK TIME_MMSysTimeThread(LPVOID arg)
{ {
LPWINE_MM_IDATA iData = (LPWINE_MM_IDATA)arg; LPWINE_MM_IDATA iData = (LPWINE_MM_IDATA)arg;
DWORD rc; int sleep_time, ret;
char readme[16];
struct pollfd pfd;
pfd.fd = TIME_fdWake[0];
pfd.events = POLLIN;
TRACE("Starting main winmm thread\n"); TRACE("Starting main winmm thread\n");
@ -231,17 +243,21 @@ static DWORD CALLBACK TIME_MMSysTimeThread(LPVOID arg)
while (! TIME_TimeToDie) while (! TIME_TimeToDie)
{ {
int sleep_time = TIME_MMSysTimeCallback(iData); sleep_time = TIME_MMSysTimeCallback(iData);
if (sleep_time == 0) if (sleep_time == 0)
continue; continue;
rc = WaitForSingleObject(TIME_hWakeEvent, (sleep_time == -1) ? INFINITE : (DWORD)sleep_time); if ((ret = poll(&pfd, 1, sleep_time)) < 0)
if (rc != WAIT_TIMEOUT && rc != WAIT_OBJECT_0) {
{ if (errno != EINTR && errno != EAGAIN)
FIXME("Unexpected error %d(%d) in timer thread\n", rc, GetLastError()); {
break; ERR("Unexpected error in poll: %s(%d)\n", strerror(errno), errno);
} break;
}
}
while (ret > 0) ret = read(TIME_fdWake[0], readme, sizeof(readme));
} }
TRACE("Exiting main winmm thread\n"); TRACE("Exiting main winmm thread\n");
return 0; return 0;
@ -253,10 +269,17 @@ static DWORD CALLBACK TIME_MMSysTimeThread(LPVOID arg)
void TIME_MMTimeStart(void) void TIME_MMTimeStart(void)
{ {
if (!TIME_hMMTimer) { if (!TIME_hMMTimer) {
TIME_TimersList = NULL; TIME_TimersList = NULL;
TIME_hWakeEvent = CreateEventW(NULL, FALSE, FALSE, NULL); if (pipe(TIME_fdWake) < 0)
{
TIME_fdWake[0] = TIME_fdWake[1] = -1;
ERR("Cannot create pipe: %s\n", strerror(errno));
} else {
fcntl(TIME_fdWake[0], F_SETFL, O_NONBLOCK);
fcntl(TIME_fdWake[1], F_SETFL, O_NONBLOCK);
}
TIME_TimeToDie = FALSE; TIME_TimeToDie = FALSE;
TIME_hMMTimer = CreateThread(NULL, 0, TIME_MMSysTimeThread, &WINMM_IData, 0, NULL); TIME_hMMTimer = CreateThread(NULL, 0, TIME_MMSysTimeThread, &WINMM_IData, 0, NULL);
SetThreadPriority(TIME_hMMTimer, THREAD_PRIORITY_TIME_CRITICAL); SetThreadPriority(TIME_hMMTimer, THREAD_PRIORITY_TIME_CRITICAL);
InitializeCriticalSection(&TIME_cbcrst); InitializeCriticalSection(&TIME_cbcrst);
TIME_cbcrst.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": WINMM.TIME_cbcrst"); TIME_cbcrst.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": WINMM.TIME_cbcrst");
@ -269,16 +292,17 @@ void TIME_MMTimeStart(void)
void TIME_MMTimeStop(void) void TIME_MMTimeStop(void)
{ {
if (TIME_hMMTimer) { if (TIME_hMMTimer) {
const char a='a';
TIME_TimeToDie = TRUE; TIME_TimeToDie = TRUE;
SetEvent(TIME_hWakeEvent); write(TIME_fdWake[1], &a, sizeof(a));
/* FIXME: in the worst case, we're going to wait 65 seconds here :-( */ WaitForSingleObject(TIME_hMMTimer, INFINITE);
WaitForSingleObject(TIME_hMMTimer, INFINITE); close(TIME_fdWake[0]);
close(TIME_fdWake[1]);
CloseHandle(TIME_hMMTimer); TIME_fdWake[0] = TIME_fdWake[1] = -1;
CloseHandle(TIME_hWakeEvent); CloseHandle(TIME_hMMTimer);
TIME_hMMTimer = 0; TIME_hMMTimer = 0;
TIME_cbcrst.DebugInfo->Spare[0] = 0; TIME_cbcrst.DebugInfo->Spare[0] = 0;
DeleteCriticalSection(&TIME_cbcrst); DeleteCriticalSection(&TIME_cbcrst);
TIME_TimersList = NULL; TIME_TimersList = NULL;
@ -309,6 +333,7 @@ WORD TIME_SetEventInternal(UINT wDelay, UINT wResol,
WORD wNewID = 0; WORD wNewID = 0;
LPWINE_TIMERENTRY lpNewTimer; LPWINE_TIMERENTRY lpNewTimer;
LPWINE_TIMERENTRY lpTimer; LPWINE_TIMERENTRY lpTimer;
const char c = 'c';
TRACE("(%u, %u, %p, %08X, %04X);\n", wDelay, wResol, lpFunc, dwUser, wFlags); TRACE("(%u, %u, %p, %08X, %04X);\n", wDelay, wResol, lpFunc, dwUser, wFlags);
@ -344,7 +369,7 @@ WORD TIME_SetEventInternal(UINT wDelay, UINT wResol,
LeaveCriticalSection(&WINMM_IData.cs); LeaveCriticalSection(&WINMM_IData.cs);
/* Wake the service thread in case there is work to be done */ /* Wake the service thread in case there is work to be done */
SetEvent(TIME_hWakeEvent); write(TIME_fdWake[1], &c, sizeof(c));
TRACE("=> %u\n", wNewID + 1); TRACE("=> %u\n", wNewID + 1);