2015-06-10 18:15:31 +02:00
|
|
|
#ifndef OPENMW_COMPONENTS_SCENEUTIL_WORKQUEUE_H
|
|
|
|
#define OPENMW_COMPONENTS_SCENEUTIL_WORKQUEUE_H
|
|
|
|
|
|
|
|
#include <osg/Referenced>
|
|
|
|
#include <osg/ref_ptr>
|
|
|
|
|
2018-12-24 02:49:36 -05:00
|
|
|
#include <atomic>
|
2020-06-24 19:28:54 +02:00
|
|
|
#include <condition_variable>
|
2022-10-09 10:39:43 +00:00
|
|
|
#include <deque>
|
2020-06-24 19:28:54 +02:00
|
|
|
#include <mutex>
|
|
|
|
#include <thread>
|
2022-10-09 10:39:43 +00:00
|
|
|
#include <vector>
|
2015-06-10 18:15:31 +02:00
|
|
|
|
|
|
|
namespace SceneUtil
|
|
|
|
{
|
|
|
|
|
2016-02-06 22:29:06 +01:00
|
|
|
class WorkItem : public osg::Referenced
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
/// Override in a derived WorkItem to perform actual work.
|
2016-02-06 22:29:06 +01:00
|
|
|
virtual void doWork() {}
|
|
|
|
|
|
|
|
bool isDone() const;
|
2015-06-10 18:15:31 +02:00
|
|
|
|
2016-02-06 22:29:06 +01: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 18:15:31 +02:00
|
|
|
|
2017-02-09 01:08:07 +01: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 19:28:54 +02:00
|
|
|
private:
|
|
|
|
std::atomic_bool mDone{ false };
|
|
|
|
std::mutex mMutex;
|
|
|
|
std::condition_variable mCondition;
|
2015-06-10 18:15:31 +02:00
|
|
|
};
|
|
|
|
|
2016-02-06 22:29:06 +01:00
|
|
|
class WorkThread;
|
2015-06-10 18:15:31 +02:00
|
|
|
|
|
|
|
/// @brief A work queue that users can push work items onto, to be completed by one or more background threads.
|
2016-02-06 22:29:06 +01:00
|
|
|
/// @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 18:15:31 +02:00
|
|
|
{
|
|
|
|
public:
|
2021-05-25 13:44:40 +02:00
|
|
|
WorkQueue(std::size_t workerThreads);
|
2015-06-10 18:15:31 +02:00
|
|
|
~WorkQueue();
|
|
|
|
|
2021-05-25 13:44:40 +02:00
|
|
|
void start(std::size_t workerThreads);
|
|
|
|
|
|
|
|
void stop();
|
|
|
|
|
2015-06-10 18:15:31 +02:00
|
|
|
/// Add a new work item to the back of the queue.
|
2016-02-06 22:29:06 +01:00
|
|
|
/// @par The work item's waitTillDone() method may be used by the caller to wait until the work is complete.
|
2016-03-29 00:25:51 +02:00
|
|
|
/// @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 18:15:31 +02: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 10:21:12 +04:00
|
|
|
/// If the workqueue is in the process of being destroyed, may return nullptr.
|
2016-02-06 22:29:06 +01:00
|
|
|
/// @par Used internally by the WorkThread.
|
|
|
|
osg::ref_ptr<WorkItem> removeWorkItem();
|
2015-06-10 18:15:31 +02:00
|
|
|
|
2017-02-22 02:18:18 +01:00
|
|
|
unsigned int getNumItems() const;
|
|
|
|
|
|
|
|
unsigned int getNumActiveThreads() const;
|
|
|
|
|
2015-06-10 18:15:31 +02:00
|
|
|
private:
|
|
|
|
bool mIsReleased;
|
2016-03-29 00:25:51 +02:00
|
|
|
std::deque<osg::ref_ptr<WorkItem>> mQueue;
|
2015-06-10 18:15:31 +02:00
|
|
|
|
2020-06-24 19:28:54 +02:00
|
|
|
mutable std::mutex mMutex;
|
|
|
|
std::condition_variable mCondition;
|
2015-06-10 18:15:31 +02:00
|
|
|
|
2020-06-24 19:28:54 +02:00
|
|
|
std::vector<std::unique_ptr<WorkThread>> mThreads;
|
2015-06-10 18:15:31 +02:00
|
|
|
};
|
|
|
|
|
2016-02-06 22:29:06 +01:00
|
|
|
/// Internally used by WorkQueue.
|
2020-06-24 19:28:54 +02:00
|
|
|
class WorkThread
|
2016-02-06 22:29:06 +01:00
|
|
|
{
|
|
|
|
public:
|
2020-06-24 19:28:54 +02:00
|
|
|
WorkThread(WorkQueue& workQueue);
|
2016-02-06 22:29:06 +01:00
|
|
|
|
2020-06-24 19:28:54 +02:00
|
|
|
~WorkThread();
|
2016-02-06 22:29:06 +01:00
|
|
|
|
2017-02-22 02:18:18 +01:00
|
|
|
bool isActive() const;
|
|
|
|
|
2016-02-06 22:29:06 +01:00
|
|
|
private:
|
|
|
|
WorkQueue* mWorkQueue;
|
2018-12-24 02:49:36 -05:00
|
|
|
std::atomic<bool> mActive;
|
2020-06-24 19:28:54 +02:00
|
|
|
std::thread mThread;
|
|
|
|
|
|
|
|
void run();
|
2016-02-06 22:29:06 +01:00
|
|
|
};
|
2015-06-10 18:15:31 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|