#include #include "OgrePlatform.hpp" #include #include #include #include "OgreMaterial.hpp" #include "OgreGpuProgram.hpp" #include "OgreMaterialSerializer.hpp" #include "../../Main/MaterialInstance.hpp" #include "../../Main/Factory.hpp" namespace { std::string convertLang (sh::Language lang) { if (lang == sh::Language_CG) return "cg"; else if (lang == sh::Language_HLSL) return "hlsl"; else if (lang == sh::Language_GLSL) return "glsl"; throw std::runtime_error ("invalid language, valid are: cg, hlsl, glsl"); } } namespace sh { OgreMaterialSerializer* OgrePlatform::sSerializer = 0; OgrePlatform::OgrePlatform(const std::string& resourceGroupName, const std::string& basePath) : Platform(basePath) , mResourceGroup(resourceGroupName) { Ogre::MaterialManager::getSingleton().addListener(this); if (supportsShaderSerialization()) Ogre::GpuProgramManager::getSingletonPtr()->setSaveMicrocodesToCache(true); sSerializer = new OgreMaterialSerializer(); } OgreMaterialSerializer& OgrePlatform::getSerializer() { assert(sSerializer); return *sSerializer; } OgrePlatform::~OgrePlatform () { delete sSerializer; } bool OgrePlatform::isProfileSupported (const std::string& profile) { return Ogre::GpuProgramManager::getSingleton().isSyntaxSupported(profile); } bool OgrePlatform::supportsShaderSerialization () { // Not very reliable in OpenGL mode (requires extension), and somehow doesn't work on linux even if the extension is present return Ogre::Root::getSingleton ().getRenderSystem ()->getName ().find("OpenGL") == std::string::npos; } bool OgrePlatform::supportsMaterialQueuedListener () { return true; } boost::shared_ptr OgrePlatform::createMaterial (const std::string& name) { OgreMaterial* material = new OgreMaterial(name, mResourceGroup); return boost::shared_ptr (material); } boost::shared_ptr OgrePlatform::createGpuProgram ( GpuProgramType type, const std::string& compileArguments, const std::string& name, const std::string& profile, const std::string& source, Language lang) { OgreGpuProgram* prog = new OgreGpuProgram (type, compileArguments, name, profile, source, convertLang(lang), mResourceGroup); return boost::shared_ptr (static_cast(prog)); } Ogre::Technique* OgrePlatform::handleSchemeNotFound ( unsigned short schemeIndex, const Ogre::String &schemeName, Ogre::Material *originalMaterial, unsigned short lodIndex, const Ogre::Renderable *rend) { MaterialInstance* m = fireMaterialRequested(originalMaterial->getName(), schemeName, lodIndex); if (m) { OgreMaterial* _m = static_cast(m->getMaterial()); return _m->getOgreTechniqueForConfiguration (schemeName, lodIndex); } else return 0; // material does not belong to us } void OgrePlatform::serializeShaders (const std::string& file) { std::fstream output; output.open(file.c_str(), std::ios::out | std::ios::binary); Ogre::DataStreamPtr shaderCache (OGRE_NEW Ogre::FileStreamDataStream(file, &output, false)); Ogre::GpuProgramManager::getSingleton().saveMicrocodeCache(shaderCache); } void OgrePlatform::deserializeShaders (const std::string& file) { std::ifstream inp; inp.open(file.c_str(), std::ios::in | std::ios::binary); Ogre::DataStreamPtr shaderCache(OGRE_NEW Ogre::FileStreamDataStream(file, &inp, false)); Ogre::GpuProgramManager::getSingleton().loadMicrocodeCache(shaderCache); } void OgrePlatform::setSharedParameter (const std::string& name, PropertyValuePtr value) { Ogre::GpuSharedParametersPtr params; if (mSharedParameters.find(name) == mSharedParameters.end()) { params = Ogre::GpuProgramManager::getSingleton().createSharedParameters(name); Ogre::GpuConstantType type; if (typeid(*value) == typeid(Vector4)) type = Ogre::GCT_FLOAT4; else if (typeid(*value) == typeid(Vector3)) type = Ogre::GCT_FLOAT3; else if (typeid(*value) == typeid(Vector2)) type = Ogre::GCT_FLOAT2; else if (typeid(*value) == typeid(FloatValue)) type = Ogre::GCT_FLOAT1; else if (typeid(*value) == typeid(IntValue)) type = Ogre::GCT_INT1; else assert(0); params->addConstantDefinition(name, type); mSharedParameters[name] = params; } else params = mSharedParameters.find(name)->second; Ogre::Vector4 v (1.0, 1.0, 1.0, 1.0); if (typeid(*value) == typeid(Vector4)) { Vector4 vec = retrieveValue(value, NULL); v.x = vec.mX; v.y = vec.mY; v.z = vec.mZ; v.w = vec.mW; } else if (typeid(*value) == typeid(Vector3)) { Vector3 vec = retrieveValue(value, NULL); v.x = vec.mX; v.y = vec.mY; v.z = vec.mZ; } else if (typeid(*value) == typeid(Vector2)) { Vector2 vec = retrieveValue(value, NULL); v.x = vec.mX; v.y = vec.mY; } else if (typeid(*value) == typeid(FloatValue)) v.x = retrieveValue(value, NULL).get(); else if (typeid(*value) == typeid(IntValue)) v.x = static_cast(retrieveValue(value, NULL).get()); else throw std::runtime_error ("unsupported property type for shared parameter \"" + name + "\""); params->setNamedConstant(name, v); } }