140 lines
3.4 KiB
C
140 lines
3.4 KiB
C
/*
|
|
* This implements immediate mode over the top of glDrawArrays
|
|
* current problems:
|
|
*
|
|
* 1. Calling glNormal(); glVertex(); glVertex(); glVertex(); will break.
|
|
* 2. Mixing with glXPointer stuff will break badly
|
|
* 3. This is entirely untested.
|
|
*/
|
|
|
|
#include "../include/gl.h"
|
|
#include "private.h"
|
|
|
|
static GLboolean IMMEDIATE_MODE_ACTIVE = GL_FALSE;
|
|
static GLenum ACTIVE_POLYGON_MODE = GL_TRIANGLES;
|
|
|
|
static AlignedVector VERTICES;
|
|
static AlignedVector COLOURS;
|
|
static AlignedVector TEXCOORDS;
|
|
static AlignedVector NORMALS;
|
|
|
|
|
|
static GLfloat NORMAL[3] = {0.0f, 0.0f, 1.0f};
|
|
static GLfloat COLOR[4] = {1.0f, 1.0f, 1.0f, 1.0f};
|
|
static GLfloat TEXCOORD[2] = {0.0f, 0.0f};
|
|
|
|
|
|
void initImmediateMode() {
|
|
aligned_vector_init(&VERTICES, sizeof(GLfloat));
|
|
aligned_vector_init(&COLOURS, sizeof(GLfloat));
|
|
aligned_vector_init(&TEXCOORDS, sizeof(GLfloat));
|
|
aligned_vector_init(&NORMALS, sizeof(GLfloat));
|
|
}
|
|
|
|
GLubyte checkImmediateModeInactive(const char* func) {
|
|
/* Returns 1 on error */
|
|
if(IMMEDIATE_MODE_ACTIVE) {
|
|
_glKosThrowError(GL_INVALID_OPERATION, func);
|
|
_glKosPrintError();
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void APIENTRY glBegin(GLenum mode) {
|
|
if(IMMEDIATE_MODE_ACTIVE) {
|
|
_glKosThrowError(GL_INVALID_OPERATION, __func__);
|
|
_glKosPrintError();
|
|
return;
|
|
}
|
|
|
|
IMMEDIATE_MODE_ACTIVE = GL_TRUE;
|
|
ACTIVE_POLYGON_MODE = mode;
|
|
}
|
|
|
|
void APIENTRY glColor4f(GLfloat r, GLfloat g, GLfloat b, GLfloat a) {
|
|
COLOR[0] = r;
|
|
COLOR[1] = g;
|
|
COLOR[2] = b;
|
|
COLOR[3] = a;
|
|
}
|
|
|
|
void APIENTRY glColor4fv(const GLfloat* v) {
|
|
glColor4f(v[0], v[1], v[2], v[3]);
|
|
}
|
|
|
|
void APIENTRY glColor3f(GLfloat r, GLfloat g, GLfloat b) {
|
|
static float a = 1.0f;
|
|
glColor4f(r, g, b, a);
|
|
}
|
|
|
|
void APIENTRY glColor3fv(const GLfloat* v) {
|
|
glColor3f(v[0], v[1], v[2]);
|
|
}
|
|
|
|
void APIENTRY glVertex3f(GLfloat x, GLfloat y, GLfloat z) {
|
|
aligned_vector_push_back(&VERTICES, &x, 1);
|
|
aligned_vector_push_back(&VERTICES, &y, 1);
|
|
aligned_vector_push_back(&VERTICES, &z, 1);
|
|
|
|
|
|
/* Push back the stashed colour, normal and texcoord */
|
|
aligned_vector_push_back(&COLOURS, COLOR, 4);
|
|
aligned_vector_push_back(&TEXCOORDS, TEXCOORD, 2);
|
|
aligned_vector_push_back(&NORMALS, NORMAL, 3);
|
|
}
|
|
|
|
void APIENTRY glVertex3fv(const GLfloat* v) {
|
|
glVertex3f(v[0], v[1], v[2]);
|
|
}
|
|
|
|
void APIENTRY glVertex4f(GLfloat x, GLfloat y, GLfloat z, GLfloat w) {
|
|
glVertex3f(x, y, z);
|
|
}
|
|
|
|
void APIENTRY glVertex4fv(const GLfloat* v) {
|
|
glVertex4f(v[0], v[1], v[2], v[3]);
|
|
}
|
|
|
|
void APIENTRY glTexCoord2f(GLfloat u, GLfloat v) {
|
|
TEXCOORD[0] = u;
|
|
TEXCOORD[1] = v;
|
|
}
|
|
|
|
void APIENTRY glTexCoord2fv(const GLfloat* v) {
|
|
glTexCoord2f(v[0], v[1]);
|
|
}
|
|
|
|
void APIENTRY glNormal3f(GLfloat x, GLfloat y, GLfloat z) {
|
|
NORMAL[0] = x;
|
|
NORMAL[1] = y;
|
|
NORMAL[2] = z;
|
|
}
|
|
|
|
void APIENTRY glNormal3fv(const GLfloat* v) {
|
|
glNormal3f(v[0], v[1], v[2]);
|
|
}
|
|
|
|
void APIENTRY glEnd() {
|
|
IMMEDIATE_MODE_ACTIVE = GL_FALSE;
|
|
|
|
/* FIXME: Push pointer state */
|
|
|
|
glVertexPointer(3, GL_FLOAT, 0, VERTICES.data);
|
|
glColorPointer(4, GL_FLOAT, 0, COLOURS.data);
|
|
glNormalPointer(GL_FLOAT, 0, NORMALS.data);
|
|
|
|
glClientActiveTextureARB(GL_TEXTURE0);
|
|
glTexCoordPointer(2, GL_FLOAT, 0, TEXCOORDS.data);
|
|
|
|
glDrawArrays(ACTIVE_POLYGON_MODE, 0, VERTICES.size / 3);
|
|
|
|
aligned_vector_clear(&VERTICES);
|
|
aligned_vector_clear(&COLOURS);
|
|
aligned_vector_clear(&TEXCOORDS);
|
|
aligned_vector_clear(&NORMALS);
|
|
|
|
/* FIXME: Pop pointers */
|
|
}
|