mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 09:26:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			368 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			368 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "pass.hpp"
 | |
| 
 | |
| #include <unordered_set>
 | |
| #include <string>
 | |
| #include <sstream>
 | |
| 
 | |
| #include <osg/Program>
 | |
| #include <osg/Shader>
 | |
| #include <osg/State>
 | |
| #include <osg/StateSet>
 | |
| #include <osg/BindImageTexture>
 | |
| #include <osg/FrameBufferObject>
 | |
| 
 | |
| #include <components/misc/stringops.hpp>
 | |
| #include <components/sceneutil/util.hpp>
 | |
| #include <components/sceneutil/lightmanager.hpp>
 | |
| #include <components/sceneutil/clearcolor.hpp>
 | |
| #include <components/resource/scenemanager.hpp>
 | |
| #include <components/stereo/multiview.hpp>
 | |
| 
 | |
| #include "technique.hpp"
 | |
| #include "stateupdater.hpp"
 | |
| 
 | |
| namespace
 | |
| {
 | |
|     constexpr char s_DefaultVertex[] = R"GLSL(
 | |
| #if OMW_USE_BINDINGS
 | |
|     omw_In vec2 omw_Vertex;
 | |
| #endif
 | |
| omw_Out vec2 omw_TexCoord;
 | |
| 
 | |
| void main()
 | |
| {
 | |
|     omw_Position = vec4(omw_Vertex.xy, 0.0, 1.0);
 | |
|     omw_TexCoord = omw_Position.xy * 0.5 + 0.5;
 | |
| })GLSL";
 | |
| 
 | |
|     constexpr char s_DefaultVertexMultiview[] = R"GLSL(
 | |
| layout(num_views = 2) in;
 | |
| #if OMW_USE_BINDINGS
 | |
|     omw_In vec2 omw_Vertex;
 | |
| #endif
 | |
| omw_Out vec2 omw_TexCoord;
 | |
| 
 | |
| void main()
 | |
| {
 | |
|     omw_Position = vec4(omw_Vertex.xy, 0.0, 1.0);
 | |
|     omw_TexCoord = omw_Position.xy * 0.5 + 0.5;
 | |
| })GLSL";
 | |
| 
 | |
| }
 | |
| 
 | |
| namespace fx
 | |
| {
 | |
|     Pass::Pass(Pass::Type type, Pass::Order order, bool ubo)
 | |
|         : mCompiled(false)
 | |
|         , mType(type)
 | |
|         , mOrder(order)
 | |
|         , mLegacyGLSL(true)
 | |
|         , mUBO(ubo)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     std::string Pass::getPassHeader(Technique& technique, std::string_view preamble, bool fragOut)
 | |
|     {
 | |
|         std::string header = R"GLSL(
 | |
| #version @version @profile
 | |
| @extensions
 | |
| 
 | |
| @uboStruct
 | |
| 
 | |
| #define OMW_REVERSE_Z @reverseZ
 | |
| #define OMW_RADIAL_FOG @radialFog
 | |
| #define OMW_EXPONENTIAL_FOG @exponentialFog
 | |
| #define OMW_HDR @hdr
 | |
| #define OMW_NORMALS @normals
 | |
| #define OMW_USE_BINDINGS @useBindings
 | |
| #define OMW_MULTIVIEW @multiview
 | |
| #define omw_In @in
 | |
| #define omw_Out @out
 | |
| #define omw_Position @position
 | |
| #define omw_Texture1D @texture1D
 | |
| #define omw_Texture2D @texture2D
 | |
| #define omw_Texture3D @texture3D
 | |
| #define omw_Vertex @vertex
 | |
| #define omw_FragColor @fragColor
 | |
| 
 | |
| @fragBinding
 | |
| 
 | |
| uniform @builtinSampler omw_SamplerLastShader;
 | |
| uniform @builtinSampler omw_SamplerLastPass;
 | |
| uniform @builtinSampler omw_SamplerDepth;
 | |
| uniform @builtinSampler omw_SamplerNormals;
 | |
| 
 | |
| uniform vec4 omw_PointLights[@pointLightCount];
 | |
| uniform int omw_PointLightsCount;
 | |
| 
 | |
| #if OMW_MULTIVIEW
 | |
| uniform mat4 projectionMatrixMultiView[2];
 | |
| uniform mat4 invProjectionMatrixMultiView[2];
 | |
| #endif
 | |
| 
 | |
| int omw_GetPointLightCount()
 | |
| {
 | |
|     return omw_PointLightsCount;
 | |
| }
 | |
| 
 | |
| vec3 omw_GetPointLightWorldPos(int index)
 | |
| {
 | |
|     return omw_PointLights[(index * 3)].xyz;
 | |
| }
 | |
| 
 | |
| vec3 omw_GetPointLightDiffuse(int index)
 | |
| {
 | |
|     return omw_PointLights[(index * 3) + 1].xyz;
 | |
| }
 | |
| 
 | |
| vec3 omw_GetPointLightAttenuation(int index)
 | |
| {
 | |
|     return omw_PointLights[(index * 3) + 2].xyz;
 | |
| }
 | |
| 
 | |
| float omw_GetPointLightRadius(int index)
 | |
| {
 | |
|     return omw_PointLights[(index * 3) + 2].w;
 | |
| }
 | |
| 
 | |
| #if @ubo
 | |
|     layout(std140) uniform _data { _omw_data omw; };
 | |
| #else
 | |
|     uniform _omw_data omw;
 | |
| #endif
 | |
| 
 | |
| 
 | |
| mat4 omw_ProjectionMatrix()
 | |
| {
 | |
| #if OMW_MULTIVIEW
 | |
|     return projectionMatrixMultiView[gl_ViewID_OVR];
 | |
| #else
 | |
|     return omw.projectionMatrix;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| mat4 omw_InvProjectionMatrix()
 | |
| {
 | |
| #if OMW_MULTIVIEW
 | |
|     return invProjectionMatrixMultiView[gl_ViewID_OVR];
 | |
| #else
 | |
|     return omw.invProjectionMatrix;
 | |
| #endif
 | |
| }
 | |
| 
 | |
|     float omw_GetDepth(vec2 uv)
 | |
|     {
 | |
| #if OMW_MULTIVIEW
 | |
|         float depth = omw_Texture2D(omw_SamplerDepth, vec3(uv, gl_ViewID_OVR)).r;
 | |
| #else
 | |
|         float depth = omw_Texture2D(omw_SamplerDepth, uv).r;
 | |
| #endif
 | |
| #if OMW_REVERSE_Z
 | |
|         return 1.0 - depth;
 | |
| #else
 | |
|         return depth;
 | |
| #endif
 | |
|     }
 | |
| 
 | |
|     vec4 omw_GetLastShader(vec2 uv)
 | |
|     {
 | |
| #if OMW_MULTIVIEW
 | |
|         return omw_Texture2D(omw_SamplerLastShader, vec3(uv, gl_ViewID_OVR));
 | |
| #else
 | |
|         return omw_Texture2D(omw_SamplerLastShader, uv);
 | |
| #endif
 | |
|     }
 | |
| 
 | |
|     vec4 omw_GetLastPass(vec2 uv)
 | |
|     {
 | |
| #if OMW_MULTIVIEW
 | |
|         return omw_Texture2D(omw_SamplerLastPass, vec3(uv, gl_ViewID_OVR));
 | |
| #else
 | |
|         return omw_Texture2D(omw_SamplerLastPass, uv);
 | |
| #endif
 | |
|     }
 | |
| 
 | |
|     vec3 omw_GetNormals(vec2 uv)
 | |
|     {
 | |
| #if OMW_MULTIVIEW
 | |
|         return omw_Texture2D(omw_SamplerNormals, vec3(uv, gl_ViewID_OVR)).rgb * 2.0 - 1.0;
 | |
| #else
 | |
|         return omw_Texture2D(omw_SamplerNormals, uv).rgb * 2.0 - 1.0;
 | |
| #endif
 | |
|     }
 | |
| 
 | |
|     vec3 omw_GetWorldPosFromUV(vec2 uv)
 | |
|     {
 | |
|         float depth = omw_GetDepth(uv);
 | |
| #if (OMW_REVERSE_Z == 1)
 | |
|         float flippedDepth = 1.0 - depth;
 | |
| #else
 | |
|         float flippedDepth = omw_Texture2D(omw_SamplerDepth, uv).r * 2.0 - 1.0;
 | |
| #endif
 | |
|         vec4 clip_space = vec4(uv * 2.0 - 1.0, flippedDepth, 1.0);
 | |
|         vec4 world_space = omw.invViewMatrix * (omw.invProjectionMatrix * clip_space);
 | |
|         return world_space.xyz / world_space.w;
 | |
|     }
 | |
| 
 | |
|     float omw_GetLinearDepth(vec2 uv)
 | |
|     {
 | |
| #if (OMW_REVERSE_Z == 1)
 | |
|         float depth = omw_GetDepth(uv);
 | |
|         float dist = omw.near * omw.far / (omw.far + depth * (omw.near - omw.far));
 | |
| #else
 | |
|         float depth = omw_GetDepth(uv) * 2.0 - 1.0;
 | |
|         float dist = 2.0 * omw.near * omw.far / (omw.far + omw.near - depth * (omw.far - omw.near));
 | |
| #endif
 | |
| 
 | |
|         return dist;
 | |
|     }
 | |
| 
 | |
| float omw_EstimateFogCoverageFromUV(vec2 uv)
 | |
|     {
 | |
| #if OMW_RADIAL_FOG
 | |
|         vec3 uvPos = omw_GetWorldPosFromUV(uv);
 | |
|         float dist = length(uvPos - omw.eyePos.xyz);
 | |
| #else
 | |
|         float dist = omw_GetLinearDepth(uv);
 | |
| #endif
 | |
| #if OMW_EXPONENTIAL_FOG
 | |
|         float fogValue = 1.0 - exp(-2.0 * max(0.0, dist - omw.fogNear/2.0) / (omw.fogFar - omw.fogNear/2.0));
 | |
| #else
 | |
|         float fogValue = clamp((dist - omw.fogNear) / (omw.fogFar - omw.fogNear), 0.0, 1.0);
 | |
| #endif
 | |
| 
 | |
|         return fogValue;
 | |
|     }
 | |
| 
 | |
| #if OMW_HDR
 | |
|     uniform sampler2D omw_EyeAdaptation;
 | |
| #endif
 | |
| 
 | |
|     float omw_GetEyeAdaptation()
 | |
|     {
 | |
| #if OMW_HDR
 | |
|         return omw_Texture2D(omw_EyeAdaptation, vec2(0.5, 0.5)).r;
 | |
| #else
 | |
|         return 1.0;
 | |
| #endif
 | |
|     }
 | |
| )GLSL";
 | |
| 
 | |
|         std::stringstream extBlock;
 | |
|         for (const auto& extension : technique.getGLSLExtensions())
 | |
|             extBlock << "#ifdef " << extension << '\n' << "\t#extension " << extension << ": enable" << '\n' << "#endif" << '\n';
 | |
| 
 | |
|         const std::vector<std::pair<std::string,std::string>> defines = {
 | |
|             {"@pointLightCount", std::to_string(SceneUtil::PPLightBuffer::sMaxPPLightsArraySize)},
 | |
|             {"@version", std::to_string(technique.getGLSLVersion())},
 | |
|             {"@multiview", Stereo::getMultiview() ? "1" : "0"},
 | |
|             {"@builtinSampler", Stereo::getMultiview() ? "sampler2DArray" : "sampler2D"},
 | |
|             {"@profile", technique.getGLSLProfile()},
 | |
|             {"@extensions", extBlock.str()},
 | |
|             {"@uboStruct", StateUpdater::getStructDefinition()},
 | |
|             {"@ubo", mUBO ? "1" : "0"},
 | |
|             {"@normals", technique.getNormals() ? "1" : "0"},
 | |
|             {"@reverseZ", SceneUtil::AutoDepth::isReversed() ? "1" : "0"},
 | |
|             {"@radialFog", Settings::Manager::getBool("radial fog", "Fog") ? "1" : "0"},
 | |
|             {"@exponentialFog", Settings::Manager::getBool("exponential fog", "Fog") ? "1" : "0"},
 | |
|             {"@hdr", technique.getHDR() ? "1" : "0"},
 | |
|             {"@in", mLegacyGLSL ? "varying" : "in"},
 | |
|             {"@out", mLegacyGLSL ? "varying" : "out"},
 | |
|             {"@position", "gl_Position"},
 | |
|             {"@texture1D", mLegacyGLSL ? "texture1D" : "texture"},
 | |
|             {"@texture2D", mLegacyGLSL ? "texture2D" : "texture"},
 | |
|             {"@texture3D", mLegacyGLSL ? "texture3D" : "texture"},
 | |
|             {"@vertex", mLegacyGLSL ? "gl_Vertex" : "_omw_Vertex"},
 | |
|             {"@fragColor", mLegacyGLSL ? "gl_FragColor" : "_omw_FragColor"},
 | |
|             {"@useBindings", mLegacyGLSL ? "0" : "1"},
 | |
|             {"@fragBinding", mLegacyGLSL ? "" : "out vec4 omw_FragColor;"}
 | |
|         };
 | |
| 
 | |
|         for (const auto& [define, value]: defines)
 | |
|             for (size_t pos = header.find(define); pos != std::string::npos; pos = header.find(define))
 | |
|                 header.replace(pos, define.size(), value);
 | |
| 
 | |
|         for (const auto& target : mRenderTargets)
 | |
|             header.append("uniform sampler2D " + std::string(target) + ";");
 | |
| 
 | |
|         for (auto& uniform : technique.getUniformMap())
 | |
|             if (auto glsl = uniform->getGLSL())
 | |
|                 header.append(glsl.value());
 | |
| 
 | |
|         header.append(preamble);
 | |
| 
 | |
|         return header;
 | |
|     }
 | |
| 
 | |
|     void Pass::prepareStateSet(osg::StateSet* stateSet, const std::string& name) const
 | |
|     {
 | |
|         osg::ref_ptr<osg::Program> program = new osg::Program;
 | |
|         if (mType == Type::Pixel)
 | |
|         {
 | |
|             program->addShader(new osg::Shader(*mVertex));
 | |
|             program->addShader(new osg::Shader(*mFragment));
 | |
|         }
 | |
|         else if (mType == Type::Compute)
 | |
|         {
 | |
|             program->addShader(new osg::Shader(*mCompute));
 | |
|         }
 | |
| 
 | |
|         if (mUBO)
 | |
|             program->addBindUniformBlock("_data", static_cast<int>(Resource::SceneManager::UBOBinding::PostProcessor));
 | |
| 
 | |
|         program->setName(name);
 | |
| 
 | |
|         if (!mLegacyGLSL)
 | |
|         {
 | |
|             program->addBindFragDataLocation("_omw_FragColor", 0);
 | |
|             program->addBindAttribLocation("_omw_Vertex", 0);
 | |
|         }
 | |
| 
 | |
|         stateSet->setAttribute(program);
 | |
| 
 | |
|         if (mBlendSource && mBlendDest)
 | |
|             stateSet->setAttributeAndModes(new osg::BlendFunc(mBlendSource.value(), mBlendDest.value()));
 | |
| 
 | |
|         if (mBlendEq)
 | |
|             stateSet->setAttributeAndModes(new osg::BlendEquation(mBlendEq.value()));
 | |
| 
 | |
|         if (mClearColor)
 | |
|             stateSet->setAttributeAndModes(new SceneUtil::ClearColor(mClearColor.value(), GL_COLOR_BUFFER_BIT));
 | |
|     }
 | |
| 
 | |
|     void Pass::dirty()
 | |
|     {
 | |
|         mVertex = nullptr;
 | |
|         mFragment = nullptr;
 | |
|         mCompute = nullptr;
 | |
|         mCompiled = false;
 | |
|     }
 | |
| 
 | |
|     void Pass::compile(Technique& technique, std::string_view preamble)
 | |
|     {
 | |
|         if (mCompiled)
 | |
|             return;
 | |
| 
 | |
|         mLegacyGLSL = technique.getGLSLVersion() != 330;
 | |
| 
 | |
|         if (mType == Type::Pixel)
 | |
|         {
 | |
|             if (!mVertex)
 | |
|                 mVertex = new osg::Shader(osg::Shader::VERTEX, Stereo::getMultiview() ? s_DefaultVertexMultiview : s_DefaultVertex);
 | |
| 
 | |
|             mVertex->setShaderSource(getPassHeader(technique, preamble).append(mVertex->getShaderSource()));
 | |
|             mFragment->setShaderSource(getPassHeader(technique, preamble, true).append(mFragment->getShaderSource()));
 | |
| 
 | |
|             mVertex->setName(mName);
 | |
|             mFragment->setName(mName);
 | |
|         }
 | |
|         else if (mType == Type::Compute)
 | |
|         {
 | |
|             mCompute->setShaderSource(getPassHeader(technique, preamble).append(mCompute->getShaderSource()));
 | |
|             mCompute->setName(mName);
 | |
|         }
 | |
| 
 | |
|         mCompiled = true;
 | |
|     }
 | |
| 
 | |
| }
 |