]> git.tdb.fi Git - libs/gl.git/blob - source/effects/shadowmap.cpp
Use persistent cameras in effects
[libs/gl.git] / source / effects / shadowmap.cpp
1 #include <cmath>
2 #include <cstdlib>
3 #include "camera.h"
4 #include "light.h"
5 #include "renderer.h"
6 #include "resources.h"
7 #include "scene.h"
8 #include "shadowmap.h"
9 #include "tests.h"
10
11 using namespace std;
12
13 namespace Msp {
14 namespace GL {
15
16 ShadowMap::ShadowMap(Resources &resources, unsigned s, Renderable &r, const Light &l):
17         Effect(r),
18         size(s),
19         light(l),
20         sampler(resources.get<Sampler>("_linear_clamp_shadow.samp")),
21         radius(1),
22         depth_bias(4),
23         rendered(false)
24 {
25         depth_buf.storage(DEPTH_COMPONENT32F, size, size, 1);
26         fbo.attach(DEPTH_ATTACHMENT, depth_buf, 0);
27         fbo.require_complete();
28
29         set_darkness(0.7);
30 }
31
32 void ShadowMap::set_target(const Vector3 &t, float r)
33 {
34         target = t;
35         radius = r;
36 }
37
38 void ShadowMap::set_darkness(float d)
39 {
40         if(d<0.0f || d>1.0f)
41                 throw invalid_argument("ShadowMap::set_darkness");
42
43         shdata.uniform("shadow_darkness", d);
44 }
45
46 void ShadowMap::set_depth_bias(float b)
47 {
48         if(b<0.0f)
49                 throw invalid_argument("ShadowMap::set_depth_bias");
50
51         depth_bias = b;
52 }
53
54 void ShadowMap::setup_frame(Renderer &renderer)
55 {
56         if(rendered)
57                 return;
58
59         rendered = true;
60         renderable.setup_frame(renderer);
61
62         shadow_camera.set_object_matrix(*light.get_matrix());
63         shadow_camera.set_position(target);
64         // TODO support point and spot lights with a frustum projection.
65         // Omnidirectional lights also need a cube shadow map.
66         shadow_camera.set_orthographic(radius*2, radius*2);
67         shadow_camera.set_depth_clip(-radius, radius);
68
69         shadow_matrix = shadow_camera.get_object_matrix();
70         shadow_matrix.scale(radius*2, radius*2, -radius*2);
71         shadow_matrix.translate(-0.5, -0.5, depth_bias/size-0.5);
72         shadow_matrix.invert();
73
74         BindRestore bind_fbo(fbo);
75         Bind bind_depth(DepthTest::lequal());
76         fbo.clear(DEPTH_BUFFER_BIT);
77
78         Renderer::Push push(renderer);
79         renderer.set_camera(shadow_camera);
80
81         renderer.render(renderable, "shadow");
82 }
83
84 void ShadowMap::finish_frame()
85 {
86         renderable.finish_frame();
87         rendered = false;
88 }
89
90 void ShadowMap::render(Renderer &renderer, Tag tag) const
91 {
92         if(!enabled_passes.count(tag))
93                 return renderer.render(renderable, tag);
94
95         if(const Camera *camera = renderer.get_camera())
96                 /* Multiply by camera's object matrix to form a matrix that transforms
97                 from eye space to shadow space. */
98                 shdata.uniform("shd_eye_matrix", shadow_matrix*camera->get_object_matrix());
99         else
100                 shdata.uniform("shd_eye_matrix", shadow_matrix);
101
102         Renderer::Push _push_rend(renderer);
103
104         renderer.set_texture("shadow_map", &depth_buf, &sampler);
105         renderer.add_shader_data(shdata);
106         renderer.render(renderable, tag);
107 }
108
109 } // namespace GL
110 } // namespace Msp