mirror of
https://github.com/Xahau/xahaud.git
synced 2026-04-29 15:37:46 +00:00
201 lines
4.8 KiB
C++
201 lines
4.8 KiB
C++
//
|
|
// Copyright (c) 2015-2016 Vinnie Falco (vinnie dot falco at gmail dot com)
|
|
//
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
//
|
|
|
|
#ifndef BASIC_SECONDS_CLOCK_HPP
|
|
#define BASIC_SECONDS_CLOCK_HPP
|
|
|
|
#include "chrono_util.hpp"
|
|
|
|
#include <algorithm>
|
|
#include <chrono>
|
|
#include <condition_variable>
|
|
#include <mutex>
|
|
#include <thread>
|
|
#include <vector>
|
|
|
|
namespace detail {
|
|
|
|
class seconds_clock_worker
|
|
{
|
|
public:
|
|
virtual void sample() = 0;
|
|
};
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Updates the clocks
|
|
class seconds_clock_thread
|
|
{
|
|
public:
|
|
using mutex = std::mutex;
|
|
using cond_var = std::condition_variable;
|
|
using lock_guard = std::lock_guard <mutex>;
|
|
using unique_lock = std::unique_lock <mutex>;
|
|
using clock_type = std::chrono::steady_clock;
|
|
using seconds = std::chrono::seconds;
|
|
using thread = std::thread;
|
|
using workers = std::vector <seconds_clock_worker*>;
|
|
|
|
bool stop_;
|
|
mutex m_;
|
|
cond_var cond_;
|
|
workers workers_;
|
|
thread thread_;
|
|
|
|
seconds_clock_thread()
|
|
: stop_(false)
|
|
{
|
|
thread_ = thread{
|
|
&seconds_clock_thread::run, this};
|
|
}
|
|
|
|
~seconds_clock_thread()
|
|
{
|
|
stop();
|
|
}
|
|
|
|
void add(seconds_clock_worker& w)
|
|
{
|
|
lock_guard lock{m_};
|
|
workers_.push_back(&w);
|
|
}
|
|
|
|
void remove(seconds_clock_worker& w)
|
|
{
|
|
lock_guard lock{m_};
|
|
workers_.erase(std::find(
|
|
workers_.begin(), workers_.end(), &w));
|
|
}
|
|
|
|
void stop()
|
|
{
|
|
if(thread_.joinable())
|
|
{
|
|
{
|
|
lock_guard lock{m_};
|
|
stop_ = true;
|
|
}
|
|
cond_.notify_all();
|
|
thread_.join();
|
|
}
|
|
}
|
|
|
|
void run()
|
|
{
|
|
unique_lock lock{m_};
|
|
for(;;)
|
|
{
|
|
for(auto iter : workers_)
|
|
iter->sample();
|
|
|
|
using namespace std::chrono;
|
|
clock_type::time_point const when(
|
|
floor <seconds>(
|
|
clock_type::now().time_since_epoch()) +
|
|
milliseconds(900));
|
|
|
|
if(cond_.wait_until(lock, when, [this]{ return stop_; }))
|
|
return;
|
|
}
|
|
}
|
|
|
|
static seconds_clock_thread& instance()
|
|
{
|
|
static seconds_clock_thread singleton;
|
|
return singleton;
|
|
}
|
|
};
|
|
|
|
} // detail
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
/** Called before main exits to terminate the utility thread.
|
|
This is a workaround for Visual Studio 2013:
|
|
http://connect.microsoft.com/VisualStudio/feedback/details/786016/creating-a-global-c-object-that-used-thread-join-in-its-destructor-causes-a-lockup
|
|
http://stackoverflow.com/questions/10915233/stdthreadjoin-hangs-if-called-after-main-exits-when-using-vs2012-rc
|
|
*/
|
|
inline
|
|
void
|
|
basic_seconds_clock_main_hook()
|
|
{
|
|
#ifdef _MSC_VER
|
|
detail::seconds_clock_thread::instance().stop();
|
|
#endif
|
|
}
|
|
|
|
/** A clock whose minimum resolution is one second.
|
|
|
|
The purpose of this class is to optimize the performance of the now()
|
|
member function call. It uses a dedicated thread that wakes up at least
|
|
once per second to sample the requested trivial clock.
|
|
|
|
@tparam Clock A type meeting these requirements:
|
|
http://en.cppreference.com/w/cpp/concept/Clock
|
|
*/
|
|
template<class Clock>
|
|
class basic_seconds_clock
|
|
{
|
|
public:
|
|
using rep = typename Clock::rep;
|
|
using period = typename Clock::period;
|
|
using duration = typename Clock::duration;
|
|
using time_point = typename Clock::time_point;
|
|
|
|
static bool const is_steady = Clock::is_steady;
|
|
|
|
static time_point now()
|
|
{
|
|
// Make sure the thread is constructed before the
|
|
// worker otherwise we will crash during destruction
|
|
// of objects with static storage duration.
|
|
struct initializer
|
|
{
|
|
initializer()
|
|
{
|
|
detail::seconds_clock_thread::instance();
|
|
}
|
|
};
|
|
static initializer init;
|
|
|
|
struct worker : detail::seconds_clock_worker
|
|
{
|
|
time_point m_now;
|
|
std::mutex m_;
|
|
|
|
worker()
|
|
: m_now(Clock::now())
|
|
{
|
|
detail::seconds_clock_thread::instance().add(*this);
|
|
}
|
|
|
|
~worker()
|
|
{
|
|
detail::seconds_clock_thread::instance().remove(*this);
|
|
}
|
|
|
|
time_point now()
|
|
{
|
|
std::lock_guard<std::mutex> lock{m_};
|
|
return m_now;
|
|
}
|
|
|
|
void sample()
|
|
{
|
|
std::lock_guard<std::mutex> lock{m_};
|
|
m_now = Clock::now();
|
|
}
|
|
};
|
|
|
|
static worker w;
|
|
|
|
return w.now();
|
|
}
|
|
};
|
|
|
|
#endif
|