mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 03:26:37 +00:00 
			
		
		
		
	As we don't reconfigure all shaders without shadows when we disable them indoors (as it'd probably add a hitch to transitioning in and out) we need to set up dummy state so the shaders don't do anything illegal. This hadn't had symptoms for most objects as when indoors, nearly everything would be drawn first in one of the water RTTs, which had dummy state to disable shadows already. This wasn't true of the water plane itself, though, yet somehow it took until just now for anyone to report that. This resolves vtastek's issue where the water would be invisible indoors
		
			
				
	
	
		
			296 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			296 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* This file is based on OpenSceneGraph's include/osgShadow/ViewDependentShadowMap.
 | |
|  * Where applicable, any changes made are covered by OpenMW's GPL 3 license, not the OSGPL.
 | |
|  * The original copyright notice is listed below.
 | |
|  */
 | |
| 
 | |
| /* -*-c++-*- OpenSceneGraph - Copyright (C) 1998-2011 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.
 | |
| */
 | |
| 
 | |
| #ifndef COMPONENTS_SCENEUTIL_MWSHADOWTECHNIQUE_H
 | |
| #define COMPONENTS_SCENEUTIL_MWSHADOWTECHNIQUE_H 1
 | |
| 
 | |
| #include <array>
 | |
| #include <mutex>
 | |
| 
 | |
| #include <osg/Camera>
 | |
| #include <osg/Material>
 | |
| #include <osg/MatrixTransform>
 | |
| #include <osg/LightSource>
 | |
| #include <osg/PolygonOffset>
 | |
| 
 | |
| #include <osgShadow/ShadowTechnique>
 | |
| 
 | |
| #include <components/shader/shadermanager.hpp>
 | |
| #include <components/terrain/quadtreeworld.hpp>
 | |
| 
 | |
| namespace SceneUtil {
 | |
| 
 | |
|     /** ViewDependentShadowMap provides an base implementation of view dependent shadow mapping techniques.*/
 | |
|     class MWShadowTechnique : public osgShadow::ShadowTechnique
 | |
|     {
 | |
|     public:
 | |
|         MWShadowTechnique();
 | |
| 
 | |
|         MWShadowTechnique(const MWShadowTechnique& vdsm, const osg::CopyOp& copyop = osg::CopyOp::SHALLOW_COPY);
 | |
| 
 | |
|         META_Object(SceneUtil, MWShadowTechnique);
 | |
| 
 | |
|         /** initialize the ShadowedScene and local cached data structures.*/
 | |
|         void init() override;
 | |
| 
 | |
|         /** run the update traversal of the ShadowedScene and update any loca chached data structures.*/
 | |
|         void update(osg::NodeVisitor& nv) override;
 | |
| 
 | |
|         /** run the cull traversal of the ShadowedScene and set up the rendering for this ShadowTechnique.*/
 | |
|         void cull(osgUtil::CullVisitor& cv) override;
 | |
| 
 | |
|         /** Resize any per context GLObject buffers to specified size. */
 | |
|         void resizeGLObjectBuffers(unsigned int maxSize) override;
 | |
| 
 | |
|         /** If State is non-zero, this function releases any associated OpenGL objects for
 | |
|         * the specified graphics context. Otherwise, releases OpenGL objects
 | |
|         * for all graphics contexts. */
 | |
|         void releaseGLObjects(osg::State* = 0) const override;
 | |
| 
 | |
|         /** Clean scene graph from any shadow technique specific nodes, state and drawables.*/
 | |
|         void cleanSceneGraph() override;
 | |
| 
 | |
|         virtual void enableShadows();
 | |
| 
 | |
|         virtual void disableShadows(bool setDummyState = false);
 | |
| 
 | |
|         virtual void enableDebugHUD();
 | |
| 
 | |
|         virtual void disableDebugHUD();
 | |
| 
 | |
|         virtual void setSplitPointUniformLogarithmicRatio(double ratio);
 | |
| 
 | |
|         virtual void setSplitPointDeltaBias(double bias);
 | |
| 
 | |
|         virtual void setPolygonOffset(float factor, float units);
 | |
| 
 | |
|         virtual void setShadowFadeStart(float shadowFadeStart);
 | |
| 
 | |
|         virtual void enableFrontFaceCulling();
 | |
| 
 | |
|         virtual void disableFrontFaceCulling();
 | |
| 
 | |
|         virtual void setupCastingShader(Shader::ShaderManager &shaderManager);
 | |
| 
 | |
|         class ComputeLightSpaceBounds : public osg::NodeVisitor, public osg::CullStack
 | |
|         {
 | |
|         public:
 | |
|             ComputeLightSpaceBounds(osg::Viewport* viewport, const osg::Matrixd& projectionMatrix, osg::Matrixd& viewMatrix);
 | |
| 
 | |
|             void apply(osg::Node& node) override;
 | |
| 
 | |
|             void apply(osg::Drawable& drawable) override;
 | |
| 
 | |
|             void apply(Terrain::QuadTreeWorld& quadTreeWorld);
 | |
| 
 | |
|             void apply(osg::Billboard&) override;
 | |
| 
 | |
|             void apply(osg::Projection&) override;
 | |
| 
 | |
|             void apply(osg::Transform& transform) override;
 | |
| 
 | |
|             void apply(osg::Camera&) override;
 | |
| 
 | |
|             using osg::NodeVisitor::apply;
 | |
| 
 | |
|             void updateBound(const osg::BoundingBox& bb);
 | |
| 
 | |
|             void update(const osg::Vec3& v);
 | |
| 
 | |
|             osg::BoundingBox _bb;
 | |
|         };
 | |
| 
 | |
|         struct Frustum
 | |
|         {
 | |
|             Frustum(osgUtil::CullVisitor* cv, double minZNear, double maxZFar);
 | |
| 
 | |
|             osg::Matrixd projectionMatrix;
 | |
|             osg::Matrixd modelViewMatrix;
 | |
| 
 | |
|             typedef std::vector<osg::Vec3d> Vertices;
 | |
|             Vertices corners;
 | |
| 
 | |
|             typedef std::vector<unsigned int> Indices;
 | |
|             typedef std::vector<Indices> Faces;
 | |
|             Faces faces;
 | |
| 
 | |
|             typedef std::vector<Indices> Edges;
 | |
|             Edges edges;
 | |
| 
 | |
|             osg::Vec3d eye;
 | |
|             osg::Vec3d centerNearPlane;
 | |
|             osg::Vec3d centerFarPlane;
 | |
|             osg::Vec3d center;
 | |
|             osg::Vec3d frustumCenterLine;
 | |
|         };
 | |
| 
 | |
|         // forward declare
 | |
|         class ViewDependentData;
 | |
| 
 | |
|         struct LightData : public osg::Referenced
 | |
|         {
 | |
|             LightData(ViewDependentData* vdd);
 | |
| 
 | |
|             virtual void setLightData(osg::RefMatrix* lm, const osg::Light* l, const osg::Matrixd& modelViewMatrix);
 | |
| 
 | |
|             ViewDependentData*                  _viewDependentData;
 | |
| 
 | |
|             osg::ref_ptr<osg::RefMatrix>        lightMatrix;
 | |
|             osg::ref_ptr<const osg::Light>      light;
 | |
| 
 | |
|             osg::Vec4d                          lightPos;
 | |
|             osg::Vec3d                          lightPos3;
 | |
|             osg::Vec3d                          lightDir;
 | |
|             bool                                directionalLight;
 | |
| 
 | |
|             typedef std::vector<unsigned int> ActiveTextureUnits;
 | |
|             ActiveTextureUnits                   textureUnits;
 | |
|         };
 | |
| 
 | |
|         typedef std::list< osg::ref_ptr<LightData> > LightDataList;
 | |
| 
 | |
|         struct ShadowData : public osg::Referenced
 | |
|         {
 | |
|             ShadowData(ViewDependentData* vdd);
 | |
| 
 | |
|             virtual void releaseGLObjects(osg::State* = 0) const;
 | |
| 
 | |
|             ViewDependentData*                  _viewDependentData;
 | |
| 
 | |
|             unsigned int                        _textureUnit;
 | |
|             osg::ref_ptr<osg::Texture2D>        _texture;
 | |
|             osg::ref_ptr<osg::TexGen>           _texgen;
 | |
|             osg::ref_ptr<osg::Camera>           _camera;
 | |
|         };
 | |
| 
 | |
|         typedef std::list< osg::ref_ptr<ShadowData> > ShadowDataList;
 | |
| 
 | |
| 
 | |
|         class ViewDependentData : public osg::Referenced
 | |
|         {
 | |
|         public:
 | |
|             ViewDependentData(MWShadowTechnique* vdsm);
 | |
| 
 | |
|             const MWShadowTechnique* getViewDependentShadowMap() const { return _viewDependentShadowMap; }
 | |
| 
 | |
|             LightDataList& getLightDataList() { return _lightDataList; }
 | |
| 
 | |
|             ShadowDataList& getShadowDataList() { return _shadowDataList; }
 | |
| 
 | |
|             osg::StateSet* getStateSet(unsigned int traversalNumber) { return _stateset[traversalNumber % 2].get(); }
 | |
| 
 | |
|             virtual void releaseGLObjects(osg::State* = 0) const;
 | |
| 
 | |
|         protected:
 | |
|             virtual ~ViewDependentData() {}
 | |
| 
 | |
|             MWShadowTechnique*          _viewDependentShadowMap;
 | |
| 
 | |
|             std::array<osg::ref_ptr<osg::StateSet>, 2> _stateset;
 | |
| 
 | |
|             LightDataList               _lightDataList;
 | |
|             ShadowDataList              _shadowDataList;
 | |
|         };
 | |
| 
 | |
|         virtual ViewDependentData* createViewDependentData(osgUtil::CullVisitor* cv);
 | |
| 
 | |
|         ViewDependentData* getViewDependentData(osgUtil::CullVisitor* cv);
 | |
| 
 | |
| 
 | |
| 
 | |
|         virtual void createShaders();
 | |
| 
 | |
|         virtual bool selectActiveLights(osgUtil::CullVisitor* cv, ViewDependentData* vdd) const;
 | |
| 
 | |
|         virtual osg::Polytope computeLightViewFrustumPolytope(Frustum& frustum, LightData& positionedLight);
 | |
| 
 | |
|         virtual bool computeShadowCameraSettings(Frustum& frustum, LightData& positionedLight, osg::Matrixd& projectionMatrix, osg::Matrixd& viewMatrix);
 | |
| 
 | |
|         virtual bool cropShadowCameraToMainFrustum(Frustum& frustum, osg::Camera* camera, double viewNear, double viewFar, std::vector<osg::Plane>& planeList);
 | |
| 
 | |
|         virtual bool adjustPerspectiveShadowMapCameraSettings(osgUtil::RenderStage* renderStage, Frustum& frustum, LightData& positionedLight, osg::Camera* camera, double viewNear, double viewFar);
 | |
| 
 | |
|         virtual bool assignTexGenSettings(osgUtil::CullVisitor* cv, osg::Camera* camera, unsigned int textureUnit, osg::TexGen* texgen);
 | |
| 
 | |
|         virtual void cullShadowReceivingScene(osgUtil::CullVisitor* cv) const;
 | |
| 
 | |
|         virtual void cullShadowCastingScene(osgUtil::CullVisitor* cv, osg::Camera* camera) const;
 | |
| 
 | |
|         virtual osg::StateSet* selectStateSetForRenderingShadow(ViewDependentData& vdd, unsigned int traversalNumber) const;
 | |
| 
 | |
|     protected:
 | |
|         virtual ~MWShadowTechnique();
 | |
| 
 | |
|         typedef std::map< osgUtil::CullVisitor*, osg::ref_ptr<ViewDependentData> >  ViewDependentDataMap;
 | |
|         mutable std::mutex                      _viewDependentDataMapMutex;
 | |
|         ViewDependentDataMap                    _viewDependentDataMap;
 | |
| 
 | |
|         osg::ref_ptr<osg::StateSet>             _shadowRecievingPlaceholderStateSet;
 | |
| 
 | |
|         osg::ref_ptr<osg::StateSet>             _shadowCastingStateSet;
 | |
|         osg::ref_ptr<osg::PolygonOffset>        _polygonOffset;
 | |
|         osg::ref_ptr<osg::Texture2D>            _fallbackBaseTexture;
 | |
|         osg::ref_ptr<osg::Texture2D>            _fallbackShadowMapTexture;
 | |
| 
 | |
|         typedef std::vector< osg::ref_ptr<osg::Uniform> > Uniforms;
 | |
|         std::array<Uniforms, 2>                 _uniforms;
 | |
|         osg::ref_ptr<osg::Program>              _program;
 | |
| 
 | |
|         bool                                    _enableShadows;
 | |
|         bool                                    mSetDummyStateWhenDisabled;
 | |
| 
 | |
|         double                                  _splitPointUniformLogRatio = 0.5;
 | |
|         double                                  _splitPointDeltaBias = 0.0;
 | |
| 
 | |
|         float                                   _polygonOffsetFactor = 1.1;
 | |
|         float                                   _polygonOffsetUnits = 4.0;
 | |
| 
 | |
|         bool                                    _useFrontFaceCulling = true;
 | |
| 
 | |
|         float                                   _shadowFadeStart = 0.0;
 | |
| 
 | |
|         class DebugHUD final : public osg::Referenced
 | |
|         {
 | |
|         public:
 | |
|             DebugHUD(int numberOfShadowMapsPerLight);
 | |
| 
 | |
|             void draw(osg::ref_ptr<osg::Texture2D> texture, unsigned int shadowMapNumber, const osg::Matrixd &matrix, osgUtil::CullVisitor& cv);
 | |
| 
 | |
|             void releaseGLObjects(osg::State* state = 0) const;
 | |
| 
 | |
|             void setFrustumVertices(osg::ref_ptr<osg::Vec3Array> vertices, unsigned int traversalNumber);
 | |
|         protected:
 | |
|             void addAnotherShadowMap();
 | |
| 
 | |
|             static const int sDebugTextureUnit = 0;
 | |
| 
 | |
|             std::vector<osg::ref_ptr<osg::Camera>> mDebugCameras;
 | |
|             osg::ref_ptr<osg::Program> mDebugProgram;
 | |
|             std::vector<osg::ref_ptr<osg::Node>> mDebugGeometry;
 | |
|             std::vector<osg::ref_ptr<osg::Group>> mFrustumTransforms;
 | |
|             std::array<std::vector<osg::ref_ptr<osg::Uniform>>, 2> mFrustumUniforms;
 | |
|             std::array<osg::ref_ptr<osg::Geometry>, 2> mFrustumGeometries;
 | |
|         };
 | |
| 
 | |
|         osg::ref_ptr<DebugHUD>                  _debugHud;
 | |
|         osg::ref_ptr<osg::Program>              _castingProgram;
 | |
|     };
 | |
| 
 | |
| }
 | |
| 
 | |
| #endif
 |