// Matrix Construct
//
// Copyright (C) Matrix Construct Developers, Authors & Contributors
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice is present in all copies. The
// full license for this software is available in the LICENSE file.

#pragma once
#define HAVE_IRCD_UTIL_TIMER_H

namespace ircd::util
{
	struct timer;
}

struct ircd::util::timer
{
	IRCD_OVERLOAD(nostart)
	using clock = std::chrono::steady_clock;

	nanoseconds accumulator;
	clock::time_point start;

	bool stopped() const;
	template<class duration = std::chrono::seconds> duration get() const;
	template<class duration = std::chrono::seconds> duration at() const;
	void cont();
	void stop();

	timer(const std::function<void ()> &);
	timer(nostart_t);
	timer();
};


/// Default construction will start the timer.
inline
ircd::util::timer::timer()
:accumulator{0ns}
,start{clock::now()}
{
}

/// timer(timer::nostart)
inline
ircd::util::timer::timer(nostart_t)
:accumulator{0ns}
,start{clock::time_point::min()}
{
}

inline
ircd::util::timer::timer(const std::function<void ()> &func)
:timer{}
{
	func();
	stop();
}

inline void
ircd::util::timer::stop()
{
	if(stopped())
		return;

	const auto now(clock::now());
	accumulator += std::chrono::duration_cast<decltype(accumulator)>(now - start);
	start = clock::time_point::min();
}

inline void
ircd::util::timer::cont()
{
	if(!stopped())
	{
		const auto now(clock::now());
		accumulator += std::chrono::duration_cast<decltype(accumulator)>(now - start);
	}

	start = clock::now();
}

template<class duration>
duration
ircd::util::timer::at()
const
{
	const auto now(clock::now());
	return duration_cast<duration>(now - start);
}

template<class duration>
duration
ircd::util::timer::get()
const
{
	return std::chrono::duration_cast<duration>(accumulator);
}

inline bool
ircd::util::timer::stopped()
const
{
	return start == clock::time_point::min();
}