From 5f92a0fcd0213c83a6de294fcf9aba3e0d1162df Mon Sep 17 00:00:00 2001 From: Bigfoot71 Date: Wed, 12 Mar 2025 21:42:29 +0100 Subject: [PATCH] impl face culling --- src/external/rlsw.h | 50 ++++++++++++++++++++++++++++++++++++--------- 1 file changed, 40 insertions(+), 10 deletions(-) diff --git a/src/external/rlsw.h b/src/external/rlsw.h index c4d10a9fb..3e1340569 100644 --- a/src/external/rlsw.h +++ b/src/external/rlsw.h @@ -145,7 +145,7 @@ typedef enum { SW_LINES = GL_LINES, SW_TRIANGLES = GL_TRIANGLES, SW_QUADS = GL_QUADS, -} SWfill; +} SWdraw; typedef enum { SW_FRONT = GL_FRONT, @@ -232,7 +232,9 @@ void swViewport(int x, int y, int width, int height); void swClearColor(float r, float g, float b, float a); void swClear(void); -void swBegin(SWfill mode); +void swCullFace(SWface face); + +void swBegin(SWdraw mode); void swEnd(void); void swVertex2i(int x, int y); @@ -270,7 +272,7 @@ void swNormal3f(float x, float y, float z); void swNormal3fv(const float* v); void swBindArray(SWarray type, void *buffer); -void swDrawArrays(SWfill mode, int offset, int count); +void swDrawArrays(SWdraw mode, int offset, int count); uint32_t swLoadTexture(const void *data, int width, int height, int format, int mipmapCount); void swUnloadTexture(uint32_t id); @@ -370,7 +372,7 @@ typedef struct { sw_vertex_t vertexBuffer[4]; // Buffer used for storing primitive vertices, used for processing and rendering int vertexCounter; // Number of vertices in 'ctx.vertexBuffer' - SWfill fillMode; // Current polygon filling mode (e.g., lines, triangles) + SWdraw drawMode; // Current polygon filling mode (e.g., lines, triangles) float pointSize; // Rasterized point size float lineWidth; // Rasterized line width @@ -936,6 +938,18 @@ static inline void sw_triangle_project_and_clip(sw_vertex_t polygon[SW_MAX_CLIPP sw_vec4_transform(v->homogeneous, v->position, RLSW.matMVP); } + if (RLSW.stateFlags & SW_STATE_CULL_FACE) { + float x0 = polygon[0].homogeneous[0], y0 = polygon[0].homogeneous[1]; + float x1 = polygon[1].homogeneous[0], y1 = polygon[1].homogeneous[1]; + float x2 = polygon[2].homogeneous[0], y2 = polygon[2].homogeneous[1]; + + float sgnArea = (x1 - x0) * (y2 - y0) - (x2 - x0) * (y1 - y0); + if ((RLSW.cullFace == SW_BACK && sgnArea >= 0) || (RLSW.cullFace == SW_FRONT && sgnArea <= 0)) { + *vertexCounter = 0; + return; + } + } + if (sw_triangle_clip_w(polygon, vertexCounter) && sw_triangle_clip_xyz(polygon, vertexCounter)) { for (int i = 0; i < *vertexCounter; i++) { sw_vertex_t *v = polygon + i; @@ -965,7 +979,7 @@ static inline void sw_triangle_project_and_clip(sw_vertex_t polygon[SW_MAX_CLIPP } } -#define DEFINE_TRIANGLE_RASTER_SCANLINE(FUNC_NAME, ENABLE_TEXTURE, ENABLE_DEPTH_TEST) \ +#define DEFINE_TRIANGLE_RASTER_SCANLINE(FUNC_NAME, ENABLE_TEXTURE, ENABLE_DEPTH_TEST) \ static inline void FUNC_NAME(const sw_texture_t* tex, const sw_vertex_t* start, \ const sw_vertex_t* end, float yDu, float yDv) \ { \ @@ -1504,6 +1518,11 @@ static inline bool sw_is_texture_wrap_valid(int wrap) return (wrap == SW_REPEAT || wrap == SW_CLAMP_TO_EDGE || SW_MIRRORED_REPEAT); } +static inline bool sw_is_face_valid(int face) +{ + return (face == SW_FRONT || face == SW_BACK); +} + /* === Public Implementation === */ @@ -1546,6 +1565,8 @@ void swInit(int w, int h) RLSW.vertexBuffer[0].normal[1] = 0.0f; RLSW.vertexBuffer[0].normal[2] = 1.0f; + RLSW.cullFace = SW_BACK; + static const float defTex[3*2*2] = { 1.0f, 1.0f, 1.0f, @@ -1889,14 +1910,23 @@ void swClear(void) } } -void swBegin(SWfill mode) +void swCullFace(SWface face) +{ + if (!sw_is_face_valid(face)) { + RLSW.errCode = SW_INVALID_ENUM; + return; + } + RLSW.cullFace = face; +} + +void swBegin(SWdraw mode) { if (mode < SW_POINTS || mode > SW_QUADS) { RLSW.errCode = SW_INVALID_ENUM; return; } RLSW.vertexCounter = 0; - RLSW.fillMode = mode; + RLSW.drawMode = mode; } void swEnd(void) @@ -1960,7 +1990,7 @@ void swVertex4fv(const float* v) RLSW.vertexCounter++; int neededVertices = 0; - switch (RLSW.fillMode) { + switch (RLSW.drawMode) { case SW_POINTS: neededVertices = 1; break; @@ -1981,7 +2011,7 @@ void swVertex4fv(const float* v) sw_matrix_mul(RLSW.matMVP, RLSW.matModel, RLSW.matView); sw_matrix_mul(RLSW.matMVP, RLSW.matMVP, RLSW.matProjection); - switch (RLSW.fillMode) { + switch (RLSW.drawMode) { case SW_POINTS: break; case SW_LINES: @@ -2259,7 +2289,7 @@ void swBindArray(SWarray type, void *buffer) } } -void swDrawArrays(SWfill mode, int offset, int count) +void swDrawArrays(SWdraw mode, int offset, int count) { if (RLSW.array.positions == 0) { RLSW.errCode = SW_INVALID_OPERATION;