/* * Copyright (c) 2019-2020, Sergey Bugaev * Copyright (c) 2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include static pthread_mutex_t s_mutex = PTHREAD_MUTEX_INITIALIZER; static pthread_cond_t s_condition = PTHREAD_COND_INITIALIZER; static Queue>* s_all_actions; static Threading::Thread* s_background_thread; static intptr_t background_thread_func() { Vector> actions; while (true) { pthread_mutex_lock(&s_mutex); while (s_all_actions->is_empty()) pthread_cond_wait(&s_condition, &s_mutex); while (!s_all_actions->is_empty()) actions.append(s_all_actions->dequeue()); pthread_mutex_unlock(&s_mutex); for (auto& action : actions) action(); actions.clear(); } } static void init() { s_all_actions = new Queue>; s_background_thread = &Threading::Thread::construct(background_thread_func, "Background Thread"sv).leak_ref(); s_background_thread->start(); } Threading::Thread& Threading::BackgroundActionBase::background_thread() { if (s_background_thread == nullptr) init(); return *s_background_thread; } void Threading::BackgroundActionBase::enqueue_work(Function work) { if (s_all_actions == nullptr) init(); pthread_mutex_lock(&s_mutex); s_all_actions->enqueue(move(work)); pthread_cond_broadcast(&s_condition); pthread_mutex_unlock(&s_mutex); }