From eb39e88682e2ef297096d22c9930d1fb2cb76103 Mon Sep 17 00:00:00 2001 From: Magnus Norddahl Date: Mon, 19 Feb 2018 02:01:33 +0100 Subject: [PATCH] - clean up the main.fp light handling code so that a single ApplyDynLights function applies all dynamic light --- wadsrc/static/shaders/glsl/main.fp | 375 ++++++++++++++++------------- 1 file changed, 210 insertions(+), 165 deletions(-) diff --git a/wadsrc/static/shaders/glsl/main.fp b/wadsrc/static/shaders/glsl/main.fp index b868f86c3..a6c02654d 100644 --- a/wadsrc/static/shaders/glsl/main.fp +++ b/wadsrc/static/shaders/glsl/main.fp @@ -221,39 +221,26 @@ float shadowmapAttenuation(vec4 lightpos, float shadowIndex) #endif } -#endif - -//=========================================================================== -// -// Standard lambertian diffuse light calculation -// -//=========================================================================== - -float diffuseContribution(vec3 lightDirection, vec3 normal) +float shadowAttenuation(vec4 lightpos, float lightcolorA) { - return max(dot(normal, lightDirection), 0.0f); + float shadowIndex = abs(lightcolorA) - 1.0; + return shadowmapAttenuation(lightpos, shadowIndex); } -//=========================================================================== -// -// Blinn specular light calculation -// -//=========================================================================== +#else -float blinnSpecularContribution(float diffuseContribution, vec3 lightDirection, vec3 faceNormal, float glossiness, float specularLevel) +float shadowAttenuation(vec4 lightpos, float lightcolorA) { - if (diffuseContribution > 0.0f) - { - vec3 viewDir = normalize(uCameraPos.xyz - pixelpos.xyz); - vec3 halfDir = normalize(lightDirection + viewDir); - float specAngle = max(dot(halfDir, faceNormal), 0.0f); - float phExp = glossiness * 4.0f; - return specularLevel * pow(specAngle, phExp); - } - else - { - return 0.0f; - } + return 1.0; +} + +#endif + +float spotLightAttenuation(vec4 lightpos, vec3 spotdir, float lightCosInnerAngle, float lightCosOuterAngle) +{ + vec3 lightDirection = normalize(lightpos.xyz - pixelpos.xyz); + float cosDir = dot(lightDirection, spotdir); + return smoothstep(lightCosOuterAngle, lightCosInnerAngle, cosDir); } //=========================================================================== @@ -262,7 +249,7 @@ float blinnSpecularContribution(float diffuseContribution, vec3 lightDirection, // //=========================================================================== -#if defined(SPECULAR) || defined(PBR) +#if defined(SPECULAR) || defined(PBR) // To do: create define for when normal map is present mat3 cotangent_frame(vec3 n, vec3 p, vec2 uv) { // get edge vectors of the pixel triangle @@ -313,46 +300,118 @@ vec3 ApplyNormalMap() #endif //=========================================================================== +// Dynamic light material modes begin // -// Calculates the brightness of a dynamic point light -// +// To do: move each of the following #if blocks needs to its own file //=========================================================================== -vec2 pointLightAttenuation(vec4 lightpos, float lightcolorA) +#if !defined(NUM_UBO_LIGHTS) && !defined(SHADER_STORAGE_LIGHTS) // Legacy light mode (no lights[] array) + +vec3 ApplyDynLights(vec3 material, vec3 color) { - float attenuation = max(lightpos.w - distance(pixelpos.xyz, lightpos.xyz),0.0) / lightpos.w; - if (attenuation == 0.0) return vec2(0.0); -#ifdef SUPPORTS_SHADOWMAPS - float shadowIndex = abs(lightcolorA) - 1.0; - attenuation *= shadowmapAttenuation(lightpos, shadowIndex); -#endif - if (lightcolorA >= 0.0) // Sign bit is the attenuated light flag + return material * clamp(color + desaturate(uDynLightColor).rgb, 0.0, 1.4); +} + +#elif defined(SPECULAR) // Specular light mode + +vec2 lightAttenuation(int i, vec3 normal, vec3 viewdir, float lightcolorA) +{ + vec4 lightpos = lights[i]; + vec4 lightspot1 = lights[i+2]; + vec4 lightspot2 = lights[i+3]; + + float lightdistance = distance(lightpos.xyz, pixelpos.xyz); + if (lightpos.w < lightdistance) + return vec2(0.0); // Early out lights touching surface but not this fragment + + float attenuation = clamp((lightpos.w - lightdistance) / lightpos.w, 0.0, 1.0); + + if (lightspot1.w == 1.0) + attenuation *= spotLightAttenuation(lightpos, lightspot1.xyz, lightspot2.x, lightspot2.y); + + vec3 lightdir = normalize(lightpos.xyz - pixelpos.xyz); + + if (lightcolorA < 0.0) // Sign bit is the attenuated light flag + attenuation *= clamp(dot(normal, lightdir), 0.0, 1.0); + + if (attenuation > 0.0) // Skip shadow map test if possible + attenuation *= shadowAttenuation(lightpos, lightcolorA); + + if (attenuation <= 0.0) + return vec2(0.0); + + float glossiness = uSpecularMaterial.x; + float specularLevel = uSpecularMaterial.y; + + vec3 halfdir = normalize(viewdir + lightdir); + float specAngle = clamp(dot(halfdir, normal), 0.0f, 1.0f); + float phExp = glossiness * 4.0f; + return vec2(attenuation, attenuation * specularLevel * pow(specAngle, phExp)); +} + +vec3 ApplyDynLights(vec3 material, vec3 color) +{ + if (uLightIndex >= 0) { - return vec2(attenuation, 0.0); + vec4 dynlight = uDynLightColor; + vec4 specular = vec4(0.0, 0.0, 0.0, 1.0); + + vec3 normal = ApplyNormalMap(); + vec3 viewdir = normalize(uCameraPos.xyz - pixelpos.xyz); + + ivec4 lightRange = ivec4(lights[uLightIndex]) + ivec4(uLightIndex + 1); + + if (lightRange.z > lightRange.x) + { + // modulated lights + for(int i=lightRange.x; i lightRange.z) + { + vec4 addlight = vec4(0.0,0.0,0.0,0.0); + + // additive lights + for(int i=lightRange.z; i= 0) { ivec4 lightRange = ivec4(lights[uLightIndex]) + ivec4(uLightIndex + 1); @@ -458,7 +506,7 @@ vec3 applyLight(vec3 albedo, vec3 ambientLight) vec3 L = normalize(lightpos.xyz - worldpos); vec3 H = normalize(V + L); - float attenuation = quadraticDistanceAttenuation(lightpos) * shadowAttenuation(lightpos, lightcolor.a); + float attenuation = quadraticDistanceAttenuation(lightpos); if (lightspot1.w == 1.0) attenuation *= spotLightAttenuation(lightpos, lightspot1.xyz, lightspot2.x, lightspot2.y); if (lightcolor.a < 0.0) @@ -498,7 +546,7 @@ vec3 applyLight(vec3 albedo, vec3 ambientLight) vec3 L = normalize(lightpos.xyz - worldpos); vec3 H = normalize(V + L); - float attenuation = quadraticDistanceAttenuation(lightpos) * shadowAttenuation(lightpos, lightcolor.a); + float attenuation = quadraticDistanceAttenuation(lightpos); if (lightspot1.w == 1.0) attenuation *= spotLightAttenuation(lightpos, lightspot1.xyz, lightspot2.x, lightspot2.y); if (lightcolor.a < 0.0) @@ -527,7 +575,6 @@ vec3 applyLight(vec3 albedo, vec3 ambientLight) } } } -#endif // Pretend we sampled the sector light level from an irradiance map @@ -555,8 +602,94 @@ vec3 applyLight(vec3 albedo, vec3 ambientLight) return pow(color, vec3(1.0 / 2.2)); } +#else // Normal dynlight mode + +vec3 lightContribution(int i, vec3 normal) +{ + vec4 lightpos = lights[i]; + vec4 lightcolor = lights[i+1]; + vec4 lightspot1 = lights[i+2]; + vec4 lightspot2 = lights[i+3]; + + float lightdistance = distance(lightpos.xyz, pixelpos.xyz); + if (lightpos.w < lightdistance) + return vec3(0.0); // Early out lights touching surface but not this fragment + + float attenuation = clamp((lightpos.w - lightdistance) / lightpos.w, 0.0, 1.0); + + if (lightspot1.w == 1.0) + attenuation *= spotLightAttenuation(lightpos, lightspot1.xyz, lightspot2.x, lightspot2.y); + + if (lightcolor.a < 0.0) // Sign bit is the attenuated light flag + { + vec3 lightdir = normalize(lightpos.xyz - pixelpos.xyz); + attenuation *= clamp(dot(normal, lightdir), 0.0, 1.0); + } + + if (attenuation > 0.0) // Skip shadow map test if possible + { + attenuation *= shadowAttenuation(lightpos, lightcolor.a); + return lightcolor.rgb * attenuation; + } + else + { + return vec3(0.0); + } +} + +vec3 ApplyDynLights(vec3 material, vec3 color) +{ + if (uLightIndex >= 0) + { + vec4 dynlight = uDynLightColor; + vec3 normal = ApplyNormalMap(); + + ivec4 lightRange = ivec4(lights[uLightIndex]) + ivec4(uLightIndex + 1); + + if (lightRange.z > lightRange.x) + { + // modulated lights + for(int i=lightRange.x; i lightRange.z) + { + vec4 addlight = vec4(0.0,0.0,0.0,0.0); + + // additive lights + for(int i=lightRange.z; i= 0) - { - ivec4 lightRange = ivec4(lights[uLightIndex]) + ivec4(uLightIndex + 1); - if (lightRange.z > lightRange.x) - { - // - // modulated lights - // - for(int i=lightRange.x; i= 0) - { - ivec4 lightRange = ivec4(lights[uLightIndex]) + ivec4(uLightIndex + 1); - if (lightRange.w > lightRange.z) - { - vec4 addlight = vec4(0.0,0.0,0.0,0.0); - - // - // additive lights - these can be done after the alpha test. - // - for(int i=lightRange.z; i