/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ #pragma once #include "cmConfigure.h" // IWYU pragma: keep #include <cstdint> #include <string> #include <utility> #include <vector> #include <cm/memory> // -- Types class cmWorkerPoolInternal; /** @class cmWorkerPool * @brief Thread pool with job queue */ class cmWorkerPool { public: /** * Return value and output of an external process. */ struct ProcessResultT { void reset(); bool error() const { return (this->ExitStatus != 0) || (this->TermSignal != 0) || !this->ErrorMessage.empty(); } std::int64_t ExitStatus = 0; int TermSignal = 0; std::string StdOut; std::string StdErr; std::string ErrorMessage; }; /** * Abstract job class for concurrent job processing. */ class JobT { public: JobT(JobT const&) = delete; JobT& operator=(JobT const&) = delete; /** * Virtual destructor. */ virtual ~JobT(); /** * Fence job flag * * Fence jobs require that: * - all jobs before in the queue have been processed * - no jobs later in the queue will be processed before this job was * processed */ bool IsFence() const { return this->Fence_; } protected: /** * Protected default constructor */ JobT(bool fence = false) : Fence_(fence) { } /** * Abstract processing interface that must be implement in derived classes. */ virtual void Process() = 0; /** * Get the worker pool. * Only valid during the JobT::Process() call! */ cmWorkerPool* Pool() const { return this->Pool_; } /** * Get the user data. * Only valid during the JobT::Process() call! */ void* UserData() const { return this->Pool_->UserData(); } /** * Get the worker index. * This is the index of the thread processing this job and is in the range * [0..ThreadCount). * Concurrently processing jobs will never have the same WorkerIndex(). * Only valid during the JobT::Process() call! */ unsigned int WorkerIndex() const { return this->WorkerIndex_; } /** * Run an external read only process. * Use only during JobT::Process() call! */ bool RunProcess(ProcessResultT& result, std::vector<std::string> const& command, std::string const& workingDirectory); private: //! Needs access to Work() friend class cmWorkerPoolInternal; //! Worker thread entry method. void Work(cmWorkerPool* pool, unsigned int workerIndex) { this->Pool_ = pool; this->WorkerIndex_ = workerIndex; this->Process(); } cmWorkerPool* Pool_ = nullptr; unsigned int WorkerIndex_ = 0; bool Fence_ = false; }; /** * Job handle type */ using JobHandleT = std::unique_ptr<JobT>; /** * Fence job base class */ class JobFenceT : public JobT { public: JobFenceT() : JobT(true) { } //! Does nothing void Process() override {} }; /** * Fence job that aborts the worker pool. * * Useful as the last job in the job queue. */ class JobEndT : JobFenceT { public: //! Does nothing void Process() override { this->Pool()->Abort(); } }; // -- Methods cmWorkerPool(); ~cmWorkerPool(); /** * Number of worker threads. */ unsigned int ThreadCount() const { return this->ThreadCount_; } /** * Set the number of worker threads. * * Calling this method during Process() has no effect. */ void SetThreadCount(unsigned int threadCount); /** * Blocking function that starts threads to process all Jobs in the queue. * * This method blocks until a job calls the Abort() method. * @arg threadCount Number of threads to process jobs. * @arg userData Common user data pointer available in all Jobs. */ bool Process(void* userData = nullptr); /** * User data reference passed to Process(). * * Only valid during Process(). */ void* UserData() const { return this->UserData_; } // -- Job processing interface /** * Clears the job queue and aborts all worker threads. * * This method is thread safe and can be called from inside a job. */ void Abort(); /** * Push job to the queue. * * This method is thread safe and can be called from inside a job or before * Process(). */ bool PushJob(JobHandleT&& jobHandle); /** * Push job to the queue * * This method is thread safe and can be called from inside a job or before * Process(). */ template <class T, typename... Args> bool EmplaceJob(Args&&... args) { return this->PushJob(cm::make_unique<T>(std::forward<Args>(args)...)); } private: void* UserData_ = nullptr; unsigned int ThreadCount_ = 1; std::unique_ptr<cmWorkerPoolInternal> Int_; };