2017-02-11 16:14:06 +00:00
!!ver 100 150
!!permu TESS
2012-10-07 18:38:22 +00:00
!!permu DELUXE
2012-05-09 15:30:53 +00:00
!!permu FULLBRIGHT
!!permu FOG
2012-07-05 13:08:07 +00:00
!!permu LIGHTSTYLED
2012-10-07 18:38:22 +00:00
!!permu BUMP
2012-11-27 03:23:19 +00:00
!!permu SPECULAR
2015-05-03 19:57:46 +00:00
!!permu REFLECTCUBEMASK
2012-05-09 15:30:53 +00:00
!!cvarf r_glsl_offsetmapping_scale
2017-02-19 00:15:42 +00:00
!!cvardf r_tessellation_level=5
2018-11-27 16:48:19 +00:00
!!samps !EIGHTBIT diffuse specular normalmap fullbright reflectmask reflectcube
//diffuse gives us alpha, and prevents dlight from bugging out when there's no diffuse.
!!samps =EIGHTBIT paletted 1 specular diffuse
!!samps lightmap deluxemap
!!samps =LIGHTSTYLED lightmap1 lightmap2 lightmap3 deluxemap deluxemap1 deluxemap2 deluxemap3
2012-05-09 15:30:53 +00:00
2015-03-03 00:14:43 +00:00
#include "sys/defs.h"
2012-05-09 15:30:53 +00:00
//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"
2012-07-05 13:08:07 +00:00
2017-02-11 16:14:06 +00:00
#if !defined(TESS_CONTROL_SHADER)
#if defined(OFFSETMAPPING) || defined(SPECULAR) || defined(REFLECTCUBEMASK)
varying vec3 eyevector;
#endif
2015-05-03 19:57:46 +00:00
2017-08-29 02:29:06 +00:00
#if defined(REFLECTCUBEMASK) || defined(BUMPMODELSPACE)
2017-02-11 16:14:06 +00:00
varying mat3 invsurface;
#endif
varying vec2 tc;
#ifdef VERTEXLIT
varying vec4 vc;
2016-07-12 00:40:13 +00:00
#else
2017-02-11 16:14:06 +00:00
#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
varying vec2 lm0;
#endif
2016-07-12 00:40:13 +00:00
#endif
2012-07-05 13:08:07 +00:00
#endif
2012-05-09 15:30:53 +00:00
#ifdef VERTEX_SHADER
2017-02-11 16:14:06 +00:00
#ifdef TESS
varying vec3 vertex, normal;
#endif
2012-05-09 15:30:53 +00:00
void main ()
{
2015-05-03 19:57:46 +00:00
#if defined(OFFSETMAPPING) || defined(SPECULAR) || defined(REFLECTCUBEMASK)
2012-05-09 15:30:53 +00:00
vec3 eyeminusvertex = e_eyepos - v_position.xyz;
2014-01-13 02:42:25 +00:00
eyevector.x = dot(eyeminusvertex, v_svector.xyz);
2012-07-05 13:08:07 +00:00
eyevector.y = dot(eyeminusvertex, v_tvector.xyz);
2012-05-09 15:30:53 +00:00
eyevector.z = dot(eyeminusvertex, v_normal.xyz);
2015-05-03 19:57:46 +00:00
#endif
2017-08-29 02:29:06 +00:00
#if defined(REFLECTCUBEMASK) || defined(BUMPMODELSPACE)
2015-05-03 19:57:46 +00:00
invsurface[0] = v_svector;
invsurface[1] = v_tvector;
invsurface[2] = v_normal;
2012-05-09 15:30:53 +00:00
#endif
tc = v_texcoord;
2017-08-14 16:38:44 +00:00
#ifdef FLOW
tc.s += e_time * -0.5;
#endif
2016-07-12 00:40:13 +00:00
#ifdef VERTEXLIT
#ifdef LIGHTSTYLED
//FIXME, only one colour.
vc = v_colour * e_lmscale[0];
#else
vc = v_colour * e_lmscale;
#endif
#else
2015-02-07 22:34:22 +00:00
lm0 = v_lmcoord;
2012-07-05 13:08:07 +00:00
#ifdef LIGHTSTYLED
2015-02-07 22:34:22 +00:00
lm1 = v_lmcoord2;
lm2 = v_lmcoord3;
lm3 = v_lmcoord4;
2016-07-12 00:40:13 +00:00
#endif
2012-07-05 13:08:07 +00:00
#endif
2012-05-09 15:30:53 +00:00
gl_Position = ftetransform();
2017-02-11 16:14:06 +00:00
#ifdef TESS
vertex = v_position;
normal = v_normal;
#endif
}
#endif
#if defined(TESS_CONTROL_SHADER)
layout(vertices = 3) out;
in vec3 vertex[];
out vec3 t_vertex[];
in vec3 normal[];
out vec3 t_normal[];
#if defined(OFFSETMAPPING) || defined(SPECULAR) || defined(REFLECTCUBEMASK)
in vec3 eyevector[];
out vec3 t_eyevector[];
#endif
#ifdef REFLECTCUBEMASK
in mat3 invsurface[];
out mat3 t_invsurface[];
#endif
in vec2 tc[];
out vec2 t_tc[];
#ifdef VERTEXLIT
in vec4 vc[];
out vec4 t_vc[];
#else
in vec2 lm0[];
out vec2 t_lm0[];
#ifdef LIGHTSTYLED
in vec2 lm1[], lm2[], lm3[];
out vec2 t_lm1[], t_lm2[], t_lm3[];
#endif
#endif
void main()
{
//the control shader needs to pass stuff through
#define id gl_InvocationID
t_vertex[id] = vertex[id];
t_normal[id] = normal[id];
#ifdef REFLECTCUBEMASK
t_invsurface[id] = invsurface[id];
#endif
t_tc[id] = tc[id];
#ifdef VERTEXLIT
t_vc[id] = vc[id];
#else
t_lm0[id] = lm0[id];
#ifdef LIGHTSTYLED
t_lm1[id] = lm1[id];
t_lm2[id] = lm2[id];
t_lm3[id] = lm3[id];
#endif
#endif
#if defined(SPECULAR) || defined(OFFSETMAPPING) || defined(REFLECTCUBEMASK)
t_eyevector[id] = eyevector[id];
#endif
2017-02-19 00:15:42 +00:00
gl_TessLevelOuter[0] = float(r_tessellation_level);
gl_TessLevelOuter[1] = float(r_tessellation_level);
gl_TessLevelOuter[2] = float(r_tessellation_level);
gl_TessLevelInner[0] = float(r_tessellation_level);
2017-02-11 16:14:06 +00:00
}
#endif
#if defined(TESS_EVALUATION_SHADER)
layout(triangles) in;
in vec3 t_vertex[];
in vec3 t_normal[];
#if defined(OFFSETMAPPING) || defined(SPECULAR) || defined(REFLECTCUBEMASK)
in vec3 t_eyevector[];
#endif
#ifdef REFLECTCUBEMASK
in mat3 t_invsurface[];
#endif
in vec2 t_tc[];
#ifdef VERTEXLIT
in vec4 t_vc[];
#else
#ifdef LIGHTSTYLED
//we could use an offset, but that would still need to be per-surface which would break batches
//fixme: merge attributes?
in vec2 t_lm0[], t_lm1[], t_lm2[], t_lm3[];
#else
in vec2 t_lm0[];
#endif
#endif
#define LERP(a) (gl_TessCoord.x*a[0] + gl_TessCoord.y*a[1] + gl_TessCoord.z*a[2])
void main()
{
#define factor 1.0
tc = LERP(t_tc);
#ifdef VERTEXLIT
vc = LERP(t_vc);
#else
lm0 = LERP(t_lm0);
#ifdef LIGHTSTYLED
lm1 = LERP(t_lm1);
lm2 = LERP(t_lm2);
lm3 = LERP(t_lm3);
#endif
#endif
vec3 w = LERP(t_vertex);
vec3 t0 = w - dot(w-t_vertex[0],t_normal[0])*t_normal[0];
vec3 t1 = w - dot(w-t_vertex[1],t_normal[1])*t_normal[1];
vec3 t2 = w - dot(w-t_vertex[2],t_normal[2])*t_normal[2];
w = w*(1.0-factor) + factor*(gl_TessCoord.x*t0+gl_TessCoord.y*t1+gl_TessCoord.z*t2);
#if defined(PCF) || defined(SPOT) || defined(CUBE)
//for texture projections/shadowmapping on dlights
vtexprojcoord = (l_cubematrix*vec4(w.xyz, 1.0));
#endif
//FIXME: we should be recalcing these here, instead of just lerping them
#ifdef REFLECTCUBEMASK
invsurface = LERP(t_invsurface);
#endif
#if defined(SPECULAR) || defined(OFFSETMAPPING) || defined(REFLECTCUBEMASK)
eyevector = LERP(t_eyevector);
#endif
gl_Position = m_modelviewprojection * vec4(w,1.0);
2012-05-09 15:30:53 +00:00
}
#endif
2012-07-05 13:08:07 +00:00
2017-02-11 16:14:06 +00:00
2012-05-09 15:30:53 +00:00
#ifdef FRAGMENT_SHADER
2015-03-03 00:14:43 +00:00
#define s_colourmap s_t0
2012-07-05 13:08:07 +00:00
2012-05-09 15:30:53 +00:00
#ifdef OFFSETMAPPING
#include "sys/offsetmapping.h"
#endif
void main ()
{
2012-10-07 18:38:22 +00:00
//adjust texture coords for offsetmapping
2012-05-09 15:30:53 +00:00
#ifdef OFFSETMAPPING
2015-02-07 22:34:22 +00:00
vec2 tcoffsetmap = offsetmap(s_normalmap, tc, eyevector);
2012-05-09 15:30:53 +00:00
#define tc tcoffsetmap
#endif
2012-10-07 18:38:22 +00:00
2015-02-07 22:34:22 +00:00
#if defined(EIGHTBIT) && !defined(LIGHTSTYLED)
//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.
2016-12-13 11:50:15 +00:00
#if __VERSION__ >= 130
2016-11-25 08:14:54 +00:00
vec2 lmsize = vec2(textureSize(s_lightmap0, 0));
#else
#define lmsize vec2(128.0,2048.0)
#endif
#define texelstolightmap (16.0)
vec2 lmcoord0 = floor(lm0 * lmsize*texelstolightmap)/(lmsize*texelstolightmap);
2015-02-07 22:34:22 +00:00
#define lm0 lmcoord0
#endif
2012-10-07 18:38:22 +00:00
//yay, regular texture!
2015-02-07 22:34:22 +00:00
gl_FragColor = texture2D(s_diffuse, tc);
2012-10-07 18:38:22 +00:00
2015-05-03 19:57:46 +00:00
#if defined(BUMP) && (defined(DELUXE) || defined(SPECULAR) || defined(REFLECTCUBEMASK))
2015-02-07 22:34:22 +00:00
vec3 norm = normalize(texture2D(s_normalmap, tc).rgb - 0.5);
2015-05-03 19:57:46 +00:00
#elif defined(SPECULAR) || defined(DELUXE) || defined(REFLECTCUBEMASK)
2012-11-27 03:23:19 +00:00
vec3 norm = vec3(0, 0, 1); //specular lighting expects this to exist.
#endif
2012-10-07 18:38:22 +00:00
//modulate that by the lightmap(s) including deluxemap(s)
2016-07-12 00:40:13 +00:00
#ifdef VERTEXLIT
#ifdef LIGHTSTYLED
vec3 lightmaps = vc.rgb;
2012-10-07 18:38:22 +00:00
#else
2016-07-12 00:40:13 +00:00
vec3 lightmaps = vc.rgb;
2012-10-07 18:38:22 +00:00
#endif
2018-07-05 16:21:44 +00:00
#define deluxe vec3(0.0,0.0,1.0)
2012-07-05 13:08:07 +00:00
#else
2016-07-12 00:40:13 +00:00
#ifdef LIGHTSTYLED
2018-07-05 16:21:44 +00:00
#define deluxe vec3(0.0,0.0,1.0)
2016-07-12 00:40:13 +00:00
vec3 lightmaps;
#ifdef DELUXE
2018-07-05 16:21:44 +00:00
lightmaps = texture2D(s_lightmap0, lm0).rgb * e_lmscale[0].rgb * dot(norm, 2.0*texture2D(s_deluxemap0, lm0).rgb-0.5);
lightmaps += texture2D(s_lightmap1, lm1).rgb * e_lmscale[1].rgb * dot(norm, 2.0*texture2D(s_deluxemap1, lm1).rgb-0.5);
lightmaps += texture2D(s_lightmap2, lm2).rgb * e_lmscale[2].rgb * dot(norm, 2.0*texture2D(s_deluxemap2, lm2).rgb-0.5);
lightmaps += texture2D(s_lightmap3, lm3).rgb * e_lmscale[3].rgb * dot(norm, 2.0*texture2D(s_deluxemap3, lm3).rgb-0.5);
2016-07-12 00:40:13 +00:00
#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;
#endif
#else
vec3 lightmaps = (texture2D(s_lightmap, lm0) * e_lmscale).rgb;
//modulate by the bumpmap dot light
#ifdef DELUXE
2018-07-05 16:21:44 +00:00
vec3 deluxe = (texture2D(s_deluxemap, lm0).rgb-0.5);
2017-08-29 02:29:06 +00:00
#ifdef BUMPMODELSPACE
2018-07-05 16:21:44 +00:00
deluxe = normalize(deluxe*invsurface);
2017-10-12 12:02:25 +00:00
#else
2018-10-27 04:32:18 +00:00
deluxe = normalize(deluxe);
2018-07-05 16:21:44 +00:00
lightmaps *= 2.0 / max(0.25, deluxe.z); //counter the darkening from deluxemaps
2017-08-29 02:29:06 +00:00
#endif
2018-07-05 16:21:44 +00:00
lightmaps *= dot(norm, deluxe);
2017-08-29 02:29:06 +00:00
#else
2018-07-05 16:21:44 +00:00
#define deluxe vec3(0.0,0.0,1.0)
2016-07-12 00:40:13 +00:00
#endif
2012-11-27 03:23:19 +00:00
#endif
#endif
2015-02-07 22:34:22 +00:00
//add in specular, if applicable.
2012-11-27 03:23:19 +00:00
#ifdef SPECULAR
2015-02-07 22:34:22 +00:00
vec4 specs = texture2D(s_specular, tc);
2018-07-05 16:21:44 +00:00
vec3 halfdir = normalize(normalize(eyevector) + deluxe); //this norm should be the deluxemap info instead
2017-10-12 12:02:25 +00:00
float spec = pow(max(dot(halfdir, norm), 0.0), FTE_SPECULAR_EXPONENT * specs.a);
2017-08-29 02:29:06 +00:00
spec *= FTE_SPECULAR_MULTIPLIER;
2012-11-27 03:23:19 +00:00
//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;
2012-07-05 13:08:07 +00:00
#endif
2015-05-03 19:57:46 +00:00
#ifdef REFLECTCUBEMASK
2017-10-12 12:02:25 +00:00
vec3 rtc = reflect(normalize(-eyevector), norm);
2015-05-03 19:57:46 +00:00
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;
#endif
2015-02-07 22:34:22 +00:00
#ifdef 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.
2015-03-03 00:14:43 +00:00
float pal = texture2D(s_paletted, tc).r; //the palette index. hopefully not interpolated.
2015-02-07 22:34:22 +00:00
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
2012-11-27 03:23:19 +00:00
//now we have our diffuse+specular terms, modulate by lightmap values.
gl_FragColor.rgb *= lightmaps.rgb;
2012-10-07 18:38:22 +00:00
//add on the fullbright
2012-05-09 15:30:53 +00:00
#ifdef FULLBRIGHT
2015-02-07 22:34:22 +00:00
gl_FragColor.rgb += texture2D(s_fullbright, tc).rgb;
2012-05-09 15:30:53 +00:00
#endif
2015-02-07 22:34:22 +00:00
#endif
2012-10-07 18:38:22 +00:00
//entity modifiers
2012-05-09 15:30:53 +00:00
gl_FragColor = gl_FragColor * e_colourident;
2012-10-07 18:38:22 +00:00
2017-08-29 02:29:06 +00:00
#if defined(MASK)
#if defined(MASKLT)
if (gl_FragColor.a < MASK)
discard;
#else
if (gl_FragColor.a >= MASK)
discard;
#endif
2018-04-15 02:48:23 +00:00
gl_FragColor.a = 1.0; //alpha blending AND alpha testing usually looks stupid, plus it screws up our fog.
2017-08-29 02:29:06 +00:00
#endif
2012-10-07 18:38:22 +00:00
//and finally hide it all if we're fogged.
2012-05-09 15:30:53 +00:00
#ifdef FOG
gl_FragColor = fog4(gl_FragColor);
#endif
}
#endif
2017-10-12 12:02:25 +00:00