Added setting of the timer resolution in the Win32 implementation of Sleep and rewrote the Unix Sleep implementation to use nanosleep instead of pthread_cond_timedwait to prevent spurious wakeups from causing the function to return too early.

This commit is contained in:
binary1248 2013-01-11 05:34:23 +01:00
parent 91e1743516
commit 20db4969c5
3 changed files with 22 additions and 30 deletions

View File

@ -74,6 +74,9 @@ endif()
if(LINUX) if(LINUX)
set(SYSTEM_EXT_LIBS ${SYSTEM_EXT_LIBS} rt) set(SYSTEM_EXT_LIBS ${SYSTEM_EXT_LIBS} rt)
endif() endif()
if(WINDOWS)
set(SYSTEM_EXT_LIBS ${SYSTEM_EXT_LIBS} winmm)
endif()
# define the sfml-system target # define the sfml-system target
sfml_add_library(sfml-system sfml_add_library(sfml-system

View File

@ -26,9 +26,8 @@
// Headers // Headers
//////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////
#include <SFML/System/Unix/SleepImpl.hpp> #include <SFML/System/Unix/SleepImpl.hpp>
#include <pthread.h> #include <errno.h>
#include <unistd.h> #include <time.h>
#include <sys/time.h>
namespace sf namespace sf
@ -38,38 +37,17 @@ namespace priv
//////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////
void sleepImpl(Time time) void sleepImpl(Time time)
{ {
// usleep is not reliable enough (it might block the
// whole process instead of just the current thread)
// so we must use pthread_cond_timedwait instead
// this implementation is inspired from Qt
Uint64 usecs = time.asMicroseconds(); Uint64 usecs = time.asMicroseconds();
// get the current time // construct the time to wait
timeval tv;
gettimeofday(&tv, NULL);
// construct the time limit (current time + time to wait)
timespec ti; timespec ti;
ti.tv_nsec = (tv.tv_usec + (usecs % 1000000)) * 1000; ti.tv_nsec = (usecs % 1000000) * 1000;
ti.tv_sec = tv.tv_sec + (usecs / 1000000) + (ti.tv_nsec / 1000000000); ti.tv_sec = usecs / 1000000;
ti.tv_nsec %= 1000000000;
// create a mutex and thread condition
pthread_mutex_t mutex;
pthread_mutex_init(&mutex, 0);
pthread_cond_t condition;
pthread_cond_init(&condition, 0);
// wait... // wait...
pthread_mutex_lock(&mutex); while ((nanosleep(&ti, &ti) == -1) && (errno == EINTR))
pthread_cond_timedwait(&condition, &mutex, &ti); {
pthread_mutex_unlock(&mutex); }
// destroy the mutex and condition
pthread_cond_destroy(&condition);
pthread_mutex_destroy(&mutex);
} }
} // namespace priv } // namespace priv

View File

@ -36,7 +36,18 @@ namespace priv
//////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////
void sleepImpl(Time time) void sleepImpl(Time time)
{ {
// get the supported timer resolutions on this system
TIMECAPS tc;
timeGetDevCaps(&tc, sizeof(TIMECAPS));
// set the timer resolution to the minimum for the Sleep call
timeBeginPeriod(tc.wPeriodMin);
// wait...
::Sleep(time.asMilliseconds()); ::Sleep(time.asMilliseconds());
// reset the timer resolution back to the system default
timeEndPeriod(tc.wPeriodMin);
} }
} // namespace priv } // namespace priv