2015-06-10 18:15:31 +02:00
|
|
|
#include "workqueue.hpp"
|
|
|
|
|
2018-08-14 19:42:41 +04:00
|
|
|
#include <components/debug/debuglog.hpp>
|
2016-02-06 22:29:06 +01:00
|
|
|
|
2020-06-24 19:28:54 +02:00
|
|
|
#include <numeric>
|
|
|
|
|
2015-06-10 18:15:31 +02:00
|
|
|
namespace SceneUtil
|
|
|
|
{
|
|
|
|
|
2016-02-06 22:29:06 +01:00
|
|
|
void WorkItem::waitTillDone()
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
if (mDone)
|
2015-06-10 18:15:31 +02:00
|
|
|
return;
|
|
|
|
|
2020-06-24 19:28:54 +02:00
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
|
|
|
while (!mDone)
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
mCondition.wait(lock);
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-06 22:29:06 +01:00
|
|
|
void WorkItem::signalDone()
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2015-06-14 21:04:59 +02:00
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
|
|
|
mDone = true;
|
2015-06-14 21:04:59 +02:00
|
|
|
}
|
2020-06-24 19:28:54 +02:00
|
|
|
mCondition.notify_all();
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
|
2016-02-06 22:29:06 +01:00
|
|
|
bool WorkItem::isDone() const
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
return mDone;
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
WorkQueue::WorkQueue(int workerThreads)
|
|
|
|
: mIsReleased(false)
|
|
|
|
{
|
|
|
|
for (int i=0; i<workerThreads; ++i)
|
2020-06-24 19:28:54 +02:00
|
|
|
mThreads.emplace_back(std::make_unique<WorkThread>(*this));
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
WorkQueue::~WorkQueue()
|
|
|
|
{
|
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
2016-01-03 18:20:34 +01:00
|
|
|
while (!mQueue.empty())
|
2016-03-29 00:25:51 +02:00
|
|
|
mQueue.pop_back();
|
2015-06-10 18:15:31 +02:00
|
|
|
mIsReleased = true;
|
2020-06-24 19:28:54 +02:00
|
|
|
mCondition.notify_all();
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
|
2020-06-24 19:28:54 +02:00
|
|
|
mThreads.clear();
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
|
2016-03-29 00:25:51 +02:00
|
|
|
void WorkQueue::addWorkItem(osg::ref_ptr<WorkItem> item, bool front)
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2016-02-06 22:29:06 +01:00
|
|
|
if (item->isDone())
|
|
|
|
{
|
2018-08-14 19:42:41 +04:00
|
|
|
Log(Debug::Error) << "Error: trying to add a work item that is already completed";
|
2016-02-06 22:29:06 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-24 19:28:54 +02:00
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
2016-03-29 00:25:51 +02:00
|
|
|
if (front)
|
|
|
|
mQueue.push_front(item);
|
|
|
|
else
|
|
|
|
mQueue.push_back(item);
|
2020-06-24 19:28:54 +02:00
|
|
|
mCondition.notify_one();
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
|
2016-02-06 22:29:06 +01:00
|
|
|
osg::ref_ptr<WorkItem> WorkQueue::removeWorkItem()
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
2016-01-03 18:20:34 +01:00
|
|
|
while (mQueue.empty() && !mIsReleased)
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
mCondition.wait(lock);
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
2016-02-22 19:06:12 +01:00
|
|
|
if (!mQueue.empty())
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2016-02-06 22:29:06 +01:00
|
|
|
osg::ref_ptr<WorkItem> item = mQueue.front();
|
2016-03-29 00:25:51 +02:00
|
|
|
mQueue.pop_front();
|
2015-06-10 18:15:31 +02:00
|
|
|
return item;
|
|
|
|
}
|
|
|
|
else
|
2018-10-09 10:21:12 +04:00
|
|
|
return nullptr;
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
|
2017-02-22 02:18:18 +01:00
|
|
|
unsigned int WorkQueue::getNumItems() const
|
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
2017-02-22 02:18:18 +01:00
|
|
|
return mQueue.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int WorkQueue::getNumActiveThreads() const
|
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
return std::accumulate(mThreads.begin(), mThreads.end(), 0u,
|
|
|
|
[] (auto r, const auto& t) { return r + t->isActive(); });
|
2017-02-22 02:18:18 +01:00
|
|
|
}
|
|
|
|
|
2020-06-24 19:28:54 +02:00
|
|
|
WorkThread::WorkThread(WorkQueue& workQueue)
|
|
|
|
: mWorkQueue(&workQueue)
|
2017-10-15 17:03:11 +02:00
|
|
|
, mActive(false)
|
2020-06-24 19:28:54 +02:00
|
|
|
, mThread([this] { run(); })
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
WorkThread::~WorkThread()
|
2015-06-10 18:15:31 +02:00
|
|
|
{
|
2020-06-24 19:28:54 +02:00
|
|
|
mThread.join();
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WorkThread::run()
|
|
|
|
{
|
|
|
|
while (true)
|
|
|
|
{
|
2016-02-06 22:29:06 +01:00
|
|
|
osg::ref_ptr<WorkItem> item = mWorkQueue->removeWorkItem();
|
2015-06-10 18:15:31 +02:00
|
|
|
if (!item)
|
|
|
|
return;
|
2017-02-22 02:18:18 +01:00
|
|
|
mActive = true;
|
2015-06-10 18:15:31 +02:00
|
|
|
item->doWork();
|
2016-02-06 22:29:06 +01:00
|
|
|
item->signalDone();
|
2017-02-22 02:18:18 +01:00
|
|
|
mActive = false;
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-22 02:18:18 +01:00
|
|
|
bool WorkThread::isActive() const
|
|
|
|
{
|
|
|
|
return mActive;
|
|
|
|
}
|
|
|
|
|
2015-06-10 18:15:31 +02:00
|
|
|
}
|