#pragma once #include #include #include #include #include #include "sync/lockable.hpp" class Pool : Lockable { using task_t = std::function; public: using sptr = std::shared_ptr; Pool(size_t n = std::thread::hardware_concurrency()) : alive(true) { threads.reserve(n); for(size_t i = 0; i < n; ++i) threads.emplace_back([this]()->void { loop(); }); } Pool(Pool&) = delete; Pool(Pool&&) = delete; ~Pool() { alive.store(false, std::memory_order_seq_cst); cond.notify_all(); for(auto& thread : threads) thread.join(); } void run(task_t f) { { auto lock = acquire_unique(); tasks.push(f); } cond.notify_one(); } private: std::vector threads; std::queue tasks; std::atomic alive; std::mutex mutex; std::condition_variable cond; void loop() { while(true) { task_t task; { auto lock = acquire_unique(); cond.wait(lock, [this] { return !this->alive || !this->tasks.empty(); }); if(!alive && tasks.empty()) return; task = std::move(tasks.front()); tasks.pop(); } task(); } } };