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

123 lines
2.2 KiB
C++
Raw Normal View History

2015-06-10 16:15:31 +00:00
#include "workqueue.hpp"
namespace SceneUtil
{
void WorkTicket::waitTillDone()
{
if (mDone > 0)
return;
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(mMutex);
while (mDone == 0)
{
mCondition.wait(&mMutex);
}
}
void WorkTicket::signalDone()
{
2015-06-14 19:04:59 +00:00
{
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(mMutex);
mDone.exchange(1);
}
2015-06-10 16:15:31 +00:00
mCondition.broadcast();
}
WorkItem::WorkItem()
: mTicket(new WorkTicket)
{
2015-06-10 17:02:33 +00:00
mTicket->setThreadSafeRefUnref(true);
2015-06-10 16:15:31 +00:00
}
WorkItem::~WorkItem()
{
}
void WorkItem::doWork()
{
mTicket->signalDone();
}
osg::ref_ptr<WorkTicket> WorkItem::getTicket()
{
return mTicket;
}
WorkQueue::WorkQueue(int workerThreads)
: mIsReleased(false)
{
for (int i=0; i<workerThreads; ++i)
{
WorkThread* thread = new WorkThread(this);
mThreads.push_back(thread);
thread->startThread();
}
}
WorkQueue::~WorkQueue()
{
{
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(mMutex);
2016-01-03 17:20:34 +00:00
while (!mQueue.empty())
2015-06-10 16:15:31 +00:00
{
WorkItem* item = mQueue.front();
delete item;
mQueue.pop();
}
mIsReleased = true;
mCondition.broadcast();
}
for (unsigned int i=0; i<mThreads.size(); ++i)
{
mThreads[i]->join();
delete mThreads[i];
}
}
2015-06-10 17:02:33 +00:00
osg::ref_ptr<WorkTicket> WorkQueue::addWorkItem(WorkItem *item)
2015-06-10 16:15:31 +00:00
{
2015-06-10 17:02:33 +00:00
osg::ref_ptr<WorkTicket> ticket = item->getTicket();
2015-06-10 16:15:31 +00:00
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(mMutex);
mQueue.push(item);
mCondition.signal();
return ticket;
}
WorkItem *WorkQueue::removeWorkItem()
{
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(mMutex);
2016-01-03 17:20:34 +00:00
while (mQueue.empty() && !mIsReleased)
2015-06-10 16:15:31 +00:00
{
mCondition.wait(&mMutex);
}
if (mQueue.size())
{
WorkItem* item = mQueue.front();
mQueue.pop();
return item;
}
else
return NULL;
}
WorkThread::WorkThread(WorkQueue *workQueue)
: mWorkQueue(workQueue)
{
}
void WorkThread::run()
{
while (true)
{
WorkItem* item = mWorkQueue->removeWorkItem();
if (!item)
return;
item->doWork();
delete item;
}
}
}