250 lines
6.6 KiB
GLSL
250 lines
6.6 KiB
GLSL
/* clang-format off */
|
|
#[modes]
|
|
|
|
mode_background =
|
|
mode_half_res = #define USE_HALF_RES_PASS
|
|
mode_quarter_res = #define USE_QUARTER_RES_PASS
|
|
mode_cubemap = #define USE_CUBEMAP_PASS
|
|
mode_cubemap_half_res = #define USE_CUBEMAP_PASS \n#define USE_HALF_RES_PASS
|
|
mode_cubemap_quarter_res = #define USE_CUBEMAP_PASS \n#define USE_QUARTER_RES_PASS
|
|
|
|
#[specializations]
|
|
|
|
USE_MULTIVIEW = false
|
|
USE_INVERTED_Y = true
|
|
APPLY_TONEMAPPING = true
|
|
|
|
#[vertex]
|
|
|
|
layout(location = 0) in vec2 vertex_attrib;
|
|
|
|
out vec2 uv_interp;
|
|
/* clang-format on */
|
|
|
|
void main() {
|
|
#ifdef USE_INVERTED_Y
|
|
uv_interp = vertex_attrib;
|
|
#else
|
|
// We're doing clockwise culling so flip the order
|
|
uv_interp = vec2(vertex_attrib.x, vertex_attrib.y * -1.0);
|
|
#endif
|
|
gl_Position = vec4(uv_interp, -1.0, 1.0);
|
|
}
|
|
|
|
/* clang-format off */
|
|
#[fragment]
|
|
|
|
#define M_PI 3.14159265359
|
|
|
|
#include "tonemap_inc.glsl"
|
|
|
|
in vec2 uv_interp;
|
|
|
|
/* clang-format on */
|
|
|
|
uniform samplerCube radiance; //texunit:-1
|
|
#ifdef USE_CUBEMAP_PASS
|
|
uniform samplerCube half_res; //texunit:-2
|
|
uniform samplerCube quarter_res; //texunit:-3
|
|
#elif defined(USE_MULTIVIEW)
|
|
uniform sampler2DArray half_res; //texunit:-2
|
|
uniform sampler2DArray quarter_res; //texunit:-3
|
|
#else
|
|
uniform sampler2D half_res; //texunit:-2
|
|
uniform sampler2D quarter_res; //texunit:-3
|
|
#endif
|
|
|
|
layout(std140) uniform GlobalShaderUniformData { //ubo:1
|
|
vec4 global_shader_uniforms[MAX_GLOBAL_SHADER_UNIFORMS];
|
|
};
|
|
|
|
struct DirectionalLightData {
|
|
vec4 direction_energy;
|
|
vec4 color_size;
|
|
bool enabled;
|
|
};
|
|
|
|
layout(std140) uniform DirectionalLights { //ubo:4
|
|
DirectionalLightData data[MAX_DIRECTIONAL_LIGHT_DATA_STRUCTS];
|
|
}
|
|
directional_lights;
|
|
|
|
/* clang-format off */
|
|
|
|
#ifdef MATERIAL_UNIFORMS_USED
|
|
layout(std140) uniform MaterialUniforms{ //ubo:3
|
|
|
|
#MATERIAL_UNIFORMS
|
|
|
|
};
|
|
#endif
|
|
/* clang-format on */
|
|
#GLOBALS
|
|
|
|
#ifdef USE_CUBEMAP_PASS
|
|
#define AT_CUBEMAP_PASS true
|
|
#else
|
|
#define AT_CUBEMAP_PASS false
|
|
#endif
|
|
|
|
#ifdef USE_HALF_RES_PASS
|
|
#define AT_HALF_RES_PASS true
|
|
#else
|
|
#define AT_HALF_RES_PASS false
|
|
#endif
|
|
|
|
#ifdef USE_QUARTER_RES_PASS
|
|
#define AT_QUARTER_RES_PASS true
|
|
#else
|
|
#define AT_QUARTER_RES_PASS false
|
|
#endif
|
|
|
|
// mat4 is a waste of space, but we don't have an easy way to set a mat3 uniform for now
|
|
uniform mat4 orientation;
|
|
uniform vec4 projection;
|
|
uniform vec3 position;
|
|
uniform float time;
|
|
uniform float sky_energy_multiplier;
|
|
uniform float luminance_multiplier;
|
|
|
|
uniform float fog_aerial_perspective;
|
|
uniform vec4 fog_light_color;
|
|
uniform float fog_sun_scatter;
|
|
uniform bool fog_enabled;
|
|
uniform float fog_density;
|
|
uniform float fog_sky_affect;
|
|
uniform uint directional_light_count;
|
|
|
|
#ifdef USE_MULTIVIEW
|
|
layout(std140) uniform MultiviewData { // ubo:11
|
|
highp mat4 projection_matrix_view[MAX_VIEWS];
|
|
highp mat4 inv_projection_matrix_view[MAX_VIEWS];
|
|
highp vec4 eye_offset[MAX_VIEWS];
|
|
}
|
|
multiview_data;
|
|
#endif
|
|
|
|
layout(location = 0) out vec4 frag_color;
|
|
|
|
#ifdef USE_DEBANDING
|
|
// https://www.iryoku.com/next-generation-post-processing-in-call-of-duty-advanced-warfare
|
|
vec3 interleaved_gradient_noise(vec2 pos) {
|
|
const vec3 magic = vec3(0.06711056f, 0.00583715f, 52.9829189f);
|
|
float res = fract(magic.z * fract(dot(pos, magic.xy))) * 2.0 - 1.0;
|
|
return vec3(res, -res, res) / 255.0;
|
|
}
|
|
#endif
|
|
|
|
#if !defined(DISABLE_FOG)
|
|
vec4 fog_process(vec3 view, vec3 sky_color) {
|
|
vec3 fog_color = mix(fog_light_color.rgb, sky_color, fog_aerial_perspective);
|
|
|
|
if (fog_sun_scatter > 0.001) {
|
|
vec4 sun_scatter = vec4(0.0);
|
|
float sun_total = 0.0;
|
|
for (uint i = 0u; i < directional_light_count; i++) {
|
|
vec3 light_color = directional_lights.data[i].color_size.xyz * directional_lights.data[i].direction_energy.w;
|
|
float light_amount = pow(max(dot(view, directional_lights.data[i].direction_energy.xyz), 0.0), 8.0);
|
|
fog_color += light_color * light_amount * fog_sun_scatter;
|
|
}
|
|
}
|
|
|
|
return vec4(fog_color, 1.0);
|
|
}
|
|
#endif // !DISABLE_FOG
|
|
|
|
void main() {
|
|
vec3 cube_normal;
|
|
#ifdef USE_MULTIVIEW
|
|
// In multiview our projection matrices will contain positional and rotational offsets that we need to properly unproject.
|
|
vec4 unproject = vec4(uv_interp.xy, -1.0, 1.0); // unproject at the far plane
|
|
vec4 unprojected = multiview_data.inv_projection_matrix_view[ViewIndex] * unproject;
|
|
cube_normal = unprojected.xyz / unprojected.w;
|
|
|
|
// Unproject will give us the position between the eyes, need to re-offset.
|
|
cube_normal += multiview_data.eye_offset[ViewIndex].xyz;
|
|
#else
|
|
cube_normal.z = -1.0;
|
|
cube_normal.x = (uv_interp.x + projection.x) / projection.y;
|
|
cube_normal.y = (-uv_interp.y - projection.z) / projection.w;
|
|
#endif
|
|
cube_normal = mat3(orientation) * cube_normal;
|
|
cube_normal = normalize(cube_normal);
|
|
|
|
vec2 uv = gl_FragCoord.xy; // uv_interp * 0.5 + 0.5;
|
|
|
|
vec2 panorama_coords = vec2(atan(cube_normal.x, -cube_normal.z), acos(cube_normal.y));
|
|
|
|
if (panorama_coords.x < 0.0) {
|
|
panorama_coords.x += M_PI * 2.0;
|
|
}
|
|
|
|
panorama_coords /= vec2(M_PI * 2.0, M_PI);
|
|
|
|
vec3 color = vec3(0.0, 0.0, 0.0);
|
|
float alpha = 1.0; // Only available to subpasses
|
|
vec4 half_res_color = vec4(1.0);
|
|
vec4 quarter_res_color = vec4(1.0);
|
|
vec4 custom_fog = vec4(0.0);
|
|
|
|
#ifdef USE_CUBEMAP_PASS
|
|
#ifdef USES_HALF_RES_COLOR
|
|
half_res_color = texture(samplerCube(half_res, SAMPLER_LINEAR_WITH_MIPMAPS_CLAMP), cube_normal);
|
|
#endif
|
|
#ifdef USES_QUARTER_RES_COLOR
|
|
quarter_res_color = texture(samplerCube(quarter_res, SAMPLER_LINEAR_WITH_MIPMAPS_CLAMP), cube_normal);
|
|
#endif
|
|
#else
|
|
#ifdef USES_HALF_RES_COLOR
|
|
#ifdef USE_MULTIVIEW
|
|
half_res_color = textureLod(sampler2DArray(half_res, SAMPLER_LINEAR_CLAMP), vec3(uv, ViewIndex), 0.0);
|
|
#else
|
|
half_res_color = textureLod(sampler2D(half_res, SAMPLER_LINEAR_CLAMP), uv, 0.0);
|
|
#endif
|
|
#endif
|
|
#ifdef USES_QUARTER_RES_COLOR
|
|
#ifdef USE_MULTIVIEW
|
|
quarter_res_color = textureLod(sampler2DArray(quarter_res, SAMPLER_LINEAR_CLAMP), vec3(uv, ViewIndex), 0.0);
|
|
#else
|
|
quarter_res_color = textureLod(sampler2D(quarter_res, SAMPLER_LINEAR_CLAMP), uv, 0.0);
|
|
#endif
|
|
#endif
|
|
#endif
|
|
|
|
{
|
|
#CODE : SKY
|
|
}
|
|
|
|
color *= sky_energy_multiplier;
|
|
|
|
// Convert to Linear for tonemapping so color matches scene shader better
|
|
color = srgb_to_linear(color);
|
|
|
|
#if !defined(DISABLE_FOG) && !defined(USE_CUBEMAP_PASS)
|
|
|
|
// Draw "fixed" fog before volumetric fog to ensure volumetric fog can appear in front of the sky.
|
|
if (fog_enabled) {
|
|
vec4 fog = fog_process(cube_normal, color.rgb);
|
|
color.rgb = mix(color.rgb, fog.rgb, fog.a * fog_sky_affect);
|
|
}
|
|
|
|
if (custom_fog.a > 0.0) {
|
|
color.rgb = mix(color.rgb, custom_fog.rgb, custom_fog.a);
|
|
}
|
|
|
|
#endif // DISABLE_FOG
|
|
|
|
color *= exposure;
|
|
#ifdef APPLY_TONEMAPPING
|
|
color = apply_tonemapping(color, white);
|
|
#endif
|
|
color = linear_to_srgb(color);
|
|
|
|
frag_color.rgb = color * luminance_multiplier;
|
|
frag_color.a = alpha;
|
|
|
|
#ifdef USE_DEBANDING
|
|
frag_color.rgb += interleaved_gradient_noise(gl_FragCoord.xy) * sky_energy_multiplier * luminance_multiplier;
|
|
#endif
|
|
}
|