#pragma once #include "moodycamel/blockingconcurrentqueue.h" #include #include #include #include #include #include #include #include #include #include #include struct ThreadPoolTraits : public moodycamel::ConcurrentQueueDefaultTraits { static const int MAX_SEMA_SPINS = 1; }; class ThreadPool { private: using Func = std::function; using Queue = moodycamel::BlockingConcurrentQueue; public: explicit ThreadPool(std::size_t const thread_count_ = std::thread::hardware_concurrency(), std::size_t const queue_depth_ = 1, std::string const name_ = {}) : _queue(queue_depth_,thread_count_,thread_count_), _name(get_thread_name(name_)) { syslog(LOG_DEBUG, "threadpool: spawning %zu threads of queue depth %zu named '%s'", thread_count_, queue_depth_, _name.c_str()); sigset_t oldset; sigset_t newset; sigfillset(&newset); pthread_sigmask(SIG_BLOCK,&newset,&oldset); _threads.reserve(thread_count_); for(std::size_t i = 0; i < thread_count_; ++i) { int rv; pthread_t t; rv = pthread_create(&t,NULL,ThreadPool::start_routine,this); if(rv != 0) { syslog(LOG_WARNING, "threadpool: error spawning thread - %d (%s)", rv, strerror(rv)); continue; } if(!_name.empty()) pthread_setname_np(t,_name.c_str()); _threads.push_back(t); } pthread_sigmask(SIG_SETMASK,&oldset,NULL); if(_threads.empty()) throw std::runtime_error("threadpool: failed to spawn any threads"); } ~ThreadPool() { syslog(LOG_DEBUG, "threadpool: destroying %zu threads named '%s'", _threads.size(), _name.c_str()); auto func = []() { pthread_exit(NULL); }; for(std::size_t i = 0; i < _threads.size(); i++) _queue.enqueue(func); for(auto t : _threads) pthread_cancel(t); for(auto t : _threads) pthread_join(t,NULL); } private: static std::string get_thread_name(std::string const name_) { if(!name_.empty()) return name_; char name[16]; pthread_getname_np(pthread_self(),name,sizeof(name)); return name; } static void* start_routine(void *arg_) { ThreadPool *btp = static_cast(arg_); ThreadPool::Func func; ThreadPool::Queue &q = btp->_queue; moodycamel::ConsumerToken ctok(btp->_queue); while(true) { q.wait_dequeue(ctok,func); func(); } return NULL; } public: int add_thread(std::string const name_ = {}) { int rv; pthread_t t; sigset_t oldset; sigset_t newset; std::string name; name = (name_.empty() ? _name : name_); sigfillset(&newset); pthread_sigmask(SIG_BLOCK,&newset,&oldset); rv = pthread_create(&t,NULL,ThreadPool::start_routine,this); pthread_sigmask(SIG_SETMASK,&oldset,NULL); if(rv != 0) { syslog(LOG_WARNING, "threadpool: error spawning thread - %d (%s)", rv, strerror(rv)); return -rv; } if(!name.empty()) pthread_setname_np(t,name.c_str()); { std::lock_guard lg(_threads_mutex); _threads.push_back(t); } syslog(LOG_DEBUG, "threadpool: 1 thread added to pool '%s' named '%s'", _name.c_str(), name.c_str()); return 0; } int remove_thread(void) { { std::lock_guard lg(_threads_mutex); if(_threads.size() <= 1) return -EINVAL; } std::promise promise; auto func = [&]() { pthread_t t; t = pthread_self(); promise.set_value(t); { std::lock_guard lg(_threads_mutex); for(auto i = _threads.begin(); i != _threads.end(); ++i) { if(*i != t) continue; _threads.erase(i); break; } } char name[16]; pthread_getname_np(t,name,sizeof(name)); syslog(LOG_DEBUG, "threadpool: 1 thread removed from pool '%s' named '%s'", _name.c_str(), name); pthread_exit(NULL); }; enqueue_work(func); pthread_join(promise.get_future().get(),NULL); return 0; } int set_threads(std::size_t const count_) { int diff; { std::lock_guard lg(_threads_mutex); diff = ((int)count_ - (int)_threads.size()); } for(auto i = diff; i > 0; --i) add_thread(); for(auto i = diff; i < 0; ++i) remove_thread(); return diff; } public: template void enqueue_work(moodycamel::ProducerToken &ptok_, FuncType &&f_) { timespec ts = {0,10}; while(true) { if(_queue.try_enqueue(ptok_,f_)) return; ::nanosleep(&ts,NULL); ts.tv_nsec += 10; } } template void enqueue_work(FuncType &&f_) { timespec ts = {0,10}; while(true) { if(_queue.try_enqueue(f_)) return; ::nanosleep(&ts,NULL); ts.tv_nsec += 10; } } template [[nodiscard]] std::future::type> enqueue_task(FuncType&& f_) { using TaskReturnType = typename std::result_of::type; using Promise = std::promise; auto promise = std::make_shared(); auto future = promise->get_future(); auto work = [=]() { auto rv = f_(); promise->set_value(rv); }; timespec ts = {0,10}; while(true) { if(_queue.try_enqueue(work)) break; ::nanosleep(&ts,NULL); ts.tv_nsec += 10; } return future; } public: std::vector threads() const { std::lock_guard lg(_threads_mutex); return _threads; } moodycamel::ProducerToken ptoken() { return moodycamel::ProducerToken(_queue); } private: Queue _queue; private: std::string const _name; std::vector _threads; mutable std::mutex _threads_mutex; };