TimerThread corrections

master
Denes Matetelki 14 years ago
parent efab283199
commit 097db4c80d

@ -48,7 +48,7 @@ public:
const timespec expiration, const timespec expiration,
const timespec periodTime = timespec_ctor() ); const timespec periodTime = timespec_ctor() );
bool removeTimerUser ( void* timerUser ); bool removeTimerUser( TimerUser* timerUser );
// override to signal as well // override to signal as well
void stop(); void stop();
@ -56,9 +56,11 @@ public:
private: private:
void notifyAndRemove( const timespec t );
void* run( void ); void* run( void );
// facory function // ctor function
inline static timespec timespec_ctor() { inline static timespec timespec_ctor() {
timespec tmp = { 0, 0 }; timespec tmp = { 0, 0 };
return tmp; return tmp;

@ -4,12 +4,11 @@
#include "ScopedLock.hpp" #include "ScopedLock.hpp"
#include <errno.h> // ETIMEDOUT #include <errno.h> // ETIMEDOUT
#include <pthread.h> // pthread_attr_t, sched_param
TimerThread::TimerThread() TimerThread::TimerThread()
: Thread(true) : Thread()
, m_mutex() , m_mutex()
, m_condVar(m_mutex) , m_condVar(m_mutex)
, m_users() , m_users()
@ -29,19 +28,11 @@ void TimerThread::addTimerUser(TimerUser* user,
const time_t periodTime) const time_t periodTime)
{ {
TRACE; TRACE;
ScopedLock sl( m_mutex );
if ( not m_isRunning ) return;
timespec expirationTS = { expiration, 0 };
timespec periodTimeTS = { periodTime, 0 };
UserEntry userEntry = { periodTimeTS, user }; timespec periodTimeTS = { periodTime, 0 };
m_users.insert( std::pair<timespec, UserEntry>( expirationTS, userEntry ) ); timespec expirationTS = { expiration, 0 };
std::string s;
s = std::string("adding user: ") + stringify(user);
LOG(Logger::FINEST, s.c_str());
m_condVar.signal(); addTimerUser( user, expirationTS, periodTimeTS );
} }
@ -52,35 +43,30 @@ void TimerThread::addTimerUser( TimerUser* user,
TRACE; TRACE;
ScopedLock sl( m_mutex ); ScopedLock sl( m_mutex );
if ( not m_isRunning ) return; if ( not m_isRunning ) return;
UserEntry userEntry = { periodTime, user };
m_users.insert( std::pair<timespec, UserEntry>( expiration, userEntry ) );
std::string s; timespec ts;
s = std::string("adding user: ") + stringify(user); clock_gettime( CLOCK_REALTIME, &ts );
LOG(Logger::FINEST, s.c_str()); ts = timespecAdd ( ts, expiration );
UserEntry userEntry = { periodTime, user };
m_users.insert( std::pair<timespec, UserEntry>( ts, userEntry ) );
m_condVar.signal(); m_condVar.signal();
} }
bool TimerThread::removeTimerUser ( void* timerUser ) bool TimerThread::removeTimerUser( TimerUser* timerUser )
{ {
TRACE; TRACE;
ScopedLock sl( m_mutex ); ScopedLock sl( m_mutex );
if ( not m_isRunning ) return false; if ( not m_isRunning ) return false;
std::multimap<timespec, UserEntry>::iterator it, tmp; std::multimap<timespec, UserEntry>::iterator it, tmp;
bool found(false); bool found(false);
for ( it = m_users.begin(); it != m_users.end(); ) { for ( it = m_users.begin(); it != m_users.end(); ) {
std::string s;
s = stringify(it->second.user) + " = ? = " + stringify(timerUser);
LOG(Logger::FINEST, "Checking user to delete...does it match? ");
LOG(Logger::FINEST, s.c_str());
/// @todo solve the abstract pointer problem
if ( (it->second.user) == timerUser ) { if ( (it->second.user) == timerUser ) {
tmp = it++; tmp = it++;
LOG(Logger::FINEST, "Removing a user.");
m_users.erase(tmp); m_users.erase(tmp);
m_condVar.signal(); m_condVar.signal();
found = true; // one user can be registered multiple times found = true; // one user can be registered multiple times
@ -91,7 +77,7 @@ bool TimerThread::removeTimerUser ( void* timerUser )
return found; return found;
} }
// override to signal as well
void TimerThread::stop() void TimerThread::stop()
{ {
TRACE; TRACE;
@ -101,62 +87,78 @@ void TimerThread::stop()
} }
void* TimerThread::run( void ) void TimerThread::notifyAndRemove( const timespec t )
{ {
TRACE; TRACE;
timespec nextExpiration; ScopedLock sl( m_mutex );
if ( not m_isRunning ) return;
timespec ts;
std::multimap<timespec, UserEntry, timespec_cmp> tmp; std::multimap<timespec, UserEntry, timespec_cmp> tmp;
std::multimap<timespec, UserEntry>::iterator it; std::multimap<timespec, UserEntry>::iterator it;
std::pair<std::multimap<timespec, UserEntry>::iterator, std::pair<std::multimap<timespec, UserEntry>::iterator,
std::multimap<timespec, UserEntry>::iterator> ret; std::multimap<timespec, UserEntry>::iterator> ret;
ret = m_users.equal_range( t );
/// @todo modify key values in multimap, must be a better way
tmp.clear();
for ( it = ret.first; it != ret.second; it++ ) {
it->second.user->timerExpired();
if ( it->second.periodTime.tv_sec != 0 or it->second.periodTime.tv_nsec != 0) {
clock_gettime( CLOCK_REALTIME, &ts );
ts = timespecAdd( ts, it->second.periodTime );
tmp.insert(std::pair<timespec, UserEntry>( ts, it->second ) );
m_condVar.signal();
}
}
m_users.erase( t );
m_users.insert( tmp.begin(), tmp.end() );
}
void* TimerThread::run( void )
{
TRACE;
timespec nextExpiration;
timespec ts, wait;
while( m_isRunning ) { while( m_isRunning ) {
m_mutex.lock(); m_mutex.lock();
while ( m_users.empty() and m_isRunning ) { while ( m_users.empty() and m_isRunning ) {
LOG(Logger::FINEST, "Waiting for a user, since the map is empty.");
m_condVar.wait(); m_condVar.wait();
} }
m_mutex.unlock(); m_mutex.unlock();
if ( not m_isRunning) return 0; if ( not m_isRunning) return 0;
nextExpiration = m_users.begin()->first; nextExpiration = m_users.begin()->first;
clock_gettime( CLOCK_REALTIME, &ts );
m_mutex.lock(); wait = timespecSubstract ( nextExpiration, ts );
// timer deleted / added, get nextExpiration again if ( wait.tv_sec == 0 and wait.tv_nsec == 0 ) {
if ( m_condVar.wait( nextExpiration.tv_sec, notifyAndRemove ( nextExpiration );
nextExpiration.tv_nsec ) != ETIMEDOUT ) {
LOG(Logger::FINEST, "Abort sleep: user has been added/removed.");
m_mutex.unlock();
continue; continue;
} }
m_mutex.unlock();
// notify & remove
m_mutex.lock(); m_mutex.lock();
ret = m_users.equal_range( nextExpiration );
/// @todo modify key values in multimap, must be a better way if ( m_condVar.wait( wait.tv_sec,
tmp.clear(); wait.tv_nsec ) != ETIMEDOUT ) {
for ( it = ret.first; it != ret.second; it++ ) { m_mutex.unlock();
LOG(Logger::FINEST, "Notifying one user."); continue;
it->second.user->timerExpired();
if ( it->second.periodTime.tv_sec != 0 or it->second.periodTime.tv_nsec != 0) {
LOG(Logger::FINEST, "Periodic, re-adding.");
tmp.insert(std::pair<timespec, UserEntry>(
it->second.periodTime, it->second ) );
}
} }
m_users.erase( nextExpiration );
m_users.insert( tmp.begin(), tmp.end() );
m_mutex.unlock(); m_mutex.unlock();
notifyAndRemove( nextExpiration );
} }
// end ...
if ( not m_users.empty() ) { if ( not m_users.empty() ) {
std::multimap<timespec, UserEntry>::iterator it;
for ( it = m_users.begin(); it != m_users.end(); it++ ) { for ( it = m_users.begin(); it != m_users.end(); it++ ) {
LOG(Logger::FINEST, "Calling timerDestroyed on one user.");
it->second.user->timerDestroyed(); it->second.user->timerDestroyed();
} }
} }

Loading…
Cancel
Save