jack2/posix/JackPosixProcessSync.cpp

178 lines
5.4 KiB
C++

/*
Copyright (C) 2004-2008 Grame
This program 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 program 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 program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include "JackPosixProcessSync.h"
#include "JackError.h"
namespace Jack
{
void JackPosixProcessSync::Signal()
{
int res = pthread_cond_signal(&fCond);
if (res != 0) {
jack_error("JackPosixProcessSync::Signal error err = %s", strerror(res));
}
}
// TO DO : check thread consistency?
void JackPosixProcessSync::LockedSignal()
{
int res = pthread_mutex_lock(&fMutex);
if (res != 0) {
jack_error("JackPosixProcessSync::LockedSignal error err = %s", strerror(res));
}
res = pthread_cond_signal(&fCond);
if (res != 0) {
jack_error("JackPosixProcessSync::LockedSignal error err = %s", strerror(res));
}
res = pthread_mutex_unlock(&fMutex);
if (res != 0) {
jack_error("JackPosixProcessSync::LockedSignal error err = %s", strerror(res));
}
}
void JackPosixProcessSync::SignalAll()
{
int res = pthread_cond_broadcast(&fCond);
if (res != 0) {
jack_error("JackPosixProcessSync::SignalAll error err = %s", strerror(res));
}
}
// TO DO : check thread consistency?
void JackPosixProcessSync::LockedSignalAll()
{
int res = pthread_mutex_lock(&fMutex);
if (res != 0) {
jack_error("JackPosixProcessSync::LockedSignalAll error err = %s", strerror(res));
}
res = pthread_cond_broadcast(&fCond);
if (res != 0) {
jack_error("JackPosixProcessSync::LockedSignalAll error err = %s", strerror(res));
}
res = pthread_mutex_unlock(&fMutex);
if (res != 0) {
jack_error("JackPosixProcessSync::LockedSignalAll error err = %s", strerror(res));
}
}
void JackPosixProcessSync::Wait()
{
ThrowIf(!pthread_equal(pthread_self(), fOwner), JackException("JackPosixProcessSync::Wait: a thread has to have locked a mutex before it can wait"));
fOwner = 0;
int res = pthread_cond_wait(&fCond, &fMutex);
if (res != 0) {
jack_error("JackPosixProcessSync::Wait error err = %s", strerror(res));
} else {
fOwner = pthread_self();
}
}
// TO DO : check thread consistency?
void JackPosixProcessSync::LockedWait()
{
int res;
res = pthread_mutex_lock(&fMutex);
if (res != 0) {
jack_error("JackPosixProcessSync::LockedWait error err = %s", strerror(res));
}
if ((res = pthread_cond_wait(&fCond, &fMutex)) != 0) {
jack_error("JackPosixProcessSync::LockedWait error err = %s", strerror(res));
}
res = pthread_mutex_unlock(&fMutex);
if (res != 0) {
jack_error("JackPosixProcessSync::LockedWait error err = %s", strerror(res));
}
}
bool JackPosixProcessSync::TimedWait(long usec)
{
ThrowIf(!pthread_equal(pthread_self(), fOwner), JackException("JackPosixProcessSync::TimedWait: a thread has to have locked a mutex before it can wait"));
fOwner = 0;
struct timeval T0, T1;
timespec time;
struct timeval now;
int res;
jack_log("JackPosixProcessSync::TimedWait time out = %ld", usec);
gettimeofday(&T0, 0);
gettimeofday(&now, 0);
unsigned int next_date_usec = now.tv_usec + usec;
time.tv_sec = now.tv_sec + (next_date_usec / 1000000);
time.tv_nsec = (next_date_usec % 1000000) * 1000;
res = pthread_cond_timedwait(&fCond, &fMutex, &time);
if (res != 0 && res != ETIMEDOUT) {
jack_error("JackPosixProcessSync::TimedWait error usec = %ld err = %s", usec, strerror(res));
} else {
fOwner = pthread_self();
}
gettimeofday(&T1, 0);
jack_log("JackPosixProcessSync::TimedWait finished delta = %5.1lf",
(1e6 * T1.tv_sec - 1e6 * T0.tv_sec + T1.tv_usec - T0.tv_usec));
return (res == 0);
}
// TO DO : check thread consistency?
bool JackPosixProcessSync::LockedTimedWait(long usec)
{
struct timeval T0, T1;
timespec time;
struct timeval now;
int res1, res2;
res1 = pthread_mutex_lock(&fMutex);
if (res1 != 0) {
jack_error("JackPosixProcessSync::LockedTimedWait error err = %s", usec, strerror(res1));
}
jack_log("JackPosixProcessSync::TimedWait time out = %ld", usec);
gettimeofday(&T0, 0);
gettimeofday(&now, 0);
unsigned int next_date_usec = now.tv_usec + usec;
time.tv_sec = now.tv_sec + (next_date_usec / 1000000);
time.tv_nsec = (next_date_usec % 1000000) * 1000;
res2 = pthread_cond_timedwait(&fCond, &fMutex, &time);
if (res2 != 0) {
jack_error("JackPosixProcessSync::LockedTimedWait error usec = %ld err = %s", usec, strerror(res2));
}
gettimeofday(&T1, 0);
res1 = pthread_mutex_unlock(&fMutex);
if (res1 != 0) {
jack_error("JackPosixProcessSync::LockedTimedWait error err = %s", usec, strerror(res1));
}
jack_log("JackPosixProcessSync::TimedWait finished delta = %5.1lf",
(1e6 * T1.tv_sec - 1e6 * T0.tv_sec + T1.tv_usec - T0.tv_usec));
return (res2 == 0);
}
} // end of namespace