1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-22 09:53:54 +00:00
openmw/components/sceneutil/workqueue.hpp

100 lines
2.7 KiB
C++
Raw Normal View History

2015-06-10 16:15:31 +00:00
#ifndef OPENMW_COMPONENTS_SCENEUTIL_WORKQUEUE_H
#define OPENMW_COMPONENTS_SCENEUTIL_WORKQUEUE_H
#include <osg/Referenced>
#include <osg/ref_ptr>
#include <atomic>
2015-06-10 16:15:31 +00:00
#include <queue>
2020-06-24 17:28:54 +00:00
#include <thread>
#include <mutex>
#include <condition_variable>
2015-06-10 16:15:31 +00:00
namespace SceneUtil
{
class WorkItem : public osg::Referenced
2015-06-10 16:15:31 +00:00
{
public:
/// Override in a derived WorkItem to perform actual work.
virtual void doWork() {}
bool isDone() const;
2015-06-10 16:15:31 +00:00
/// Wait until the work is completed. Usually called from the main thread.
void waitTillDone();
/// Internal use by the WorkQueue.
void signalDone();
2015-06-10 16:15:31 +00:00
/// Set abort flag in order to return from doWork() as soon as possible. May not be respected by all WorkItems.
virtual void abort() {}
2020-06-24 17:28:54 +00:00
private:
std::atomic_bool mDone {false};
std::mutex mMutex;
std::condition_variable mCondition;
2015-06-10 16:15:31 +00:00
};
class WorkThread;
2015-06-10 16:15:31 +00:00
/// @brief A work queue that users can push work items onto, to be completed by one or more background threads.
/// @note Work items will be processed in the order that they were given in, however
/// if multiple work threads are involved then it is possible for a later item to complete before earlier items.
class WorkQueue : public osg::Referenced
2015-06-10 16:15:31 +00:00
{
public:
WorkQueue(std::size_t workerThreads);
2015-06-10 16:15:31 +00:00
~WorkQueue();
void start(std::size_t workerThreads);
void stop();
2015-06-10 16:15:31 +00:00
/// Add a new work item to the back of the queue.
/// @par The work item's waitTillDone() method may be used by the caller to wait until the work is complete.
/// @param front If true, add item to the front of the queue. If false (default), add to the back.
void addWorkItem(osg::ref_ptr<WorkItem> item, bool front=false);
2015-06-10 16:15:31 +00:00
/// Get the next work item from the front of the queue. If the queue is empty, waits until a new item is added.
2018-10-09 06:21:12 +00:00
/// If the workqueue is in the process of being destroyed, may return nullptr.
/// @par Used internally by the WorkThread.
osg::ref_ptr<WorkItem> removeWorkItem();
2015-06-10 16:15:31 +00:00
unsigned int getNumItems() const;
unsigned int getNumActiveThreads() const;
2015-06-10 16:15:31 +00:00
private:
bool mIsReleased;
std::deque<osg::ref_ptr<WorkItem> > mQueue;
2015-06-10 16:15:31 +00:00
2020-06-24 17:28:54 +00:00
mutable std::mutex mMutex;
std::condition_variable mCondition;
2015-06-10 16:15:31 +00:00
2020-06-24 17:28:54 +00:00
std::vector<std::unique_ptr<WorkThread>> mThreads;
2015-06-10 16:15:31 +00:00
};
/// Internally used by WorkQueue.
2020-06-24 17:28:54 +00:00
class WorkThread
{
public:
2020-06-24 17:28:54 +00:00
WorkThread(WorkQueue& workQueue);
2020-06-24 17:28:54 +00:00
~WorkThread();
bool isActive() const;
private:
WorkQueue* mWorkQueue;
std::atomic<bool> mActive;
2020-06-24 17:28:54 +00:00
std::thread mThread;
void run();
};
2015-06-10 16:15:31 +00:00
}
#endif