X-Git-Url: http://git.tdb.fi/?p=libs%2Fcore.git;a=blobdiff_plain;f=source%2Ftime%2Ftimer.cpp;h=e0af57cc78fcc41363632e169e33a8219653e9f1;hp=9fc82f57e872bf0b0c1f1dcc3ee8f974003eafd5;hb=be8ea216d23bf36bdfb2d3e302638782575fc136;hpb=3d1b0b44b2d75ed7d97b3588eefe61a9b511365c diff --git a/source/time/timer.cpp b/source/time/timer.cpp index 9fc82f5..e0af57c 100644 --- a/source/time/timer.cpp +++ b/source/time/timer.cpp @@ -1,9 +1,5 @@ -/* -This file is part of libmspcore -Copyright © 2006 Mikko Rasa, Mikkosoft Productions -Distributed under the LGPL -*/ - +#include +#include #include "timer.h" #include "utils.h" @@ -13,66 +9,138 @@ namespace Msp { namespace Time { Timer::Timer(): - slots(slot_compare) + sem(1), + blocking(false) { } Timer::~Timer() { - while(!slots.empty()) - { - delete slots.top(); - slots.pop(); - } + for(vector::iterator i=slots.begin(); i!=slots.end(); ++i) + delete i->slot; } Timer::Slot &Timer::add(const TimeDelta &td) { - Slot *s=new Slot(td); - mutex.lock(); - slots.push(s); - mutex.unlock(); - sem.signal(); + Slot *s = new Slot(td); + MutexLock l(mutex); + slots.push_back(s); + push_heap(slots.begin(), slots.end()); + if(blocking) + sem.signal(); return *s; } Timer::Slot &Timer::add(const TimeStamp &ts) { - Slot *s=new Slot(ts); - mutex.lock(); - slots.push(s); - mutex.unlock(); - sem.signal(); + Slot *s = new Slot(ts); + MutexLock l(mutex); + slots.push_back(s); + push_heap(slots.begin(), slots.end()); + if(blocking) + sem.signal(); return *s; } +void Timer::cancel(Slot &slot) +{ + MutexLock l(mutex); + for(vector::iterator i=slots.begin(); i!=slots.end(); ++i) + if(i->slot==&slot) + { + delete i->slot; + slots.erase(i); + make_heap(slots.begin(), slots.end()); + return; + } +} + void Timer::tick(bool block) { - if(slots.empty()) + if(block) + tick(); + else + tick(zero); +} + +void Timer::tick() +{ + do_tick(-sec); +} + +void Timer::tick(const TimeDelta &timeout) +{ + if(timeout=zero) + deadline = now()+timeout; + + Slot *next = 0; { - if(block) - sem.wait(); - return; + MutexLock l(mutex); + while(1) + { + TimeStamp stamp; + TimeStamp t = now(); + if(!slots.empty()) + { + next = slots.begin()->slot; + stamp = next->get_timeout(); + if(stamp<=t) + break; + } + + if(timeout && (!deadline || tget_timeout(); - const TimeStamp t=now(); - if(stamp<=t || (block && sem.wait(stamp-t)==1)) + try { - slots.pop(); if(next->signal_timeout.emit() && next->increment()) - slots.push(next); + { + MutexLock l(mutex); + slots.push_back(next); + push_heap(slots.begin(), slots.end()); + } else delete next; } + catch(...) + { + delete next; + throw; + } } -bool Timer::slot_compare(Slot *a, Slot *b) +TimeStamp Timer::get_next_timeout() const { - return *a<*b; + if(slots.empty()) + return TimeStamp(); + return slots.begin()->slot->get_timeout(); } @@ -89,13 +157,18 @@ bool Timer::Slot::increment() { if(!interval) return false; - timeout+=interval; + timeout += interval; return true; } -bool Timer::Slot::operator<(const Slot &other) const + +Timer::SlotProxy::SlotProxy(Slot *s): + slot(s) +{ } + +bool Timer::SlotProxy::operator<(const SlotProxy &sp) const { - return timeoutget_timeout()>sp.slot->get_timeout(); } } // namespace Time