/* * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. * */ #ifndef sys_Timer #define sys_Timer #include "qpid/sys/TimerWarnings.h" #include "qpid/sys/Monitor.h" #include "qpid/sys/Mutex.h" #include "qpid/sys/Thread.h" #include "qpid/sys/Runnable.h" #include "qpid/RefCounted.h" #include "qpid/CommonImportExport.h" #include #include #include namespace qpid { namespace sys { class Timer; class TimerTask : public RefCounted { friend class Timer; friend class TimerTaskCallbackScope; friend bool operator<(const boost::intrusive_ptr&, const boost::intrusive_ptr&); std::string name; AbsTime sortTime; Duration period; AbsTime nextFireTime; qpid::sys::Monitor stateMonitor; enum {WAITING, CALLING, CANCELLED} state; bool prepareToFire(); void finishFiring(); bool readyToFire() const; void fireTask(); public: /** Create a periodic TimerTask * * This TimerTask type will trigger after the specified duration * and can also be retriggered again after the same duration * by using setupNextFire() after it has been fired. * * Before it has been triggered you can use restart() to push off * triggering the TimerTask by the specified duration. */ QPID_COMMON_EXTERN TimerTask(Duration period, const std::string& name); /** Create a TimerTask that fires at a given absolute time * * This is a once only Timer and cannot be restarted after it has fired. */ QPID_COMMON_EXTERN TimerTask(AbsTime fireTime, const std::string& name); QPID_COMMON_EXTERN virtual ~TimerTask(); /** Adjust a periodic TimerTask for next firing time * * Called after a TimerTask has been triggered - probably in the fire() * callback function itself. This will set up a TimerTask for the next * triggering. * * Note that the TimerTask will need to be added again to the Timer. */ QPID_COMMON_EXTERN void setupNextFire(); /** Restart a TimerTask so to delay it being firing * * This can be called either with the TimerTask already added to a Timer * or after the task has been triggered. It has the effect of delaying * the task triggering by the initially specified duration. */ QPID_COMMON_EXTERN void restart(); /** Cancel a TimerTask so that it is no longer triggered * * After cancelling the only thing you can do nothing further * with a TimerTask. * * The Timer will delete the cancelled TimerTask. */ QPID_COMMON_EXTERN void cancel(); std::string getName() const { return name; } protected: // Must be overridden with callback virtual void fire() = 0; }; // For the priority_queue order bool operator<(const boost::intrusive_ptr& a, const boost::intrusive_ptr& b); class Timer : private Runnable { qpid::sys::Monitor monitor; std::priority_queue > tasks; qpid::sys::Thread runner; bool active; // Runnable interface void run(); public: QPID_COMMON_EXTERN Timer(); QPID_COMMON_EXTERN virtual ~Timer(); /** Add an TimerTask to the Timer queue * * Once a TimerTask has been triggered by (calling its fire() function) * the TimerTask is no longer on the Timer queue and needs to be added again. * * Note that TimerTasks must never be added more than once to a Timer * and must never be added simultaneuosly to multiple Timers. */ QPID_COMMON_EXTERN virtual void add(boost::intrusive_ptr task); /** Start the Timer * * This will start a new thread that runs the Timer and the fire callbacks. */ QPID_COMMON_EXTERN virtual void start(); /** Stop the Timer * * This will stop the Timer and its thread. */ QPID_COMMON_EXTERN virtual void stop(); protected: QPID_COMMON_EXTERN virtual void fire(boost::intrusive_ptr task); // Allow derived classes to change the late/overran thresholds. Duration late; Duration overran; Duration lateCancel; TimerWarnings warn; }; }} #endif