2023-11-18 14:50:43 +01:00
|
|
|
#include <hex/api/task_manager.hpp>
|
2021-12-16 23:48:52 +01:00
|
|
|
|
2023-11-21 14:38:01 +01:00
|
|
|
#include <hex/api/localization_manager.hpp>
|
2022-09-03 23:56:57 +02:00
|
|
|
#include <hex/helpers/logger.hpp>
|
2021-12-16 23:48:52 +01:00
|
|
|
|
2022-07-29 18:49:43 +02:00
|
|
|
#include <algorithm>
|
2024-01-10 20:13:53 +01:00
|
|
|
#include <ranges>
|
2022-07-29 18:49:43 +02:00
|
|
|
|
2023-12-22 23:39:38 +01:00
|
|
|
#include <jthread.hpp>
|
|
|
|
|
2023-06-21 09:29:57 +02:00
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
#include <windows.h>
|
|
|
|
#include <processthreadsapi.h>
|
2024-01-30 16:32:48 +01:00
|
|
|
#include <hex/helpers/utils.hpp>
|
2023-06-21 09:29:57 +02:00
|
|
|
#else
|
|
|
|
#include <pthread.h>
|
|
|
|
#endif
|
|
|
|
|
2024-02-23 17:47:40 +01:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
struct SourceLocationWrapper {
|
|
|
|
std::source_location location;
|
|
|
|
|
|
|
|
bool operator==(const SourceLocationWrapper &other) const {
|
|
|
|
return location.file_name() == other.location.file_name() &&
|
|
|
|
location.function_name() == other.location.function_name() &&
|
|
|
|
location.column() == other.location.column() &&
|
|
|
|
location.line() == other.location.line();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
template<>
|
|
|
|
struct std::hash<SourceLocationWrapper> {
|
|
|
|
std::size_t operator()(const SourceLocationWrapper& s) const noexcept {
|
|
|
|
auto h1 = std::hash<std::string>{}(s.location.file_name());
|
|
|
|
auto h2 = std::hash<std::string>{}(s.location.function_name());
|
|
|
|
auto h3 = std::hash<u32>{}(s.location.column());
|
|
|
|
auto h4 = std::hash<u32>{}(s.location.line());
|
|
|
|
return (h1 << 0) ^ (h2 << 1) ^ (h3 << 2) ^ (h4 << 3);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-12-16 23:48:52 +01:00
|
|
|
namespace hex {
|
|
|
|
|
2023-07-26 13:50:51 +02:00
|
|
|
namespace {
|
2022-08-17 16:15:36 +02:00
|
|
|
|
2024-02-23 17:47:40 +01:00
|
|
|
std::recursive_mutex s_deferredCallsMutex, s_tasksFinishedMutex;
|
2023-07-26 13:50:51 +02:00
|
|
|
|
|
|
|
std::list<std::shared_ptr<Task>> s_tasks, s_taskQueue;
|
|
|
|
std::list<std::function<void()>> s_deferredCalls;
|
2024-02-23 17:47:40 +01:00
|
|
|
std::unordered_map<SourceLocationWrapper, std::function<void()>> s_onceDeferredCalls;
|
2023-07-26 13:50:51 +02:00
|
|
|
std::list<std::function<void()>> s_tasksFinishedCallbacks;
|
|
|
|
|
|
|
|
std::mutex s_queueMutex;
|
|
|
|
std::condition_variable s_jobCondVar;
|
|
|
|
std::vector<std::jthread> s_workers;
|
|
|
|
|
2024-01-10 20:13:53 +01:00
|
|
|
thread_local std::array<char, 256> s_currentThreadName;
|
2024-01-21 18:39:13 +01:00
|
|
|
thread_local Task* s_currentTask = nullptr;
|
2024-01-09 21:46:54 +01:00
|
|
|
|
2023-07-26 13:50:51 +02:00
|
|
|
}
|
2022-08-17 16:15:36 +02:00
|
|
|
|
|
|
|
|
2024-08-03 11:32:17 +02:00
|
|
|
Task::Task(const UnlocalizedString &unlocalizedName, u64 maxValue, bool background, std::function<void(Task &)> function)
|
|
|
|
: m_unlocalizedName(std::move(unlocalizedName)), m_maxValue(maxValue), m_function(std::move(function)), m_background(background) { }
|
2022-08-17 16:15:36 +02:00
|
|
|
|
|
|
|
Task::Task(hex::Task &&other) noexcept {
|
2022-10-04 23:37:48 +02:00
|
|
|
{
|
2023-12-19 13:10:25 +01:00
|
|
|
std::scoped_lock thisLock(m_mutex);
|
2022-10-04 23:37:48 +02:00
|
|
|
std::scoped_lock otherLock(other.m_mutex);
|
2022-02-01 18:09:40 +01:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_function = std::move(other.m_function);
|
2024-08-03 11:32:17 +02:00
|
|
|
m_unlocalizedName = std::move(other.m_unlocalizedName);
|
2022-10-04 23:37:48 +02:00
|
|
|
}
|
2022-02-01 18:09:40 +01:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_maxValue = u64(other.m_maxValue);
|
|
|
|
m_currValue = u64(other.m_currValue);
|
2022-08-17 16:15:36 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_finished = bool(other.m_finished);
|
|
|
|
m_hadException = bool(other.m_hadException);
|
|
|
|
m_interrupted = bool(other.m_interrupted);
|
|
|
|
m_shouldInterrupt = bool(other.m_shouldInterrupt);
|
2021-12-16 23:48:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Task::~Task() {
|
2022-09-14 13:51:48 +02:00
|
|
|
if (!this->isFinished())
|
|
|
|
this->interrupt();
|
2021-12-16 23:48:52 +01:00
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
void Task::update(u64 value) {
|
2023-12-06 11:04:35 +01:00
|
|
|
// Update the current progress value of the task
|
2023-12-19 13:10:25 +01:00
|
|
|
m_currValue.store(value, std::memory_order_relaxed);
|
2022-07-29 18:49:43 +02:00
|
|
|
|
2023-12-06 11:04:35 +01:00
|
|
|
// Check if the task has been interrupted by the main thread and if yes,
|
|
|
|
// throw an exception that is generally not caught by the task
|
2023-12-19 13:10:25 +01:00
|
|
|
if (m_shouldInterrupt.load(std::memory_order_relaxed)) [[unlikely]]
|
2022-08-17 16:15:36 +02:00
|
|
|
throw TaskInterruptor();
|
|
|
|
}
|
|
|
|
|
2024-02-21 00:06:52 +01:00
|
|
|
void Task::update() const {
|
|
|
|
if (m_shouldInterrupt.load(std::memory_order_relaxed)) [[unlikely]]
|
|
|
|
throw TaskInterruptor();
|
|
|
|
}
|
|
|
|
|
2024-02-23 17:47:40 +01:00
|
|
|
void Task::increment() {
|
|
|
|
m_currValue.fetch_add(1, std::memory_order_relaxed);
|
|
|
|
|
|
|
|
if (m_shouldInterrupt.load(std::memory_order_relaxed)) [[unlikely]]
|
|
|
|
throw TaskInterruptor();
|
|
|
|
}
|
|
|
|
|
2024-02-21 00:06:52 +01:00
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
void Task::setMaxValue(u64 value) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_maxValue = value;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Task::interrupt() {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_shouldInterrupt = true;
|
2022-09-13 16:05:41 +02:00
|
|
|
|
2023-12-06 11:04:35 +01:00
|
|
|
// Call the interrupt callback on the current thread if one is set
|
2023-12-19 13:10:25 +01:00
|
|
|
if (m_interruptCallback)
|
|
|
|
m_interruptCallback();
|
2022-09-13 16:05:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Task::setInterruptCallback(std::function<void()> callback) {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_interruptCallback = std::move(callback);
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
2022-10-04 23:37:48 +02:00
|
|
|
bool Task::isBackgroundTask() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_background;
|
2022-09-19 16:09:22 +02:00
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
bool Task::isFinished() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_finished;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Task::hadException() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_hadException;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
2022-10-28 14:32:28 +02:00
|
|
|
bool Task::shouldInterrupt() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_shouldInterrupt;
|
2022-10-28 14:32:28 +02:00
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
bool Task::wasInterrupted() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_interrupted;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Task::clearException() {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_hadException = false;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
2022-09-03 23:56:57 +02:00
|
|
|
std::string Task::getExceptionMessage() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
std::scoped_lock lock(m_mutex);
|
2022-09-03 23:56:57 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_exceptionMessage;
|
2022-09-03 23:56:57 +02:00
|
|
|
}
|
|
|
|
|
2024-08-03 11:32:17 +02:00
|
|
|
const UnlocalizedString &Task::getUnlocalizedName() {
|
|
|
|
return m_unlocalizedName;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
u64 Task::getValue() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_currValue;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
u64 Task::getMaxValue() const {
|
2023-12-19 13:10:25 +01:00
|
|
|
return m_maxValue;
|
2022-07-29 18:49:43 +02:00
|
|
|
}
|
|
|
|
|
2021-12-16 23:48:52 +01:00
|
|
|
void Task::finish() {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_finished = true;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Task::interruption() {
|
2023-12-19 13:10:25 +01:00
|
|
|
m_interrupted = true;
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
2022-09-03 23:56:57 +02:00
|
|
|
void Task::exception(const char *message) {
|
2023-12-19 13:10:25 +01:00
|
|
|
std::scoped_lock lock(m_mutex);
|
2022-02-01 18:09:40 +01:00
|
|
|
|
2023-12-06 11:04:35 +01:00
|
|
|
// Store information about the caught exception
|
2023-12-19 13:10:25 +01:00
|
|
|
m_exceptionMessage = message;
|
|
|
|
m_hadException = true;
|
2021-12-16 23:48:52 +01:00
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
|
|
|
|
bool TaskHolder::isRunning() const {
|
2024-01-30 11:21:34 +01:00
|
|
|
const auto &task = m_task.lock();
|
2023-12-06 11:04:35 +01:00
|
|
|
if (!task)
|
2022-10-05 12:02:40 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return !task->isFinished();
|
2022-01-09 21:27:59 +01:00
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
bool TaskHolder::hadException() const {
|
2024-01-30 11:21:34 +01:00
|
|
|
const auto &task = m_task.lock();
|
2023-12-06 11:04:35 +01:00
|
|
|
if (!task)
|
2022-10-05 12:02:40 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return !task->hadException();
|
2021-12-16 23:48:52 +01:00
|
|
|
}
|
|
|
|
|
2022-10-28 14:32:28 +02:00
|
|
|
bool TaskHolder::shouldInterrupt() const {
|
2024-01-30 11:21:34 +01:00
|
|
|
const auto &task = m_task.lock();
|
2023-12-06 11:04:35 +01:00
|
|
|
if (!task)
|
2022-10-28 14:32:28 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return !task->shouldInterrupt();
|
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
bool TaskHolder::wasInterrupted() const {
|
2024-01-30 11:21:34 +01:00
|
|
|
const auto &task = m_task.lock();
|
2023-12-06 11:04:35 +01:00
|
|
|
if (!task)
|
2022-10-05 12:02:40 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return !task->wasInterrupted();
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
2022-01-09 21:27:59 +01:00
|
|
|
|
2023-11-10 20:47:08 +01:00
|
|
|
void TaskHolder::interrupt() const {
|
2024-01-30 11:21:34 +01:00
|
|
|
const auto &task = m_task.lock();
|
2023-12-06 11:04:35 +01:00
|
|
|
if (!task)
|
2022-10-05 12:02:40 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
task->interrupt();
|
2021-12-16 23:48:52 +01:00
|
|
|
}
|
|
|
|
|
2023-04-08 00:58:53 +02:00
|
|
|
u32 TaskHolder::getProgress() const {
|
2024-01-30 11:21:34 +01:00
|
|
|
const auto &task = m_task.lock();
|
2023-12-06 11:04:35 +01:00
|
|
|
if (!task)
|
2024-01-30 11:21:34 +01:00
|
|
|
return 0;
|
2023-12-06 11:04:35 +01:00
|
|
|
|
|
|
|
// If the max value is 0, the task has no progress
|
2023-12-05 14:32:28 +01:00
|
|
|
if (task->getMaxValue() == 0)
|
|
|
|
return 0;
|
|
|
|
|
2023-12-06 11:04:35 +01:00
|
|
|
// Calculate the progress of the task from 0 to 100
|
2023-11-10 20:47:08 +01:00
|
|
|
return u32((task->getValue() * 100) / task->getMaxValue());
|
2023-04-08 00:58:53 +02:00
|
|
|
}
|
|
|
|
|
2022-10-04 23:37:48 +02:00
|
|
|
void TaskManager::init() {
|
2023-06-06 11:29:58 +02:00
|
|
|
const auto threadCount = std::thread::hardware_concurrency();
|
|
|
|
|
|
|
|
log::debug("Initializing task manager thread pool with {} workers.", threadCount);
|
|
|
|
|
2023-12-06 11:04:35 +01:00
|
|
|
// Create worker threads
|
2023-12-22 23:39:38 +01:00
|
|
|
for (u32 i = 0; i < threadCount; i++) {
|
|
|
|
s_workers.emplace_back([](const std::stop_token &stopToken) {
|
|
|
|
while (true) {
|
|
|
|
std::shared_ptr<Task> task;
|
|
|
|
|
|
|
|
// Set the thread name to "Idle Task" while waiting for a task
|
2024-01-09 10:39:06 +01:00
|
|
|
TaskManager::setCurrentThreadName("Idle Task");
|
2023-12-22 23:39:38 +01:00
|
|
|
|
|
|
|
{
|
|
|
|
// Wait for a task to be added to the queue
|
|
|
|
std::unique_lock lock(s_queueMutex);
|
|
|
|
s_jobCondVar.wait(lock, [&] {
|
|
|
|
return !s_taskQueue.empty() || stopToken.stop_requested();
|
|
|
|
});
|
|
|
|
|
|
|
|
// Check if the thread should exit
|
|
|
|
if (stopToken.stop_requested())
|
|
|
|
break;
|
|
|
|
|
|
|
|
// Grab the next task from the queue
|
|
|
|
task = std::move(s_taskQueue.front());
|
|
|
|
s_taskQueue.pop_front();
|
2024-01-21 18:39:13 +01:00
|
|
|
|
|
|
|
s_currentTask = task.get();
|
2023-12-22 23:39:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Set the thread name to the name of the task
|
2024-08-03 11:32:17 +02:00
|
|
|
TaskManager::setCurrentThreadName(Lang(task->m_unlocalizedName));
|
2023-12-22 23:39:38 +01:00
|
|
|
|
|
|
|
// Execute the task
|
|
|
|
task->m_function(*task);
|
|
|
|
|
2024-08-03 11:32:17 +02:00
|
|
|
log::debug("Task '{}' finished", task->m_unlocalizedName.get());
|
2023-12-22 23:39:38 +01:00
|
|
|
} catch (const Task::TaskInterruptor &) {
|
|
|
|
// Handle the task being interrupted by user request
|
|
|
|
task->interruption();
|
|
|
|
} catch (const std::exception &e) {
|
2024-08-03 11:32:17 +02:00
|
|
|
log::error("Exception in task '{}': {}", task->m_unlocalizedName.get(), e.what());
|
2023-12-22 23:39:38 +01:00
|
|
|
|
|
|
|
// Handle the task throwing an uncaught exception
|
|
|
|
task->exception(e.what());
|
|
|
|
} catch (...) {
|
2024-08-03 11:32:17 +02:00
|
|
|
log::error("Exception in task '{}'", task->m_unlocalizedName.get());
|
2023-12-22 23:39:38 +01:00
|
|
|
|
|
|
|
// Handle the task throwing an uncaught exception of unknown type
|
|
|
|
task->exception("Unknown Exception");
|
|
|
|
}
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
s_currentTask = nullptr;
|
2023-12-22 23:39:38 +01:00
|
|
|
task->finish();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2022-10-04 23:37:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void TaskManager::exit() {
|
2023-12-06 11:04:35 +01:00
|
|
|
// Interrupt all tasks
|
2024-01-30 11:21:34 +01:00
|
|
|
for (const auto &task : s_tasks) {
|
2022-10-04 23:37:48 +02:00
|
|
|
task->interrupt();
|
2023-12-06 11:04:35 +01:00
|
|
|
}
|
2022-10-04 23:37:48 +02:00
|
|
|
|
2023-12-06 11:04:35 +01:00
|
|
|
// Ask worker threads to exit after finishing their task
|
2023-07-26 13:50:51 +02:00
|
|
|
for (auto &thread : s_workers)
|
2022-10-04 23:37:48 +02:00
|
|
|
thread.request_stop();
|
|
|
|
|
2023-12-06 11:04:35 +01:00
|
|
|
// Wake up all the idle worker threads so they can exit
|
2022-10-04 23:37:48 +02:00
|
|
|
s_jobCondVar.notify_all();
|
|
|
|
|
2023-12-06 11:04:35 +01:00
|
|
|
// Wait for all worker threads to exit
|
2023-07-26 13:50:51 +02:00
|
|
|
s_workers.clear();
|
2023-12-06 11:04:35 +01:00
|
|
|
|
|
|
|
s_tasks.clear();
|
|
|
|
s_taskQueue.clear();
|
2024-01-04 21:33:17 +01:00
|
|
|
|
|
|
|
s_deferredCalls.clear();
|
2024-02-23 17:47:40 +01:00
|
|
|
s_onceDeferredCalls.clear();
|
2024-01-04 21:33:17 +01:00
|
|
|
s_tasksFinishedCallbacks.clear();
|
2022-10-04 23:37:48 +02:00
|
|
|
}
|
|
|
|
|
2024-08-03 11:32:17 +02:00
|
|
|
TaskHolder TaskManager::createTask(const UnlocalizedString &unlocalizedName, u64 maxValue, bool background, std::function<void(Task&)> function) {
|
2023-12-06 11:04:35 +01:00
|
|
|
std::scoped_lock lock(s_queueMutex);
|
|
|
|
|
|
|
|
// Construct new task
|
2024-08-03 11:32:17 +02:00
|
|
|
auto task = std::make_shared<Task>(std::move(unlocalizedName), maxValue, background, std::move(function));
|
2023-12-06 11:04:35 +01:00
|
|
|
|
2022-10-05 21:25:52 +02:00
|
|
|
s_tasks.emplace_back(task);
|
2023-12-06 11:04:35 +01:00
|
|
|
|
|
|
|
// Add task to the queue for the worker to pick up
|
|
|
|
s_taskQueue.emplace_back(std::move(task));
|
2022-10-05 21:25:52 +02:00
|
|
|
|
2022-10-04 23:37:48 +02:00
|
|
|
s_jobCondVar.notify_one();
|
2022-08-17 16:15:36 +02:00
|
|
|
|
|
|
|
return TaskHolder(s_tasks.back());
|
2022-01-09 21:27:59 +01:00
|
|
|
}
|
|
|
|
|
2022-10-05 21:25:52 +02:00
|
|
|
|
2024-08-03 11:32:17 +02:00
|
|
|
TaskHolder TaskManager::createTask(const UnlocalizedString &unlocalizedName, u64 maxValue, std::function<void(Task &)> function) {
|
|
|
|
log::debug("Creating task {}", unlocalizedName.get());
|
|
|
|
return createTask(std::move(unlocalizedName), maxValue, false, std::move(function));
|
2023-12-06 11:04:35 +01:00
|
|
|
}
|
2022-09-19 16:09:22 +02:00
|
|
|
|
2024-08-03 22:01:18 +02:00
|
|
|
TaskHolder TaskManager::createTask(const UnlocalizedString &unlocalizedName, u64 maxValue, std::function<void()> function) {
|
|
|
|
log::debug("Creating task {}", unlocalizedName.get());
|
|
|
|
return createTask(std::move(unlocalizedName), maxValue, false,
|
|
|
|
[function = std::move(function)](Task&) {
|
|
|
|
function();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-08-03 11:32:17 +02:00
|
|
|
TaskHolder TaskManager::createBackgroundTask(const UnlocalizedString &unlocalizedName, std::function<void(Task &)> function) {
|
|
|
|
log::debug("Creating background task {}", unlocalizedName.get());
|
|
|
|
return createTask(std::move(unlocalizedName), 0, true, std::move(function));
|
2022-09-19 16:09:22 +02:00
|
|
|
}
|
|
|
|
|
2024-08-03 22:01:18 +02:00
|
|
|
TaskHolder TaskManager::createBackgroundTask(const UnlocalizedString &unlocalizedName, std::function<void()> function) {
|
|
|
|
log::debug("Creating background task {}", unlocalizedName.get());
|
|
|
|
return createTask(std::move(unlocalizedName), 0, true,
|
|
|
|
[function = std::move(function)](Task&) {
|
|
|
|
function();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
void TaskManager::collectGarbage() {
|
2023-04-06 12:44:25 +02:00
|
|
|
{
|
2023-12-06 11:04:35 +01:00
|
|
|
std::scoped_lock lock(s_queueMutex);
|
|
|
|
std::erase_if(s_tasks, [](const auto &task) {
|
|
|
|
return task->isFinished() && !task->hadException();
|
|
|
|
});
|
2023-04-06 12:44:25 +02:00
|
|
|
}
|
2022-12-27 11:35:20 +01:00
|
|
|
|
|
|
|
if (s_tasks.empty()) {
|
2023-12-06 11:04:35 +01:00
|
|
|
std::scoped_lock lock(s_deferredCallsMutex);
|
2022-12-27 11:35:20 +01:00
|
|
|
for (auto &call : s_tasksFinishedCallbacks)
|
|
|
|
call();
|
|
|
|
s_tasksFinishedCallbacks.clear();
|
|
|
|
}
|
2023-08-26 12:54:52 +02:00
|
|
|
|
2021-12-16 23:48:52 +01:00
|
|
|
}
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
Task& TaskManager::getCurrentTask() {
|
|
|
|
return *s_currentTask;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-02-10 23:31:05 +01:00
|
|
|
const std::list<std::shared_ptr<Task>>& TaskManager::getRunningTasks() {
|
2022-08-17 16:15:36 +02:00
|
|
|
return s_tasks;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t TaskManager::getRunningTaskCount() {
|
2023-12-06 11:04:35 +01:00
|
|
|
std::scoped_lock lock(s_queueMutex);
|
2022-10-04 23:37:48 +02:00
|
|
|
|
2024-01-30 11:21:34 +01:00
|
|
|
return std::ranges::count_if(s_tasks, [](const auto &task){
|
2022-10-04 23:37:48 +02:00
|
|
|
return !task->isBackgroundTask();
|
|
|
|
});
|
2022-08-17 16:15:36 +02:00
|
|
|
}
|
|
|
|
|
2022-12-27 11:35:20 +01:00
|
|
|
size_t TaskManager::getRunningBackgroundTaskCount() {
|
2023-12-06 11:04:35 +01:00
|
|
|
std::scoped_lock lock(s_queueMutex);
|
2022-12-27 11:35:20 +01:00
|
|
|
|
2024-01-30 11:21:34 +01:00
|
|
|
return std::ranges::count_if(s_tasks, [](const auto &task){
|
2022-12-27 11:35:20 +01:00
|
|
|
return task->isBackgroundTask();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
|
|
|
|
void TaskManager::doLater(const std::function<void()> &function) {
|
|
|
|
std::scoped_lock lock(s_deferredCallsMutex);
|
|
|
|
|
|
|
|
s_deferredCalls.push_back(function);
|
|
|
|
}
|
|
|
|
|
2024-02-23 17:47:40 +01:00
|
|
|
void TaskManager::doLaterOnce(const std::function<void()> &function, std::source_location location) {
|
|
|
|
std::scoped_lock lock(s_deferredCallsMutex);
|
|
|
|
|
|
|
|
s_onceDeferredCalls[SourceLocationWrapper{ location }] = function;
|
|
|
|
}
|
|
|
|
|
2022-08-17 16:15:36 +02:00
|
|
|
void TaskManager::runDeferredCalls() {
|
|
|
|
std::scoped_lock lock(s_deferredCallsMutex);
|
|
|
|
|
2024-03-01 20:54:27 +01:00
|
|
|
while (!s_deferredCalls.empty()) {
|
|
|
|
auto callback = s_deferredCalls.front();
|
|
|
|
s_deferredCalls.pop_front();
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
while (!s_onceDeferredCalls.empty()) {
|
|
|
|
auto node = s_onceDeferredCalls.extract(s_onceDeferredCalls.begin());
|
|
|
|
node.mapped()();
|
|
|
|
}
|
2022-02-01 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2022-12-27 11:35:20 +01:00
|
|
|
void TaskManager::runWhenTasksFinished(const std::function<void()> &function) {
|
2022-12-28 23:06:27 +01:00
|
|
|
std::scoped_lock lock(s_tasksFinishedMutex);
|
2022-12-27 11:35:20 +01:00
|
|
|
|
2024-05-17 21:56:43 +02:00
|
|
|
for (const auto &task : s_tasks) {
|
|
|
|
task->interrupt();
|
|
|
|
}
|
|
|
|
|
2022-12-27 11:35:20 +01:00
|
|
|
s_tasksFinishedCallbacks.push_back(function);
|
|
|
|
}
|
|
|
|
|
2024-01-09 10:39:06 +01:00
|
|
|
void TaskManager::setCurrentThreadName(const std::string &name) {
|
2024-01-10 20:13:53 +01:00
|
|
|
std::ranges::fill(s_currentThreadName, '\0');
|
|
|
|
std::ranges::copy(name | std::views::take(255), s_currentThreadName.begin());
|
|
|
|
|
2024-01-09 10:39:06 +01:00
|
|
|
#if defined(OS_WINDOWS)
|
2024-01-09 13:43:34 +01:00
|
|
|
using SetThreadDescriptionFunc = HRESULT(WINAPI*)(HANDLE hThread, PCWSTR lpThreadDescription);
|
|
|
|
|
|
|
|
static auto setThreadDescription = reinterpret_cast<SetThreadDescriptionFunc>(
|
|
|
|
reinterpret_cast<uintptr_t>(
|
|
|
|
::GetProcAddress(
|
2024-06-28 11:12:17 +02:00
|
|
|
::GetModuleHandleW(L"Kernel32.dll"),
|
2024-01-09 13:43:34 +01:00
|
|
|
"SetThreadDescription"
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (setThreadDescription != nullptr) {
|
|
|
|
const auto longName = hex::utf8ToUtf16(name);
|
|
|
|
setThreadDescription(::GetCurrentThread(), longName.c_str());
|
|
|
|
} else {
|
|
|
|
struct THREADNAME_INFO {
|
|
|
|
DWORD dwType;
|
|
|
|
LPCSTR szName;
|
|
|
|
DWORD dwThreadID;
|
|
|
|
DWORD dwFlags;
|
|
|
|
};
|
|
|
|
|
|
|
|
THREADNAME_INFO info = { };
|
|
|
|
info.dwType = 0x1000;
|
|
|
|
info.szName = name.c_str();
|
|
|
|
info.dwThreadID = ::GetCurrentThreadId();
|
|
|
|
info.dwFlags = 0;
|
|
|
|
|
|
|
|
constexpr static DWORD MS_VC_EXCEPTION = 0x406D1388;
|
|
|
|
RaiseException(MS_VC_EXCEPTION, 0, sizeof(info) / sizeof(ULONG_PTR), reinterpret_cast<ULONG_PTR*>(&info));
|
|
|
|
}
|
2024-01-09 10:39:06 +01:00
|
|
|
#elif defined(OS_LINUX)
|
|
|
|
pthread_setname_np(pthread_self(), name.c_str());
|
|
|
|
#elif defined(OS_WEB)
|
|
|
|
hex::unused(name);
|
|
|
|
#elif defined(OS_MACOS)
|
|
|
|
pthread_setname_np(name.c_str());
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string TaskManager::getCurrentThreadName() {
|
2024-01-10 20:13:53 +01:00
|
|
|
return s_currentThreadName.data();
|
2024-01-09 10:39:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-05-20 13:38:12 +02:00
|
|
|
}
|