TDME2 1.9.121
ShadowMapping.cpp
Go to the documentation of this file.
2
3#include <string>
4#include <vector>
5
6#include <tdme/tdme.h>
13#include <tdme/engine/Engine.h>
14#include <tdme/engine/Light.h>
15#include <tdme/math/Matrix4x4.h>
16#include <tdme/math/Vector3.h>
17#include <tdme/math/Vector4.h>
19
20using std::string;
21using std::to_string;
22using std::vector;
23
28using tdme::engine::subsystems::shadowmapping::ShadowMapping_RunState;
37
38ShadowMapping::ShadowMapping(Engine* engine, Renderer* renderer, EntityRenderer* entityRenderer)
39{
40 this->engine = engine;
41 this->renderer = renderer;
42 this->entityRenderer = entityRenderer;
44 for (auto i = 0; i < shadowMaps.size(); i++) {
45 shadowMaps[i] = nullptr;
46 }
48 runState = ShadowMapping_RunState::NONE;
49}
50
52 for (auto i = 0; i < shadowMaps.size(); i++) {
53 if (shadowMaps[i] != nullptr) delete shadowMaps[i];
54 }
55}
56
58{
59 return engine;
60}
61
62void ShadowMapping::reshape(int32_t width, int32_t height)
63{
64}
65
67{
68 runState = ShadowMapping_RunState::CREATE;
69 // disable color rendering, we only want to write to the Z-Buffer
70 renderer->setColorMask(false, false, false, false);
71 // render backfaces only, avoid self-shadowing
73 // render to shadow maps
74 for (auto i = 0; i < engine->getLightCount(); i++) {
75 auto light = engine->getLightAt(i);
76 if (light->isEnabled() == true &&
77 light->getSpotDirection().computeLengthSquared() > Math::square(Math::EPSILON)) {
78 // create shadow map for light, if required
79 if (shadowMaps[i] == nullptr) {
80 auto shadowMap = new ShadowMap(this, Engine::getShadowMapWidth(), Engine::getShadowMapHeight());
81 shadowMap->initialize();
82 shadowMaps[i] = shadowMap;
83 }
84 // render
86 shadowMaps[i]->createShadowMap(light);
88 } else {
89 if (shadowMaps[i] != nullptr) {
90 // dispose shadow map
91 shadowMaps[i]->dispose();
92 delete shadowMaps[i];
93 shadowMaps[i] = nullptr;
94 }
95 }
96 }
97 // restore disable color rendering
98 renderer->setColorMask(true, true, true, true);
99 // restore render backfaces only
101 //
102 runState = ShadowMapping_RunState::NONE;
103}
104
106 return shadowMaps[idx];
107}
108
109void ShadowMapping::renderShadowMaps(const vector<Object3D*>& visibleObjects)
110{
111 // only render for objects that receives shadows
112 for (auto object: visibleObjects) {
113 if (object->isReceivesShadows() == false) continue;
114 visibleObjectsReceivingShadows.push_back(object);
115 }
116
117 //
118 auto contextCount = renderer->isSupportingMultithreadedRendering() == true?Engine::getThreadCount():1;
119 //
120 runState = ShadowMapping_RunState::RENDER;
121 // render using shadow mapping program
122 auto shader = Engine::getShadowMapRenderShader();
123 // do not allow writing to depth buffer
125 // user shader program
126 shader->useProgram(engine);
127 // render each shadow map
129
130 for (auto i = 0; i < shadowMaps.size(); i++) {
131 // skip on unused shadow mapping
132 if (shadowMaps[i] == nullptr) continue;
133
134 //
135 auto shadowMap = shadowMaps[i];
136 // set light to render
137 shader->setRenderLightId(i);
138
139 // setup depth textures to contexts
140 for (auto j = 0; j < contextCount; j++) {
141 // use default context
142 auto contextIdx = j;
143 // set up light shader uniforms
144 shadowMap->updateDepthBiasMVPMatrix(contextIdx);
145 //
146 // bind shadow map texture on shadow map texture unit
147 auto textureUnit = renderer->getTextureUnit(contextIdx);
149 shadowMap->bindDepthBufferTexture(contextIdx);
150 // switch back to texture last unit
151 renderer->setTextureUnit(contextIdx, textureUnit);
152 }
153
154 // only opaque face entities as shadows will not be produced on transparent faces
155 for (auto i = 0; i < Entity::RENDERPASS_MAX; i++) {
156 auto renderPass = static_cast<Entity::RenderPass>(Math::pow(2, i));
158 renderPass,
160 false,
161 EntityRenderer::RENDERTYPE_NORMALS |
162 EntityRenderer::RENDERTYPE_TEXTUREARRAYS_DIFFUSEMASKEDTRANSPARENCY |
163 EntityRenderer::RENDERTYPE_TEXTURES_DIFFUSEMASKEDTRANSPARENCY |
164 EntityRenderer::RENDERTYPE_SHADOWMAPPING
165 );
166 }
167 }
168
169 //
171
172 // unuse shader program
173 shader->unUseProgram();
174
175 // restore depth textures of contexts
176 for (auto j = 0; j < contextCount; j++) {
177 // use default context
178 auto contextIdx = j;
179 auto textureUnit = renderer->getTextureUnit(contextIdx);
181 renderer->bindTexture(contextIdx, renderer->ID_NONE);
182 renderer->setTextureUnit(contextIdx, textureUnit);
183 }
184
185 // restore render defaults
188
189 //
191
192 //
193 runState = ShadowMapping_RunState::NONE;
194}
195
197{
198 // dispose shadow mappings
199 for (auto i = 0; i < shadowMaps.size(); i++) {
200 if (shadowMaps[i] != nullptr) {
201 shadowMaps[i]->dispose();
202 delete shadowMaps[i];
203 shadowMaps[i] = nullptr;
204 }
205 }
206}
207
208void ShadowMapping::startObjectTransformations(int contextIdx, Matrix4x4& transformationsMatrix)
209{
210 if (runState != ShadowMapping_RunState::RENDER)
211 return;
212
213 // retrieve current model view matrix and put it on stack
214 Matrix4x4 tmpMatrix;
216 // set up new model view matrix
217 tmpMatrix.set(depthBiasMVPMatrix);
218 depthBiasMVPMatrix.set(transformationsMatrix).multiply(tmpMatrix);
219
220 //
221 updateDepthBiasMVPMatrix(contextIdx);
222}
223
225{
226 if (runState != ShadowMapping_RunState::RENDER)
227 return;
228 // set up new model view matrix
230}
231
233{
234 if (runState == ShadowMapping_RunState::NONE) return;
235
236 //
237 switch(runState) {
238 case ShadowMapping_RunState::CREATE:
240 break;
241 case ShadowMapping_RunState::RENDER:
243 break;
244 default:
245 Console::println(string("ShadowMapping::updateTextureMatrix(): unsupported run state '" + to_string(runState)));
246 }
247}
248
250{
251 if (runState == ShadowMapping_RunState::NONE) return;
252
253 //
254 switch(runState) {
255 case ShadowMapping_RunState::CREATE:
257 break;
258 case ShadowMapping_RunState::RENDER:
260 break;
261 default:
262 Console::println(string("ShadowMapping::updateMatrices(): unsupported run state '" + to_string(runState)));
263 }
264}
265
266void ShadowMapping::updateMaterial(int contextIdx) {
267 if (runState == ShadowMapping_RunState::NONE)
268 return;
269
270 //
271 switch(runState) {
272 case ShadowMapping_RunState::CREATE:
274 break;
275 case ShadowMapping_RunState::RENDER:
277 break;
278 default:
279 Console::println(string("ShadowMapping::updateMaterial(): unsupported run state '" + to_string(runState)));
280 }
281}
282
283void ShadowMapping::updateLight(int contextIdx, int32_t lightId) {
284 if (runState == ShadowMapping_RunState::RENDER) {
285 Engine::getShadowMapRenderShader()->updateLight(contextIdx, lightId);
286 }
287}
288
289void ShadowMapping::bindTexture(int contextIdx, int32_t textureId) {
290 if (runState == ShadowMapping_RunState::NONE)
291 return;
292
293 //
294 switch(runState) {
295 case ShadowMapping_RunState::CREATE:
296 Engine::getShadowMapCreationShader()->bindTexture(contextIdx, textureId);
297 break;
298 case ShadowMapping_RunState::RENDER:
299 Engine::getShadowMapRenderShader()->bindTexture(contextIdx, textureId);
300 break;
301 default:
302 Console::println(string("ShadowMapping::bindTexture(): unsupported run state '" + to_string(runState)));
303 }
304}
305
306void ShadowMapping::updateDepthBiasMVPMatrix(int contextIdx, Matrix4x4& depthBiasMVPMatrix)
307{
308 if (runState != ShadowMapping_RunState::RENDER)
309 return;
310
311 // copy matrix
312 this->depthBiasMVPMatrix.set(depthBiasMVPMatrix);
313 // upload
315}
316
318{
319 if (runState != ShadowMapping_RunState::RENDER)
320 return;
321
322 // upload
324}
325
326void ShadowMapping::setShader(int contextIdx, const string& id) {
327 if (runState == ShadowMapping_RunState::NONE)
328 return;
329
330 //
331 switch(runState) {
332 case ShadowMapping_RunState::CREATE:
334 break;
335 case ShadowMapping_RunState::RENDER:
337 break;
338 default:
339 Console::println(string("ShadowMapping::setShader(): unsupported run state '" + to_string(runState)));
340 }
341}
342
344 if (runState == ShadowMapping_RunState::NONE)
345 return;
346
347 //
348 switch(runState) {
349 case ShadowMapping_RunState::CREATE:
351 break;
352 case ShadowMapping_RunState::RENDER:
354 break;
355 default:
356 Console::println(string("ShadowMapping::updateShaderParameters(): unsupported run state '" + to_string(runState)));
357 }
358}
359
Engine main class.
Definition: Engine.h:122
int32_t getLightCount()
Definition: Engine.h:941
static ShadowMapRenderShader * getShadowMapRenderShader()
Definition: Engine.h:407
Light * getLightAt(int32_t idx)
Returns light at idx (0 <= idx < 8)
Definition: Engine.h:950
static int32_t getShadowMapWidth()
Definition: Engine.h:639
static ShadowMapCreationShader * getShadowMapCreationShader()
Definition: Engine.h:400
static int32_t getShadowMapHeight()
Definition: Engine.h:646
static int getThreadCount()
Definition: Engine.h:579
static constexpr int RENDERPASS_MAX
Definition: Entity.h:56
Light representation.
Definition: Light.h:32
virtual void setTextureUnit(int contextIdx, int32_t textureUnit)=0
Sets up texture unit.
virtual void enableDepthBufferWriting()=0
Enable depth buffer writing.
virtual void setCullFace(int32_t cullFace)=0
Sets up which face will be culled.
virtual int32_t getTextureUnit(int contextIdx)=0
Get texture unit.
virtual void enableBlending()=0
Enables blending.
virtual void disableBlending()=0
Disables blending.
virtual void setColorMask(bool red, bool green, bool blue, bool alpha)=0
Set up GL rendering colormask.
virtual void bindTexture(int contextIdx, int32_t textureId)=0
Binds a texture with given id or unbinds when using ID_NONE.
virtual void disableDepthBufferWriting()=0
Disable depth buffer writing.
void render(Entity::RenderPass renderPass, const vector< Object3D * > &objects, bool renderTransparentFaces, int32_t renderTypes)
Renders all given objects.
void useProgram(Engine *engine)
Use shadow map creation shader program.
void setShader(int contextIdx, const string &id)
Set shader.
void updateShaderParameters(int contextIdx)
Update shader parameters.
void bindTexture(int contextIdx, int32_t textureId)
Bind texture.
void unUseProgram()
Unuse shadow map creation shader program.
void updateLight(int contextIdx, int32_t lightId)
Update light.
void setDepthBiasMVPMatrix(int contextIdx, const Matrix4x4 &depthBiasMVPMatrix)
Set up program depth bias mvp matrix.
void setShader(int contextIdx, const string &id)
Set shader.
void updateTextureMatrix(int contextIdx)
Update up texture matrix.
void updateShaderParameters(int contextIdx)
Update shader parameters.
void bindTexture(int contextIdx, int32_t textureId)
Bind texture.
void updateDepthBiasMVPMatrix(int contextIdx, Matrix4x4 &depthBiasMVPMatrix)
Update depth bias mvp matrix with given matrix.
void reshape(int32_t width, int32_t height)
Reshape shadow maps.
void updateLight(int contextIdx, int32_t lightId)
Update light.
void updateMatrices(int contextIdx)
Update matrices.
void renderShadowMaps(const vector< Object3D * > &visibleObjects)
Render shadow maps to world.
ShadowMap * getShadowMap(int idx)
Get shadow map.
void setShader(int contextIdx, const string &id)
Set shader.
void startObjectTransformations(int contextIdx, Matrix4x4 &transformationsMatrix)
Start object transformations.
void updateTextureMatrix(int contextIdx)
Update texture matrix.
void updateShaderParameters(int contextIdx)
Update shader parameters.
void updateMaterial(int contextIdx)
Update material.
void endObjectTransformations()
End object transformations.
void bindTexture(int contextIdx, int32_t textureId)
Bind texture.
4x4 3D Matrix class
Definition: Matrix4x4.h:24
Matrix4x4 & identity()
Setup identity matrix.
Definition: Matrix4x4.h:326
Matrix4x4 & set(float r0c0, float r1c0, float r2c0, float r3c0, float r0c1, float r1c1, float r2c1, float r3c1, float r0c2, float r1c2, float r2c2, float r3c2, float r0c3, float r1c3, float r2c3, float r3c3)
Set up matrix by values.
Definition: Matrix4x4.h:95
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Definition: Matrix4x4.h:351
3D vector 3 class
Definition: Vector3.h:22
3D vector 4 class
Definition: Vector4.h:19
Console class.
Definition: Console.h:26