2020-12-26 14:12:05 -08:00
|
|
|
// Copyright (c) FIRST and other WPILib contributors.
|
|
|
|
|
// Open Source Software; you can modify and/or share it under the terms of
|
|
|
|
|
// the WPILib BSD license file in the root directory of this project.
|
2017-08-18 21:35:53 -07:00
|
|
|
|
2020-10-22 20:55:12 -07:00
|
|
|
#include <algorithm>
|
2017-08-18 21:35:53 -07:00
|
|
|
#include <atomic>
|
|
|
|
|
#include <chrono>
|
|
|
|
|
#include <cstdio>
|
|
|
|
|
#include <thread>
|
|
|
|
|
|
2021-06-06 16:13:58 -07:00
|
|
|
#include <fmt/format.h>
|
2018-04-29 23:33:19 -07:00
|
|
|
#include <wpi/timestamp.h>
|
2017-08-27 00:11:52 -07:00
|
|
|
|
2017-08-18 21:35:53 -07:00
|
|
|
#include "MockHooksInternal.h"
|
2019-11-10 20:54:25 -08:00
|
|
|
#include "NotifierInternal.h"
|
2020-10-22 20:55:12 -07:00
|
|
|
#include "hal/simulation/NotifierData.h"
|
2017-08-18 21:35:53 -07:00
|
|
|
|
|
|
|
|
static std::atomic<bool> programStarted{false};
|
|
|
|
|
|
|
|
|
|
static std::atomic<uint64_t> programStartTime{0};
|
2019-11-10 20:54:25 -08:00
|
|
|
static std::atomic<uint64_t> programPauseTime{0};
|
2020-06-29 21:52:23 -07:00
|
|
|
static std::atomic<uint64_t> programStepTime{0};
|
2017-08-18 21:35:53 -07:00
|
|
|
|
2020-12-28 01:19:59 -08:00
|
|
|
namespace hal::init {
|
2020-12-28 12:58:06 -08:00
|
|
|
void InitializeMockHooks() {
|
|
|
|
|
wpi::SetNowImpl(GetFPGATime);
|
|
|
|
|
}
|
2020-12-28 01:19:59 -08:00
|
|
|
} // namespace hal::init
|
2017-12-10 19:38:53 -08:00
|
|
|
|
2017-08-18 21:35:53 -07:00
|
|
|
namespace hal {
|
2019-11-10 20:54:25 -08:00
|
|
|
void RestartTiming() {
|
2020-08-06 23:03:42 -07:00
|
|
|
programStartTime = wpi::NowDefault();
|
2020-06-29 21:52:23 -07:00
|
|
|
programStepTime = 0;
|
2020-12-28 12:58:06 -08:00
|
|
|
if (programPauseTime != 0) {
|
|
|
|
|
programPauseTime = programStartTime.load();
|
|
|
|
|
}
|
2019-11-10 20:54:25 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void PauseTiming() {
|
2020-12-28 12:58:06 -08:00
|
|
|
if (programPauseTime == 0) {
|
|
|
|
|
programPauseTime = wpi::NowDefault();
|
|
|
|
|
}
|
2019-11-10 20:54:25 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ResumeTiming() {
|
|
|
|
|
if (programPauseTime != 0) {
|
2020-08-06 23:03:42 -07:00
|
|
|
programStartTime += wpi::NowDefault() - programPauseTime;
|
2019-11-10 20:54:25 -08:00
|
|
|
programPauseTime = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-28 12:58:06 -08:00
|
|
|
bool IsTimingPaused() {
|
|
|
|
|
return programPauseTime != 0;
|
|
|
|
|
}
|
2019-11-10 20:54:25 -08:00
|
|
|
|
2020-12-28 12:58:06 -08:00
|
|
|
void StepTiming(uint64_t delta) {
|
|
|
|
|
programStepTime += delta;
|
|
|
|
|
}
|
2017-08-18 21:35:53 -07:00
|
|
|
|
2020-08-06 23:03:42 -07:00
|
|
|
uint64_t GetFPGATime() {
|
2019-11-10 20:54:25 -08:00
|
|
|
uint64_t curTime = programPauseTime;
|
2020-12-28 12:58:06 -08:00
|
|
|
if (curTime == 0) {
|
|
|
|
|
curTime = wpi::NowDefault();
|
|
|
|
|
}
|
2020-06-29 21:52:23 -07:00
|
|
|
return curTime + programStepTime - programStartTime;
|
2017-08-18 21:35:53 -07:00
|
|
|
}
|
|
|
|
|
|
2020-12-28 12:58:06 -08:00
|
|
|
double GetFPGATimestamp() {
|
|
|
|
|
return GetFPGATime() * 1.0e-6;
|
|
|
|
|
}
|
2017-08-18 21:35:53 -07:00
|
|
|
|
2020-12-28 12:58:06 -08:00
|
|
|
void SetProgramStarted() {
|
|
|
|
|
programStarted = true;
|
|
|
|
|
}
|
|
|
|
|
bool GetProgramStarted() {
|
|
|
|
|
return programStarted;
|
|
|
|
|
}
|
2017-08-18 21:35:53 -07:00
|
|
|
} // namespace hal
|
|
|
|
|
|
|
|
|
|
using namespace hal;
|
|
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
|
void HALSIM_WaitForProgramStart(void) {
|
|
|
|
|
int count = 0;
|
|
|
|
|
while (!programStarted) {
|
|
|
|
|
count++;
|
2021-06-06 16:13:58 -07:00
|
|
|
fmt::print("Waiting for program start signal: {}\n", count);
|
2017-08-18 21:35:53 -07:00
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(500));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-28 12:58:06 -08:00
|
|
|
void HALSIM_SetProgramStarted(void) {
|
|
|
|
|
SetProgramStarted();
|
|
|
|
|
}
|
2017-08-18 21:35:53 -07:00
|
|
|
|
2020-12-28 12:58:06 -08:00
|
|
|
HAL_Bool HALSIM_GetProgramStarted(void) {
|
|
|
|
|
return GetProgramStarted();
|
|
|
|
|
}
|
2018-07-26 01:30:29 -07:00
|
|
|
|
2020-12-28 12:58:06 -08:00
|
|
|
void HALSIM_RestartTiming(void) {
|
|
|
|
|
RestartTiming();
|
|
|
|
|
}
|
2019-11-10 20:54:25 -08:00
|
|
|
|
|
|
|
|
void HALSIM_PauseTiming(void) {
|
|
|
|
|
PauseTiming();
|
|
|
|
|
PauseNotifiers();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HALSIM_ResumeTiming(void) {
|
|
|
|
|
ResumeTiming();
|
|
|
|
|
ResumeNotifiers();
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-28 12:58:06 -08:00
|
|
|
HAL_Bool HALSIM_IsTimingPaused(void) {
|
|
|
|
|
return IsTimingPaused();
|
|
|
|
|
}
|
2019-11-10 20:54:25 -08:00
|
|
|
|
|
|
|
|
void HALSIM_StepTiming(uint64_t delta) {
|
2020-10-15 20:18:15 -07:00
|
|
|
WaitNotifiers();
|
2020-10-22 20:55:12 -07:00
|
|
|
|
|
|
|
|
while (delta > 0) {
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
uint64_t curTime = HAL_GetFPGATime(&status);
|
|
|
|
|
uint64_t nextTimeout = HALSIM_GetNextNotifierTimeout();
|
2022-12-31 12:00:45 -08:00
|
|
|
uint64_t step = (std::min)(delta, nextTimeout - curTime);
|
2020-10-22 20:55:12 -07:00
|
|
|
|
|
|
|
|
StepTiming(step);
|
|
|
|
|
delta -= step;
|
|
|
|
|
|
|
|
|
|
WakeupWaitNotifiers();
|
|
|
|
|
}
|
2020-09-27 13:27:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HALSIM_StepTimingAsync(uint64_t delta) {
|
2019-11-10 20:54:25 -08:00
|
|
|
StepTiming(delta);
|
|
|
|
|
WakeupNotifiers();
|
|
|
|
|
}
|
2017-10-16 19:56:08 -07:00
|
|
|
} // extern "C"
|