Use nanosleep instead of pthread_cond_timedwait for thread sleeping
There's a comment saying nanosleep's availability is questionable, but the information of what systems don't have that is now lost in time. It's quite likely that they were older, Unix systems we no longer support anyway. nanosleep comes from POSIX.1b-1993, which is merged into POSIX.1-2001, so chances are that it's supported almost everywhere where Qt runs (except for Windows anyway). Change-Id: I4fd18f8715c43a42429000f3b3d2c3b7343f94b4 Reviewed-by: Lars Knoll <lars.knoll@nokia.com> Reviewed-by: João Abecasis <joao.abecasis@nokia.com>
This commit is contained in:
parent
88c7c35b21
commit
caa22ff8ad
@ -312,6 +312,7 @@ static inline pid_t qt_safe_waitpid(pid_t pid, int *status, int options)
|
|||||||
|
|
||||||
// in qelapsedtimer_mac.cpp or qtimestamp_unix.cpp
|
// in qelapsedtimer_mac.cpp or qtimestamp_unix.cpp
|
||||||
timeval qt_gettime() Q_DECL_NOTHROW;
|
timeval qt_gettime() Q_DECL_NOTHROW;
|
||||||
|
void qt_nanosleep(timespec amount);
|
||||||
|
|
||||||
Q_CORE_EXPORT int qt_safe_select(int nfds, fd_set *fdread, fd_set *fdwrite, fd_set *fdexcept,
|
Q_CORE_EXPORT int qt_safe_select(int nfds, fd_set *fdread, fd_set *fdwrite, fd_set *fdexcept,
|
||||||
const struct timeval *tv);
|
const struct timeval *tv);
|
||||||
|
@ -454,59 +454,27 @@ void QThread::yieldCurrentThread()
|
|||||||
sched_yield();
|
sched_yield();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
static timespec makeTimespec(time_t secs, long nsecs)
|
||||||
\internal
|
|
||||||
helper function to do thread sleeps, since usleep()/nanosleep()
|
|
||||||
aren't reliable enough (in terms of behavior and availability)
|
|
||||||
*/
|
|
||||||
static void thread_sleep(struct timespec *ti)
|
|
||||||
{
|
{
|
||||||
pthread_mutex_t mtx;
|
struct timespec ts;
|
||||||
pthread_cond_t cnd;
|
ts.tv_sec = secs;
|
||||||
|
ts.tv_nsec = nsecs;
|
||||||
pthread_mutex_init(&mtx, 0);
|
return ts;
|
||||||
pthread_cond_init(&cnd, 0);
|
|
||||||
|
|
||||||
pthread_mutex_lock(&mtx);
|
|
||||||
(void) pthread_cond_timedwait(&cnd, &mtx, ti);
|
|
||||||
pthread_mutex_unlock(&mtx);
|
|
||||||
|
|
||||||
pthread_cond_destroy(&cnd);
|
|
||||||
pthread_mutex_destroy(&mtx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void QThread::sleep(unsigned long secs)
|
void QThread::sleep(unsigned long secs)
|
||||||
{
|
{
|
||||||
struct timeval tv;
|
qt_nanosleep(makeTimespec(secs, 0));
|
||||||
gettimeofday(&tv, 0);
|
|
||||||
struct timespec ti;
|
|
||||||
ti.tv_sec = tv.tv_sec + secs;
|
|
||||||
ti.tv_nsec = (tv.tv_usec * 1000);
|
|
||||||
thread_sleep(&ti);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void QThread::msleep(unsigned long msecs)
|
void QThread::msleep(unsigned long msecs)
|
||||||
{
|
{
|
||||||
struct timeval tv;
|
qt_nanosleep(makeTimespec(msecs / 1000, msecs % 1000 * 1000 * 1000));
|
||||||
gettimeofday(&tv, 0);
|
|
||||||
struct timespec ti;
|
|
||||||
|
|
||||||
ti.tv_nsec = (tv.tv_usec + (msecs % 1000) * 1000) * 1000;
|
|
||||||
ti.tv_sec = tv.tv_sec + (msecs / 1000) + (ti.tv_nsec / 1000000000);
|
|
||||||
ti.tv_nsec %= 1000000000;
|
|
||||||
thread_sleep(&ti);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void QThread::usleep(unsigned long usecs)
|
void QThread::usleep(unsigned long usecs)
|
||||||
{
|
{
|
||||||
struct timeval tv;
|
qt_nanosleep(makeTimespec(usecs / 1000 / 1000, usecs % (1000*1000) * 1000));
|
||||||
gettimeofday(&tv, 0);
|
|
||||||
struct timespec ti;
|
|
||||||
|
|
||||||
ti.tv_nsec = (tv.tv_usec + (usecs % 1000000)) * 1000;
|
|
||||||
ti.tv_sec = tv.tv_sec + (usecs / 1000000) + (ti.tv_nsec / 1000000000);
|
|
||||||
ti.tv_nsec %= 1000000000;
|
|
||||||
thread_sleep(&ti);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef QT_HAS_THREAD_PRIORITY_SCHEDULING
|
#ifdef QT_HAS_THREAD_PRIORITY_SCHEDULING
|
||||||
|
@ -39,8 +39,12 @@
|
|||||||
**
|
**
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
|
// ask for the latest POSIX, just in case
|
||||||
|
#define _POSIX_C_SOURCE 200809L
|
||||||
|
|
||||||
#include "qelapsedtimer.h"
|
#include "qelapsedtimer.h"
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
|
#include <time.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
#include <mach/mach_time.h>
|
#include <mach/mach_time.h>
|
||||||
@ -83,6 +87,15 @@ timeval qt_gettime() Q_DECL_NOTHROW
|
|||||||
return tv;
|
return tv;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void qt_nanosleep(timespec amount)
|
||||||
|
{
|
||||||
|
// Mac doesn't have clock_nanosleep, but it does have nanosleep.
|
||||||
|
// nanosleep is POSIX.1-1993
|
||||||
|
|
||||||
|
int r;
|
||||||
|
EINTR_LOOP(r, nanosleep(&amount, &amount));
|
||||||
|
}
|
||||||
|
|
||||||
void QElapsedTimer::start() Q_DECL_NOTHROW
|
void QElapsedTimer::start() Q_DECL_NOTHROW
|
||||||
{
|
{
|
||||||
t1 = mach_absolute_time();
|
t1 = mach_absolute_time();
|
||||||
|
@ -144,6 +144,20 @@ timeval qt_gettime() Q_DECL_NOTHROW
|
|||||||
return tv;
|
return tv;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void qt_nanosleep(timespec amount)
|
||||||
|
{
|
||||||
|
// We'd like to use clock_nanosleep.
|
||||||
|
//
|
||||||
|
// But clock_nanosleep is from POSIX.1-2001 and both are *not*
|
||||||
|
// affected by clock changes when using relative sleeps, even for
|
||||||
|
// CLOCK_REALTIME.
|
||||||
|
//
|
||||||
|
// nanosleep is POSIX.1-1993
|
||||||
|
|
||||||
|
int r;
|
||||||
|
EINTR_LOOP(r, nanosleep(&amount, &amount));
|
||||||
|
}
|
||||||
|
|
||||||
static qint64 elapsedAndRestart(qint64 sec, qint64 frac,
|
static qint64 elapsedAndRestart(qint64 sec, qint64 frac,
|
||||||
qint64 *nowsec, qint64 *nowfrac)
|
qint64 *nowsec, qint64 *nowfrac)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user