blob: 64b43e32c93a3e074b3852dbd32eff43c1c0c317 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
* Copyright (c) 2013-2018 Regents of the University of California.
*
* This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
*
* ndn-cxx library is free software: you can redistribute it and/or modify it under the
* terms of the GNU Lesser General Public License as published by the Free Software
* Foundation, either version 3 of the License, or (at your option) any later version.
*
* ndn-cxx library is distributed in the hope that it will be useful, but WITHOUT ANY
* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
*
* You should have received copies of the GNU General Public License and GNU Lesser
* General Public License along with ndn-cxx, e.g., in COPYING.md file. If not, see
* <http://www.gnu.org/licenses/>.
*
* See AUTHORS.md for complete list of ndn-cxx authors and contributors.
*/
#include "time.hpp"
#include "time-custom-clock.hpp"
#include <boost/date_time/posix_time/posix_time.hpp>
#include <sstream>
namespace ndn {
namespace time {
static shared_ptr<CustomSystemClock> g_systemClock;
static shared_ptr<CustomSteadyClock> g_steadyClock;
// this function is declared in time-custom-clock.hpp
void
setCustomClocks(shared_ptr<CustomSteadyClock> steadyClock,
shared_ptr<CustomSystemClock> systemClock)
{
g_systemClock = std::move(systemClock);
g_steadyClock = std::move(steadyClock);
}
/////////////////////////////////////////////////////////////////////////////////////////////
system_clock::time_point
system_clock::now() noexcept
{
if (g_systemClock == nullptr) {
// optimized default version
return time_point(boost::chrono::system_clock::now().time_since_epoch());
}
else {
return g_systemClock->getNow();
}
}
std::time_t
system_clock::to_time_t(const system_clock::time_point& t) noexcept
{
return duration_cast<seconds>(t.time_since_epoch()).count();
}
system_clock::time_point
system_clock::from_time_t(std::time_t t) noexcept
{
return time_point(seconds(t));
}
/////////////////////////////////////////////////////////////////////////////////////////////
#ifdef __APPLE__
// Note that on OS X platform boost::steady_clock is not truly monotonic, so we use
// system_clock instead. Refer to https://svn.boost.org/trac/boost/ticket/7719)
typedef boost::chrono::system_clock base_steady_clock;
#else
typedef boost::chrono::steady_clock base_steady_clock;
#endif
steady_clock::time_point
steady_clock::now() noexcept
{
if (g_steadyClock == nullptr) {
// optimized default version
return time_point(base_steady_clock::now().time_since_epoch());
}
else {
return g_steadyClock->getNow();
}
}
steady_clock::duration
steady_clock::to_wait_duration(steady_clock::duration d)
{
if (g_steadyClock == nullptr) {
// optimized default version
return d;
}
else {
return g_steadyClock->toWaitDuration(d);
}
}
/////////////////////////////////////////////////////////////////////////////////////////////
const system_clock::TimePoint&
getUnixEpoch()
{
static constexpr system_clock::TimePoint epoch(seconds::zero());
return epoch;
}
milliseconds
toUnixTimestamp(const system_clock::TimePoint& point)
{
return duration_cast<milliseconds>(point - getUnixEpoch());
}
system_clock::TimePoint
fromUnixTimestamp(milliseconds duration)
{
return getUnixEpoch() + duration;
}
static boost::posix_time::ptime
convertToPosixTime(const system_clock::TimePoint& timePoint)
{
namespace bpt = boost::posix_time;
static bpt::ptime epoch(boost::gregorian::date(1970, 1, 1));
using BptResolution =
#if defined(BOOST_DATE_TIME_HAS_NANOSECONDS)
nanoseconds;
#elif defined(BOOST_DATE_TIME_HAS_MICROSECONDS)
microseconds;
#else
milliseconds;
#endif
constexpr auto unitsPerHour = duration_cast<BptResolution>(1_h).count();
auto sinceEpoch = duration_cast<BptResolution>(timePoint - getUnixEpoch()).count();
return epoch + bpt::time_duration(sinceEpoch / unitsPerHour, 0, 0, sinceEpoch % unitsPerHour);
}
std::string
toIsoString(const system_clock::TimePoint& timePoint)
{
return boost::posix_time::to_iso_string(convertToPosixTime(timePoint));
}
static system_clock::TimePoint
convertToTimePoint(const boost::posix_time::ptime& ptime)
{
namespace bpt = boost::posix_time;
static bpt::ptime epoch(boost::gregorian::date(1970, 1, 1));
// .total_seconds() has an issue with large dates until Boost 1.66, see #4478.
// time_t overflows for large dates on 32-bit platforms (Y2038 problem).
auto sinceEpoch = ptime - epoch;
auto point = system_clock::TimePoint(seconds(sinceEpoch.ticks() / bpt::time_duration::ticks_per_second()));
return point + microseconds(sinceEpoch.total_microseconds() % 1000000);
}
system_clock::TimePoint
fromIsoString(const std::string& isoString)
{
return convertToTimePoint(boost::posix_time::from_iso_string(isoString));
}
std::string
toString(const system_clock::TimePoint& timePoint,
const std::string& format/* = "%Y-%m-%d %H:%M:%S"*/,
const std::locale& locale/* = std::locale("C")*/)
{
namespace bpt = boost::posix_time;
std::ostringstream os;
auto* facet = new bpt::time_facet(format.data());
os.imbue(std::locale(locale, facet));
os << convertToPosixTime(timePoint);
return os.str();
}
system_clock::TimePoint
fromString(const std::string& timePointStr,
const std::string& format/* = "%Y-%m-%d %H:%M:%S"*/,
const std::locale& locale/* = std::locale("C")*/)
{
namespace bpt = boost::posix_time;
std::istringstream is(timePointStr);
auto* facet = new bpt::time_input_facet(format);
is.imbue(std::locale(locale, facet));
bpt::ptime ptime;
is >> ptime;
return convertToTimePoint(ptime);
}
} // namespace time
} // namespace ndn
namespace boost {
namespace chrono {
template<class CharT>
std::basic_string<CharT>
clock_string<ndn::time::system_clock, CharT>::since()
{
if (ndn::time::g_systemClock == nullptr) {
// optimized default version
return clock_string<system_clock, CharT>::since();
}
else {
return ndn::time::g_systemClock->getSince();
}
}
template<class CharT>
std::basic_string<CharT>
clock_string<ndn::time::steady_clock, CharT>::since()
{
if (ndn::time::g_steadyClock == nullptr) {
// optimized default version
return clock_string<ndn::time::base_steady_clock, CharT>::since();
}
else {
return ndn::time::g_steadyClock->getSince();
}
}
template struct clock_string<ndn::time::system_clock, char>;
template struct clock_string<ndn::time::steady_clock, char>;
} // namespace chrono
} // namespace boost