2022-01-10 15:36:14 +00:00
|
|
|
/*
|
|
|
|
DingusPPC - The Experimental PowerPC Macintosh emulator
|
2023-08-01 15:36:29 +00:00
|
|
|
Copyright (C) 2018-23 divingkatae and maximum
|
2022-01-10 15:36:14 +00:00
|
|
|
(theweirdo) spatium
|
|
|
|
|
|
|
|
(Contact divingkatae#1017 or powermax#2286 on Discord for more info)
|
|
|
|
|
|
|
|
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 3 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, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef TIMER_MANAGER_H
|
|
|
|
#define TIMER_MANAGER_H
|
|
|
|
|
2024-03-17 02:24:40 +00:00
|
|
|
#include <atomic>
|
2022-02-16 22:52:43 +00:00
|
|
|
#include <algorithm>
|
2022-01-10 15:36:14 +00:00
|
|
|
#include <cinttypes>
|
|
|
|
#include <functional>
|
|
|
|
#include <memory>
|
|
|
|
#include <queue>
|
|
|
|
#include <vector>
|
2023-11-23 19:56:36 +00:00
|
|
|
#include <mutex>
|
2022-01-10 15:36:14 +00:00
|
|
|
|
2024-08-27 11:43:18 +00:00
|
|
|
#define NS_PER_SEC 1000000000
|
|
|
|
#define USEC_PER_SEC 1000000
|
|
|
|
#define NS_PER_USEC 1000
|
|
|
|
#define NS_PER_MSEC 1000000
|
|
|
|
#define ONE_BILLION_NS 1000000000
|
2022-01-10 15:36:14 +00:00
|
|
|
|
2022-10-20 11:02:48 +00:00
|
|
|
#define USECS_TO_NSECS(us) (us) * NS_PER_USEC
|
2023-04-01 15:20:53 +00:00
|
|
|
#define MSECS_TO_NSECS(ms) (ms) * NS_PER_MSEC
|
2022-01-10 15:36:14 +00:00
|
|
|
|
2024-11-10 06:25:48 +00:00
|
|
|
typedef std::function<void()> timer_cb;
|
2022-01-10 15:36:14 +00:00
|
|
|
|
|
|
|
/** Extend std::priority_queue as suggested here:
|
|
|
|
https://stackoverflow.com/a/36711682
|
|
|
|
to be able to remove arbitrary elements.
|
|
|
|
*/
|
|
|
|
template <typename T, class Container = std::vector<T>, class Compare = std::less<typename Container::value_type>>
|
|
|
|
class my_priority_queue : public std::priority_queue<T, Container, Compare> {
|
|
|
|
public:
|
|
|
|
bool remove_by_id(const uint32_t id){
|
2023-11-23 19:56:36 +00:00
|
|
|
std::lock_guard<std::recursive_mutex> lk(mtx);
|
|
|
|
auto el = this->top();
|
|
|
|
if (el->id == id) {
|
|
|
|
std::priority_queue<T, Container, Compare>::pop();
|
|
|
|
return true;
|
|
|
|
}
|
2022-01-10 15:36:14 +00:00
|
|
|
auto it = std::find_if(
|
|
|
|
this->c.begin(), this->c.end(), [id](const T& el) { return el->id == id; });
|
|
|
|
if (it != this->c.end()) {
|
|
|
|
this->c.erase(it);
|
|
|
|
std::make_heap(this->c.begin(), this->c.end(), this->comp);
|
|
|
|
return true;
|
|
|
|
}
|
2023-11-23 19:56:36 +00:00
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
void push(T val)
|
|
|
|
{
|
|
|
|
std::lock_guard<std::recursive_mutex> lk(mtx);
|
|
|
|
std::priority_queue<T, Container, Compare>::push(val);
|
|
|
|
};
|
|
|
|
|
|
|
|
T pop()
|
|
|
|
{
|
|
|
|
std::lock_guard<std::recursive_mutex> lk(mtx);
|
|
|
|
T val = std::priority_queue<T, Container, Compare>::top();
|
|
|
|
std::priority_queue<T, Container, Compare>::pop();
|
|
|
|
return val;
|
2022-01-10 15:36:14 +00:00
|
|
|
};
|
2023-11-23 19:56:36 +00:00
|
|
|
|
|
|
|
std::recursive_mutex& get_mtx()
|
|
|
|
{
|
|
|
|
return mtx;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::recursive_mutex mtx;
|
2022-01-10 15:36:14 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct TimerInfo {
|
|
|
|
uint32_t id;
|
|
|
|
uint64_t timeout_ns; // timer expiry
|
|
|
|
uint64_t interval_ns; // 0 for one-shot timers
|
|
|
|
timer_cb cb; // timer callback
|
|
|
|
} TimerInfo;
|
|
|
|
|
|
|
|
// Custom comparator for sorting our timer queue in ascending order
|
|
|
|
class MyGtComparator {
|
|
|
|
public:
|
2024-11-10 06:25:48 +00:00
|
|
|
bool operator()(const std::shared_ptr<TimerInfo>& l, const std::shared_ptr<TimerInfo>& r) {
|
2022-01-10 15:36:14 +00:00
|
|
|
return l.get()->timeout_ns > r.get()->timeout_ns;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class TimerManager {
|
|
|
|
public:
|
|
|
|
static TimerManager* get_instance() {
|
|
|
|
if (!timer_manager) {
|
|
|
|
timer_manager = new TimerManager();
|
|
|
|
}
|
|
|
|
return timer_manager;
|
|
|
|
};
|
|
|
|
|
2022-02-05 16:10:57 +00:00
|
|
|
// callback for retrieving current time
|
2024-11-10 06:25:48 +00:00
|
|
|
void set_time_now_cb(const std::function<uint64_t()> &cb) {
|
2022-01-10 15:36:14 +00:00
|
|
|
this->get_time_now = cb;
|
|
|
|
};
|
|
|
|
|
|
|
|
// callback for acknowledging time changes
|
|
|
|
void set_notify_changes_cb(const timer_cb &cb) {
|
|
|
|
this->notify_timer_changes = cb;
|
|
|
|
};
|
|
|
|
|
2022-02-13 02:02:17 +00:00
|
|
|
// return current virtual time in nanoseconds
|
|
|
|
uint64_t current_time_ns() { return get_time_now(); };
|
|
|
|
|
2022-01-10 15:36:14 +00:00
|
|
|
// creating and cancelling timers
|
|
|
|
uint32_t add_oneshot_timer(uint64_t timeout, timer_cb cb);
|
2023-11-03 09:30:32 +00:00
|
|
|
uint32_t add_immediate_timer(timer_cb cb);
|
2022-01-10 15:36:14 +00:00
|
|
|
uint32_t add_cyclic_timer(uint64_t interval, timer_cb cb);
|
2023-08-01 15:36:29 +00:00
|
|
|
uint32_t add_cyclic_timer(uint64_t interval, uint64_t delay, timer_cb cb);
|
2022-01-10 15:36:14 +00:00
|
|
|
void cancel_timer(uint32_t id);
|
|
|
|
|
2023-10-18 02:48:51 +00:00
|
|
|
uint64_t process_timers();
|
2022-01-10 15:36:14 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
static TimerManager* timer_manager;
|
|
|
|
TimerManager(){}; // private constructor to implement a singleton
|
|
|
|
|
|
|
|
// timer queue
|
2024-11-10 06:25:48 +00:00
|
|
|
my_priority_queue<std::shared_ptr<TimerInfo>, std::vector<std::shared_ptr<TimerInfo>>, MyGtComparator> timer_queue;
|
2022-01-10 15:36:14 +00:00
|
|
|
|
2024-11-10 06:25:48 +00:00
|
|
|
std::function<uint64_t()> get_time_now;
|
|
|
|
std::function<void()> notify_timer_changes;
|
2022-02-05 16:10:57 +00:00
|
|
|
|
2023-11-23 19:56:36 +00:00
|
|
|
std::atomic<uint32_t> id{0};
|
|
|
|
bool cb_active = false; // true if a timer callback is executing // FIXME: Do we need this? It gets written in main thread and read in audio thread.
|
2022-01-10 15:36:14 +00:00
|
|
|
};
|
|
|
|
|
2022-02-05 16:10:57 +00:00
|
|
|
#endif // TIMER_MANAGER_H
|