/* -*-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 fragment programs which can only // by completely deleted once the appropriate OpenGL context // is set. typedef std::list FragmentProgramObjectList; typedef std::map DeletedFragmentProgramObjectCache; #ifdef THREAD_SAFE_GLOBJECT_DELETE_LISTS static OpenThreads::Mutex s_mutex_deletedFragmentProgramObjectCache; #endif static DeletedFragmentProgramObjectCache s_deletedFragmentProgramObjectCache; void FragmentProgram::deleteFragmentProgramObject(unsigned int contextID,GLuint handle) { if (handle!=0) { #ifdef THREAD_SAFE_GLOBJECT_DELETE_LISTS OpenThreads::ScopedLock lock(s_mutex_deletedFragmentProgramObjectCache); #endif // insert the handle into the cache for the appropriate context. s_deletedFragmentProgramObjectCache[contextID].push_back(handle); } } void FragmentProgram::flushDeletedFragmentProgramObjects(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_deletedFragmentProgramObjectCache); #endif DeletedFragmentProgramObjectCache::iterator citr = s_deletedFragmentProgramObjectCache.find(contextID); if (citr!=s_deletedFragmentProgramObjectCache.end()) { const Extensions* extensions = getExtensions(contextID,true); FragmentProgramObjectList& vpol = citr->second; for(FragmentProgramObjectList::iterator titr=vpol.begin(); titr!=vpol.end() && elapsedTimeglDeletePrograms( 1L, &(*titr ) ); titr = vpol.erase(titr); elapsedTime = timer.delta_s(start_tick,timer.tick()); } } } availableTime -= elapsedTime; } FragmentProgram::FragmentProgram() { } FragmentProgram::FragmentProgram(const FragmentProgram& vp,const CopyOp& copyop): osg::StateAttribute(vp,copyop) { _fragmentProgram = vp._fragmentProgram; 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 FragmentProgram::~FragmentProgram() { dirtyFragmentProgramObject(); } void FragmentProgram::dirtyFragmentProgramObject() { for(unsigned int i=0;i<_fragmentProgramIDList.size();++i) { if (_fragmentProgramIDList[i] != 0) { FragmentProgram::deleteFragmentProgramObject(i,_fragmentProgramIDList[i]); _fragmentProgramIDList[i] = 0; } } } void FragmentProgram::apply(State& state) const { const unsigned int contextID = state.getContextID(); const Extensions* extensions = getExtensions(contextID,true); if (!extensions->isFragmentProgramSupported()) return; GLuint& fragmentProgramId=getFragmentProgramID(state.getContextID()); // Fragment Program if (fragmentProgramId != 0) { extensions->glBindProgram( GL_FRAGMENT_PROGRAM_ARB, fragmentProgramId ); } else if (!_fragmentProgram.empty()) { glGetError(); // Reset Error flags. extensions->glGenPrograms( 1, &fragmentProgramId ); extensions->glBindProgram( GL_FRAGMENT_PROGRAM_ARB, fragmentProgramId ); extensions->glProgramString( GL_FRAGMENT_PROGRAM_ARB, GL_PROGRAM_FORMAT_ASCII_ARB, _fragmentProgram.length(), _fragmentProgram.c_str()); // Check for errors GLint errorposition; glGetIntegerv(GL_PROGRAM_ERROR_POSITION_ARB, &errorposition); if (errorposition != -1) { notify(osg::FATAL) << "FragmentProgram: " << glGetString(GL_PROGRAM_ERROR_STRING_ARB) << std::endl; std::string::size_type start = _fragmentProgram.rfind('\n', errorposition); std::string::size_type stop = _fragmentProgram.find('\n', errorposition); if (start!=std::string::npos && stop!=std::string::npos) { notify(osg::FATAL) << " : " << _fragmentProgram.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_FRAGMENT_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 FragmentProgram::releaseGLObjects(State* state) const { if (!state) const_cast(this)->dirtyFragmentProgramObject(); else { unsigned int contextID = state->getContextID(); if (_fragmentProgramIDList[contextID] != 0) { FragmentProgram::deleteFragmentProgramObject(contextID,_fragmentProgramIDList[contextID]); _fragmentProgramIDList[contextID] = 0; } } } typedef buffered_value< ref_ptr > BufferedExtensions; static BufferedExtensions s_extensions; FragmentProgram::Extensions* FragmentProgram::getExtensions(unsigned int contextID,bool createIfNotInitalized) { if (!s_extensions[contextID] && createIfNotInitalized) s_extensions[contextID] = new Extensions; return s_extensions[contextID].get(); } void FragmentProgram::setExtensions(unsigned int contextID,Extensions* extensions) { s_extensions[contextID] = extensions; } FragmentProgram::Extensions::Extensions() { setupGLExtenions(); } FragmentProgram::Extensions::Extensions(const Extensions& rhs): Referenced() { _isFragmentProgramSupported = rhs._isFragmentProgramSupported; _glBindProgram = rhs._glBindProgram; _glGenPrograms = rhs._glGenPrograms; _glDeletePrograms = rhs._glDeletePrograms; _glProgramString = rhs._glProgramString; _glProgramLocalParameter4fv = rhs._glProgramLocalParameter4fv; } void FragmentProgram::Extensions::lowestCommonDenominator(const Extensions& rhs) { if (!rhs._isFragmentProgramSupported) _isFragmentProgramSupported = 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 FragmentProgram::Extensions::setupGLExtenions() { _isFragmentProgramSupported = isGLExtensionSupported("GL_ARB_fragment_program"); _glBindProgram = osg::getGLExtensionFuncPtr("glBindProgramARB"); _glGenPrograms = osg::getGLExtensionFuncPtr("glGenProgramsARB"); _glDeletePrograms = osg::getGLExtensionFuncPtr("glDeleteProgramsARB"); _glProgramString = osg::getGLExtensionFuncPtr("glProgramStringARB"); _glProgramLocalParameter4fv = osg::getGLExtensionFuncPtr("glProgramLocalParameter4fvARB"); } void FragmentProgram::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"<