2019-03-18 05:52:32 +00:00
|
|
|
// Copyright 2019, Collabora, Ltd.
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of a steady, convertible clock.
|
|
|
|
* @author Ryan Pavlik <ryan.pavlik@collabora.com>
|
2019-04-06 11:28:56 +00:00
|
|
|
* @ingroup aux_util
|
2019-03-18 05:52:32 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "u_time.h"
|
|
|
|
|
|
|
|
#include <chrono>
|
|
|
|
#include <new>
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
using namespace std::chrono;
|
|
|
|
|
|
|
|
struct MatchingTimePoints
|
|
|
|
{
|
2019-06-10 13:15:47 +00:00
|
|
|
system_clock::time_point sys = system_clock::now();
|
|
|
|
steady_clock::time_point steady = steady_clock::now();
|
2019-03-18 05:52:32 +00:00
|
|
|
// high_resolution_clock::time_point highRes;
|
|
|
|
|
|
|
|
timepoint_ns
|
2019-09-29 10:43:23 +00:00
|
|
|
getTimestamp(time_state const &prevState);
|
2019-03-18 05:52:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct time_state
|
|
|
|
{
|
2019-06-10 13:15:47 +00:00
|
|
|
MatchingTimePoints lastTimePoints = {};
|
|
|
|
timepoint_ns lastTime = {};
|
2019-03-18 05:52:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
timepoint_ns
|
2019-09-29 10:43:23 +00:00
|
|
|
MatchingTimePoints::getTimestamp(time_state const &prevState)
|
2019-03-18 05:52:32 +00:00
|
|
|
{
|
|
|
|
//! @todo right now just doing steady clock for simplicity.
|
|
|
|
//! @todo Eventually need to make the high-res clock steady.
|
|
|
|
auto elapsed = steady - prevState.lastTimePoints.steady;
|
|
|
|
return prevState.lastTime + duration_cast<nanoseconds>(elapsed).count();
|
|
|
|
}
|
|
|
|
|
2019-11-15 20:28:24 +00:00
|
|
|
extern "C" struct time_state *
|
2019-03-18 05:52:32 +00:00
|
|
|
time_state_create()
|
|
|
|
{
|
|
|
|
|
2019-09-29 10:43:23 +00:00
|
|
|
time_state *state = new (std::nothrow) time_state;
|
2019-03-18 05:52:32 +00:00
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
2019-11-15 20:28:24 +00:00
|
|
|
extern "C" void
|
|
|
|
time_state_destroy(struct time_state **state_ptr)
|
2019-03-18 05:52:32 +00:00
|
|
|
{
|
2019-11-15 20:30:01 +00:00
|
|
|
struct time_state *state = *state_ptr;
|
|
|
|
|
|
|
|
if (state == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-03-18 05:52:32 +00:00
|
|
|
delete state;
|
2019-11-15 20:30:01 +00:00
|
|
|
*state_ptr = NULL;
|
2019-03-18 05:52:32 +00:00
|
|
|
}
|
|
|
|
|
2019-11-15 20:28:24 +00:00
|
|
|
extern "C" timepoint_ns
|
2019-09-29 10:43:23 +00:00
|
|
|
time_state_get_now(struct time_state const *state)
|
2019-03-18 05:52:32 +00:00
|
|
|
{
|
|
|
|
assert(state != NULL);
|
|
|
|
auto now = MatchingTimePoints();
|
|
|
|
|
|
|
|
return now.getTimestamp(*state);
|
|
|
|
}
|
|
|
|
|
2019-11-15 20:28:24 +00:00
|
|
|
extern "C" timepoint_ns
|
2019-09-29 10:43:23 +00:00
|
|
|
time_state_get_now_and_update(struct time_state *state)
|
2019-03-18 05:52:32 +00:00
|
|
|
{
|
|
|
|
assert(state != NULL);
|
|
|
|
auto now = MatchingTimePoints();
|
|
|
|
|
|
|
|
auto timestamp = now.getTimestamp(*state);
|
|
|
|
|
|
|
|
// Update the state
|
|
|
|
state->lastTimePoints = now;
|
|
|
|
state->lastTime = timestamp;
|
|
|
|
|
|
|
|
return timestamp;
|
|
|
|
}
|
|
|
|
|
2019-11-15 20:28:24 +00:00
|
|
|
extern "C" void
|
2019-09-29 10:43:23 +00:00
|
|
|
time_state_to_timespec(struct time_state const *state,
|
2019-03-18 05:52:32 +00:00
|
|
|
timepoint_ns timestamp,
|
2019-09-29 10:43:23 +00:00
|
|
|
struct timespec *out)
|
2019-03-18 05:52:32 +00:00
|
|
|
{
|
|
|
|
assert(state != NULL);
|
|
|
|
assert(out != NULL);
|
|
|
|
// Subject to some jitter, roughly up to the magnitude of the actual
|
|
|
|
// resolution difference between the used time source and the system
|
|
|
|
// clock, as well as the non-simultaneity of the calls in
|
|
|
|
// MatchingTimePoints::getNow()
|
|
|
|
auto sinceLastUpdate = nanoseconds{timestamp - state->lastTime};
|
|
|
|
auto equivSystemTimepoint = state->lastTimePoints.sys + sinceLastUpdate;
|
|
|
|
|
|
|
|
// System clock epoch is same as unix epoch for all known
|
|
|
|
// implementations, but not strictly standard-required yet, see
|
|
|
|
// wg21.link/p0355
|
|
|
|
auto sinceEpoch = equivSystemTimepoint.time_since_epoch();
|
|
|
|
auto secondsSinceEpoch = duration_cast<seconds>(sinceEpoch);
|
|
|
|
sinceEpoch -= secondsSinceEpoch;
|
|
|
|
out->tv_sec = secondsSinceEpoch.count();
|
|
|
|
out->tv_nsec = duration_cast<nanoseconds>(sinceEpoch).count();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-11-15 20:28:24 +00:00
|
|
|
extern "C" timepoint_ns
|
2019-09-29 10:43:23 +00:00
|
|
|
time_state_from_timespec(struct time_state const *state,
|
|
|
|
const struct timespec *timespecTime)
|
2019-03-18 05:52:32 +00:00
|
|
|
{
|
|
|
|
assert(state != NULL);
|
|
|
|
assert(timespecTime != NULL);
|
|
|
|
auto sinceEpoch =
|
|
|
|
seconds{timespecTime->tv_sec} + nanoseconds{timespecTime->tv_nsec};
|
|
|
|
|
|
|
|
|
|
|
|
// System clock epoch is same as unix epoch for all known
|
|
|
|
// implementations, but not strictly standard-required yet, see
|
|
|
|
// wg21.link/p0355
|
|
|
|
auto systemTimePoint = time_point<system_clock, nanoseconds>{
|
|
|
|
duration_cast<system_clock::duration>(sinceEpoch)};
|
|
|
|
|
|
|
|
// duration between last update and the supplied timespec
|
|
|
|
auto sinceLastUpdate = state->lastTimePoints.sys - systemTimePoint;
|
|
|
|
|
|
|
|
// Offset the last timestamp by that duration.
|
|
|
|
return state->lastTime +
|
|
|
|
duration_cast<nanoseconds>(sinceLastUpdate).count();
|
2020-02-27 13:46:10 +00:00
|
|
|
}
|