2007-11-15 03:18:48 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* "$Id: rsthreads.cc,v 1.1 2007-02-19 20:08:30 rmf24 Exp $"
|
|
|
|
*
|
|
|
|
* RetroShare C++ Interface.
|
|
|
|
*
|
|
|
|
* Copyright 2004-2007 by Robert Fernie.
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License Version 2 as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
|
|
* USA.
|
|
|
|
*
|
|
|
|
* Please report all bugs and problems to "retroshare@lunamutt.com".
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "rsthreads.h"
|
2014-10-24 21:31:58 +00:00
|
|
|
#include <unistd.h> // for usleep()
|
|
|
|
#include <errno.h> // for errno
|
2011-03-30 22:32:13 +00:00
|
|
|
#include <iostream>
|
2013-10-21 11:00:49 +00:00
|
|
|
#include <time.h>
|
2013-08-31 08:16:09 +00:00
|
|
|
|
2016-06-03 19:03:10 +02:00
|
|
|
int __attribute__((weak)) pthread_setname_np(pthread_t __target_thread, const char *__buf) ;
|
|
|
|
|
2011-07-04 22:59:39 +00:00
|
|
|
#ifdef RSMUTEX_DEBUG
|
|
|
|
#include <stdio.h>
|
2014-10-28 20:37:48 +00:00
|
|
|
#include <sys/time.h>
|
2011-07-04 22:59:39 +00:00
|
|
|
#endif
|
|
|
|
|
2008-11-02 11:38:11 +00:00
|
|
|
/*******
|
|
|
|
* #define DEBUG_THREADS 1
|
2013-08-31 08:16:09 +00:00
|
|
|
* #define RSMUTEX_ABORT 1 // Catch wrong pthreads mode.
|
2008-11-02 11:38:11 +00:00
|
|
|
*******/
|
2016-05-12 10:26:38 -04:00
|
|
|
#define THREAD_DEBUG std::cerr << "[this=" << (void*)this << ", caller thread ID: " << std::hex << pthread_self() << ", thread ID: " << mTid << std::dec << "] "
|
2008-11-02 11:38:11 +00:00
|
|
|
|
2013-08-31 08:16:09 +00:00
|
|
|
#ifdef RSMUTEX_ABORT
|
|
|
|
#include <stdlib.h>
|
|
|
|
#endif
|
|
|
|
|
2008-11-02 11:38:11 +00:00
|
|
|
#ifdef DEBUG_THREADS
|
|
|
|
#include <iostream>
|
|
|
|
#endif
|
|
|
|
|
2014-12-21 20:52:34 +00:00
|
|
|
void *RsThread::rsthread_init(void* p)
|
2007-11-15 03:18:48 +00:00
|
|
|
{
|
|
|
|
RsThread *thread = (RsThread *) p;
|
|
|
|
if (!thread)
|
|
|
|
{
|
2014-12-21 20:52:34 +00:00
|
|
|
return NULL;
|
2007-11-15 03:18:48 +00:00
|
|
|
}
|
2015-05-22 20:54:38 +00:00
|
|
|
// tell the OS to free the thread resources when this function exits
|
|
|
|
// it is a replacement for pthread_join()
|
|
|
|
pthread_detach(pthread_self());
|
|
|
|
|
2016-05-04 22:34:03 -04:00
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-04 22:51:33 -04:00
|
|
|
std::cerr << "[Thread ID:" << std::hex << pthread_self() << std::dec << "] thread is started. Calling runloop()..." << std::endl;
|
2016-05-04 22:34:03 -04:00
|
|
|
#endif
|
|
|
|
|
2015-05-22 20:54:38 +00:00
|
|
|
thread -> runloop();
|
2014-12-21 20:52:34 +00:00
|
|
|
return NULL;
|
2007-11-15 03:18:48 +00:00
|
|
|
}
|
2016-05-05 20:52:10 -04:00
|
|
|
RsThread::RsThread()
|
2015-05-22 20:54:38 +00:00
|
|
|
{
|
|
|
|
#ifdef WINDOWS_SYS
|
|
|
|
memset (&mTid, 0, sizeof(mTid));
|
|
|
|
#else
|
|
|
|
mTid = 0;
|
|
|
|
#endif
|
2016-05-12 10:26:38 -04:00
|
|
|
// The thread is certainly not running. This avoids to lock down when calling shutdown on a thread that has never started.
|
|
|
|
|
|
|
|
#ifdef DEBUG_THREADS
|
|
|
|
THREAD_DEBUG << "[Thread ID:" << std::hex << pthread_self() << std::dec << "] thread object created. Initing stopped=1, should_stop=0" << std::endl;
|
|
|
|
#endif
|
|
|
|
mHasStoppedSemaphore.set(1) ;
|
|
|
|
mShouldStopSemaphore.set(0) ;
|
2015-05-22 20:54:38 +00:00
|
|
|
}
|
2016-06-01 14:58:12 +02:00
|
|
|
|
2015-05-22 20:54:38 +00:00
|
|
|
bool RsThread::isRunning()
|
|
|
|
{
|
|
|
|
// do we need a mutex for this ?
|
2016-05-05 20:52:10 -04:00
|
|
|
int sval = mHasStoppedSemaphore.value() ;
|
2015-05-22 20:54:38 +00:00
|
|
|
|
|
|
|
return !sval ;
|
|
|
|
}
|
|
|
|
|
2015-05-25 15:11:42 +00:00
|
|
|
bool RsThread::shouldStop()
|
|
|
|
{
|
2016-05-05 20:52:10 -04:00
|
|
|
int sval = mShouldStopSemaphore.value() ;
|
2015-05-25 15:11:42 +00:00
|
|
|
return sval > 0;
|
|
|
|
}
|
|
|
|
|
2015-05-22 20:54:38 +00:00
|
|
|
void RsTickingThread::shutdown()
|
2015-05-22 18:16:49 +00:00
|
|
|
{
|
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << "pqithreadstreamer::shutdown()" << std::endl;
|
2015-05-22 18:16:49 +00:00
|
|
|
#endif
|
|
|
|
|
2016-05-05 20:52:10 -04:00
|
|
|
int sval = mHasStoppedSemaphore.value() ;
|
2015-05-22 18:16:49 +00:00
|
|
|
|
|
|
|
if(sval > 0)
|
|
|
|
{
|
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << " thread not running. Quit." << std::endl;
|
2015-05-22 18:16:49 +00:00
|
|
|
#endif
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
|
2015-05-25 15:11:42 +00:00
|
|
|
ask_for_stop() ;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RsThread::ask_for_stop()
|
|
|
|
{
|
2015-05-22 18:16:49 +00:00
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << " calling stop" << std::endl;
|
2015-05-22 18:16:49 +00:00
|
|
|
#endif
|
2016-05-11 12:42:59 -04:00
|
|
|
mShouldStopSemaphore.set(1);
|
2015-05-22 18:16:49 +00:00
|
|
|
}
|
|
|
|
|
2015-05-22 20:54:38 +00:00
|
|
|
void RsTickingThread::fullstop()
|
2015-05-22 18:16:49 +00:00
|
|
|
{
|
|
|
|
shutdown() ;
|
|
|
|
|
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << " waiting stop" << std::endl;
|
2015-05-22 18:16:49 +00:00
|
|
|
#endif
|
2016-05-11 22:18:02 -04:00
|
|
|
if(pthread_equal(mTid,pthread_self()))
|
|
|
|
{
|
2016-05-12 10:26:38 -04:00
|
|
|
THREAD_DEBUG << "(WW) RsTickingThread::fullstop() called by same thread. This is unexpected." << std::endl;
|
2016-05-11 22:18:02 -04:00
|
|
|
return ;
|
|
|
|
}
|
|
|
|
|
2016-05-12 10:26:38 -04:00
|
|
|
mHasStoppedSemaphore.wait_no_relock(); // Wait for semaphore value to become 1, but does not decrement it when obtained.
|
2015-05-22 18:16:49 +00:00
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << " finished!" << std::endl;
|
2015-05-22 18:16:49 +00:00
|
|
|
#endif
|
|
|
|
}
|
2016-06-01 14:58:12 +02:00
|
|
|
|
|
|
|
void RsThread::start(const std::string &threadName)
|
2007-11-15 03:18:48 +00:00
|
|
|
{
|
|
|
|
pthread_t tid;
|
2014-12-21 20:52:34 +00:00
|
|
|
void *data = (void *)this ;
|
2007-11-15 03:18:48 +00:00
|
|
|
|
2015-05-22 18:16:49 +00:00
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-12 10:26:38 -04:00
|
|
|
THREAD_DEBUG << "pqithreadstreamer::start() initing should_stop=0" << std::endl;
|
2009-08-04 23:22:44 +00:00
|
|
|
#endif
|
2016-05-05 20:52:10 -04:00
|
|
|
mShouldStopSemaphore.set(0) ;
|
2009-08-04 23:22:44 +00:00
|
|
|
|
2014-12-21 20:52:34 +00:00
|
|
|
int err ;
|
2011-03-30 22:32:13 +00:00
|
|
|
|
2015-05-18 08:51:23 +00:00
|
|
|
// pthread_create is a memory barrier
|
2015-05-22 18:16:49 +00:00
|
|
|
// -> the new thread will see mIsRunning() = true
|
|
|
|
|
2014-12-21 20:52:34 +00:00
|
|
|
if( 0 == (err=pthread_create(&tid, 0, &rsthread_init, data)))
|
2016-06-01 14:58:12 +02:00
|
|
|
{
|
2014-12-21 20:52:34 +00:00
|
|
|
mTid = tid;
|
2016-06-01 14:58:12 +02:00
|
|
|
|
|
|
|
// set name
|
2016-06-03 19:03:10 +02:00
|
|
|
|
|
|
|
if(pthread_setname_np)
|
|
|
|
if(!threadName.empty())
|
|
|
|
{
|
|
|
|
// thread names are restricted to 16 characters including the terminating null byte
|
|
|
|
if(threadName.length() > 15)
|
|
|
|
{
|
2016-06-01 14:58:12 +02:00
|
|
|
#ifdef DEBUG_THREADS
|
2016-06-03 19:03:10 +02:00
|
|
|
THREAD_DEBUG << "RsThread::start called with to long name '" << name << "' truncating..." << std::endl;
|
2016-06-01 14:58:12 +02:00
|
|
|
#endif
|
2016-06-03 19:03:10 +02:00
|
|
|
pthread_setname_np(mTid, threadName.substr(0, 15).c_str());
|
|
|
|
} else {
|
|
|
|
pthread_setname_np(mTid, threadName.c_str());
|
|
|
|
}
|
|
|
|
}
|
2016-06-01 14:58:12 +02:00
|
|
|
}
|
2014-12-21 20:52:34 +00:00
|
|
|
else
|
|
|
|
{
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << "Fatal error: pthread_create could not create a thread. Error returned: " << err << " !!!!!!!" << std::endl;
|
2016-06-01 14:58:12 +02:00
|
|
|
mHasStoppedSemaphore.set(1) ;
|
2007-11-15 03:18:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-21 20:52:34 +00:00
|
|
|
|
2010-05-29 15:14:25 +00:00
|
|
|
|
2016-05-04 22:23:36 -04:00
|
|
|
RsTickingThread::RsTickingThread()
|
2009-08-04 23:22:44 +00:00
|
|
|
{
|
2016-05-04 22:23:36 -04:00
|
|
|
#ifdef DEBUG_THREADS
|
|
|
|
THREAD_DEBUG << "RsTickingThread::RsTickingThread()" << std::endl;
|
|
|
|
#endif
|
2009-08-04 23:22:44 +00:00
|
|
|
}
|
|
|
|
|
2015-05-25 15:11:42 +00:00
|
|
|
void RsSingleJobThread::runloop()
|
|
|
|
{
|
2016-05-12 10:26:38 -04:00
|
|
|
mHasStoppedSemaphore.set(0) ;
|
2015-05-25 15:11:42 +00:00
|
|
|
run() ;
|
|
|
|
}
|
|
|
|
|
2015-05-22 20:54:38 +00:00
|
|
|
void RsTickingThread::runloop()
|
2009-08-04 23:22:44 +00:00
|
|
|
{
|
2016-05-12 10:26:38 -04:00
|
|
|
mHasStoppedSemaphore.set(0) ; // first time we are 100% the thread is actually running.
|
|
|
|
|
2015-05-22 18:16:49 +00:00
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-12 10:26:38 -04:00
|
|
|
THREAD_DEBUG << "RsTickingThread::runloop(). Setting stopped=0" << std::endl;
|
2015-05-22 18:16:49 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
while(1)
|
|
|
|
{
|
2015-05-25 15:11:42 +00:00
|
|
|
if(shouldStop())
|
2015-05-22 18:16:49 +00:00
|
|
|
{
|
|
|
|
#ifdef DEBUG_THREADS
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << "pqithreadstreamer::runloop(): asked to stop. setting hasStopped=1, and returning. Thread ends." << std::endl;
|
2015-05-22 18:16:49 +00:00
|
|
|
#endif
|
2016-05-11 12:42:59 -04:00
|
|
|
mHasStoppedSemaphore.set(1);
|
2015-05-22 18:16:49 +00:00
|
|
|
return ;
|
|
|
|
}
|
|
|
|
|
|
|
|
data_tick();
|
|
|
|
}
|
2011-03-03 23:30:08 +00:00
|
|
|
}
|
|
|
|
|
2008-07-23 22:01:59 +00:00
|
|
|
RsQueueThread::RsQueueThread(uint32_t min, uint32_t max, double relaxFactor )
|
2015-05-22 18:16:49 +00:00
|
|
|
:mMinSleep(min), mMaxSleep(max), mRelaxFactor(relaxFactor)
|
2008-07-23 22:01:59 +00:00
|
|
|
{
|
2015-05-22 18:16:49 +00:00
|
|
|
mLastSleep = (uint32_t)mMinSleep ;
|
|
|
|
mLastWork = time(NULL) ;
|
2008-07-23 22:01:59 +00:00
|
|
|
}
|
|
|
|
|
2015-05-22 18:16:49 +00:00
|
|
|
void RsQueueThread::data_tick()
|
2008-07-23 22:01:59 +00:00
|
|
|
{
|
2015-05-22 18:16:49 +00:00
|
|
|
bool doneWork = false;
|
|
|
|
while(workQueued() && doWork())
|
|
|
|
{
|
|
|
|
doneWork = true;
|
|
|
|
}
|
|
|
|
time_t now = time(NULL);
|
|
|
|
if (doneWork)
|
|
|
|
{
|
|
|
|
mLastWork = now;
|
|
|
|
mLastSleep = (uint32_t) (mMinSleep + (mLastSleep - mMinSleep) / 2.0);
|
2016-05-12 10:26:38 -04:00
|
|
|
#ifdef DEBUG_TICKING
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << "RsQueueThread::data_tick() done work: sleeping for: " << mLastSleep << " ms" << std::endl;
|
2008-11-02 11:38:11 +00:00
|
|
|
#endif
|
|
|
|
|
2015-05-22 18:16:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
uint32_t deltaT = now - mLastWork;
|
|
|
|
double frac = deltaT / mRelaxFactor;
|
|
|
|
|
|
|
|
mLastSleep += (uint32_t)
|
|
|
|
((mMaxSleep-mMinSleep) * (frac + 0.05));
|
|
|
|
if (mLastSleep > mMaxSleep)
|
|
|
|
{
|
|
|
|
mLastSleep = mMaxSleep;
|
|
|
|
}
|
2016-05-12 10:26:38 -04:00
|
|
|
#ifdef DEBUG_TICKING
|
2016-05-04 22:23:36 -04:00
|
|
|
THREAD_DEBUG << "RsQueueThread::data_tick() no work: sleeping for: " << mLastSleep << " ms" << std::endl;
|
2008-11-02 11:38:11 +00:00
|
|
|
#endif
|
2015-05-22 18:16:49 +00:00
|
|
|
}
|
|
|
|
usleep(mLastSleep * 1000); // mLastSleep msec
|
2008-07-23 22:01:59 +00:00
|
|
|
}
|
2011-07-04 22:59:39 +00:00
|
|
|
|
2014-10-28 20:37:48 +00:00
|
|
|
void RsMutex::unlock()
|
|
|
|
{
|
|
|
|
#ifdef RSTHREAD_SELF_LOCKING_GUARD
|
|
|
|
if(--_cnt == 0)
|
|
|
|
{
|
|
|
|
#endif
|
|
|
|
_thread_id = 0 ;
|
|
|
|
pthread_mutex_unlock(&realMutex);
|
|
|
|
|
|
|
|
#ifdef RSTHREAD_SELF_LOCKING_GUARD
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-07-04 22:59:39 +00:00
|
|
|
void RsMutex::lock()
|
|
|
|
{
|
|
|
|
#ifdef RSMUTEX_DEBUG
|
2014-11-05 20:07:15 +00:00
|
|
|
pthread_t owner = _thread_id ;
|
2011-07-04 22:59:39 +00:00
|
|
|
#endif
|
|
|
|
|
2013-08-31 08:16:09 +00:00
|
|
|
int retval = 0;
|
2011-07-04 22:59:39 +00:00
|
|
|
#ifdef RSTHREAD_SELF_LOCKING_GUARD
|
|
|
|
if(!trylock())
|
|
|
|
if(!pthread_equal(_thread_id,pthread_self()))
|
|
|
|
#endif
|
2013-08-31 08:16:09 +00:00
|
|
|
retval = pthread_mutex_lock(&realMutex);
|
|
|
|
|
|
|
|
switch(retval)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EINVAL:
|
|
|
|
std::cerr << "RsMutex::lock() pthread_mutex_lock returned EINVAL";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EBUSY:
|
|
|
|
std::cerr << "RsMutex::lock() pthread_mutex_lock returned EBUSY";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EAGAIN:
|
|
|
|
std::cerr << "RsMutex::lock() pthread_mutex_lock returned EAGAIN";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EDEADLK:
|
|
|
|
std::cerr << "RsMutex::lock() pthread_mutex_lock returned EDEADLK";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EPERM:
|
|
|
|
std::cerr << "RsMutex::lock() pthread_mutex_lock returned EPERM";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
std::cerr << "RsMutex::lock() pthread_mutex_lock returned UNKNOWN ERROR";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Here is some debugging code - to catch failed locking attempts.
|
|
|
|
* Major bug is it is ever triggered.
|
|
|
|
*/
|
|
|
|
#ifdef RSMUTEX_ABORT
|
|
|
|
|
|
|
|
if (retval != 0)
|
|
|
|
{
|
|
|
|
#ifdef RSMUTEX_DEBUG
|
|
|
|
std::cerr << "RsMutex::lock() name: " << name << std::endl;
|
|
|
|
#endif
|
|
|
|
std::cerr << "RsMutex::lock() pthread_mutex_lock returned an Error. Aborting()";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
#endif
|
2011-07-04 22:59:39 +00:00
|
|
|
|
2013-08-31 08:16:09 +00:00
|
|
|
|
2011-07-04 22:59:39 +00:00
|
|
|
_thread_id = pthread_self() ;
|
|
|
|
#ifdef RSTHREAD_SELF_LOCKING_GUARD
|
|
|
|
++_cnt ;
|
|
|
|
#endif
|
2014-10-28 20:37:48 +00:00
|
|
|
}
|
2011-07-04 22:59:39 +00:00
|
|
|
#ifdef RSMUTEX_DEBUG
|
2014-11-11 19:47:05 +00:00
|
|
|
double RsStackMutex::getCurrentTS()
|
2014-10-28 20:37:48 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
#ifndef WINDOWS_SYS
|
|
|
|
struct timeval cts_tmp;
|
|
|
|
gettimeofday(&cts_tmp, NULL);
|
|
|
|
double cts = (cts_tmp.tv_sec) + ((double) cts_tmp.tv_usec) / 1000000.0;
|
|
|
|
#else
|
|
|
|
struct _timeb timebuf;
|
|
|
|
_ftime( &timebuf);
|
|
|
|
double cts = (timebuf.time) + ((double) timebuf.millitm) / 1000.0;
|
2011-07-04 22:59:39 +00:00
|
|
|
#endif
|
2014-10-28 20:37:48 +00:00
|
|
|
return cts;
|
2011-07-04 22:59:39 +00:00
|
|
|
}
|
2014-10-28 20:37:48 +00:00
|
|
|
#endif
|
|
|
|
|
2014-11-11 19:47:05 +00:00
|
|
|
|