2017-11-12 08:06:40 +00:00
|
|
|
/*
|
|
|
|
** i_time.cpp
|
|
|
|
** Implements the timer
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
2017-11-24 03:04:42 +00:00
|
|
|
** Copyright 1998-2016 Randy Heit
|
|
|
|
** Copyright 2017 Magnus Norddahl
|
2017-11-12 08:06:40 +00:00
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <chrono>
|
|
|
|
#include <thread>
|
|
|
|
#include "i_time.h"
|
|
|
|
#include "doomdef.h"
|
2017-11-16 03:28:18 +00:00
|
|
|
#include "c_cvars.h"
|
|
|
|
#include "doomstat.h"
|
2017-11-12 08:06:40 +00:00
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Tick time functions
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2017-11-12 11:57:19 +00:00
|
|
|
static uint64_t FirstFrameStartTime;
|
|
|
|
static uint64_t CurrentFrameStartTime;
|
|
|
|
static uint64_t FreezeTime;
|
2017-11-12 08:06:40 +00:00
|
|
|
|
2017-11-16 03:28:18 +00:00
|
|
|
static double TimeScale = 1.0;
|
|
|
|
|
|
|
|
CUSTOM_CVAR(Float, i_timescale, 1.0f, CVAR_NOINITCALL)
|
|
|
|
{
|
2017-11-29 13:41:09 +00:00
|
|
|
if (netgame)
|
2017-11-16 03:28:18 +00:00
|
|
|
{
|
|
|
|
Printf("Time scale cannot be changed in net games.\n");
|
|
|
|
self = 1.0f;
|
|
|
|
}
|
2017-11-29 13:41:09 +00:00
|
|
|
else if (self >= 0.05f)
|
2017-11-16 03:28:18 +00:00
|
|
|
{
|
|
|
|
I_FreezeTime(true);
|
|
|
|
TimeScale = self;
|
|
|
|
I_FreezeTime(false);
|
|
|
|
}
|
2017-11-29 13:41:09 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
Printf("Time scale must be at least 0.05!\n");
|
|
|
|
}
|
2017-11-16 03:28:18 +00:00
|
|
|
}
|
|
|
|
|
2017-11-12 11:57:19 +00:00
|
|
|
static uint64_t GetClockTimeNS()
|
2017-11-12 08:06:40 +00:00
|
|
|
{
|
|
|
|
using namespace std::chrono;
|
2017-11-29 12:43:17 +00:00
|
|
|
return (uint64_t)((duration_cast<microseconds>(steady_clock::now().time_since_epoch()).count()) * (uint64_t)(TimeScale * 1000));
|
2017-11-12 11:57:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static uint64_t MSToNS(unsigned int ms)
|
|
|
|
{
|
|
|
|
return static_cast<uint64_t>(ms) * 1'000'000;
|
|
|
|
}
|
|
|
|
|
2017-11-14 20:52:54 +00:00
|
|
|
static uint64_t NSToMS(uint64_t ns)
|
2017-11-12 11:57:19 +00:00
|
|
|
{
|
2017-11-14 20:52:54 +00:00
|
|
|
return static_cast<uint64_t>(ns / 1'000'000);
|
2017-11-12 11:57:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int NSToTic(uint64_t ns)
|
|
|
|
{
|
|
|
|
return static_cast<int>(ns * TICRATE / 1'000'000'000);
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint64_t TicToNS(int tic)
|
|
|
|
{
|
|
|
|
return static_cast<uint64_t>(tic) * 1'000'000'000 / TICRATE;
|
2017-11-12 08:06:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void I_SetFrameTime()
|
|
|
|
{
|
|
|
|
// Must only be called once per frame/swapbuffers.
|
|
|
|
//
|
|
|
|
// Caches all timing information for the current rendered frame so that any
|
2017-11-12 23:38:04 +00:00
|
|
|
// calls to I_GetTime or I_GetTimeFrac will return
|
2017-11-12 08:06:40 +00:00
|
|
|
// the same time.
|
|
|
|
|
|
|
|
if (FreezeTime == 0)
|
|
|
|
{
|
2017-11-12 11:57:19 +00:00
|
|
|
CurrentFrameStartTime = GetClockTimeNS();
|
2017-11-12 08:06:40 +00:00
|
|
|
if (FirstFrameStartTime == 0)
|
|
|
|
FirstFrameStartTime = CurrentFrameStartTime;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void I_WaitVBL(int count)
|
|
|
|
{
|
|
|
|
// I_WaitVBL is never used to actually synchronize to the vertical blank.
|
|
|
|
// Instead, it's used for delay purposes. Doom used a 70 Hz display mode,
|
|
|
|
// so that's what we use to determine how long to wait for.
|
|
|
|
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(1000 * count / 70));
|
|
|
|
I_SetFrameTime();
|
|
|
|
}
|
|
|
|
|
|
|
|
int I_WaitForTic(int prevtic)
|
|
|
|
{
|
|
|
|
// Waits until the current tic is greater than prevtic. Time must not be frozen.
|
|
|
|
|
|
|
|
int time;
|
|
|
|
while ((time = I_GetTime()) <= prevtic)
|
|
|
|
{
|
2017-11-25 09:03:37 +00:00
|
|
|
// Windows-specific note:
|
2017-11-12 08:06:40 +00:00
|
|
|
// The minimum amount of time a thread can sleep is controlled by timeBeginPeriod.
|
|
|
|
// We set this to 1 ms in DoMain.
|
2017-11-25 09:03:37 +00:00
|
|
|
|
|
|
|
const uint64_t next = FirstFrameStartTime + TicToNS(prevtic + 1);
|
|
|
|
const uint64_t now = I_nsTime();
|
|
|
|
|
|
|
|
if (next > now)
|
|
|
|
{
|
|
|
|
const uint64_t sleepTime = NSToMS(next - now);
|
|
|
|
|
|
|
|
if (sleepTime > 2)
|
|
|
|
{
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(sleepTime - 2));
|
|
|
|
}
|
|
|
|
}
|
2017-11-12 08:06:40 +00:00
|
|
|
|
|
|
|
I_SetFrameTime();
|
|
|
|
}
|
|
|
|
|
|
|
|
return time;
|
|
|
|
}
|
|
|
|
|
2017-11-12 23:54:32 +00:00
|
|
|
uint64_t I_nsTime()
|
2017-11-12 11:57:19 +00:00
|
|
|
{
|
2017-11-12 22:39:48 +00:00
|
|
|
return GetClockTimeNS();
|
2017-11-12 11:57:19 +00:00
|
|
|
}
|
|
|
|
|
2017-11-14 20:52:54 +00:00
|
|
|
uint64_t I_msTime()
|
2017-11-12 11:57:19 +00:00
|
|
|
{
|
2017-11-12 23:54:32 +00:00
|
|
|
return NSToMS(I_nsTime());
|
2017-11-12 11:57:19 +00:00
|
|
|
}
|
|
|
|
|
2017-12-07 08:01:57 +00:00
|
|
|
uint64_t I_msTimeFS() // from "start"
|
|
|
|
{
|
2017-12-09 11:09:39 +00:00
|
|
|
return (FirstFrameStartTime == 0) ? 0 : NSToMS(I_nsTime() - FirstFrameStartTime);
|
2017-12-07 08:01:57 +00:00
|
|
|
}
|
|
|
|
|
2017-11-12 08:06:40 +00:00
|
|
|
int I_GetTime()
|
|
|
|
{
|
2017-11-24 19:38:55 +00:00
|
|
|
return NSToTic(CurrentFrameStartTime - FirstFrameStartTime);
|
2017-11-12 08:06:40 +00:00
|
|
|
}
|
|
|
|
|
2017-11-12 23:54:32 +00:00
|
|
|
double I_GetTimeFrac()
|
2017-11-12 08:06:40 +00:00
|
|
|
{
|
2017-11-12 11:57:19 +00:00
|
|
|
int currentTic = NSToTic(CurrentFrameStartTime - FirstFrameStartTime);
|
|
|
|
uint64_t ticStartTime = FirstFrameStartTime + TicToNS(currentTic);
|
|
|
|
uint64_t ticNextTime = FirstFrameStartTime + TicToNS(currentTic + 1);
|
2017-11-12 08:06:40 +00:00
|
|
|
|
|
|
|
return (CurrentFrameStartTime - ticStartTime) / (double)(ticNextTime - ticStartTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
void I_FreezeTime(bool frozen)
|
|
|
|
{
|
|
|
|
if (frozen)
|
|
|
|
{
|
2018-03-04 13:11:45 +00:00
|
|
|
assert(FreezeTime == 0);
|
2017-11-12 11:57:19 +00:00
|
|
|
FreezeTime = GetClockTimeNS();
|
2017-11-12 08:06:40 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-03-04 13:11:45 +00:00
|
|
|
assert(FreezeTime != 0);
|
2017-11-12 11:57:19 +00:00
|
|
|
FirstFrameStartTime += GetClockTimeNS() - FreezeTime;
|
2017-11-12 08:06:40 +00:00
|
|
|
FreezeTime = 0;
|
|
|
|
I_SetFrameTime();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|