#include #include #include #include #include #include #include "private.h" #include "platform.h" static int TNL_EFFECTS, TNL_LIGHTING, TNL_TEXTURE, TNL_COLOR; #define ITERATE(count) \ GLuint i = count; \ while(i--) void _glTnlLoadMatrix(void) { /* If we're lighting, then we need to do some work in * eye-space, so we only transform vertices by the modelview * matrix, and then later multiply by projection. * * If we're not doing lighting though we can optimise by taking * vertices straight to clip-space */ if(TNL_LIGHTING) { _glMatrixLoadModelView(); } else { _glMatrixLoadModelViewProjection(); } } static void updateEffects(void) { TNL_EFFECTS = TNL_LIGHTING | TNL_TEXTURE | TNL_COLOR; } static void transformVertices(SubmissionTarget* target) { TRACE(); /* Perform modelview transform, storing W */ Vertex* it = _glSubmissionTargetStart(target); uint32_t count = target->count; ITERATE(count) { TransformVertex(it->xyz[0], it->xyz[1], it->xyz[2], it->w, it->xyz, &it->w); it++; } } static void mat_transform_normal3(VertexExtra* extra, const uint32_t count) { ITERATE(count) { TransformNormalNoMod(extra->nxyz, extra->nxyz); extra++; } } static void lightingEffect(SubmissionTarget* target) { /* Perform lighting calculations and manipulate the colour */ Vertex* vertex = _glSubmissionTargetStart(target); VertexExtra* extra = aligned_vector_at(target->extras, 0); _glMatrixLoadNormal(); mat_transform_normal3(extra, target->count); _glPerformLighting(vertex, extra, target->count); } void _glTnlUpdateLighting(void) { TNL_LIGHTING = _glIsLightingEnabled(); updateEffects(); } static void textureEffect(SubmissionTarget* target) { Matrix4x4* m = _glGetTextureMatrix(); UploadMatrix4x4(m); float coords[4]; float* ptr = (float*)m; Vertex* it = _glSubmissionTargetStart(target); uint32_t count = target->count; ITERATE(count) { TransformVertex(it->uv[0], it->uv[1], 0.0f, 1.0f, coords, &coords[3]); it->uv[0] = coords[0]; it->uv[1] = coords[1]; it++; } } void _glTnlUpdateTextureMatrix(void) { Matrix4x4* m = _glGetTextureMatrix(); TNL_TEXTURE = !_glIsIdentity(m); updateEffects(); } static void colorEffect(SubmissionTarget* target) { Matrix4x4* m = _glGetColorMatrix(); UploadMatrix4x4(m); float coords[4]; Vertex* it = _glSubmissionTargetStart(target); uint32_t count = target->count; ITERATE(count) { TransformVertex(it->bgra[2], it->bgra[1], it->bgra[0], it->bgra[3], coords, &coords[3]); it->bgra[2] = clamp(coords[0], 0, 255); it->bgra[1] = clamp(coords[1], 0, 255); it->bgra[0] = clamp(coords[2], 0, 255); it->bgra[3] = clamp(coords[3], 0, 255); it++; } } void _glTnlUpdateColorMatrix(void) { Matrix4x4* m = _glGetColorMatrix(); TNL_COLOR = !_glIsIdentity(m); updateEffects(); } void _glTnlApplyEffects(SubmissionTarget* target) { if (!TNL_EFFECTS) return; if (TNL_LIGHTING) lightingEffect(target); if (TNL_TEXTURE) textureEffect(target); if (TNL_COLOR) colorEffect(target); if (TNL_LIGHTING) { /* OK eye-space work done, now move into clip space */ _glMatrixLoadProjection(); transformVertices(target); } }