2006-05-16 14:52:36 +00:00
|
|
|
|
2009-05-08 07:32:33 +00:00
|
|
|
#include "graphics/tinygl/zgl.h"
|
2014-08-07 15:54:53 +00:00
|
|
|
#include "graphics/tinygl/zdirtyrect.h"
|
2008-01-26 11:47:23 +00:00
|
|
|
|
2009-05-25 13:19:29 +00:00
|
|
|
namespace TinyGL {
|
2005-01-12 15:20:02 +00:00
|
|
|
|
2009-05-25 13:19:29 +00:00
|
|
|
void glopNormal(GLContext *c, GLParam *p) {
|
2014-06-08 13:20:27 +00:00
|
|
|
c->current_normal.X = p[1].f;
|
|
|
|
c->current_normal.Y = p[2].f;
|
|
|
|
c->current_normal.Z = p[3].f;
|
|
|
|
c->current_normal.W = 0.0f;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2009-05-25 13:19:29 +00:00
|
|
|
void glopTexCoord(GLContext *c, GLParam *p) {
|
2014-06-08 13:20:27 +00:00
|
|
|
c->current_tex_coord.X = p[1].f;
|
|
|
|
c->current_tex_coord.Y = p[2].f;
|
|
|
|
c->current_tex_coord.Z = p[3].f;
|
|
|
|
c->current_tex_coord.W = p[4].f;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2009-05-25 13:19:29 +00:00
|
|
|
void glopEdgeFlag(GLContext *c, GLParam *p) {
|
2006-05-16 14:52:36 +00:00
|
|
|
c->current_edge_flag = p[1].i;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2014-02-20 17:46:49 +00:00
|
|
|
void glopColor(GLContext *c, GLParam *p) {
|
2014-06-08 13:20:27 +00:00
|
|
|
c->current_color.X = p[1].f;
|
|
|
|
c->current_color.Y = p[2].f;
|
|
|
|
c->current_color.Z = p[3].f;
|
|
|
|
c->current_color.W = p[4].f;
|
2006-05-16 14:52:36 +00:00
|
|
|
c->longcurrent_color[0] = p[5].ui;
|
|
|
|
c->longcurrent_color[1] = p[6].ui;
|
|
|
|
c->longcurrent_color[2] = p[7].ui;
|
2014-06-19 15:30:28 +00:00
|
|
|
c->longcurrent_color[3] = p[8].ui;
|
2006-05-16 14:52:36 +00:00
|
|
|
|
|
|
|
if (c->color_material_enabled) {
|
2009-05-25 13:19:29 +00:00
|
|
|
GLParam q[7];
|
2006-05-16 14:52:36 +00:00
|
|
|
q[0].op = OP_Material;
|
|
|
|
q[1].i = c->current_color_material_mode;
|
|
|
|
q[2].i = c->current_color_material_type;
|
|
|
|
q[3].f = p[1].f;
|
|
|
|
q[4].f = p[2].f;
|
|
|
|
q[5].f = p[3].f;
|
|
|
|
q[6].f = p[4].f;
|
|
|
|
glopMaterial(c, q);
|
|
|
|
}
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
void gl_eval_viewport(GLContext *c) {
|
|
|
|
GLViewport *v;
|
|
|
|
float zsize = (1 << (ZB_Z_BITS + ZB_POINT_Z_FRAC_BITS));
|
2005-01-12 15:20:02 +00:00
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
v = &c->viewport;
|
2005-01-12 15:20:02 +00:00
|
|
|
|
2014-06-09 17:53:40 +00:00
|
|
|
v->trans.X = (float)(((v->xsize - 0.5) / 2.0) + v->xmin);
|
|
|
|
v->trans.Y = (float)(((v->ysize - 0.5) / 2.0) + v->ymin);
|
|
|
|
v->trans.Z = (float)(((zsize - 0.5) / 2.0) + ((1 << ZB_POINT_Z_FRAC_BITS)) / 2);
|
2005-01-12 15:20:02 +00:00
|
|
|
|
2014-06-09 17:53:40 +00:00
|
|
|
v->scale.X = (float)((v->xsize - 0.5) / 2.0);
|
|
|
|
v->scale.Y = (float)(-(v->ysize - 0.5) / 2.0);
|
|
|
|
v->scale.Z = (float)(-((zsize - 0.5) / 2.0));
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2009-05-25 13:19:29 +00:00
|
|
|
void glopBegin(GLContext *c, GLParam *p) {
|
2006-05-16 14:52:36 +00:00
|
|
|
int type;
|
|
|
|
|
|
|
|
assert(c->in_begin == 0);
|
|
|
|
|
|
|
|
type = p[1].i;
|
|
|
|
c->begin_type = type;
|
|
|
|
c->in_begin = 1;
|
|
|
|
c->vertex_n = 0;
|
|
|
|
c->vertex_cnt = 0;
|
|
|
|
|
|
|
|
if (c->matrix_model_projection_updated) {
|
|
|
|
if (c->lighting_enabled) {
|
|
|
|
// precompute inverse modelview
|
2014-06-06 22:55:08 +00:00
|
|
|
c->matrix_model_view_inv = *c->matrix_stack_ptr[0];
|
|
|
|
c->matrix_model_view_inv.invert();
|
|
|
|
c->matrix_model_view_inv.transpose();
|
2006-05-16 14:52:36 +00:00
|
|
|
} else {
|
|
|
|
// precompute projection matrix
|
2014-05-28 18:27:00 +00:00
|
|
|
c->matrix_model_projection = (*c->matrix_stack_ptr[1]) * (*c->matrix_stack_ptr[0]);
|
2006-05-16 14:52:36 +00:00
|
|
|
// test to accelerate computation
|
|
|
|
c->matrix_model_projection_no_w_transform = 0;
|
2014-06-08 13:20:27 +00:00
|
|
|
if (c->matrix_model_projection._m[3][0] == 0.0 && c->matrix_model_projection._m[3][1] == 0.0 && c->matrix_model_projection._m[3][2] == 0.0)
|
2006-05-16 14:52:36 +00:00
|
|
|
c->matrix_model_projection_no_w_transform = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// test if the texture matrix is not Identity
|
2014-06-10 15:14:54 +00:00
|
|
|
c->apply_texture_matrix = !c->matrix_stack_ptr[2]->isIdentity();
|
2006-05-16 14:52:36 +00:00
|
|
|
|
|
|
|
c->matrix_model_projection_updated = 0;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
2006-05-16 14:52:36 +00:00
|
|
|
// viewport
|
|
|
|
if (c->viewport.updated) {
|
|
|
|
gl_eval_viewport(c);
|
|
|
|
c->viewport.updated = 0;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
2014-02-20 17:46:49 +00:00
|
|
|
// triangle drawing functions
|
2006-05-16 14:52:36 +00:00
|
|
|
if (c->render_mode == TGL_SELECT) {
|
|
|
|
c->draw_triangle_front = gl_draw_triangle_select;
|
|
|
|
c->draw_triangle_back = gl_draw_triangle_select;
|
|
|
|
} else {
|
|
|
|
switch (c->polygon_mode_front) {
|
|
|
|
case TGL_POINT:
|
|
|
|
c->draw_triangle_front = gl_draw_triangle_point;
|
|
|
|
break;
|
|
|
|
case TGL_LINE:
|
|
|
|
c->draw_triangle_front = gl_draw_triangle_line;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
c->draw_triangle_front = gl_draw_triangle_fill;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (c->polygon_mode_back) {
|
|
|
|
case TGL_POINT:
|
|
|
|
c->draw_triangle_back = gl_draw_triangle_point;
|
|
|
|
break;
|
|
|
|
case TGL_LINE:
|
|
|
|
c->draw_triangle_back = gl_draw_triangle_line;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
c->draw_triangle_back = gl_draw_triangle_fill;
|
|
|
|
break;
|
|
|
|
}
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
// coords, tranformation, clip code and projection
|
|
|
|
// TODO : handle all cases
|
|
|
|
static inline void gl_vertex_transform(GLContext *c, GLVertex *v) {
|
2014-05-28 18:27:00 +00:00
|
|
|
Matrix4 *m;
|
2006-05-16 14:52:36 +00:00
|
|
|
|
|
|
|
if (c->lighting_enabled) {
|
2014-02-20 17:46:49 +00:00
|
|
|
// eye coordinates needed for lighting
|
2006-05-16 14:52:36 +00:00
|
|
|
|
2014-05-28 18:27:00 +00:00
|
|
|
m = c->matrix_stack_ptr[0];
|
2014-06-09 17:53:40 +00:00
|
|
|
m->transform3x4(v->coord,v->ec);
|
2006-05-16 14:52:36 +00:00
|
|
|
|
|
|
|
// projection coordinates
|
2014-05-28 18:27:00 +00:00
|
|
|
m = c->matrix_stack_ptr[1];
|
2014-06-09 17:53:40 +00:00
|
|
|
m->transform(v->ec, v->pc);
|
2014-06-06 13:17:05 +00:00
|
|
|
|
2014-05-28 18:27:00 +00:00
|
|
|
m = &c->matrix_model_view_inv;
|
2006-05-16 14:52:36 +00:00
|
|
|
|
2014-06-10 15:14:54 +00:00
|
|
|
m->transform3x3(c->current_normal, v->normal);
|
2006-05-16 14:52:36 +00:00
|
|
|
|
|
|
|
if (c->normalize_enabled) {
|
2014-05-28 18:27:00 +00:00
|
|
|
v->normal.normalize();
|
2006-05-16 14:52:36 +00:00
|
|
|
}
|
2014-02-20 17:46:49 +00:00
|
|
|
} else {
|
2006-05-16 14:52:36 +00:00
|
|
|
// no eye coordinates needed, no normal
|
|
|
|
// NOTE: W = 1 is assumed
|
2014-05-28 18:27:00 +00:00
|
|
|
m = &c->matrix_model_projection;
|
2006-05-16 14:52:36 +00:00
|
|
|
|
2014-06-09 17:53:40 +00:00
|
|
|
m->transform3x4(v->coord, v->pc);
|
2006-05-16 14:52:36 +00:00
|
|
|
if (c->matrix_model_projection_no_w_transform) {
|
2014-06-08 13:20:27 +00:00
|
|
|
v->pc.W = (m->_m[3][3]);
|
2006-05-16 14:52:36 +00:00
|
|
|
}
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2014-06-06 22:55:08 +00:00
|
|
|
v->clip_code = gl_clipcode(v->pc.X, v->pc.Y, v->pc.Z, v->pc.W);
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2009-05-25 13:19:29 +00:00
|
|
|
void glopVertex(GLContext *c, GLParam *p) {
|
2006-05-16 14:52:36 +00:00
|
|
|
GLVertex *v;
|
2014-02-20 16:56:55 +00:00
|
|
|
int n, cnt;
|
2006-05-16 14:52:36 +00:00
|
|
|
|
|
|
|
assert(c->in_begin != 0);
|
|
|
|
|
|
|
|
n = c->vertex_n;
|
|
|
|
cnt = c->vertex_cnt;
|
|
|
|
cnt++;
|
|
|
|
c->vertex_cnt = cnt;
|
|
|
|
|
|
|
|
// quick fix to avoid crashes on large polygons
|
|
|
|
if (n >= c->vertex_max) {
|
|
|
|
GLVertex *newarray;
|
2014-02-20 17:46:49 +00:00
|
|
|
c->vertex_max <<= 1; // just double size
|
2006-05-16 14:52:36 +00:00
|
|
|
newarray = (GLVertex *)gl_malloc(sizeof(GLVertex) * c->vertex_max);
|
|
|
|
if (!newarray) {
|
2008-09-28 19:09:23 +00:00
|
|
|
error("unable to allocate GLVertex array.");
|
2006-05-16 14:52:36 +00:00
|
|
|
}
|
|
|
|
memcpy(newarray, c->vertex, n * sizeof(GLVertex));
|
|
|
|
gl_free(c->vertex);
|
|
|
|
c->vertex = newarray;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
2006-05-16 14:52:36 +00:00
|
|
|
// new vertex entry
|
|
|
|
v = &c->vertex[n];
|
|
|
|
n++;
|
2005-01-12 15:20:02 +00:00
|
|
|
|
2014-06-08 13:20:27 +00:00
|
|
|
v->coord.X = p[1].f;
|
|
|
|
v->coord.Y = p[2].f;
|
|
|
|
v->coord.Z = p[3].f;
|
|
|
|
v->coord.W = p[4].f;
|
2005-01-12 15:20:02 +00:00
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
gl_vertex_transform(c, v);
|
2005-01-12 15:20:02 +00:00
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
// color
|
2005-01-12 15:20:02 +00:00
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
if (c->lighting_enabled) {
|
|
|
|
gl_shade_vertex(c, v);
|
|
|
|
} else {
|
|
|
|
v->color = c->current_color;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
// tex coords
|
|
|
|
|
|
|
|
if (c->texture_2d_enabled) {
|
|
|
|
if (c->apply_texture_matrix) {
|
2014-06-09 17:53:40 +00:00
|
|
|
c->matrix_stack_ptr[2]->transform(c->current_tex_coord, v->tex_coord);
|
2006-05-16 14:52:36 +00:00
|
|
|
} else {
|
|
|
|
v->tex_coord = c->current_tex_coord;
|
|
|
|
}
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
2014-02-20 17:46:49 +00:00
|
|
|
// precompute the mapping to the viewport
|
2006-05-16 14:52:36 +00:00
|
|
|
if (v->clip_code == 0)
|
|
|
|
gl_transform_to_viewport(c, v);
|
|
|
|
|
2014-02-20 17:46:49 +00:00
|
|
|
// edge flag
|
2006-05-16 14:52:36 +00:00
|
|
|
|
|
|
|
v->edge_flag = c->current_edge_flag;
|
2014-07-24 17:50:11 +00:00
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
c->vertex_n = n;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
|
|
|
|
2009-05-25 13:19:29 +00:00
|
|
|
void glopEnd(GLContext *c, GLParam *) {
|
2006-05-16 14:52:36 +00:00
|
|
|
assert(c->in_begin == 1);
|
2014-08-05 11:18:47 +00:00
|
|
|
|
|
|
|
if (c->vertex_cnt > 0) {
|
2014-08-07 15:38:11 +00:00
|
|
|
tglIssueDrawCall(new Graphics::RasterizationDrawCall());
|
2014-08-05 11:18:47 +00:00
|
|
|
}
|
2014-07-24 17:50:11 +00:00
|
|
|
|
2006-05-16 14:52:36 +00:00
|
|
|
c->in_begin = 0;
|
2005-01-12 15:20:02 +00:00
|
|
|
}
|
2009-05-25 13:19:29 +00:00
|
|
|
|
|
|
|
} // end of namespace TinyGL
|