datactl/syclock.h
datactl/syclock.h
Namespaces
Name |
---|
Syntalos |
Classes
Name | |
---|---|
struct | Syntalos::symaster_clock Syntalos Master Clock. |
class | Syntalos::SyncTimer |
Defines
Name | |
---|---|
TIMER_FUNC_TIMESTAMP(T, F) | |
MTIMER_FUNC_TIMESTAMP(F) | |
FUNC_EXEC_TIMESTAMP(INIT_TIME, F) | |
FUNC_DONE_TIMESTAMP(INIT_TIME, F) |
Macros Documentation
define TIMER_FUNC_TIMESTAMP
#define TIMER_FUNC_TIMESTAMP(
T,
F
)
({ \
const auto __stime = T->timeSinceStartNsec(); \
F; \
std::chrono::round<microseconds_t>((__stime + T->timeSinceStartNsec()) / 2.0); \
})
Compute a timestamp for “when this function acquired a value”. This is assumed to be the mean between function start and end time, rounded to milliseconds. For example, if the function F acquires a timestamp’ed value, this macro should return the equivalent timestamp on our timer T. This should balance out context switches if they are not too bad, and produce a reasonably accurate result in milliseconds. It is superior to measuring our timestamp for alignment after the other timestamping function was run.
The resulting timestamp is in µs
define MTIMER_FUNC_TIMESTAMP
#define MTIMER_FUNC_TIMESTAMP(
F
)
(TIMER_FUNC_TIMESTAMP(m_syTimer, F))
define FUNC_EXEC_TIMESTAMP
#define FUNC_EXEC_TIMESTAMP(
INIT_TIME,
F
)
({ \
const auto __stime = std::chrono::duration_cast<nanoseconds_t>(symaster_clock::now() - (INIT_TIME)); \
F; \
std::chrono::round<microseconds_t>( \
(__stime + std::chrono::duration_cast<nanoseconds_t>(symaster_clock::now() - (INIT_TIME))) / 2.0); \
})
Compute a timestamp for “when this function acquired a value”. This function is equivalent to TIMER_FUNC_TIMESTAMP(), but takes a starting timepoint instead of a timer as first parameter.
The resulting timestamp is in µs
define FUNC_DONE_TIMESTAMP
#define FUNC_DONE_TIMESTAMP(
INIT_TIME,
F
)
({ \
F; \
std::chrono::duration_cast<microseconds_t>(symaster_clock::now() - (INIT_TIME)); \
})
Compute a timestamp for “when this function completed”. This macro does not return the average between starting and end-time of the function invocation, but rather just the time when it was completed.
The resulting timestamp is in µs
Source code
/*
* Copyright (C) 2016-2024 Matthias Klumpp <matthias@tenstral.net>
*
* Licensed under the GNU Lesser General Public License Version 3
*
* This program 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.
*
* This software 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 a copy of the GNU Lesser General Public License
* along with this software. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include <QLoggingCategory>
#include <QMetaType>
#include <atomic>
#include <chrono>
#include <memory>
#include <ratio>
Q_DECLARE_METATYPE(std::chrono::milliseconds);
Q_DECLARE_METATYPE(std::chrono::microseconds);
Q_DECLARE_METATYPE(std::chrono::nanoseconds);
namespace Syntalos
{
Q_DECLARE_LOGGING_CATEGORY(logTimeClock)
struct symaster_clock {
typedef std::chrono::nanoseconds duration;
typedef duration::rep rep;
typedef duration::period period;
typedef std::chrono::time_point<symaster_clock, duration> time_point;
static constexpr bool is_steady = true;
static time_point now() noexcept;
};
using symaster_timepoint = std::chrono::time_point<symaster_clock>;
using milliseconds_t = std::chrono::milliseconds;
using microseconds_t = std::chrono::duration<int64_t, std::micro>;
using nanoseconds_t = std::chrono::duration<int64_t, std::nano>;
inline milliseconds_t usecToMsec(const microseconds_t &usec)
{
return std::chrono::duration_cast<milliseconds_t>(usec);
}
inline microseconds_t msecToUsec(const milliseconds_t &msec)
{
return std::chrono::duration_cast<microseconds_t>(msec);
}
inline microseconds_t nsecToUsec(const nanoseconds_t &msec)
{
return std::chrono::duration_cast<microseconds_t>(msec);
}
inline milliseconds_t timeDiffMsec(const symaster_timepoint &timePoint1, const symaster_timepoint &timePoint2) noexcept
{
return std::chrono::duration_cast<milliseconds_t>(timePoint1 - timePoint2);
}
inline microseconds_t timeDiffUsec(const symaster_timepoint &timePoint1, const symaster_timepoint &timePoint2) noexcept
{
return std::chrono::duration_cast<microseconds_t>(timePoint1 - timePoint2);
}
inline milliseconds_t timeDiffToNowMsec(const std::chrono::time_point<symaster_clock> &timePoint) noexcept
{
return std::chrono::duration_cast<milliseconds_t>(symaster_clock::now() - timePoint);
}
inline symaster_timepoint currentTimePoint() noexcept
{
return symaster_clock::now();
}
class SyncTimer
{
public:
explicit SyncTimer();
void start() noexcept;
void startAt(const symaster_timepoint &startTimePoint) noexcept;
inline milliseconds_t timeSinceStartMsec() noexcept
{
return std::chrono::duration_cast<milliseconds_t>(symaster_clock::now() - m_startTime);
}
inline microseconds_t timeSinceStartUsec() noexcept
{
return std::chrono::duration_cast<microseconds_t>(symaster_clock::now() - m_startTime);
}
inline nanoseconds_t timeSinceStartNsec() noexcept
{
return std::chrono::duration_cast<nanoseconds_t>(symaster_clock::now() - m_startTime);
}
inline symaster_timepoint currentTimePoint() noexcept
{
return symaster_clock::now();
}
symaster_timepoint startTime() noexcept
{
return m_startTime;
}
private:
symaster_timepoint m_startTime;
bool m_started;
};
#define TIMER_FUNC_TIMESTAMP(T, F) \
({ \
const auto __stime = T->timeSinceStartNsec(); \
F; \
std::chrono::round<microseconds_t>((__stime + T->timeSinceStartNsec()) / 2.0); \
})
#define MTIMER_FUNC_TIMESTAMP(F) (TIMER_FUNC_TIMESTAMP(m_syTimer, F))
#define FUNC_EXEC_TIMESTAMP(INIT_TIME, F) \
({ \
const auto __stime = std::chrono::duration_cast<nanoseconds_t>(symaster_clock::now() - (INIT_TIME)); \
F; \
std::chrono::round<microseconds_t>( \
(__stime + std::chrono::duration_cast<nanoseconds_t>(symaster_clock::now() - (INIT_TIME))) / 2.0); \
})
#define FUNC_DONE_TIMESTAMP(INIT_TIME, F) \
({ \
F; \
std::chrono::duration_cast<microseconds_t>(symaster_clock::now() - (INIT_TIME)); \
})
} // namespace Syntalos
Updated on 2024-12-04 at 20:48:34 +0000