stk-code_catmod/data/shaders/IBL.frag

150 lines
4.8 KiB
GLSL
Raw Normal View History

2014-03-01 14:08:17 -05:00
uniform sampler2D ntex;
2014-10-05 20:19:21 -04:00
uniform sampler2D dtex;
uniform sampler2D albedo;
2014-03-01 14:08:17 -05:00
2016-09-12 15:39:52 -04:00
#ifdef GL_ES
layout (location = 0) out vec4 Diff;
layout (location = 1) out vec4 Spec;
#else
2014-03-01 14:08:17 -05:00
out vec4 Diff;
2014-10-05 20:19:21 -04:00
out vec4 Spec;
2016-09-12 15:39:52 -04:00
#endif
2014-03-01 14:08:17 -05:00
#stk_include "utils/decodeNormal.frag"
#stk_include "utils/getPosFromUVDepth.frag"
#stk_include "utils/DiffuseIBL.frag"
#stk_include "utils/SpecularIBL.frag"
2014-03-01 14:08:17 -05:00
2019-03-08 16:37:42 -05:00
vec3 getXcYcZc(int x, int y, float zC)
{
// We use perspective symetric projection matrix hence P(0,2) = P(1, 2) = 0
float xC= (2. * (float(x)) / u_screen.x - 1.) * zC / u_projection_matrix[0][0];
float yC= (2. * (float(y)) / u_screen.y - 1.) * zC / u_projection_matrix[1][1];
return vec3(xC, yC, zC);
}
float makeLinear(float f, float n, float z)
{
2019-06-26 04:25:13 -04:00
return (2.0f * n) / (f + n - z * (f - n));
2019-03-08 16:37:42 -05:00
}
vec3 CalcViewPositionFromDepth(in vec2 TexCoord, in sampler2D DepthMap)
{
// Combine UV & depth into XY & Z (NDC)
2019-06-24 17:31:51 -04:00
float z = makeLinear(1000.0, 1.0, textureLod(DepthMap, TexCoord, 0.).x);
2019-03-08 16:37:42 -05:00
vec3 rawPosition = vec3(TexCoord, z);
// Convert from (0, 1) range to (-1, 1)
2019-06-26 04:25:13 -04:00
vec4 ScreenSpacePosition = vec4( rawPosition * 2.0 - 1.0, 1.0);
2019-03-08 16:37:42 -05:00
// Undo Perspective transformation to bring into view space
vec4 ViewPosition = u_inverse_projection_matrix * ScreenSpacePosition;
// Perform perspective divide and return
return ViewPosition.xyz / ViewPosition.w;
}
2019-06-24 17:31:51 -04:00
float GetVignette(float factor)
{
vec2 inside = (gl_FragCoord.xy / u_screen) - 0.5;
2019-06-26 04:25:13 -04:00
float vignette = 1. - dot(inside, inside) * 5.0;
2019-06-24 17:31:51 -04:00
return clamp(pow(vignette, factor), 0., 1.0);
}
vec3 RayCast(vec3 dir, inout vec3 hitCoord, out float dDepth, in sampler2D DepthMap, in vec3 fallback, float spread)
2019-03-08 16:37:42 -05:00
{
dir *= 0.25f;
for(int i = 0; i < 8; ++i) {
2019-03-08 16:37:42 -05:00
hitCoord += dir;
vec4 projectedCoord = u_projection_matrix * vec4(hitCoord, 1.0);
projectedCoord.xy /= projectedCoord.w;
projectedCoord.xy = projectedCoord.xy * 0.5 + 0.5;
float depth = CalcViewPositionFromDepth(projectedCoord.xy, DepthMap).z;
dDepth = hitCoord.z - depth;
if(dDepth < 0.0)
{
2019-06-24 17:31:51 -04:00
// Texture wrapping to extand artifcially the range of the lookup texture
// FIXME can be improved to lessen the distortion
projectedCoord.y = min(.99, projectedCoord.y);
projectedCoord.x = min(.99, projectedCoord.x);
projectedCoord.x = max(.01, projectedCoord.x);
// We want only reflection on nearly horizontal surfaces
float cutout = dot(dir, vec3(0., 0., -1.));
if ((projectedCoord.x > 0.0 && projectedCoord.x < 1.0)
&& (projectedCoord.y > 0.0 && projectedCoord.y < 1.0)
2019-06-26 04:25:13 -04:00
&& (cutout > 10.0)
2019-06-24 17:31:51 -04:00
)
2019-03-08 16:37:42 -05:00
{
2019-05-31 03:17:16 -04:00
// FIXME We need to generate mipmap to take into account the gloss map
vec3 finalColor = textureLod(albedo, projectedCoord.xy, spread).rgb;
2019-06-24 17:31:51 -04:00
//return finalColor;
return mix(fallback, finalColor, GetVignette(4.));
2019-03-08 16:37:42 -05:00
}
else
{
return fallback;
}
}
}
return fallback;
}
// Main ===================================================================
2014-03-01 14:08:17 -05:00
void main(void)
{
vec2 uv = gl_FragCoord.xy / u_screen;
2014-03-01 14:08:17 -05:00
vec3 normal = normalize(DecodeNormal(2. * texture(ntex, uv).xy - 1.));
2014-10-05 20:19:21 -04:00
2014-12-07 13:10:22 -05:00
Diff = vec4(0.25 * DiffuseIBL(normal), 1.);
2014-10-05 20:19:21 -04:00
float z = texture(dtex, uv).x;
vec4 xpos = getPosFromUVDepth(vec3(uv, z), u_inverse_projection_matrix);
2014-10-05 20:19:21 -04:00
vec3 eyedir = -normalize(xpos.xyz);
2019-03-08 16:37:42 -05:00
// Extract roughness
2014-11-20 20:44:27 -05:00
float specval = texture(ntex, uv).z;
2014-12-07 13:10:22 -05:00
#ifdef GL_ES
2014-12-08 13:36:23 -05:00
Spec = vec4(.25 * SpecularIBL(normal, eyedir, specval), 1.);
#else
2019-06-24 17:31:51 -04:00
// :::::::: Compute Space Screen Reflection ::::::::::::::::::::::::::::::::::::
2019-03-08 16:37:42 -05:00
float lineardepth = textureLod(dtex, uv, 0.).x;
2019-06-24 17:31:51 -04:00
// Fallback (if the ray can't find an intersection we display the sky)
2019-03-11 16:35:55 -04:00
vec3 fallback = .25 * SpecularIBL(normal, eyedir, specval);
2019-06-24 17:31:51 -04:00
float View_Depth = makeLinear(1000.0, 1.0, lineardepth);
2019-03-08 16:37:42 -05:00
vec3 ScreenPos = xpos.xyz;
vec4 View_Pos = u_inverse_projection_matrix * vec4(ScreenPos, 1.0f);
View_Pos /= View_Pos.w;
// Reflection vector
2019-03-10 04:20:19 -04:00
vec3 reflected = normalize(reflect(eyedir, normal));
2019-03-08 16:37:42 -05:00
// Ray cast
vec3 hitPos = View_Pos.xyz;
float dDepth;
float minRayStep = 100.0f;
2019-06-24 17:31:51 -04:00
vec3 outColor = RayCast(reflected * max(minRayStep, -xpos.z),
2019-05-31 03:17:16 -04:00
hitPos, dDepth, dtex, fallback, 0.0);
// TODO temporary measure the lack of mipmaping for RTT albedo
// Implement it in proper way
outColor = mix(fallback, outColor, specval);
2019-03-08 16:37:42 -05:00
Spec = vec4(outColor.rgb, 1.0);
#endif
2019-03-08 16:37:42 -05:00
2014-03-01 14:08:17 -05:00
}