blob: f2a933b79a18a8ce7e99697ce745ac1a126c6593 [file] [log] [blame]
#pragma once
#include "logging.h"
#include <boost/circular_buffer.hpp>
#include <boost/circular_buffer/space_optimized.hpp>
#include <chrono>
#include <functional>
namespace crow
{
namespace detail
{
constexpr const size_t timerQueueTimeoutSeconds = 5;
constexpr const size_t maxSize = 100;
// fast timer queue for fixed tick value.
class TimerQueue
{
public:
TimerQueue()
{
dq.set_capacity(maxSize);
}
void cancel(size_t k)
{
size_t index = k - step;
if (index < dq.size())
{
dq[index].second = nullptr;
}
}
std::optional<size_t> add(std::function<void()> f)
{
if (dq.size() == maxSize)
{
return std::nullopt;
}
dq.push_back(
std::make_pair(std::chrono::steady_clock::now(), std::move(f)));
size_t ret = step + dq.size() - 1;
BMCWEB_LOG_DEBUG << "timer add inside: " << this << ' ' << ret;
return ret;
}
void process()
{
auto now = std::chrono::steady_clock::now();
while (!dq.empty())
{
auto& x = dq.front();
// Check expiration time only for active handlers,
// remove canceled ones immediately
if (x.second)
{
if (now - x.first <
std::chrono::seconds(timerQueueTimeoutSeconds))
{
break;
}
BMCWEB_LOG_DEBUG << "timer call: " << this << ' ' << step;
// we know that timer handlers are very simple currenty; call
// here
x.second();
}
dq.pop_front();
step++;
}
}
private:
using storage_type =
std::pair<std::chrono::time_point<std::chrono::steady_clock>,
std::function<void()>>;
boost::circular_buffer_space_optimized<storage_type,
std::allocator<storage_type>>
dq{};
// boost::circular_buffer<storage_type> dq{20};
// std::deque<storage_type> dq{};
size_t step{};
};
} // namespace detail
} // namespace crow