mirror of
https://github.com/RetroShare/RetroShare.git
synced 2025-01-03 20:01:07 -05:00
31fd6f6261
- Fixed compile with MinGW-w64 under Windows git-svn-id: http://svn.code.sf.net/p/retroshare/code/trunk@6866 b45a01b8-16f6-495d-af2f-9b41ad6348cc
267 lines
5.5 KiB
C++
267 lines
5.5 KiB
C++
|
|
/*
|
|
* "$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"
|
|
#include <unistd.h> /* for usleep() */
|
|
#include <errno.h> /* for usleep() */
|
|
#include <iostream>
|
|
#include <time.h>
|
|
|
|
#ifdef RSMUTEX_DEBUG
|
|
#include <stdio.h>
|
|
#endif
|
|
|
|
#if defined(WINDOWS_SYS) && defined(__MINGW64_VERSION_MAJOR)
|
|
#include <windows.h> // for Sleep
|
|
#endif
|
|
|
|
/*******
|
|
* #define DEBUG_THREADS 1
|
|
* #define RSMUTEX_ABORT 1 // Catch wrong pthreads mode.
|
|
*******/
|
|
|
|
#ifdef RSMUTEX_ABORT
|
|
#include <stdlib.h>
|
|
#endif
|
|
|
|
#ifdef DEBUG_THREADS
|
|
#include <iostream>
|
|
#endif
|
|
|
|
extern "C" void* rsthread_init(void* p)
|
|
{
|
|
RsThread *thread = (RsThread *) p;
|
|
if (!thread)
|
|
{
|
|
return 0;
|
|
}
|
|
thread -> run();
|
|
return 0;
|
|
}
|
|
|
|
|
|
pthread_t createThread(RsThread &thread)
|
|
{
|
|
pthread_t tid;
|
|
void *data = (void *) (&thread);
|
|
|
|
thread.mMutex.lock();
|
|
{
|
|
|
|
#if 0
|
|
int ret;
|
|
ret = pthread_attr_init(&tattr);
|
|
if (doDetached)
|
|
{
|
|
ret = pthread_attr_setdetachstate(&tattr,PTHREAD_CREATE_DETACHED);
|
|
}
|
|
else
|
|
{
|
|
ret = pthread_attr_setdetachstate(&tattr,PTHREAD_CREATE_JOINABLE);
|
|
}
|
|
|
|
pthread_create(&tid, &tattr, &rsthread_init, data);
|
|
#endif
|
|
|
|
int err ;
|
|
|
|
if( 0 == (err=pthread_create(&tid, 0, &rsthread_init, data)))
|
|
thread.mTid = tid;
|
|
else
|
|
std::cerr << "Fatal error: pthread_create could not create a thread. Error returned: " << err << " !!!!!!!" << std::endl;
|
|
}
|
|
thread.mMutex.unlock();
|
|
|
|
return tid;
|
|
|
|
}
|
|
|
|
RsThread::RsThread () : mMutex("RsThread")
|
|
{
|
|
mIsRunning = true;
|
|
|
|
#ifdef WINDOWS_SYS
|
|
memset (&mTid, 0, sizeof(mTid));
|
|
#else
|
|
mTid = 0;
|
|
#endif
|
|
}
|
|
|
|
void RsThread::join() /* waits for the the mTid thread to stop */
|
|
{
|
|
// do we need a mutex for this ?
|
|
mIsRunning = false;
|
|
|
|
void *ptr;
|
|
pthread_join(mTid, &ptr);
|
|
}
|
|
|
|
void RsThread::stop()
|
|
{
|
|
pthread_exit(NULL);
|
|
}
|
|
|
|
bool RsThread::isRunning()
|
|
{
|
|
// do we need a mutex for this ?
|
|
return mIsRunning;
|
|
}
|
|
|
|
RsQueueThread::RsQueueThread(uint32_t min, uint32_t max, double relaxFactor )
|
|
:mMinSleep(min), mMaxSleep(max), mRelaxFactor(relaxFactor)
|
|
{
|
|
mLastSleep = (uint32_t)mMinSleep ;
|
|
mLastWork = time(NULL) ;
|
|
}
|
|
|
|
void RsQueueThread::run()
|
|
{
|
|
while(isRunning())
|
|
{
|
|
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);
|
|
#ifdef DEBUG_THREADS
|
|
std::cerr << "RsQueueThread::run() done work: sleeping for: " << mLastSleep;
|
|
std::cerr << " ms";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
|
|
}
|
|
else
|
|
{
|
|
uint32_t deltaT = now - mLastWork;
|
|
double frac = deltaT / mRelaxFactor;
|
|
|
|
mLastSleep += (uint32_t)
|
|
((mMaxSleep-mMinSleep) * (frac + 0.05));
|
|
if (mLastSleep > mMaxSleep)
|
|
{
|
|
mLastSleep = mMaxSleep;
|
|
}
|
|
#ifdef DEBUG_THREADS
|
|
std::cerr << "RsQueueThread::run() no work: sleeping for: " << mLastSleep;
|
|
std::cerr << " ms";
|
|
std::cerr << std::endl;
|
|
#endif
|
|
}
|
|
#ifdef WIN32
|
|
Sleep(mLastSleep);
|
|
#else
|
|
usleep(1000 * mLastSleep);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void RsMutex::lock()
|
|
{
|
|
#ifdef RSMUTEX_DEBUG
|
|
clock_t t1 = clock();
|
|
#endif
|
|
|
|
int retval = 0;
|
|
#ifdef RSTHREAD_SELF_LOCKING_GUARD
|
|
if(!trylock())
|
|
if(!pthread_equal(_thread_id,pthread_self()))
|
|
#endif
|
|
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
|
|
|
|
|
|
_thread_id = pthread_self() ;
|
|
#ifdef RSTHREAD_SELF_LOCKING_GUARD
|
|
++_cnt ;
|
|
#endif
|
|
|
|
#ifdef RSMUTEX_DEBUG
|
|
clock_t t2 = clock();
|
|
double duration = (double) (t2 - t1) / CLOCKS_PER_SEC;
|
|
if (duration * 1000 > RSMUTEX_DEBUG) {
|
|
fprintf(stderr, "RsMutex::lock() %s --> %.3fs\n", name.c_str(), duration);
|
|
fflush(stderr);
|
|
}
|
|
#endif
|
|
}
|