/* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2003 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 #include #include #include #include #include #ifdef THREAD_SAFE_GLOBJECT_DELETE_LISTS #include #include #endif using namespace osg; // static cache of deleted vertex programs which can only // by completely deleted once the appropriate OpenGL context // is set. typedef std::list VertexProgramObjectList; typedef std::map DeletedVertexProgramObjectCache; #ifdef THREAD_SAFE_GLOBJECT_DELETE_LISTS static OpenThreads::Mutex s_mutex_deletedVertexProgramObjectCache; #endif static DeletedVertexProgramObjectCache s_deletedVertexProgramObjectCache; void VertexProgram::deleteVertexProgramObject(unsigned int contextID,GLuint handle) { if (handle!=0) { #ifdef THREAD_SAFE_GLOBJECT_DELETE_LISTS OpenThreads::ScopedLock lock(s_mutex_deletedVertexProgramObjectCache); #endif // insert the handle into the cache for the appropriate context. s_deletedVertexProgramObjectCache[contextID].push_back(handle); } } void VertexProgram::flushDeletedVertexProgramObjects(unsigned int contextID,double /*currentTime*/, double& availableTime) { // if no time available don't try to flush objects. if (availableTime<=0.0) return; const osg::Timer& timer = *osg::Timer::instance(); osg::Timer_t start_tick = timer.tick(); double elapsedTime = 0.0; { #ifdef THREAD_SAFE_GLOBJECT_DELETE_LISTS OpenThreads::ScopedLock lock(s_mutex_deletedVertexProgramObjectCache); #endif DeletedVertexProgramObjectCache::iterator citr = s_deletedVertexProgramObjectCache.find(contextID); if (citr!=s_deletedVertexProgramObjectCache.end()) { const Extensions* extensions = getExtensions(contextID,true); VertexProgramObjectList& vpol = citr->second; for(VertexProgramObjectList::iterator titr=vpol.begin(); titr!=vpol.end() && elapsedTimeglDeletePrograms( 1L, &(*titr ) ); titr = vpol.erase(titr); elapsedTime = timer.delta_s(start_tick,timer.tick()); } } } availableTime -= elapsedTime; } VertexProgram::VertexProgram() { } VertexProgram::VertexProgram(const VertexProgram& vp,const CopyOp& copyop): osg::StateAttribute(vp,copyop) { _vertexProgram = vp._vertexProgram; for( LocalParamList::const_iterator itr = vp._programLocalParameters.begin(); itr != vp._programLocalParameters.end(); ++itr ) { _programLocalParameters[itr->first] = itr->second; } for( MatrixList::const_iterator mitr = vp._matrixList.begin(); mitr != vp._matrixList.end(); ++mitr ) { _matrixList[mitr->first] = mitr->second; } } // virtual VertexProgram::~VertexProgram() { dirtyVertexProgramObject(); } void VertexProgram::dirtyVertexProgramObject() { for(unsigned int i=0;i<_vertexProgramIDList.size();++i) { if (_vertexProgramIDList[i] != 0) { VertexProgram::deleteVertexProgramObject(i,_vertexProgramIDList[i]); _vertexProgramIDList[i] = 0; } } } void VertexProgram::apply(State& state) const { const unsigned int contextID = state.getContextID(); const Extensions* extensions = getExtensions(contextID,true); if (!extensions->isVertexProgramSupported()) return; GLuint& vertexProgramId=getVertexProgramID(state.getContextID()); // Vertex Program if (vertexProgramId != 0) { extensions->glBindProgram( GL_VERTEX_PROGRAM_ARB, vertexProgramId ); } else if (!_vertexProgram.empty()) { glGetError(); // Reset Error flags. extensions->glGenPrograms( 1, &vertexProgramId ); extensions->glBindProgram( GL_VERTEX_PROGRAM_ARB, vertexProgramId ); extensions->glProgramString( GL_VERTEX_PROGRAM_ARB, GL_PROGRAM_FORMAT_ASCII_ARB, _vertexProgram.length(), _vertexProgram.c_str()); // Check for errors GLint errorposition; glGetIntegerv(GL_PROGRAM_ERROR_POSITION_ARB, &errorposition); if (errorposition != -1) { notify(osg::FATAL) << "VertexProgram: " << glGetString(GL_PROGRAM_ERROR_STRING_ARB) << std::endl; std::string::size_type start = _vertexProgram.rfind('\n', errorposition); std::string::size_type stop = _vertexProgram.find('\n', errorposition); if (start!=std::string::npos && stop!=std::string::npos) { notify(osg::FATAL) << " : " << _vertexProgram.substr(start+1, stop-start-2) << std::endl; std::string pointAtproblem(errorposition-(start+1), ' '); notify(osg::FATAL) << " : " << pointAtproblem << '^' << std::endl; } return; } } // Update local program parameters { for(LocalParamList::const_iterator itr=_programLocalParameters.begin(); itr!=_programLocalParameters.end(); ++itr) { extensions->glProgramLocalParameter4fv(GL_VERTEX_PROGRAM_ARB, (*itr).first, (*itr).second.ptr()); } } // Update matrix if (!_matrixList.empty()) { for(MatrixList::const_iterator itr = _matrixList.begin(); itr!=_matrixList.end(); ++itr) { glMatrixMode((*itr).first); glLoadMatrix((*itr).second.ptr()); } glMatrixMode(GL_MODELVIEW); // restore matrix mode } } void VertexProgram::releaseGLObjects(State* state) const { if (!state) const_cast(this)->dirtyVertexProgramObject(); else { unsigned int contextID = state->getContextID(); if (_vertexProgramIDList[contextID] != 0) { VertexProgram::deleteVertexProgramObject(contextID,_vertexProgramIDList[contextID]); _vertexProgramIDList[contextID] = 0; } } } typedef buffered_value< ref_ptr > BufferedExtensions; static BufferedExtensions s_extensions; VertexProgram::Extensions* VertexProgram::getExtensions(unsigned int contextID,bool createIfNotInitalized) { if (!s_extensions[contextID] && createIfNotInitalized) s_extensions[contextID] = new Extensions; return s_extensions[contextID].get(); } void VertexProgram::setExtensions(unsigned int contextID,Extensions* extensions) { s_extensions[contextID] = extensions; } VertexProgram::Extensions::Extensions() { setupGLExtenions(); } VertexProgram::Extensions::Extensions(const Extensions& rhs): Referenced() { _isVertexProgramSupported = rhs._isVertexProgramSupported; _glBindProgram = rhs._glBindProgram; _glGenPrograms = rhs._glGenPrograms; _glDeletePrograms = rhs._glDeletePrograms; _glProgramString = rhs._glProgramString; _glProgramLocalParameter4fv = rhs._glProgramLocalParameter4fv; } void VertexProgram::Extensions::lowestCommonDenominator(const Extensions& rhs) { if (!rhs._isVertexProgramSupported) _isVertexProgramSupported = false; if (!rhs._glBindProgram) _glBindProgram = 0; if (!rhs._glGenPrograms) _glGenPrograms = 0; if (!rhs._glDeletePrograms) _glDeletePrograms = 0; if (!rhs._glProgramString) _glProgramString = 0; if (!rhs._glProgramLocalParameter4fv) _glProgramLocalParameter4fv = 0; } void VertexProgram::Extensions::setupGLExtenions() { _isVertexProgramSupported = isGLExtensionSupported("GL_ARB_vertex_program"); _glBindProgram = osg::getGLExtensionFuncPtr("glBindProgramARB"); _glGenPrograms = osg::getGLExtensionFuncPtr("glGenProgramsARB"); _glDeletePrograms = osg::getGLExtensionFuncPtr("glDeleteProgramsARB"); _glProgramString = osg::getGLExtensionFuncPtr("glProgramStringARB"); _glProgramLocalParameter4fv = osg::getGLExtensionFuncPtr("glProgramLocalParameter4fvARB"); } void VertexProgram::Extensions::glBindProgram(GLenum target, GLuint id) const { if (_glBindProgram) { typedef void (APIENTRY * BindProgramProc) (GLenum target, GLuint id); ((BindProgramProc)_glBindProgram)(target,id); } else { notify(WARN)<<"Error: glBindProgram not supported by OpenGL driver"<