2005-07-21 16:43:24 +08:00
|
|
|
/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2005 Robert Osfield
|
|
|
|
*
|
|
|
|
* This library is open source and may be redistributed and/or modified under
|
|
|
|
* the terms of the OpenSceneGraph Public License (OSGPL) version 0.0 or
|
|
|
|
* (at your option) any later version. The full license is in LICENSE file
|
|
|
|
* included with this distribution, and on the openscenegraph.org website.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* OpenSceneGraph Public License for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <osg/GraphicsContext>
|
2005-07-25 22:28:22 +08:00
|
|
|
#include <osg/Notify>
|
|
|
|
#include <map>
|
2005-07-21 16:43:24 +08:00
|
|
|
|
|
|
|
using namespace osg;
|
|
|
|
|
|
|
|
static ref_ptr<GraphicsContext::CreateGraphicContexCallback> s_createGraphicsContextCallback;
|
|
|
|
|
|
|
|
void GraphicsContext::setCreateGraphicsContextCallback(CreateGraphicContexCallback* callback)
|
|
|
|
{
|
|
|
|
s_createGraphicsContextCallback = callback;
|
|
|
|
}
|
|
|
|
|
|
|
|
GraphicsContext::CreateGraphicContexCallback* GraphicsContext::getCreateGraphicsContextCallback()
|
|
|
|
{
|
|
|
|
return s_createGraphicsContextCallback.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
GraphicsContext* GraphicsContext::createGraphicsContext(Traits* traits)
|
|
|
|
{
|
|
|
|
if (s_createGraphicsContextCallback.valid())
|
|
|
|
return s_createGraphicsContextCallback->createGraphicsContext(traits);
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-07-25 22:28:22 +08:00
|
|
|
|
|
|
|
typedef std::map<unsigned int, unsigned int> ContextIDMap;
|
|
|
|
static ContextIDMap s_contextIDMap;
|
|
|
|
static OpenThreads::Mutex s_contextIDMapMutex;
|
|
|
|
|
|
|
|
unsigned int GraphicsContext::createNewContextID()
|
|
|
|
{
|
|
|
|
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(s_contextIDMapMutex);
|
|
|
|
|
|
|
|
// first check to see if we can reuse contextID;
|
|
|
|
for(ContextIDMap::iterator itr = s_contextIDMap.begin();
|
|
|
|
itr != s_contextIDMap.end();
|
|
|
|
++itr)
|
|
|
|
{
|
|
|
|
if (itr->second == 0)
|
|
|
|
{
|
|
|
|
|
|
|
|
// reuse contextID;
|
|
|
|
itr->second = 1;
|
|
|
|
|
|
|
|
osg::notify(osg::NOTICE)<<"GraphicsContext::createNewContextID() reusing contextID="<<itr->first<<std::endl;
|
|
|
|
|
|
|
|
return itr->first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int contextID = s_contextIDMap.size();
|
|
|
|
s_contextIDMap[contextID] = 1;
|
|
|
|
|
|
|
|
osg::notify(osg::INFO)<<"GraphicsContext::createNewContextID() creating contextID="<<contextID<<std::endl;
|
|
|
|
|
|
|
|
|
|
|
|
if ( (contextID+1) > osg::DisplaySettings::instance()->getMaxNumberOfGraphicsContexts() )
|
|
|
|
{
|
|
|
|
osg::notify(osg::INFO)<<"Updating the MaxNumberOfGraphicsContexts to "<<contextID+1<<std::endl;
|
|
|
|
|
|
|
|
// update the the maximum number of graphics contexts,
|
|
|
|
// to ensure that texture objects and display buffers are configured to the correct size.
|
|
|
|
osg::DisplaySettings::instance()->setMaxNumberOfGraphicsContexts( contextID + 1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return contextID;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsContext::incrementContextIDUsageCount(unsigned int contextID)
|
|
|
|
{
|
|
|
|
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(s_contextIDMapMutex);
|
|
|
|
|
|
|
|
osg::notify(osg::INFO)<<"GraphicsContext::incrementContextIDUsageCount("<<contextID<<") to "<<s_contextIDMap[contextID]<<std::endl;
|
|
|
|
|
|
|
|
++s_contextIDMap[contextID];
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsContext::decrementContextIDUsageCount(unsigned int contextID)
|
|
|
|
{
|
|
|
|
|
|
|
|
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(s_contextIDMapMutex);
|
|
|
|
|
|
|
|
|
|
|
|
if (s_contextIDMap[contextID]!=0)
|
|
|
|
{
|
|
|
|
--s_contextIDMap[contextID];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
osg::notify(osg::NOTICE)<<"Warning: decrementContextIDUsageCount("<<contextID<<") called on expired contextID."<<std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::notify(osg::INFO)<<"GraphicsContext::decrementContextIDUsageCount("<<contextID<<") to "<<s_contextIDMap[contextID]<<std::endl;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-07-21 16:43:24 +08:00
|
|
|
GraphicsContext::GraphicsContext():
|
2005-08-20 16:59:03 +08:00
|
|
|
_closeOnDestruction(true),
|
2005-07-21 16:43:24 +08:00
|
|
|
_threadOfLastMakeCurrent(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2005-07-25 22:28:22 +08:00
|
|
|
GraphicsContext::~GraphicsContext()
|
|
|
|
{
|
2005-08-20 16:59:03 +08:00
|
|
|
// switch off the graphics thread...
|
|
|
|
setGraphicsThread(0);
|
|
|
|
|
2005-07-25 22:28:22 +08:00
|
|
|
if (_state.valid())
|
|
|
|
{
|
|
|
|
decrementContextIDUsageCount(_state->getContextID());
|
|
|
|
}
|
2005-08-20 16:59:03 +08:00
|
|
|
|
|
|
|
if (_closeOnDestruction)
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Realise the GraphicsContext.*/
|
|
|
|
bool GraphicsContext::realize()
|
|
|
|
{
|
|
|
|
if (realizeImplementation())
|
|
|
|
{
|
|
|
|
if (_graphicsThread.valid() && !_graphicsThread->isRunning())
|
|
|
|
{
|
|
|
|
_graphicsThread->startThread();
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2005-07-25 22:28:22 +08:00
|
|
|
}
|
|
|
|
|
2005-08-20 16:59:03 +08:00
|
|
|
void GraphicsContext::close()
|
|
|
|
{
|
|
|
|
// switch off the graphics thread...
|
|
|
|
setGraphicsThread(0);
|
|
|
|
|
|
|
|
closeImplementation();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-08-19 04:17:51 +08:00
|
|
|
void GraphicsContext::makeCurrent()
|
|
|
|
{
|
2005-08-20 16:59:03 +08:00
|
|
|
osg::notify(osg::NOTICE)<<"Doing GraphicsContext::makeCurrent"<<(unsigned int)OpenThreads::Thread::CurrentThread()<<std::endl;
|
2005-08-19 04:17:51 +08:00
|
|
|
|
|
|
|
ReleaseContext_Block_MakeCurrentOperation* rcbmco = 0;
|
|
|
|
|
|
|
|
if (_graphicsThread.valid() &&
|
2005-08-20 16:59:03 +08:00
|
|
|
_threadOfLastMakeCurrent == _graphicsThread.get() &&
|
|
|
|
_threadOfLastMakeCurrent != OpenThreads::Thread::CurrentThread())
|
2005-08-19 04:17:51 +08:00
|
|
|
{
|
|
|
|
// create a relase contex, block and make current operation to stop the graphics thread while we use the graphics context for ourselves
|
|
|
|
rcbmco = new ReleaseContext_Block_MakeCurrentOperation;
|
|
|
|
_graphicsThread->add(rcbmco);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isCurrent()) _mutex.lock();
|
|
|
|
|
2005-08-20 16:59:03 +08:00
|
|
|
osg::notify(osg::NOTICE)<<"Calling GraphicsContext::makeCurrentImplementation"<<(unsigned int)OpenThreads::Thread::CurrentThread()<<std::endl;
|
|
|
|
|
2005-08-19 04:17:51 +08:00
|
|
|
makeCurrentImplementation();
|
|
|
|
|
|
|
|
_threadOfLastMakeCurrent = OpenThreads::Thread::CurrentThread();
|
|
|
|
|
|
|
|
if (rcbmco)
|
|
|
|
{
|
|
|
|
// Let the "relase contex, block and make current operation" proceed, which will now move on to trying to aquire the graphics
|
|
|
|
// contex itself with a makeCurrent(), this will then block on the GraphicsContext mutex till releaseContext() releases it.
|
|
|
|
rcbmco->release();
|
|
|
|
}
|
2005-08-20 16:59:03 +08:00
|
|
|
|
|
|
|
osg::notify(osg::NOTICE)<<"Done GraphicsContext::makeCurrentImplementation"<<(unsigned int)OpenThreads::Thread::CurrentThread()<<std::endl;
|
2005-08-19 04:17:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsContext::makeContextCurrent(GraphicsContext* readContext)
|
|
|
|
{
|
|
|
|
if (!isCurrent()) _mutex.lock();
|
|
|
|
|
|
|
|
makeContextCurrentImplementation(readContext);
|
|
|
|
|
|
|
|
_threadOfLastMakeCurrent = OpenThreads::Thread::CurrentThread();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsContext::releaseContext()
|
|
|
|
{
|
|
|
|
_mutex.unlock();
|
|
|
|
}
|
2005-08-20 16:59:03 +08:00
|
|
|
|
|
|
|
void GraphicsContext::swapBuffers()
|
|
|
|
{
|
|
|
|
if (isCurrent())
|
|
|
|
{
|
|
|
|
osg::notify(osg::NOTICE)<<"Doing GraphicsContext::swapBuffers() call to swapBuffersImplementation() "<<(unsigned int)OpenThreads::Thread::CurrentThread()<<std::endl;
|
|
|
|
swapBuffersImplementation();
|
|
|
|
}
|
|
|
|
else if (_graphicsThread.valid() &&
|
|
|
|
_threadOfLastMakeCurrent == _graphicsThread.get())
|
|
|
|
{
|
|
|
|
osg::notify(osg::NOTICE)<<"Doing GraphicsContext::swapBuffers() registering SwapBuffersOperation"<<(unsigned int)OpenThreads::Thread::CurrentThread()<<std::endl;
|
|
|
|
_graphicsThread->add(new SwapBuffersOperation);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
osg::notify(osg::NOTICE)<<"Doing GraphicsContext::swapBuffers() makeCurrent;swapBuffersImplementation;releaseContext"<<(unsigned int)OpenThreads::Thread::CurrentThread()<<std::endl;
|
|
|
|
makeCurrent();
|
|
|
|
swapBuffersImplementation();
|
|
|
|
releaseContext();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void GraphicsContext::createGraphicsThread()
|
|
|
|
{
|
|
|
|
if (!_graphicsThread)
|
|
|
|
{
|
|
|
|
setGraphicsThread(new GraphicsThread);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphicsContext::setGraphicsThread(GraphicsThread* gt)
|
|
|
|
{
|
|
|
|
if (_graphicsThread==gt) return;
|
|
|
|
|
|
|
|
if (_graphicsThread.valid())
|
|
|
|
{
|
|
|
|
// need to kill the thread in some way...
|
|
|
|
_graphicsThread->cancel();
|
|
|
|
_graphicsThread->_graphicsContext = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
_graphicsThread = gt;
|
|
|
|
|
|
|
|
if (_graphicsThread.valid())
|
|
|
|
{
|
|
|
|
_graphicsThread->_graphicsContext = this;
|
|
|
|
|
|
|
|
if (!_graphicsThread->isRunning() && isRealized())
|
|
|
|
{
|
|
|
|
_graphicsThread->startThread();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|