Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
godotengine
GitHub Repository: godotengine/godot
Path: blob/master/servers/rendering/renderer_rd/environment/fog.h
10279 views
1
/**************************************************************************/
2
/* fog.h */
3
/**************************************************************************/
4
/* This file is part of: */
5
/* GODOT ENGINE */
6
/* https://godotengine.org */
7
/**************************************************************************/
8
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
9
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
10
/* */
11
/* Permission is hereby granted, free of charge, to any person obtaining */
12
/* a copy of this software and associated documentation files (the */
13
/* "Software"), to deal in the Software without restriction, including */
14
/* without limitation the rights to use, copy, modify, merge, publish, */
15
/* distribute, sublicense, and/or sell copies of the Software, and to */
16
/* permit persons to whom the Software is furnished to do so, subject to */
17
/* the following conditions: */
18
/* */
19
/* The above copyright notice and this permission notice shall be */
20
/* included in all copies or substantial portions of the Software. */
21
/* */
22
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
23
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
24
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
25
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
26
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
27
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
28
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
29
/**************************************************************************/
30
31
#pragma once
32
33
#include "core/templates/local_vector.h"
34
#include "core/templates/rid_owner.h"
35
#include "servers/rendering/environment/renderer_fog.h"
36
#include "servers/rendering/renderer_rd/cluster_builder_rd.h"
37
#include "servers/rendering/renderer_rd/environment/gi.h"
38
#include "servers/rendering/renderer_rd/shaders/environment/volumetric_fog.glsl.gen.h"
39
#include "servers/rendering/renderer_rd/shaders/environment/volumetric_fog_process.glsl.gen.h"
40
#include "servers/rendering/renderer_rd/storage_rd/render_buffer_custom_data_rd.h"
41
#include "servers/rendering/storage/utilities.h"
42
43
#define RB_SCOPE_FOG SNAME("Fog")
44
45
namespace RendererRD {
46
47
class Fog : public RendererFog {
48
private:
49
static Fog *singleton;
50
51
static int _get_fog_shader_group();
52
static int _get_fog_variant();
53
static int _get_fog_process_variant(int p_idx);
54
55
/* FOG VOLUMES */
56
57
struct FogVolume {
58
RID material;
59
Vector3 size = Vector3(2, 2, 2);
60
61
RS::FogVolumeShape shape = RS::FOG_VOLUME_SHAPE_BOX;
62
63
Dependency dependency;
64
};
65
66
mutable RID_Owner<FogVolume, true> fog_volume_owner;
67
68
struct FogVolumeInstance {
69
RID volume;
70
Transform3D transform;
71
bool active = false;
72
};
73
74
mutable RID_Owner<FogVolumeInstance> fog_volume_instance_owner;
75
76
const int SAMPLERS_BINDING_FIRST_INDEX = 3;
77
78
/* Volumetric Fog */
79
struct VolumetricFogShader {
80
enum ShaderGroup {
81
SHADER_GROUP_BASE,
82
SHADER_GROUP_NO_ATOMICS,
83
SHADER_GROUP_VULKAN_MEMORY_MODEL,
84
SHADER_GROUP_VULKAN_MEMORY_MODEL_NO_ATOMICS,
85
};
86
87
enum FogSet {
88
FOG_SET_BASE,
89
FOG_SET_UNIFORMS,
90
FOG_SET_MATERIAL,
91
FOG_SET_MAX,
92
};
93
94
struct FogPushConstant {
95
float position[3];
96
float pad;
97
98
float size[3];
99
float pad2;
100
101
int32_t corner[3];
102
uint32_t shape;
103
104
float transform[16];
105
};
106
107
struct VolumeUBO {
108
float fog_frustum_size_begin[2];
109
float fog_frustum_size_end[2];
110
111
float fog_frustum_end;
112
float z_near;
113
float z_far;
114
float time;
115
116
int32_t fog_volume_size[3];
117
uint32_t directional_light_count;
118
119
uint32_t use_temporal_reprojection;
120
uint32_t temporal_frame;
121
float detail_spread;
122
float temporal_blend;
123
124
float to_prev_view[16];
125
float transform[16];
126
};
127
128
ShaderCompiler compiler;
129
VolumetricFogShaderRD shader;
130
RID volume_ubo;
131
132
RID default_shader;
133
RID default_material;
134
RID default_shader_rd;
135
136
RID base_uniform_set;
137
138
RID params_ubo;
139
140
enum {
141
VOLUMETRIC_FOG_PROCESS_SHADER_DENSITY,
142
VOLUMETRIC_FOG_PROCESS_SHADER_DENSITY_WITH_SDFGI,
143
VOLUMETRIC_FOG_PROCESS_SHADER_FILTER,
144
VOLUMETRIC_FOG_PROCESS_SHADER_FOG,
145
VOLUMETRIC_FOG_PROCESS_SHADER_COPY,
146
VOLUMETRIC_FOG_PROCESS_SHADER_MAX,
147
};
148
149
struct ParamsUBO {
150
float fog_frustum_size_begin[2];
151
float fog_frustum_size_end[2];
152
153
float fog_frustum_end;
154
float ambient_inject;
155
float z_far;
156
uint32_t filter_axis;
157
158
float ambient_color[3];
159
float sky_contribution;
160
161
int32_t fog_volume_size[3];
162
uint32_t directional_light_count;
163
164
float base_emission[3];
165
float base_density;
166
167
float base_scattering[3];
168
float phase_g;
169
170
float detail_spread;
171
float gi_inject;
172
uint32_t max_voxel_gi_instances;
173
uint32_t cluster_type_size;
174
175
float screen_size[2];
176
uint32_t cluster_shift;
177
uint32_t cluster_width;
178
179
uint32_t max_cluster_element_count_div_32;
180
uint32_t use_temporal_reprojection;
181
uint32_t temporal_frame;
182
float temporal_blend;
183
184
float cam_rotation[12];
185
float to_prev_view[16];
186
float radiance_inverse_xform[12];
187
};
188
189
VolumetricFogProcessShaderRD process_shader;
190
191
RID process_shader_version;
192
RID process_pipelines[VOLUMETRIC_FOG_PROCESS_SHADER_MAX];
193
194
} volumetric_fog;
195
196
Vector3i _point_get_position_in_froxel_volume(const Vector3 &p_point, float fog_end, const Vector2 &fog_near_size, const Vector2 &fog_far_size, float volumetric_fog_detail_spread, const Vector3 &fog_size, const Transform3D &p_cam_transform);
197
198
struct FogShaderData : public RendererRD::MaterialStorage::ShaderData {
199
bool valid = false;
200
RID version;
201
202
RID pipeline;
203
Vector<ShaderCompiler::GeneratedCode::Texture> texture_uniforms;
204
205
Vector<uint32_t> ubo_offsets;
206
uint32_t ubo_size = 0;
207
208
String code;
209
210
bool uses_time = false;
211
212
virtual void set_code(const String &p_Code);
213
virtual bool is_animated() const;
214
virtual bool casts_shadows() const;
215
virtual RS::ShaderNativeSourceCode get_native_source_code() const;
216
virtual Pair<ShaderRD *, RID> get_native_shader_and_version() const;
217
218
FogShaderData() {}
219
virtual ~FogShaderData();
220
};
221
222
struct FogMaterialData : public RendererRD::MaterialStorage::MaterialData {
223
FogShaderData *shader_data = nullptr;
224
RID uniform_set;
225
bool uniform_set_updated;
226
227
virtual void set_render_priority(int p_priority) {}
228
virtual void set_next_pass(RID p_pass) {}
229
virtual bool update_parameters(const HashMap<StringName, Variant> &p_parameters, bool p_uniform_dirty, bool p_textures_dirty);
230
virtual ~FogMaterialData();
231
};
232
233
RendererRD::MaterialStorage::ShaderData *_create_fog_shader_func();
234
static RendererRD::MaterialStorage::ShaderData *_create_fog_shader_funcs();
235
236
RendererRD::MaterialStorage::MaterialData *_create_fog_material_func(FogShaderData *p_shader);
237
static RendererRD::MaterialStorage::MaterialData *_create_fog_material_funcs(RendererRD::MaterialStorage::ShaderData *p_shader);
238
239
public:
240
static Fog *get_singleton() { return singleton; }
241
242
Fog();
243
~Fog();
244
245
/* FOG VOLUMES */
246
247
bool owns_fog_volume(RID p_rid) { return fog_volume_owner.owns(p_rid); }
248
249
virtual RID fog_volume_allocate() override;
250
virtual void fog_volume_initialize(RID p_rid) override;
251
virtual void fog_volume_free(RID p_rid) override;
252
Dependency *fog_volume_get_dependency(RID p_fog_volume) const;
253
254
virtual void fog_volume_set_shape(RID p_fog_volume, RS::FogVolumeShape p_shape) override;
255
virtual void fog_volume_set_size(RID p_fog_volume, const Vector3 &p_size) override;
256
virtual void fog_volume_set_material(RID p_fog_volume, RID p_material) override;
257
virtual RS::FogVolumeShape fog_volume_get_shape(RID p_fog_volume) const override;
258
RID fog_volume_get_material(RID p_fog_volume) const;
259
virtual AABB fog_volume_get_aabb(RID p_fog_volume) const override;
260
Vector3 fog_volume_get_size(RID p_fog_volume) const;
261
262
/* FOG VOLUMES INSTANCE */
263
264
bool owns_fog_volume_instance(RID p_rid) { return fog_volume_instance_owner.owns(p_rid); }
265
266
RID fog_volume_instance_create(RID p_fog_volume);
267
void fog_instance_free(RID p_rid);
268
269
void fog_volume_instance_set_transform(RID p_fog_volume_instance, const Transform3D &p_transform) {
270
Fog::FogVolumeInstance *fvi = fog_volume_instance_owner.get_or_null(p_fog_volume_instance);
271
ERR_FAIL_NULL(fvi);
272
fvi->transform = p_transform;
273
}
274
275
void fog_volume_instance_set_active(RID p_fog_volume_instance, bool p_active) {
276
Fog::FogVolumeInstance *fvi = fog_volume_instance_owner.get_or_null(p_fog_volume_instance);
277
ERR_FAIL_NULL(fvi);
278
fvi->active = p_active;
279
}
280
281
RID fog_volume_instance_get_volume(RID p_fog_volume_instance) const {
282
Fog::FogVolumeInstance *fvi = fog_volume_instance_owner.get_or_null(p_fog_volume_instance);
283
ERR_FAIL_NULL_V(fvi, RID());
284
return fvi->volume;
285
}
286
287
Vector3 fog_volume_instance_get_position(RID p_fog_volume_instance) const {
288
Fog::FogVolumeInstance *fvi = fog_volume_instance_owner.get_or_null(p_fog_volume_instance);
289
ERR_FAIL_NULL_V(fvi, Vector3());
290
return fvi->transform.get_origin();
291
}
292
293
/* Volumetric FOG */
294
class VolumetricFog : public RenderBufferCustomDataRD {
295
GDCLASS(VolumetricFog, RenderBufferCustomDataRD)
296
297
public:
298
enum {
299
MAX_TEMPORAL_FRAMES = 16
300
};
301
302
uint32_t width = 0;
303
uint32_t height = 0;
304
uint32_t depth = 0;
305
306
float length;
307
float spread;
308
309
RID light_density_map;
310
RID prev_light_density_map;
311
RID fog_map;
312
RID density_map;
313
RID light_map;
314
RID emissive_map;
315
316
RID fog_uniform_set;
317
RID copy_uniform_set;
318
319
struct {
320
RID process_uniform_set_density;
321
RID process_uniform_set;
322
RID process_uniform_set2;
323
} gi_dependent_sets;
324
325
RID sdfgi_uniform_set;
326
RID sky_uniform_set;
327
328
int last_shadow_filter = -1;
329
330
// If the device doesn't support image atomics, use storage buffers instead.
331
RD::UniformType atomic_type = RD::UNIFORM_TYPE_IMAGE;
332
333
virtual void configure(RenderSceneBuffersRD *p_render_buffers) override {}
334
virtual void free_data() override {}
335
336
bool sync_gi_dependent_sets_validity(bool p_ensure_freed = false);
337
338
void init(const Vector3i &fog_size, RID p_sky_shader);
339
~VolumetricFog();
340
};
341
342
void init_fog_shader(uint32_t p_max_directional_lights, int p_roughness_layers, bool p_is_using_radiance_cubemap_array);
343
void free_fog_shader();
344
345
struct VolumetricFogSettings {
346
Vector2i rb_size;
347
double time;
348
bool is_using_radiance_cubemap_array;
349
uint32_t max_cluster_elements;
350
bool volumetric_fog_filter_active;
351
RID shadow_sampler;
352
RID voxel_gi_buffer;
353
RID shadow_atlas_depth;
354
RID omni_light_buffer;
355
RID spot_light_buffer;
356
RID directional_shadow_depth;
357
RID directional_light_buffer;
358
359
// Objects related to our render buffer
360
Ref<VolumetricFog> vfog;
361
ClusterBuilderRD *cluster_builder;
362
GI *gi;
363
Ref<GI::SDFGI> sdfgi;
364
Ref<GI::RenderBuffersGI> rbgi;
365
RID env;
366
SkyRD *sky;
367
};
368
void volumetric_fog_update(const VolumetricFogSettings &p_settings, const Projection &p_cam_projection, const Transform3D &p_cam_transform, const Transform3D &p_prev_cam_inv_transform, RID p_shadow_atlas, int p_directional_light_count, bool p_use_directional_shadows, int p_positional_light_count, int p_voxel_gi_count, const PagedArray<RID> &p_fog_volumes);
369
};
370
371
} // namespace RendererRD
372
373