blob: ea281767188e6ce002170a1da5400ff93587a0cc [file] [log] [blame]
#pragma once
#include <chrono>
#include <functional>
#include <memory>
#include <systemd/sd-event.h>
namespace phosphor
{
namespace fan
{
namespace util
{
struct EventSourceDeleter
{
void operator()(sd_event_source* eventSource) const
{
sd_event_source_unref(eventSource);
}
};
using EventSourcePtr = std::unique_ptr<sd_event_source, EventSourceDeleter>;
using EventPtr = std::shared_ptr<sd_event>;
/**
* @class Timer
*
* This class implements a simple timer that runs an arbitrary
* function on expiration. The timeout value is set in microseconds.
* It can be stopped while it is running, and queried to see if it is
* running.
*
* If started with the 'repeating' argument, it will keep calling the
* callback function every <timeout> microseconds. If started with the
* 'oneshot' argument, it will just call the callback function one time.
*
* It needs an sd_event loop to function.
*/
class Timer
{
public:
enum class TimerType
{
oneshot,
repeating
};
Timer() = delete;
Timer(const Timer&) = delete;
Timer& operator=(const Timer&) = delete;
Timer(Timer&&) = default;
Timer& operator=(Timer&&) = default;
/**
* @brief Constructs timer object
*
* @param[in] events - sd_event pointer, previously created
* @param[in] callbackFunc - The function to call on timer expiration
*/
Timer(EventPtr& events,
std::function<void()> callbackFunc);
/**
* @brief Destructor
*/
~Timer();
/**
* @brief Starts the timer
*
* The input is an offset from the current steady clock.
*
* @param[in] usec - the timeout value in microseconds
* @param[in] type - either a oneshot, or repeating
*/
void start(std::chrono::microseconds usec,
TimerType type = TimerType::oneshot);
/**
* @brief Stop the timer
*/
void stop();
/**
* @brief Returns true if the timer is running
*/
bool running();
/**
* @brief Returns the timeout value
*
* @return - the last value sent in via start().
*
* Could be used to restart the timer with the same
* timeout. i.e. start(getTimeout());
*/
inline auto getTimeout() const
{
return timeout;
}
/**
* @brief Returns the timer type
*/
inline auto getType() const
{
return type;
}
private:
/**
* @brief Callback function when timer goes off
*
* Calls the callback function passed in by the user.
*
* @param[in] eventSource - Source of the event
* @param[in] usec - time in micro seconds
* @param[in] userData - User data pointer
*/
static int timeoutHandler(sd_event_source* eventSource,
uint64_t usec, void* userData);
/**
* @brief Gets the current time from the steady clock
*/
std::chrono::microseconds getTime();
/**
* @brief Wrapper around sd_event_source_set_enabled
*
* @param[in] action - either SD_EVENT_OFF, SD_EVENT_ON,
* or SD_EVENT_ONESHOT
*/
void setTimer(int action);
/**
* @brief Sets the expiration time for the timer
*
* Sets it to timeout microseconds in the future
*/
void setTimeout();
/**
* @brief The sd_event structure
*/
EventPtr timeEvent;
/**
* @brief Source of events
*/
EventSourcePtr eventSource;
/**
* @brief Either 'repeating' or 'oneshot'
*/
TimerType type = TimerType::oneshot;
/**
* @brief The function to call when the timer expires
*/
std::function<void()> callback;
/**
* @brief What the timer was set to run for
*
* Not cleared on timer expiration
*/
std::chrono::microseconds timeout;
};
}
}
}