!!permu FULLBRIGHT !!permu BUMP !!permu REFLECTCUBEMASK !!cvarf r_glsl_offsetmapping=0.0 !!cvarf r_glsl_offsetmapping_scale=0.04 !!cvarf gl_specular=0.3 !!cvarb r_fog_exp2=true !!samps diffuse normalmap specular fullbright lightmap !!samps deluxmap reflectmask reflectcube !!argb vertexlit=0 !!samps paletted 1 !!argb eightbit=0 !!argf mask=1.0 !!argb masklt=false !!permu FOG //!!permu DELUXE //!!permu LIGHTSTYLED //this seems to be breaking nvidia drivers if set from the engine, despite us not using it... const bool DELUXE = false; #define SPECULAR (cvar_gl_specular>0) #include "sys/defs.h" //this is what normally draws all of your walls, even with rtlights disabled //note that the '286' preset uses drawflat_walls instead. #include "sys/fog.h" layout(location=1) varying vec3 eyevector; layout(location=2) varying vec2 basetc; layout(location=3) varying vec4 vc; layout(location=4) varying mat3 invsurface; #ifdef LIGHTSTYLED //we could use an offset, but that would still need to be per-surface which would break batches //fixme: merge attributes? varying vec2 lm0, lm1, lm2, lm3; #else layout(location=0) varying vec2 lm0; #endif #ifdef VERTEX_SHADER void main () { if (OFFSETMAPPING || SPECULAR || REFLECTCUBEMASK) { vec3 eyeminusvertex = e_eyepos - v_position.xyz; eyevector.x = dot(eyeminusvertex, v_svector.xyz); eyevector.y = dot(eyeminusvertex, v_tvector.xyz); eyevector.z = dot(eyeminusvertex, v_normal.xyz); } if (REFLECTCUBEMASK) { invsurface[0] = v_svector; invsurface[1] = v_tvector; invsurface[2] = v_normal; } basetc = v_texcoord; lm0 = v_lmcoord; #ifdef LIGHTSTYLED lm1 = v_lmcoord2; lm2 = v_lmcoord3; lm3 = v_lmcoord4; #endif if (arg_vertexlit) vc = v_colour; gl_Position = ftetransform(); } #endif #ifdef FRAGMENT_SHADER //samplers //for 8bit #define s_colourmap s_t0 #include "sys/offsetmapping.h" void main () { //adjust texture coords for offsetmapping vec2 tc = basetc; if (OFFSETMAPPING) tc = offsetmap(s_normalmap, tc, eyevector); //yay, regular texture! gl_FragColor = texture2D(s_diffuse, tc); vec3 norm; if (BUMP && (DELUXE || SPECULAR || REFLECTCUBEMASK)) norm = normalize(texture2D(s_normalmap, tc).rgb - 0.5); else if (SPECULAR || DELUXE || REFLECTCUBEMASK) norm = vec3(0, 0, 1); //specular lighting expects this to exist. vec3 lightmaps; if (arg_vertexlit) lightmaps = vc.rgb * e_lmscale.rgb; else { //modulate that by the lightmap(s) including deluxemap(s) #ifdef LIGHTSTYLED if (DELUXE) { lightmaps = texture2D(s_lightmap0, lm0).rgb * e_lmscale[0].rgb * dot(norm, 2.0*texture2D(s_deluxmap0, lm0).rgb-0.5); lightmaps += texture2D(s_lightmap1, lm1).rgb * e_lmscale[1].rgb * dot(norm, 2.0*texture2D(s_deluxmap1, lm1).rgb-0.5); lightmaps += texture2D(s_lightmap2, lm2).rgb * e_lmscale[2].rgb * dot(norm, 2.0*texture2D(s_deluxmap2, lm2).rgb-0.5); lightmaps += texture2D(s_lightmap3, lm3).rgb * e_lmscale[3].rgb * dot(norm, 2.0*texture2D(s_deluxmap3, lm3).rgb-0.5); } else { lightmaps = texture2D(s_lightmap0, lm0).rgb * e_lmscale[0].rgb; lightmaps += texture2D(s_lightmap1, lm1).rgb * e_lmscale[1].rgb; lightmaps += texture2D(s_lightmap2, lm2).rgb * e_lmscale[2].rgb; lightmaps += texture2D(s_lightmap3, lm3).rgb * e_lmscale[3].rgb; } #else if (arg_eightbit) { //optional: round the lightmap coords to ensure all pixels within a texel have different lighting values either. it just looks wrong otherwise. //don't bother if its lightstyled, such cases will have unpredictable correlations anyway. //FIXME: this rounding is likely not correct with respect to software rendering. oh well. vec2 nearestlm0 = floor(lm0 * 256.0*8.0)/(256.0*8.0); lightmaps = (texture2D(s_lightmap, nearestlm0) * e_lmscale).rgb; } else lightmaps = (texture2D(s_lightmap, lm0) * e_lmscale).rgb; //modulate by the bumpmap dot light if (DELUXE) { vec3 delux = 2.0*(texture2D(s_deluxmap, lm0).rgb-0.5); lightmaps *= 1.0 / max(0.25, delux.z); //counter the darkening from deluxmaps lightmaps *= dot(norm, delux); } #endif } //add in specular, if applicable. if (SPECULAR) { vec4 specs = texture2D(s_specular, tc); vec3 halfdir; if (DELUXE) { //not lightstyled... halfdir = normalize(normalize(eyevector) + 2.0*(texture2D(s_deluxmap0, lm0).rgb-0.5)); //this norm should be the deluxemap info instead } else { halfdir = normalize(normalize(eyevector) + vec3(0.0, 0.0, 1.0)); //this norm should be the deluxemap info instead } float spec = pow(max(dot(halfdir, norm), 0.0), 32.0 * specs.a); spec *= cvar_gl_specular; //NOTE: rtlights tend to have a *4 scaler here to over-emphasise the effect because it looks cool. //As not all maps will have deluxemapping, and the double-cos from the light util makes everything far too dark anyway, //we default to something that is not garish when the light value is directly infront of every single pixel. //we can justify this difference due to the rtlight editor etc showing the *4. gl_FragColor.rgb += spec * specs.rgb; } if (REFLECTCUBEMASK) { vec3 rtc = reflect(-eyevector, norm); rtc = rtc.x*invsurface[0] + rtc.y*invsurface[1] + rtc.z*invsurface[2]; rtc = (m_model * vec4(rtc.xyz,0.0)).xyz; gl_FragColor.rgb += texture2D(s_reflectmask, tc).rgb * textureCube(s_reflectcube, rtc).rgb; } if (arg_eightbit) { //FIXME: with this extra flag, half the permutations are redundant. lightmaps *= 0.5; //counter the fact that the colourmap contains overbright values and logically ranges from 0 to 2 intead of to 1. float pal = texture2D(s_paletted, tc).r; //the palette index. hopefully not interpolated. lightmaps -= 1.0 / 128.0; //software rendering appears to round down, so make sure we favour the lower values instead of rounding to the nearest gl_FragColor.r = texture2D(s_colourmap, vec2(pal, 1.0-lightmaps.r)).r; //do 3 lookups. this is to cope with lit files, would be a waste to not support those. gl_FragColor.g = texture2D(s_colourmap, vec2(pal, 1.0-lightmaps.g)).g; //its not very softwarey, but re-palettizing is ugly. gl_FragColor.b = texture2D(s_colourmap, vec2(pal, 1.0-lightmaps.b)).b; //without lits, it should be identical. } else { //now we have our diffuse+specular terms, modulate by lightmap values. gl_FragColor.rgb *= lightmaps.rgb; //add on the fullbright if (FULLBRIGHT) gl_FragColor.rgb += texture2D(s_fullbright, tc).rgb; } //entity modifiers gl_FragColor = gl_FragColor * e_colourident; if (arg_mask != 1.0) { if (arg_masklt) { if (gl_FragColor.a < arg_mask) discard; } else { if (gl_FragColor.a >= arg_mask) discard; } } //and finally hide it all if we're fogged. #ifdef FOG gl_FragColor = fog4(gl_FragColor); #endif } #endif