GLdc/GL/flush.c

203 lines
5.3 KiB
C
Raw Normal View History

2018-05-05 19:38:55 +00:00
#include "../containers/aligned_vector.h"
#include "private.h"
2021-05-08 13:38:55 +00:00
PolyList OP_LIST;
PolyList PT_LIST;
PolyList TR_LIST;
2018-05-05 19:38:55 +00:00
2022-04-26 17:15:08 +00:00
/**
* FAST_MODE will use invW for all Z coordinates sent to the
* GPU.
*
2021-05-23 12:06:11 +00:00
* This will break orthographic mode so default is FALSE
**/
2021-05-23 12:06:11 +00:00
#define FAST_MODE GL_FALSE
2018-05-05 19:38:55 +00:00
GLboolean AUTOSORT_ENABLED = GL_FALSE;
2021-04-14 20:38:33 +00:00
PolyList* _glOpaquePolyList() {
return &OP_LIST;
}
PolyList* _glPunchThruPolyList() {
return &PT_LIST;
}
PolyList *_glTransparentPolyList() {
2018-08-04 20:00:26 +00:00
return &TR_LIST;
}
2019-03-10 12:19:41 +00:00
void APIENTRY glFlush() {
}
void APIENTRY glFinish() {
}
void APIENTRY glKosInitConfig(GLdcConfig* config) {
config->autosort_enabled = GL_FALSE;
config->fsaa_enabled = GL_FALSE;
2021-04-26 15:14:33 +00:00
config->initial_op_capacity = 1024 * 3;
config->initial_pt_capacity = 512 * 3;
config->initial_tr_capacity = 1024 * 3;
config->initial_immediate_capacity = 1024 * 3;
config->internal_palette_format = GL_RGBA8;
}
void APIENTRY glKosInitEx(GLdcConfig* config) {
TRACE();
printf("\nWelcome to GLdc! Git revision: %s\n\n", GLDC_VERSION);
2021-04-09 15:24:47 +00:00
InitGPU(config->autosort_enabled, config->fsaa_enabled);
AUTOSORT_ENABLED = config->autosort_enabled;
_glInitMatrices();
_glInitAttributePointers();
2018-09-15 10:45:38 +00:00
_glInitContext();
_glInitLights();
2019-03-13 15:43:50 +00:00
_glInitImmediateMode(config->initial_immediate_capacity);
_glInitFramebuffers();
_glSetInternalPaletteFormat(config->internal_palette_format);
_glInitTextures();
2018-05-05 19:38:55 +00:00
2021-04-09 15:24:47 +00:00
OP_LIST.list_type = GPU_LIST_OP_POLY;
PT_LIST.list_type = GPU_LIST_PT_POLY;
TR_LIST.list_type = GPU_LIST_TR_POLY;
2019-03-25 09:44:59 +00:00
aligned_vector_init(&OP_LIST.vector, sizeof(Vertex));
aligned_vector_init(&PT_LIST.vector, sizeof(Vertex));
aligned_vector_init(&TR_LIST.vector, sizeof(Vertex));
2019-03-13 15:43:50 +00:00
aligned_vector_reserve(&OP_LIST.vector, config->initial_op_capacity);
aligned_vector_reserve(&PT_LIST.vector, config->initial_pt_capacity);
aligned_vector_reserve(&TR_LIST.vector, config->initial_tr_capacity);
}
void APIENTRY glKosInit() {
GLdcConfig config;
glKosInitConfig(&config);
glKosInitEx(&config);
2018-05-05 19:38:55 +00:00
}
2021-04-26 15:14:33 +00:00
#define likely(x) __builtin_expect(!!(x), 1)
#define unlikely(x) __builtin_expect(!!(x), 0)
GL_FORCE_INLINE bool glIsVertex(const float flags) {
return flags == GPU_CMD_VERTEX_EOL || flags == GPU_CMD_VERTEX;
}
GL_FORCE_INLINE void glPerspectiveDivideStandard(void* src, uint32_t n) {
2021-04-26 15:14:33 +00:00
TRACE();
/* Perform perspective divide on each vertex */
Vertex* vertex = (Vertex*) src;
2021-05-25 19:40:43 +00:00
PREFETCH(vertex + 1);
2021-04-26 15:14:33 +00:00
const float h = GetVideoMode()->height;
while(n--) {
2021-05-25 19:40:43 +00:00
PREFETCH(vertex + 2);
2021-04-26 15:14:33 +00:00
if(likely(glIsVertex(vertex->flags))) {
const float f = MATH_Fast_Invert(vertex->w);
/* Convert to NDC and apply viewport */
2021-05-25 19:40:43 +00:00
vertex->xyz[0] = __builtin_fmaf(
2021-04-26 15:14:33 +00:00
VIEWPORT.hwidth, vertex->xyz[0] * f, VIEWPORT.x_plus_hwidth
);
2021-05-25 19:40:43 +00:00
vertex->xyz[1] = h - __builtin_fmaf(
2021-04-26 15:14:33 +00:00
VIEWPORT.hheight, vertex->xyz[1] * f, VIEWPORT.y_plus_hheight
);
2022-04-26 17:15:08 +00:00
/* Orthographic projections need to use invZ otherwise we lose
the depth information. As w == 1, and clip-space range is -w to +w
we add 1.0 to the Z to bring it into range. We add a little extra to
avoid a divide by zero.
*/
if(unlikely(vertex->w == 1.0f)) {
vertex->xyz[2] = MATH_Fast_Invert(1.0001f + vertex->xyz[2]);
} else {
vertex->xyz[2] = f;
}
2021-04-26 15:14:33 +00:00
}
++vertex;
}
}
GL_FORCE_INLINE void glPerspectiveDivideFastMode(void* src, uint32_t n) {
TRACE();
/* Perform perspective divide on each vertex */
Vertex* vertex = (Vertex*) src;
const float h = GetVideoMode()->height;
while(n--) {
2021-05-14 20:53:27 +00:00
PREFETCH(vertex + 1);
if(likely(glIsVertex(vertex->flags))) {
const float f = MATH_Fast_Invert(vertex->w);
/* Convert to NDC and apply viewport */
vertex->xyz[0] = MATH_fmac(
VIEWPORT.hwidth, vertex->xyz[0] * f, VIEWPORT.x_plus_hwidth
);
vertex->xyz[1] = h - MATH_fmac(
VIEWPORT.hheight, vertex->xyz[1] * f, VIEWPORT.y_plus_hheight
);
vertex->xyz[2] = f;
}
++vertex;
}
}
GL_FORCE_INLINE void glPerspectiveDivide(void* src, uint32_t n) {
#if FAST_MODE
glPerspectiveDivideFastMode(src, n);
#else
glPerspectiveDivideStandard(src, n);
#endif
}
void APIENTRY glKosSwapBuffers() {
TRACE();
2021-04-09 15:24:47 +00:00
SceneBegin();
SceneListBegin(GPU_LIST_OP_POLY);
2021-04-26 15:14:33 +00:00
glPerspectiveDivide(OP_LIST.vector.data, OP_LIST.vector.size);
2021-04-09 15:24:47 +00:00
SceneListSubmit(OP_LIST.vector.data, OP_LIST.vector.size);
SceneListFinish();
2018-05-05 19:38:55 +00:00
2021-04-09 15:24:47 +00:00
SceneListBegin(GPU_LIST_PT_POLY);
2021-04-26 15:14:33 +00:00
glPerspectiveDivide(PT_LIST.vector.data, PT_LIST.vector.size);
2021-04-09 15:24:47 +00:00
SceneListSubmit(PT_LIST.vector.data, PT_LIST.vector.size);
SceneListFinish();
2018-05-05 19:38:55 +00:00
2021-04-09 15:24:47 +00:00
SceneListBegin(GPU_LIST_TR_POLY);
2021-04-26 15:14:33 +00:00
glPerspectiveDivide(TR_LIST.vector.data, TR_LIST.vector.size);
2021-04-09 15:24:47 +00:00
SceneListSubmit(TR_LIST.vector.data, TR_LIST.vector.size);
SceneListFinish();
SceneFinish();
aligned_vector_clear(&OP_LIST.vector);
aligned_vector_clear(&PT_LIST.vector);
aligned_vector_clear(&TR_LIST.vector);
2018-08-19 20:10:42 +00:00
2021-04-14 20:38:33 +00:00
_glApplyScissor(true);
2018-05-05 19:38:55 +00:00
}