2018-02-03 18:22:01 -08:00
|
|
|
// 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.
|
2017-09-08 01:43:43 -07:00
|
|
|
|
|
|
|
#pragma once
|
2018-01-12 03:21:58 -08:00
|
|
|
#define HAVE_IRCD_DATE_H
|
2017-09-08 01:43:43 -07:00
|
|
|
|
|
|
|
namespace ircd
|
|
|
|
{
|
|
|
|
using microtime_t = std::pair<time_t, int32_t>;
|
2017-09-19 21:39:53 -07:00
|
|
|
using steady_point = time_point<steady_clock>;
|
|
|
|
using system_point = time_point<system_clock>;
|
|
|
|
|
2017-09-08 01:43:43 -07:00
|
|
|
microtime_t microtime();
|
2017-10-11 17:47:03 -07:00
|
|
|
string_view microtime(const mutable_buffer &);
|
2017-09-08 01:43:43 -07:00
|
|
|
std::ostream &operator<<(std::ostream &, const microtime_t &);
|
|
|
|
|
2017-09-19 21:39:53 -07:00
|
|
|
template<class unit = seconds> unit now();
|
|
|
|
template<> steady_point now();
|
|
|
|
template<> system_point now();
|
|
|
|
|
2017-09-29 22:57:29 -07:00
|
|
|
template<class unit = seconds> time_t &time(time_t &ref);
|
2017-09-19 21:39:53 -07:00
|
|
|
template<class unit = seconds> time_t time();
|
2017-09-29 22:57:29 -07:00
|
|
|
template<class unit = seconds> time_t time(time_t *const &ptr);
|
2017-10-11 17:47:03 -07:00
|
|
|
|
|
|
|
const char *const rfc7231_fmt
|
|
|
|
{
|
|
|
|
"%a, %d %b %Y %T %z"
|
|
|
|
};
|
|
|
|
|
|
|
|
IRCD_OVERLOAD(localtime)
|
|
|
|
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);
|
|
|
|
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);
|
|
|
|
template<class... args> std::string timestr(args&&...);
|
2018-02-12 15:04:54 -08:00
|
|
|
|
|
|
|
std::ostream &operator<<(std::ostream &, const steady_point &);
|
|
|
|
std::ostream &operator<<(std::ostream &, const system_point &);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::ostream &
|
|
|
|
ircd::operator<<(std::ostream &s, const system_point &tp)
|
|
|
|
{
|
|
|
|
return (s << duration_cast<seconds>(tp.time_since_epoch()).count());
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::ostream &
|
|
|
|
ircd::operator<<(std::ostream &s, const steady_point &tp)
|
|
|
|
{
|
|
|
|
return (s << duration_cast<seconds>(tp.time_since_epoch()).count());
|
2017-10-11 17:47:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
template<class... args>
|
|
|
|
std::string
|
|
|
|
ircd::timestr(args&&... a)
|
|
|
|
{
|
2018-01-29 08:37:20 -08:00
|
|
|
static const size_t max{128};
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
inline ircd::string_view
|
|
|
|
ircd::timef(const mutable_buffer &out,
|
|
|
|
const char *const &fmt)
|
|
|
|
{
|
|
|
|
const auto epoch{time()};
|
|
|
|
return timef(out, epoch, fmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ircd::string_view
|
|
|
|
ircd::timef(const mutable_buffer &out,
|
|
|
|
localtime_t,
|
|
|
|
const char *const &fmt)
|
|
|
|
{
|
|
|
|
const auto epoch{time()};
|
|
|
|
return timef(out, epoch, localtime, fmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ircd::string_view
|
|
|
|
ircd::timef(const mutable_buffer &out,
|
|
|
|
const time_t &epoch,
|
|
|
|
localtime_t,
|
|
|
|
const char *const &fmt)
|
|
|
|
{
|
|
|
|
struct tm tm;
|
|
|
|
localtime_r(&epoch, &tm);
|
|
|
|
return timef(out, tm, fmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ircd::string_view
|
|
|
|
ircd::timef(const mutable_buffer &out,
|
|
|
|
const time_t &epoch,
|
|
|
|
const char *const &fmt)
|
|
|
|
{
|
|
|
|
struct tm tm;
|
|
|
|
gmtime_r(&epoch, &tm);
|
|
|
|
return timef(out, tm, fmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ircd::string_view
|
|
|
|
ircd::timef(const mutable_buffer &out,
|
|
|
|
const struct tm &tm,
|
|
|
|
const char *const &fmt)
|
|
|
|
{
|
|
|
|
const auto len
|
|
|
|
{
|
|
|
|
strftime(data(out), size(out), fmt, &tm)
|
|
|
|
};
|
|
|
|
|
|
|
|
return { data(out), len };
|
2017-09-29 22:57:29 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
template<class unit>
|
|
|
|
time_t
|
|
|
|
ircd::time(time_t *const &ptr)
|
|
|
|
{
|
|
|
|
time_t buf, &ret{ptr? *ptr : buf};
|
|
|
|
return time<unit>(ret);
|
2017-09-08 01:43:43 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 21:39:53 -07:00
|
|
|
template<class unit>
|
2017-09-08 01:43:43 -07:00
|
|
|
time_t
|
|
|
|
ircd::time()
|
|
|
|
{
|
2017-09-29 22:57:29 -07:00
|
|
|
time_t ret;
|
|
|
|
return time<unit>(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class unit>
|
|
|
|
time_t &
|
|
|
|
ircd::time(time_t &ref)
|
|
|
|
{
|
|
|
|
ref = duration_cast<unit>(system_clock::now().time_since_epoch()).count();
|
|
|
|
return ref;
|
2017-09-19 21:39:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
template<> inline
|
|
|
|
ircd::steady_point
|
|
|
|
ircd::now()
|
|
|
|
{
|
|
|
|
return steady_clock::now();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<> inline
|
|
|
|
ircd::system_point
|
|
|
|
ircd::now()
|
|
|
|
{
|
|
|
|
return system_clock::now();
|
2017-09-08 01:43:43 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 21:39:53 -07:00
|
|
|
template<class unit>
|
|
|
|
unit
|
2017-09-08 01:43:43 -07:00
|
|
|
ircd::now()
|
|
|
|
{
|
|
|
|
const auto now
|
|
|
|
{
|
2017-09-19 21:39:53 -07:00
|
|
|
steady_clock::now()
|
2017-09-08 01:43:43 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
const auto tse
|
|
|
|
{
|
|
|
|
now.time_since_epoch()
|
|
|
|
};
|
|
|
|
|
2017-09-19 21:39:53 -07:00
|
|
|
return std::chrono::duration_cast<unit>(tse);
|
2017-09-08 01:43:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
inline std::ostream &
|
|
|
|
ircd::operator<<(std::ostream &s, const microtime_t &t)
|
|
|
|
{
|
|
|
|
char buf[64];
|
2017-10-11 17:47:03 -07:00
|
|
|
s << microtime(buf);
|
2017-09-08 01:43:43 -07:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2017-10-11 17:47:03 -07:00
|
|
|
inline ircd::string_view
|
|
|
|
ircd::microtime(const mutable_buffer &buf)
|
2017-09-08 01:43:43 -07:00
|
|
|
{
|
2017-10-11 17:47:03 -07:00
|
|
|
const auto mt{microtime()};
|
|
|
|
const auto length
|
|
|
|
{
|
|
|
|
snprintf(data(buf), size(buf), "%zd.%06d", mt.first, mt.second)
|
|
|
|
};
|
|
|
|
|
|
|
|
return string_view
|
|
|
|
{
|
|
|
|
data(buf), size_t(length)
|
|
|
|
};
|
2017-09-08 01:43:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
inline ircd::microtime_t
|
|
|
|
ircd::microtime()
|
|
|
|
{
|
|
|
|
struct timeval tv;
|
2018-01-17 21:37:11 -08:00
|
|
|
syscall(&::gettimeofday, &tv, nullptr);
|
2017-09-08 01:43:43 -07:00
|
|
|
return { tv.tv_sec, tv.tv_usec };
|
|
|
|
}
|