// Copyright (C) 2019-2020 Joel Rosdahl and other contributors // // See doc/AUTHORS.adoc for a complete list of contributors. // // 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, write to the Free Software Foundation, Inc., 51 // Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA #include "ThreadPool.hpp" ThreadPool::ThreadPool(size_t number_of_threads, size_t task_queue_max_size) : m_task_queue_max_size(task_queue_max_size) { m_worker_threads.reserve(number_of_threads); for (size_t i = 0; i < number_of_threads; ++i) { m_worker_threads.emplace_back(&ThreadPool::worker_thread_main, this); } } ThreadPool::~ThreadPool() { shut_down(); } void ThreadPool::enqueue(std::function function) { { std::unique_lock lock(m_mutex); if (m_task_queue.size() >= m_task_queue_max_size) { m_task_popped_condition.wait( lock, [this] { return m_task_queue.size() < m_task_queue_max_size; }); } m_task_queue.emplace(function); } m_task_enqueued_or_shutting_down_condition.notify_one(); } void ThreadPool::shut_down() { { std::unique_lock lock(m_mutex); if (m_shutting_down) { // Already called shut_down. return; } m_shutting_down = true; } m_task_enqueued_or_shutting_down_condition.notify_all(); for (auto& thread : m_worker_threads) { if (thread.joinable()) { thread.join(); } } } void ThreadPool::worker_thread_main() { while (true) { std::function task; { std::unique_lock lock(m_mutex); m_task_enqueued_or_shutting_down_condition.wait( lock, [this] { return m_shutting_down || !m_task_queue.empty(); }); if (m_shutting_down && m_task_queue.empty()) { return; } task = std::move(m_task_queue.front()); m_task_queue.pop(); } m_task_popped_condition.notify_all(); task(); } }