mirror of
				https://github.com/Icinga/icinga2.git
				synced 2025-10-25 01:03:54 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			116 lines
		
	
	
		
			1.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			116 lines
		
	
	
		
			1.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* Icinga 2 | (c) 2012 Icinga GmbH | GPLv2+ */
 | |
| 
 | |
| #ifndef THREADPOOL_H
 | |
| #define THREADPOOL_H
 | |
| 
 | |
| #include "base/i2-base.hpp"
 | |
| #include <boost/thread/thread.hpp>
 | |
| #include <boost/thread/mutex.hpp>
 | |
| #include <boost/thread/condition_variable.hpp>
 | |
| #include <deque>
 | |
| #include <thread>
 | |
| 
 | |
| namespace icinga
 | |
| {
 | |
| 
 | |
| #define QUEUECOUNT 4U
 | |
| 
 | |
| enum SchedulerPolicy
 | |
| {
 | |
| 	DefaultScheduler,
 | |
| 	LowLatencyScheduler
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * A thread pool.
 | |
|  *
 | |
|  * @ingroup base
 | |
|  */
 | |
| class ThreadPool
 | |
| {
 | |
| public:
 | |
| 	typedef std::function<void ()> WorkFunction;
 | |
| 
 | |
| 	ThreadPool(size_t max_threads = UINT_MAX);
 | |
| 	~ThreadPool();
 | |
| 
 | |
| 	void Start();
 | |
| 	void Stop();
 | |
| 
 | |
| 	bool Post(const WorkFunction& callback, SchedulerPolicy policy = DefaultScheduler);
 | |
| 
 | |
| private:
 | |
| 	enum ThreadState
 | |
| 	{
 | |
| 		ThreadUnspecified,
 | |
| 		ThreadDead,
 | |
| 		ThreadIdle,
 | |
| 		ThreadBusy
 | |
| 	};
 | |
| 
 | |
| 	struct WorkItem
 | |
| 	{
 | |
| 		WorkFunction Callback;
 | |
| 		double Timestamp;
 | |
| 	};
 | |
| 
 | |
| 	struct Queue;
 | |
| 
 | |
| 	struct WorkerThread
 | |
| 	{
 | |
| 		ThreadState State{ThreadDead};
 | |
| 		bool Zombie{false};
 | |
| 		double Utilization{0};
 | |
| 		double LastUpdate{0};
 | |
| 		boost::thread *Thread{nullptr};
 | |
| 
 | |
| 		WorkerThread(ThreadState state = ThreadDead)
 | |
| 			: State(state)
 | |
| 		{ }
 | |
| 
 | |
| 		void UpdateUtilization(ThreadState state = ThreadUnspecified);
 | |
| 
 | |
| 		void ThreadProc(Queue& queue);
 | |
| 	};
 | |
| 
 | |
| 	struct Queue
 | |
| 	{
 | |
| 		boost::mutex Mutex;
 | |
| 		boost::condition_variable CV;
 | |
| 		boost::condition_variable CVStarved;
 | |
| 
 | |
| 		std::deque<WorkItem> Items;
 | |
| 
 | |
| 		double WaitTime{0};
 | |
| 		double ServiceTime{0};
 | |
| 		int TaskCount{0};
 | |
| 
 | |
| 		bool Stopped{false};
 | |
| 
 | |
| 		WorkerThread Threads[16];
 | |
| 
 | |
| 		void SpawnWorker(boost::thread_group& group);
 | |
| 		void KillWorker(boost::thread_group& group);
 | |
| 	};
 | |
| 
 | |
| 	int m_ID;
 | |
| 	static int m_NextID;
 | |
| 
 | |
| 	size_t m_MaxThreads;
 | |
| 
 | |
| 	boost::thread_group m_ThreadGroup;
 | |
| 
 | |
| 	std::thread m_MgmtThread;
 | |
| 	boost::mutex m_MgmtMutex;
 | |
| 	boost::condition_variable m_MgmtCV;
 | |
| 	bool m_Stopped{true};
 | |
| 
 | |
| 	Queue m_Queues[QUEUECOUNT];
 | |
| 
 | |
| 	void ManagerThreadProc();
 | |
| };
 | |
| 
 | |
| }
 | |
| 
 | |
| #endif /* EVENTQUEUE_H */
 |