mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-25 06:35:30 +00:00
175 lines
5.2 KiB
C++
175 lines
5.2 KiB
C++
#include <stdexcept>
|
|
|
|
#include "OgrePlatform.hpp"
|
|
|
|
#include <OgreDataStream.h>
|
|
#include <OgreGpuProgramManager.h>
|
|
#include <OgreRoot.h>
|
|
|
|
#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<Material> OgrePlatform::createMaterial (const std::string& name)
|
|
{
|
|
OgreMaterial* material = new OgreMaterial(name, mResourceGroup);
|
|
return boost::shared_ptr<Material> (material);
|
|
}
|
|
|
|
boost::shared_ptr<GpuProgram> 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<GpuProgram> (static_cast<GpuProgram*>(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<OgreMaterial*>(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<Vector4>(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<Vector3>(value, NULL);
|
|
v.x = vec.mX;
|
|
v.y = vec.mY;
|
|
v.z = vec.mZ;
|
|
}
|
|
else if (typeid(*value) == typeid(Vector2))
|
|
{
|
|
Vector2 vec = retrieveValue<Vector2>(value, NULL);
|
|
v.x = vec.mX;
|
|
v.y = vec.mY;
|
|
}
|
|
else if (typeid(*value) == typeid(FloatValue))
|
|
v.x = retrieveValue<FloatValue>(value, NULL).get();
|
|
else if (typeid(*value) == typeid(IntValue))
|
|
v.x = static_cast<float>(retrieveValue<IntValue>(value, NULL).get());
|
|
else
|
|
throw std::runtime_error ("unsupported property type for shared parameter \"" + name + "\"");
|
|
params->setNamedConstant(name, v);
|
|
}
|
|
}
|