dosbox-staging/include/timer.h

148 lines
4.2 KiB
C++
Raw Normal View History

/*
* Copyright (C) 2002-2021 The DOSBox Team
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#ifndef DOSBOX_TIMER_H
#define DOSBOX_TIMER_H
#include <cassert>
2021-06-16 15:50:43 -05:00
#include <cmath>
2021-06-16 13:41:34 -05:00
#include <cstdlib>
#include <chrono>
#include <limits>
2021-06-16 13:41:34 -05:00
#include <thread>
/* underlying clock rate in HZ */
#define PIT_TICK_RATE 1193182
typedef void (*TIMER_TickHandler)(void);
2020-11-07 11:38:27 +00:00
/* Register a function that gets called every time if 1 or more ticks pass */
void TIMER_AddTickHandler(TIMER_TickHandler handler);
void TIMER_DelTickHandler(TIMER_TickHandler handler);
/* This will add 1 milliscond to all timers */
void TIMER_AddTick(void);
2021-06-19 16:25:41 -05:00
static inline int64_t GetTicks()
{
return std::chrono::duration_cast<std::chrono::milliseconds>(
std::chrono::steady_clock::now().time_since_epoch())
.count();
}
2021-06-19 16:25:41 -05:00
static inline int64_t GetTicksUs()
{
return std::chrono::duration_cast<std::chrono::microseconds>(
std::chrono::steady_clock::now().time_since_epoch())
.count();
}
2021-06-19 16:25:41 -05:00
static inline int GetTicksDiff(const int64_t new_ticks, const int64_t old_ticks)
{
assert(new_ticks >= old_ticks);
assert((new_ticks - old_ticks) <= std::numeric_limits<int>::max());
return static_cast<int>(new_ticks - old_ticks);
}
2021-06-19 16:25:41 -05:00
static inline int GetTicksSince(const int64_t old_ticks)
{
const auto now = GetTicks();
assert((now - old_ticks) <= std::numeric_limits<int>::max());
return GetTicksDiff(now, old_ticks);
}
2021-06-19 16:25:41 -05:00
static inline int GetTicksUsSince(const int64_t old_ticks)
{
const auto now = GetTicksUs();
assert((now - old_ticks) <= std::numeric_limits<int>::max());
return GetTicksDiff(now, old_ticks);
}
2021-06-19 16:25:41 -05:00
static inline void Delay(const int milliseconds)
{
std::this_thread::sleep_for(std::chrono::milliseconds(milliseconds));
}
2021-06-19 16:25:41 -05:00
static inline void DelayUs(const int microseconds)
2021-06-16 13:58:42 -05:00
{
std::this_thread::sleep_for(std::chrono::microseconds(microseconds));
}
2021-06-18 09:09:20 -05:00
// The duration to use for precise sleep
static constexpr int precise_delay_duration_us = 100;
2021-06-16 13:41:34 -05:00
2021-06-19 16:25:41 -05:00
// based on work from:
// https://blat-blatnik.github.io/computerBear/making-accurate-sleep-function/
static inline void DelayPrecise(const int milliseconds)
{
// The estimate of how long the sleep should take (microseconds)
static double estimate = 5e-5;
// Use the estimate value as the default mean time taken
static double mean = 5e-5;
static double m2 = 0;
static int64_t count = 1;
// Original code operated on seconds, convert
2021-06-16 13:41:34 -05:00
double seconds = milliseconds / 1e3;
2021-06-18 09:09:20 -05:00
// sleep as long as we can, then spinlock the rest
2021-06-19 16:25:41 -05:00
while (seconds > estimate) {
const auto start = GetTicksUs();
DelayUs(precise_delay_duration_us);
// Original code operated on seconds, convert
const double observed = GetTicksUsSince(start) / 1e6;
seconds -= observed;
++count;
const double delta = observed - mean;
mean += delta / count;
m2 += delta * (observed - mean);
const double stddev = std::sqrt(m2 / (count - 1));
estimate = mean + stddev;
}
2021-06-16 13:41:34 -05:00
// spin lock
const auto spin_start = GetTicksUs();
const int spin_remain = static_cast<int>(seconds * 1e6);
do {
std::this_thread::yield();
} while (GetTicksUsSince(spin_start) <= spin_remain);
2021-06-16 13:41:34 -05:00
}
2021-06-19 16:25:41 -05:00
static inline bool CanDelayPrecise()
2021-06-16 13:41:34 -05:00
{
2021-06-19 16:25:41 -05:00
// The tolerance to allow for sleep variation
constexpr int precise_delay_tolerance_us = precise_delay_duration_us;
2021-06-16 13:41:34 -05:00
bool is_precise = true;
for (int i=0;i<10;i++) {
const auto start = GetTicksUs();
2021-06-18 09:09:20 -05:00
DelayUs(precise_delay_duration_us);
2021-06-16 13:41:34 -05:00
const auto elapsed = GetTicksUsSince(start);
2021-06-19 16:25:41 -05:00
if (std::abs(elapsed - precise_delay_duration_us) >
precise_delay_tolerance_us)
is_precise = false;
2021-06-16 13:41:34 -05:00
}
return is_precise;
}
#endif