2024-07-19 12:56:37 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2020-01-24 04:53:40 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "types.h"
|
|
|
|
|
2024-07-19 12:56:37 +00:00
|
|
|
#include <string_view>
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
class StateWrapper;
|
2020-01-24 04:53:40 +00:00
|
|
|
|
|
|
|
// Event callback type. Second parameter is the number of cycles the event was executed "late".
|
2021-01-09 15:43:59 +00:00
|
|
|
using TimingEventCallback = void (*)(void* param, TickCount ticks, TickCount ticks_late);
|
2020-01-24 04:53:40 +00:00
|
|
|
|
|
|
|
class TimingEvent
|
|
|
|
{
|
|
|
|
public:
|
2024-07-19 12:56:37 +00:00
|
|
|
TimingEvent(const std::string_view name, TickCount period, TickCount interval, TimingEventCallback callback,
|
2021-01-09 15:43:59 +00:00
|
|
|
void* callback_param);
|
2020-01-24 04:53:40 +00:00
|
|
|
~TimingEvent();
|
|
|
|
|
2024-07-19 12:56:37 +00:00
|
|
|
ALWAYS_INLINE const std::string_view GetName() const { return m_name; }
|
2021-01-09 15:38:39 +00:00
|
|
|
ALWAYS_INLINE bool IsActive() const { return m_active; }
|
2020-01-24 04:53:40 +00:00
|
|
|
|
|
|
|
// Returns the number of ticks between each event.
|
2020-07-31 07:09:18 +00:00
|
|
|
ALWAYS_INLINE TickCount GetPeriod() const { return m_period; }
|
|
|
|
ALWAYS_INLINE TickCount GetInterval() const { return m_interval; }
|
2020-01-24 04:53:40 +00:00
|
|
|
|
|
|
|
// Includes pending time.
|
|
|
|
TickCount GetTicksSinceLastExecution() const;
|
|
|
|
TickCount GetTicksUntilNextExecution() const;
|
|
|
|
|
2021-06-14 04:55:20 +00:00
|
|
|
// Adds ticks to current execution.
|
|
|
|
void Delay(TickCount ticks);
|
|
|
|
|
2020-01-24 04:53:40 +00:00
|
|
|
void Schedule(TickCount ticks);
|
|
|
|
void SetIntervalAndSchedule(TickCount ticks);
|
|
|
|
void SetPeriodAndSchedule(TickCount ticks);
|
|
|
|
|
|
|
|
// Services the event with the current accmulated time. If force is set, when not enough time is pending to
|
|
|
|
// simulate a single cycle, the callback will still be invoked, otherwise it won't be.
|
|
|
|
void InvokeEarly(bool force = false);
|
|
|
|
|
|
|
|
// Deactivates the event, preventing it from firing again.
|
|
|
|
// Do not call within a callback, return Deactivate instead.
|
|
|
|
void Activate();
|
|
|
|
void Deactivate();
|
|
|
|
|
|
|
|
ALWAYS_INLINE void SetState(bool active)
|
|
|
|
{
|
|
|
|
if (active)
|
|
|
|
Activate();
|
|
|
|
else
|
|
|
|
Deactivate();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Directly alters the interval of the event.
|
|
|
|
void SetInterval(TickCount interval) { m_interval = interval; }
|
|
|
|
void SetPeriod(TickCount period) { m_period = period; }
|
|
|
|
|
2020-08-31 10:54:59 +00:00
|
|
|
TimingEvent* prev = nullptr;
|
|
|
|
TimingEvent* next = nullptr;
|
|
|
|
|
2021-01-09 15:43:59 +00:00
|
|
|
TimingEventCallback m_callback;
|
|
|
|
void* m_callback_param;
|
|
|
|
|
2024-08-13 13:52:25 +00:00
|
|
|
GlobalTicks m_next_run_time = 0;
|
|
|
|
GlobalTicks m_last_run_time = 0;
|
|
|
|
|
2020-01-24 04:53:40 +00:00
|
|
|
TickCount m_period;
|
|
|
|
TickCount m_interval;
|
2021-02-06 09:19:55 +00:00
|
|
|
bool m_active = false;
|
2020-01-24 04:53:40 +00:00
|
|
|
|
2024-07-19 12:56:37 +00:00
|
|
|
std::string_view m_name;
|
2020-01-24 04:53:40 +00:00
|
|
|
};
|
2020-07-31 07:09:18 +00:00
|
|
|
|
|
|
|
namespace TimingEvents {
|
|
|
|
|
2024-08-13 13:52:25 +00:00
|
|
|
GlobalTicks GetGlobalTickCounter();
|
|
|
|
GlobalTicks GetEventRunTickCounter();
|
2020-07-31 07:09:18 +00:00
|
|
|
|
|
|
|
void Initialize();
|
|
|
|
void Reset();
|
|
|
|
void Shutdown();
|
|
|
|
|
2020-08-15 04:56:20 +00:00
|
|
|
bool DoState(StateWrapper& sw);
|
2020-07-31 07:09:18 +00:00
|
|
|
|
2023-08-15 13:12:21 +00:00
|
|
|
bool IsRunningEvents();
|
2024-08-13 13:52:25 +00:00
|
|
|
void CancelRunningEvent();
|
2020-07-31 07:09:18 +00:00
|
|
|
void RunEvents();
|
2024-08-13 13:52:25 +00:00
|
|
|
void CommitLeftoverTicks();
|
2020-07-31 07:09:18 +00:00
|
|
|
|
|
|
|
void UpdateCPUDowncount();
|
|
|
|
|
2020-10-18 04:43:09 +00:00
|
|
|
TimingEvent** GetHeadEventPtr();
|
|
|
|
|
2021-01-09 15:43:59 +00:00
|
|
|
} // namespace TimingEvents
|