2018-02-03 18:22:01 -08:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
2019-07-15 09:47:34 -07:00
|
|
|
// Copyright (C) 2016-2019 Jason Volk <jason@zemos.net>
|
2018-02-03 18:22:01 -08:00
|
|
|
//
|
|
|
|
// 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.
|
2017-09-08 01:43:43 -07:00
|
|
|
|
|
|
|
#pragma once
|
2019-07-15 09:47:34 -07:00
|
|
|
#define HAVE_IRCD_TIMEDATE_H
|
2017-09-08 01:43:43 -07:00
|
|
|
|
|
|
|
namespace ircd
|
|
|
|
{
|
2018-11-14 18:14:08 -08:00
|
|
|
using high_resolution_point = time_point<high_resolution_clock>;
|
2017-09-19 21:39:53 -07:00
|
|
|
using steady_point = time_point<steady_clock>;
|
|
|
|
using system_point = time_point<system_clock>;
|
2018-11-14 18:14:08 -08:00
|
|
|
using microtime_t = std::pair<time_t, int32_t>;
|
2019-07-15 09:47:34 -07:00
|
|
|
IRCD_OVERLOAD(localtime)
|
2017-09-19 21:39:53 -07:00
|
|
|
|
2019-08-02 15:30:06 -07:00
|
|
|
// Standard time_point samples
|
|
|
|
template<class unit = seconds> unit now(); // monotonic/steady_clock
|
|
|
|
template<> steady_point now(); // monotonic/steady_clock
|
|
|
|
template<> system_point now(); // system_clock
|
2017-09-19 21:39:53 -07:00
|
|
|
|
2019-08-02 15:30:06 -07:00
|
|
|
// system_clock
|
2017-09-29 22:57:29 -07:00
|
|
|
template<class unit = seconds> time_t &time(time_t &ref);
|
|
|
|
template<class unit = seconds> time_t time(time_t *const &ptr);
|
2019-08-08 16:26:25 -07:00
|
|
|
template<class unit = seconds> time_t time();
|
2017-10-11 17:47:03 -07:00
|
|
|
|
2019-07-19 16:01:20 -07:00
|
|
|
// System microtime suite
|
2018-05-23 18:05:05 -07:00
|
|
|
microtime_t microtime();
|
|
|
|
|
2020-12-06 05:04:51 -08:00
|
|
|
// System formatted time suite (also see util/pretty.h)
|
2019-07-15 09:47:34 -07:00
|
|
|
extern const char *const rfc7231_fmt;
|
2017-10-11 17:47:03 -07:00
|
|
|
string_view timef(const mutable_buffer &out, const struct tm &tm, const char *const &fmt = rfc7231_fmt);
|
|
|
|
string_view timef(const mutable_buffer &out, const time_t &epoch, const char *const &fmt = rfc7231_fmt);
|
|
|
|
string_view timef(const mutable_buffer &out, const time_t &epoch, localtime_t, const char *const &fmt = rfc7231_fmt);
|
2018-04-13 19:23:48 -07:00
|
|
|
string_view timef(const mutable_buffer &out, const system_point &epoch, const char *const &fmt = rfc7231_fmt);
|
|
|
|
string_view timef(const mutable_buffer &out, const system_point &epoch, localtime_t, const char *const &fmt = rfc7231_fmt);
|
2017-10-11 17:47:03 -07:00
|
|
|
string_view timef(const mutable_buffer &out, localtime_t, const char *const &fmt = rfc7231_fmt);
|
|
|
|
string_view timef(const mutable_buffer &out, const char *const &fmt = rfc7231_fmt);
|
2018-05-23 18:05:05 -07:00
|
|
|
template<size_t max = 128, class... args> std::string timestr(args&&...);
|
2018-02-12 15:04:54 -08:00
|
|
|
|
2019-04-17 19:27:26 -07:00
|
|
|
// Other tools
|
2019-08-29 15:39:49 -07:00
|
|
|
auto tse(const steady_point &);
|
|
|
|
auto tse(const system_point &);
|
|
|
|
bool operator!(const steady_point &);
|
|
|
|
bool operator!(const system_point &);
|
2019-08-20 20:20:25 -07:00
|
|
|
string_view ago(const mutable_buffer &buf, const system_point &, const uint &fmt = 0);
|
2019-04-17 19:27:26 -07:00
|
|
|
string_view smalldate(const mutable_buffer &buf, const time_t <ime);
|
2019-08-03 19:57:51 -07:00
|
|
|
string_view microdate(const mutable_buffer &buf);
|
2019-08-08 16:26:25 -07:00
|
|
|
string_view microtime(const mutable_buffer &);
|
2019-04-17 19:27:26 -07:00
|
|
|
|
|
|
|
// Interface conveniences.
|
2018-05-23 18:05:05 -07:00
|
|
|
std::ostream &operator<<(std::ostream &, const microtime_t &);
|
2018-02-12 15:04:54 -08:00
|
|
|
std::ostream &operator<<(std::ostream &, const system_point &);
|
2018-05-23 17:58:11 -07:00
|
|
|
template<class rep, class period> std::ostream &operator<<(std::ostream &, const duration<rep, period> &);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class rep,
|
|
|
|
class period>
|
|
|
|
std::ostream &
|
|
|
|
ircd::operator<<(std::ostream &s,
|
|
|
|
const duration<rep, period> &duration)
|
|
|
|
{
|
|
|
|
s << duration.count();
|
|
|
|
return s;
|
2018-02-12 15:04:54 -08:00
|
|
|
}
|
|
|
|
|
2018-05-23 18:05:05 -07:00
|
|
|
/// timestr() is a passthru to timef() where you don't give the first argument
|
|
|
|
/// (the mutable_buffer). Instead of supplying a buffer an allocated
|
|
|
|
/// std::string is returned with the result. By default this string's buffer
|
|
|
|
/// is sufficiently large, but may be further tuned in the template parameter.
|
|
|
|
template<size_t max,
|
|
|
|
class... args>
|
2017-10-11 17:47:03 -07:00
|
|
|
std::string
|
|
|
|
ircd::timestr(args&&... a)
|
|
|
|
{
|
2018-01-29 08:37:20 -08:00
|
|
|
return string(max, [&](const mutable_buffer &buf)
|
2017-10-11 17:47:03 -07:00
|
|
|
{
|
2018-01-29 08:37:20 -08:00
|
|
|
return timef(buf, std::forward<args>(a)...);
|
|
|
|
});
|
2017-10-11 17:47:03 -07:00
|
|
|
}
|
|
|
|
|
2019-08-29 15:39:49 -07:00
|
|
|
//
|
|
|
|
// inline tools
|
|
|
|
//
|
|
|
|
|
|
|
|
extern inline auto
|
|
|
|
__attribute__((always_inline, gnu_inline, artificial, flatten))
|
|
|
|
ircd::tse(const system_point &sp)
|
|
|
|
{
|
|
|
|
return sp.time_since_epoch();
|
|
|
|
}
|
|
|
|
|
|
|
|
extern inline auto
|
|
|
|
__attribute__((always_inline, gnu_inline, artificial, flatten))
|
|
|
|
ircd::tse(const steady_point &sp)
|
|
|
|
{
|
|
|
|
return sp.time_since_epoch();
|
|
|
|
}
|
|
|
|
|
|
|
|
extern inline bool
|
|
|
|
__attribute__((always_inline, gnu_inline, artificial, flatten))
|
|
|
|
ircd::operator!(const system_point &sp)
|
|
|
|
{
|
|
|
|
return !tse(sp).count();
|
|
|
|
}
|
|
|
|
|
|
|
|
extern inline bool
|
|
|
|
__attribute__((always_inline, gnu_inline, artificial, flatten))
|
|
|
|
ircd::operator!(const steady_point &sp)
|
|
|
|
{
|
|
|
|
return !tse(sp).count();
|
|
|
|
}
|
|
|
|
|
2019-08-08 16:26:25 -07:00
|
|
|
//
|
|
|
|
// system_clock
|
|
|
|
//
|
|
|
|
|
2017-09-29 22:57:29 -07:00
|
|
|
template<class unit>
|
2019-08-08 16:26:25 -07:00
|
|
|
extern inline time_t
|
|
|
|
__attribute__((always_inline, gnu_inline, artificial, flatten))
|
|
|
|
ircd::time()
|
2017-09-29 22:57:29 -07:00
|
|
|
{
|
2019-08-08 16:26:25 -07:00
|
|
|
time_t ret;
|
2017-09-29 22:57:29 -07:00
|
|
|
return time<unit>(ret);
|
2017-09-08 01:43:43 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 21:39:53 -07:00
|
|
|
template<class unit>
|
2019-08-08 16:26:25 -07:00
|
|
|
extern inline time_t
|
|
|
|
__attribute__((always_inline, gnu_inline, artificial, flatten))
|
|
|
|
ircd::time(time_t *const &ptr)
|
2017-09-08 01:43:43 -07:00
|
|
|
{
|
2019-08-08 16:26:25 -07:00
|
|
|
time_t buf, &ret{ptr? *ptr : buf};
|
2017-09-29 22:57:29 -07:00
|
|
|
return time<unit>(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class unit>
|
2019-08-08 16:26:25 -07:00
|
|
|
extern inline time_t &
|
|
|
|
__attribute__((always_inline, gnu_inline, artificial, flatten))
|
2017-09-29 22:57:29 -07:00
|
|
|
ircd::time(time_t &ref)
|
|
|
|
{
|
2019-08-29 15:39:49 -07:00
|
|
|
const auto &now
|
2019-08-08 16:26:25 -07:00
|
|
|
{
|
2019-08-29 15:39:49 -07:00
|
|
|
tse(ircd::now<system_point>())
|
2019-08-08 16:26:25 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
ref = duration_cast<unit>(now).count();
|
2017-09-29 22:57:29 -07:00
|
|
|
return ref;
|
2017-09-19 21:39:53 -07:00
|
|
|
}
|
|
|
|
|
2019-09-20 23:51:46 -07:00
|
|
|
template<>
|
|
|
|
inline ircd::system_point
|
|
|
|
__attribute__((always_inline, artificial, flatten))
|
|
|
|
ircd::now<ircd::system_point>()
|
|
|
|
{
|
|
|
|
return system_clock::now();
|
|
|
|
}
|
|
|
|
|
2019-08-08 16:26:25 -07:00
|
|
|
//
|
|
|
|
// steady_clock
|
|
|
|
//
|
|
|
|
|
2017-09-19 21:39:53 -07:00
|
|
|
template<class unit>
|
2019-08-08 16:26:25 -07:00
|
|
|
extern inline unit
|
|
|
|
__attribute__((always_inline, gnu_inline, artificial, flatten))
|
2017-09-08 01:43:43 -07:00
|
|
|
ircd::now()
|
|
|
|
{
|
2019-08-29 15:39:49 -07:00
|
|
|
const auto &now
|
2017-09-08 01:43:43 -07:00
|
|
|
{
|
2019-08-29 15:39:49 -07:00
|
|
|
tse(ircd::now<steady_point>())
|
2017-09-08 01:43:43 -07:00
|
|
|
};
|
|
|
|
|
2019-08-08 16:26:25 -07:00
|
|
|
return std::chrono::duration_cast<unit>(now);
|
2017-09-08 01:43:43 -07:00
|
|
|
}
|
2019-09-20 23:51:46 -07:00
|
|
|
|
|
|
|
template<>
|
|
|
|
inline ircd::steady_point
|
|
|
|
__attribute__((always_inline, artificial, flatten))
|
|
|
|
ircd::now<ircd::steady_point>()
|
|
|
|
{
|
|
|
|
return steady_clock::now();
|
|
|
|
}
|