360 lines
11 KiB
C++
360 lines
11 KiB
C++
/**************************************************************************/
|
|
/* fog.h */
|
|
/**************************************************************************/
|
|
/* This file is part of: */
|
|
/* GODOT ENGINE */
|
|
/* https://godotengine.org */
|
|
/**************************************************************************/
|
|
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
|
|
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
|
|
/* */
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
/* a copy of this software and associated documentation files (the */
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
/* the following conditions: */
|
|
/* */
|
|
/* The above copyright notice and this permission notice shall be */
|
|
/* included in all copies or substantial portions of the Software. */
|
|
/* */
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
/**************************************************************************/
|
|
|
|
#ifndef FOG_RD_H
|
|
#define FOG_RD_H
|
|
|
|
#include "core/templates/local_vector.h"
|
|
#include "core/templates/rid_owner.h"
|
|
#include "servers/rendering/environment/renderer_fog.h"
|
|
#include "servers/rendering/renderer_rd/cluster_builder_rd.h"
|
|
#include "servers/rendering/renderer_rd/environment/gi.h"
|
|
#include "servers/rendering/renderer_rd/shaders/environment/volumetric_fog.glsl.gen.h"
|
|
#include "servers/rendering/renderer_rd/shaders/environment/volumetric_fog_process.glsl.gen.h"
|
|
#include "servers/rendering/renderer_rd/storage_rd/render_buffer_custom_data_rd.h"
|
|
#include "servers/rendering/storage/utilities.h"
|
|
|
|
#define RB_SCOPE_FOG SNAME("Fog")
|
|
|
|
namespace RendererRD {
|
|
|
|
class Fog : public RendererFog {
|
|
private:
|
|
static Fog *singleton;
|
|
|
|
/* FOG VOLUMES */
|
|
|
|
struct FogVolume {
|
|
RID material;
|
|
Vector3 size = Vector3(2, 2, 2);
|
|
|
|
RS::FogVolumeShape shape = RS::FOG_VOLUME_SHAPE_BOX;
|
|
|
|
Dependency dependency;
|
|
};
|
|
|
|
mutable RID_Owner<FogVolume, true> fog_volume_owner;
|
|
|
|
struct FogVolumeInstance {
|
|
RID volume;
|
|
Transform3D transform;
|
|
bool active = false;
|
|
};
|
|
|
|
mutable RID_Owner<FogVolumeInstance> fog_volume_instance_owner;
|
|
|
|
const int SAMPLERS_BINDING_FIRST_INDEX = 3;
|
|
|
|
/* Volumetric Fog */
|
|
struct VolumetricFogShader {
|
|
enum FogSet {
|
|
FOG_SET_BASE,
|
|
FOG_SET_UNIFORMS,
|
|
FOG_SET_MATERIAL,
|
|
FOG_SET_MAX,
|
|
};
|
|
|
|
struct FogPushConstant {
|
|
float position[3];
|
|
float pad;
|
|
|
|
float size[3];
|
|
float pad2;
|
|
|
|
int32_t corner[3];
|
|
uint32_t shape;
|
|
|
|
float transform[16];
|
|
};
|
|
|
|
struct VolumeUBO {
|
|
float fog_frustum_size_begin[2];
|
|
float fog_frustum_size_end[2];
|
|
|
|
float fog_frustum_end;
|
|
float z_near;
|
|
float z_far;
|
|
float time;
|
|
|
|
int32_t fog_volume_size[3];
|
|
uint32_t directional_light_count;
|
|
|
|
uint32_t use_temporal_reprojection;
|
|
uint32_t temporal_frame;
|
|
float detail_spread;
|
|
float temporal_blend;
|
|
|
|
float to_prev_view[16];
|
|
float transform[16];
|
|
};
|
|
|
|
ShaderCompiler compiler;
|
|
VolumetricFogShaderRD shader;
|
|
RID volume_ubo;
|
|
|
|
RID default_shader;
|
|
RID default_material;
|
|
RID default_shader_rd;
|
|
|
|
RID base_uniform_set;
|
|
|
|
RID params_ubo;
|
|
|
|
enum {
|
|
VOLUMETRIC_FOG_PROCESS_SHADER_DENSITY,
|
|
VOLUMETRIC_FOG_PROCESS_SHADER_DENSITY_WITH_SDFGI,
|
|
VOLUMETRIC_FOG_PROCESS_SHADER_FILTER,
|
|
VOLUMETRIC_FOG_PROCESS_SHADER_FOG,
|
|
VOLUMETRIC_FOG_PROCESS_SHADER_COPY,
|
|
VOLUMETRIC_FOG_PROCESS_SHADER_MAX,
|
|
};
|
|
|
|
struct ParamsUBO {
|
|
float fog_frustum_size_begin[2];
|
|
float fog_frustum_size_end[2];
|
|
|
|
float fog_frustum_end;
|
|
float ambient_inject;
|
|
float z_far;
|
|
uint32_t filter_axis;
|
|
|
|
float ambient_color[3];
|
|
float sky_contribution;
|
|
|
|
int32_t fog_volume_size[3];
|
|
uint32_t directional_light_count;
|
|
|
|
float base_emission[3];
|
|
float base_density;
|
|
|
|
float base_scattering[3];
|
|
float phase_g;
|
|
|
|
float detail_spread;
|
|
float gi_inject;
|
|
uint32_t max_voxel_gi_instances;
|
|
uint32_t cluster_type_size;
|
|
|
|
float screen_size[2];
|
|
uint32_t cluster_shift;
|
|
uint32_t cluster_width;
|
|
|
|
uint32_t max_cluster_element_count_div_32;
|
|
uint32_t use_temporal_reprojection;
|
|
uint32_t temporal_frame;
|
|
float temporal_blend;
|
|
|
|
float cam_rotation[12];
|
|
float to_prev_view[16];
|
|
float radiance_inverse_xform[12];
|
|
};
|
|
|
|
VolumetricFogProcessShaderRD process_shader;
|
|
|
|
RID process_shader_version;
|
|
RID process_pipelines[VOLUMETRIC_FOG_PROCESS_SHADER_MAX];
|
|
|
|
} volumetric_fog;
|
|
|
|
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);
|
|
|
|
struct FogShaderData : public RendererRD::MaterialStorage::ShaderData {
|
|
bool valid = false;
|
|
RID version;
|
|
|
|
RID pipeline;
|
|
Vector<ShaderCompiler::GeneratedCode::Texture> texture_uniforms;
|
|
|
|
Vector<uint32_t> ubo_offsets;
|
|
uint32_t ubo_size = 0;
|
|
|
|
String code;
|
|
|
|
bool uses_time = false;
|
|
|
|
virtual void set_code(const String &p_Code);
|
|
virtual bool is_animated() const;
|
|
virtual bool casts_shadows() const;
|
|
virtual RS::ShaderNativeSourceCode get_native_source_code() const;
|
|
|
|
FogShaderData() {}
|
|
virtual ~FogShaderData();
|
|
};
|
|
|
|
struct FogMaterialData : public RendererRD::MaterialStorage::MaterialData {
|
|
FogShaderData *shader_data = nullptr;
|
|
RID uniform_set;
|
|
bool uniform_set_updated;
|
|
|
|
virtual void set_render_priority(int p_priority) {}
|
|
virtual void set_next_pass(RID p_pass) {}
|
|
virtual bool update_parameters(const HashMap<StringName, Variant> &p_parameters, bool p_uniform_dirty, bool p_textures_dirty);
|
|
virtual ~FogMaterialData();
|
|
};
|
|
|
|
RendererRD::MaterialStorage::ShaderData *_create_fog_shader_func();
|
|
static RendererRD::MaterialStorage::ShaderData *_create_fog_shader_funcs();
|
|
|
|
RendererRD::MaterialStorage::MaterialData *_create_fog_material_func(FogShaderData *p_shader);
|
|
static RendererRD::MaterialStorage::MaterialData *_create_fog_material_funcs(RendererRD::MaterialStorage::ShaderData *p_shader);
|
|
|
|
public:
|
|
static Fog *get_singleton() { return singleton; }
|
|
|
|
Fog();
|
|
~Fog();
|
|
|
|
/* FOG VOLUMES */
|
|
|
|
bool owns_fog_volume(RID p_rid) { return fog_volume_owner.owns(p_rid); };
|
|
|
|
virtual RID fog_volume_allocate() override;
|
|
virtual void fog_volume_initialize(RID p_rid) override;
|
|
virtual void fog_volume_free(RID p_rid) override;
|
|
Dependency *fog_volume_get_dependency(RID p_fog_volume) const;
|
|
|
|
virtual void fog_volume_set_shape(RID p_fog_volume, RS::FogVolumeShape p_shape) override;
|
|
virtual void fog_volume_set_size(RID p_fog_volume, const Vector3 &p_size) override;
|
|
virtual void fog_volume_set_material(RID p_fog_volume, RID p_material) override;
|
|
virtual RS::FogVolumeShape fog_volume_get_shape(RID p_fog_volume) const override;
|
|
RID fog_volume_get_material(RID p_fog_volume) const;
|
|
virtual AABB fog_volume_get_aabb(RID p_fog_volume) const override;
|
|
Vector3 fog_volume_get_size(RID p_fog_volume) const;
|
|
|
|
/* FOG VOLUMES INSTANCE */
|
|
|
|
bool owns_fog_volume_instance(RID p_rid) { return fog_volume_instance_owner.owns(p_rid); };
|
|
|
|
RID fog_volume_instance_create(RID p_fog_volume);
|
|
void fog_instance_free(RID p_rid);
|
|
|
|
void fog_volume_instance_set_transform(RID p_fog_volume_instance, const Transform3D &p_transform) {
|
|
Fog::FogVolumeInstance *fvi = fog_volume_instance_owner.get_or_null(p_fog_volume_instance);
|
|
ERR_FAIL_COND(!fvi);
|
|
fvi->transform = p_transform;
|
|
}
|
|
|
|
void fog_volume_instance_set_active(RID p_fog_volume_instance, bool p_active) {
|
|
Fog::FogVolumeInstance *fvi = fog_volume_instance_owner.get_or_null(p_fog_volume_instance);
|
|
ERR_FAIL_COND(!fvi);
|
|
fvi->active = p_active;
|
|
}
|
|
|
|
RID fog_volume_instance_get_volume(RID p_fog_volume_instance) const {
|
|
Fog::FogVolumeInstance *fvi = fog_volume_instance_owner.get_or_null(p_fog_volume_instance);
|
|
ERR_FAIL_COND_V(!fvi, RID());
|
|
return fvi->volume;
|
|
}
|
|
|
|
Vector3 fog_volume_instance_get_position(RID p_fog_volume_instance) const {
|
|
Fog::FogVolumeInstance *fvi = fog_volume_instance_owner.get_or_null(p_fog_volume_instance);
|
|
ERR_FAIL_COND_V(!fvi, Vector3());
|
|
return fvi->transform.get_origin();
|
|
}
|
|
|
|
/* Volumetric FOG */
|
|
class VolumetricFog : public RenderBufferCustomDataRD {
|
|
GDCLASS(VolumetricFog, RenderBufferCustomDataRD)
|
|
|
|
public:
|
|
enum {
|
|
MAX_TEMPORAL_FRAMES = 16
|
|
};
|
|
|
|
uint32_t width = 0;
|
|
uint32_t height = 0;
|
|
uint32_t depth = 0;
|
|
|
|
float length;
|
|
float spread;
|
|
|
|
RID light_density_map;
|
|
RID prev_light_density_map;
|
|
RID fog_map;
|
|
RID density_map;
|
|
RID light_map;
|
|
RID emissive_map;
|
|
|
|
RID fog_uniform_set;
|
|
|
|
struct {
|
|
RID copy_uniform_set;
|
|
RID process_uniform_set_density;
|
|
RID process_uniform_set;
|
|
RID process_uniform_set2;
|
|
} gi_dependent_sets;
|
|
|
|
RID sdfgi_uniform_set;
|
|
RID sky_uniform_set;
|
|
|
|
int last_shadow_filter = -1;
|
|
|
|
virtual void configure(RenderSceneBuffersRD *p_render_buffers) override{};
|
|
virtual void free_data() override{};
|
|
|
|
bool sync_gi_dependent_sets_validity(bool p_ensure_freed = false);
|
|
|
|
void init(const Vector3i &fog_size, RID p_sky_shader);
|
|
~VolumetricFog();
|
|
};
|
|
|
|
void init_fog_shader(uint32_t p_max_directional_lights, int p_roughness_layers, bool p_is_using_radiance_cubemap_array);
|
|
void free_fog_shader();
|
|
|
|
struct VolumetricFogSettings {
|
|
Vector2i rb_size;
|
|
double time;
|
|
bool is_using_radiance_cubemap_array;
|
|
uint32_t max_cluster_elements;
|
|
bool volumetric_fog_filter_active;
|
|
RID shadow_sampler;
|
|
RID voxel_gi_buffer;
|
|
RID shadow_atlas_depth;
|
|
RID omni_light_buffer;
|
|
RID spot_light_buffer;
|
|
RID directional_shadow_depth;
|
|
RID directional_light_buffer;
|
|
|
|
// Objects related to our render buffer
|
|
Ref<VolumetricFog> vfog;
|
|
ClusterBuilderRD *cluster_builder;
|
|
GI *gi;
|
|
Ref<GI::SDFGI> sdfgi;
|
|
Ref<GI::RenderBuffersGI> rbgi;
|
|
RID env;
|
|
SkyRD *sky;
|
|
};
|
|
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);
|
|
};
|
|
|
|
} // namespace RendererRD
|
|
|
|
#endif // FOG_RD_H
|