reactos/dll/opengl/mesa/vbfill.c

1454 lines
40 KiB
C
Raw Normal View History

/* $Id: vbfill.c,v 1.22 1998/01/27 03:30:18 brianp Exp $ */
/*
* Mesa 3-D graphics library
* Version: 2.6
* Copyright (C) 1995-1997 Brian Paul
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library; if not, write to the Free
* Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
/*
* $Log: vbfill.c,v $
* Revision 1.22 1998/01/27 03:30:18 brianp
* minor tweak to FLOAT_COLOR_TO_UBYTE_COLOR macro: added an F suffix
*
* Revision 1.21 1998/01/25 16:59:13 brianp
* changed IEEE_ONE value to 0x3f7f0000 (Josh Vanderhoof)
*
* Revision 1.20 1998/01/16 01:29:29 brianp
* added DavidB's assembly language version of gl_Color3f()
*
* Revision 1.19 1998/01/09 02:40:43 brianp
* IEEE-optimized glColor[34]f[v]() commands (Josh Vanderhoof)
*
* Revision 1.18 1997/12/18 02:54:48 brianp
* now using FloatToInt() macro for better performance on x86
*
* Revision 1.17 1997/11/14 03:02:53 brianp
* clamp floating point color components to [0,1] before int conversion
*
* Revision 1.16 1997/08/13 01:31:11 brianp
* LightTwoSide is now a GLboolean
*
* Revision 1.15 1997/07/24 01:25:27 brianp
* changed precompiled header symbol from PCH to PC_HEADER
*
* Revision 1.14 1997/06/20 02:47:41 brianp
* added Color4ubv API pointer
*
* Revision 1.13 1997/06/20 02:46:49 brianp
* changed color components from GLfixed to GLubyte
*
* Revision 1.12 1997/05/28 03:26:49 brianp
* added precompiled header (PCH) support
*
* Revision 1.11 1997/05/27 03:13:41 brianp
* removed some debugging code
*
* Revision 1.10 1997/04/28 02:05:44 brianp
* renamed some vertex functions, also save color with texcoords
*
* Revision 1.9 1997/04/24 01:50:53 brianp
* optimized glColor3f, glColor3fv, glColor4fv
*
* Revision 1.8 1997/04/24 00:30:17 brianp
* optimized glTexCoord2() code
*
* Revision 1.7 1997/04/20 15:59:30 brianp
* removed VERTEX2_BIT stuff
*
* Revision 1.6 1997/04/16 23:55:33 brianp
* added optimized glTexCoord2f code
*
* Revision 1.5 1997/04/14 22:18:23 brianp
* added optimized glVertex3fv code
*
* Revision 1.4 1997/04/12 16:21:54 brianp
* added ctx->Exec.Vertex2f = vertex2_feedback; statement
*
* Revision 1.3 1997/04/12 12:23:26 brianp
* fixed 3 bugs in gl_eval_vertex
*
* Revision 1.2 1997/04/07 03:01:11 brianp
* optimized vertex[234] code
*
* Revision 1.1 1997/04/02 03:13:56 brianp
* Initial revision
*
*/
/*
* This file implements the functions for filling the vertex buffer:
* glVertex, glNormal, glColor, glIndex, glEdgeFlag, glTexCoord,
*/
#ifdef PC_HEADER
#include "all.h"
#else
#include <assert.h>
#include "context.h"
#include "light.h"
#include "clip.h"
#include "dlist.h"
#include "feedback.h"
#include "macros.h"
#include "matrix.h"
#include "mmath.h"
#include "pb.h"
#include "types.h"
#include "vb.h"
#include "vbfill.h"
#include "vbxform.h"
#include "xform.h"
#endif
/**********************************************************************/
/****** glNormal functions *****/
/**********************************************************************/
/*
* Caller: context->API.Normal3f pointer.
*/
void gl_Normal3f( GLcontext *ctx, GLfloat nx, GLfloat ny, GLfloat nz )
{
ctx->Current.Normal[0] = nx;
ctx->Current.Normal[1] = ny;
ctx->Current.Normal[2] = nz;
ctx->VB->MonoNormal = GL_FALSE;
}
/*
* Caller: context->API.Normal3fv pointer.
*/
void gl_Normal3fv( GLcontext *ctx, const GLfloat *n )
{
ctx->Current.Normal[0] = n[0];
ctx->Current.Normal[1] = n[1];
ctx->Current.Normal[2] = n[2];
ctx->VB->MonoNormal = GL_FALSE;
}
/**********************************************************************/
/****** glIndex functions *****/
/**********************************************************************/
/*
* Caller: context->API.Indexf pointer.
*/
void gl_Indexf( GLcontext *ctx, GLfloat c )
{
ctx->Current.Index = (GLuint) (GLint) c;
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Caller: context->API.Indexi pointer.
*/
void gl_Indexi( GLcontext *ctx, GLint c )
{
ctx->Current.Index = (GLuint) c;
ctx->VB->MonoColor = GL_FALSE;
}
/**********************************************************************/
/****** glColor functions *****/
/**********************************************************************/
#if defined(__i386__)
#define USE_IEEE
#endif
#if defined(USE_IEEE) && !defined(DEBUG) && 0
#define IEEE_ONE 0x3f7f0000
/*
* Optimization for:
* GLfloat f;
* GLubyte b = FloatToInt(CLAMP(f, 0, 1) * 255)
*/
#define FLOAT_COLOR_TO_UBYTE_COLOR(b, f) \
{ \
GLfloat tmp = f + 32768.0F; \
b = ((*(GLuint *)&f >= IEEE_ONE) \
? (*(GLint *)&f < 0) ? (GLubyte)0 : (GLubyte)255 \
: (GLubyte)*(GLuint *)&tmp); \
}
#else
#define FLOAT_COLOR_TO_UBYTE_COLOR(b, f) \
b = FloatToInt(CLAMP(f, 0.0F, 1.0F) * 255.0F)
#endif
/*
* Used when colors are not scaled to [0,255].
* Caller: context->API.Color3f pointer.
*/
void gl_Color3f( GLcontext *ctx, GLfloat red, GLfloat green, GLfloat blue )
{
ctx->Current.ByteColor[0] = FloatToInt(CLAMP(red , 0.0F, 1.0F) * ctx->Visual->RedScale);
ctx->Current.ByteColor[1] = FloatToInt(CLAMP(green, 0.0F, 1.0F) * ctx->Visual->GreenScale);
ctx->Current.ByteColor[2] = FloatToInt(CLAMP(blue , 0.0F, 1.0F) * ctx->Visual->BlueScale);
ctx->Current.ByteColor[3] = FloatToInt(ctx->Visual->AlphaScale);
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are scaled to [0,255].
* Caller: context->API.Color3f pointer.
*/
void gl_Color3f8bit( GLcontext *ctx, GLfloat red, GLfloat green, GLfloat blue )
{
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[0], red);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[1], green);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[2], blue);
ctx->Current.ByteColor[3] = 255;
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are not scaled to [0,255].
* Caller: context->API.Color3fv pointer.
*/
void gl_Color3fv( GLcontext *ctx, const GLfloat *c )
{
ctx->Current.ByteColor[0] = FloatToInt(CLAMP(c[0], 0.0F, 1.0F) * ctx->Visual->RedScale);
ctx->Current.ByteColor[1] = FloatToInt(CLAMP(c[1], 0.0F, 1.0F) * ctx->Visual->GreenScale);
ctx->Current.ByteColor[2] = FloatToInt(CLAMP(c[2], 0.0F, 1.0F) * ctx->Visual->BlueScale);
ctx->Current.ByteColor[3] = FloatToInt(ctx->Visual->AlphaScale);
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are scaled to [0,255].
* Caller: context->API.Color3fv pointer.
*/
void gl_Color3fv8bit( GLcontext *ctx, const GLfloat *c )
{
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[0], c[0]);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[1], c[1]);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[2], c[2]);
ctx->Current.ByteColor[3] = 255;
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are not scaled to [0,255].
* Caller: context->API.Color4f pointer.
*/
void gl_Color4f( GLcontext *ctx,
GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha )
{
ctx->Current.ByteColor[0] = FloatToInt(CLAMP(red , 0.0F, 1.0F) * ctx->Visual->RedScale);
ctx->Current.ByteColor[1] = FloatToInt(CLAMP(green, 0.0F, 1.0F) * ctx->Visual->GreenScale);
ctx->Current.ByteColor[2] = FloatToInt(CLAMP(blue , 0.0F, 1.0F) * ctx->Visual->BlueScale);
ctx->Current.ByteColor[3] = FloatToInt(CLAMP(alpha, 0.0F, 1.0F) * ctx->Visual->AlphaScale);
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are scaled to [0,255].
* Caller: context->API.Color4f pointer.
*/
void gl_Color4f8bit( GLcontext *ctx,
GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha )
{
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[0], red);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[1], green);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[2], blue);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[3], alpha);
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are not scaled to [0,255].
* Caller: context->API.Color4fv pointer.
*/
void gl_Color4fv( GLcontext *ctx, const GLfloat *c )
{
ctx->Current.ByteColor[0] = FloatToInt(CLAMP(c[0], 0.0F, 1.0F) * ctx->Visual->RedScale);
ctx->Current.ByteColor[1] = FloatToInt(CLAMP(c[1], 0.0F, 1.0F) * ctx->Visual->GreenScale);
ctx->Current.ByteColor[2] = FloatToInt(CLAMP(c[2], 0.0F, 1.0F) * ctx->Visual->BlueScale);
ctx->Current.ByteColor[3] = FloatToInt(CLAMP(c[3], 0.0F, 1.0F) * ctx->Visual->AlphaScale);
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are scaled to [0,255].
* Caller: context->API.Color4fv pointer.
*/
void gl_Color4fv8bit( GLcontext *ctx, const GLfloat *c )
{
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[0], c[0]);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[1], c[1]);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[2], c[2]);
FLOAT_COLOR_TO_UBYTE_COLOR(ctx->Current.ByteColor[3], c[3]);
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are not scaled to [0,255]
* Caller: context->API.Color4ub pointer.
*/
void gl_Color4ub( GLcontext *ctx,
GLubyte red, GLubyte green, GLubyte blue, GLubyte alpha )
{
ctx->Current.ByteColor[0] = red * ctx->Visual->RedScale * (1.0F/255.0F);
ctx->Current.ByteColor[1] = green * ctx->Visual->GreenScale * (1.0F/255.0F);
ctx->Current.ByteColor[2] = blue * ctx->Visual->BlueScale * (1.0F/255.0F);
ctx->Current.ByteColor[3] = alpha * ctx->Visual->AlphaScale * (1.0F/255.0F);
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are scaled to [0,255].
* Caller: context->API.Color4ub pointer.
*/
void gl_Color4ub8bit( GLcontext *ctx,
GLubyte red, GLubyte green, GLubyte blue, GLubyte alpha )
{
ASSIGN_4V( ctx->Current.ByteColor, red, green, blue, alpha );
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* Used when colors are not scaled to [0,255]
* Caller: context->API.Color4ub pointer.
*/
void gl_Color4ubv( GLcontext *ctx, const GLubyte *c )
{
ctx->Current.ByteColor[0] = c[0] * ctx->Visual->RedScale * (1.0F/255.0F);
ctx->Current.ByteColor[1] = c[1] * ctx->Visual->GreenScale * (1.0F/255.0F);
ctx->Current.ByteColor[2] = c[2] * ctx->Visual->BlueScale * (1.0F/255.0F);
ctx->Current.ByteColor[3] = c[3] * ctx->Visual->AlphaScale * (1.0F/255.0F);
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* This is the most efficient glColor*() command!
* Used when colors are scaled to [0,255].
* Caller: context->API.Color4ub pointer.
*/
void gl_Color4ubv8bit( GLcontext *ctx, const GLubyte *c )
{
COPY_4UBV( ctx->Current.ByteColor, c );
ASSERT( !ctx->Light.ColorMaterialEnabled );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* glColor() which modifies material(s).
* Caller: context->API.Color3f pointer.
*/
void gl_ColorMat3f( GLcontext *ctx, GLfloat red, GLfloat green, GLfloat blue )
{
GLfloat color[4];
ctx->Current.ByteColor[0] = FloatToInt(CLAMP(red , 0.0F, 1.0F) * ctx->Visual->RedScale);
ctx->Current.ByteColor[1] = FloatToInt(CLAMP(green, 0.0F, 1.0F) * ctx->Visual->GreenScale);
ctx->Current.ByteColor[2] = FloatToInt(CLAMP(blue , 0.0F, 1.0F) * ctx->Visual->BlueScale);
ctx->Current.ByteColor[3] = FloatToInt(ctx->Visual->AlphaScale);
/* update material */
ASSERT( ctx->Light.ColorMaterialEnabled );
ASSIGN_4V( color, red, green, blue, 1.0F );
gl_set_material( ctx, ctx->Light.ColorMaterialBitmask, color );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* glColor() which modifies material(s).
* Caller: context->API.Color3fv pointer.
*/
void gl_ColorMat3fv( GLcontext *ctx, const GLfloat *c )
{
GLfloat color[4];
ctx->Current.ByteColor[0] = FloatToInt(CLAMP(c[0], 0.0F, 1.0F) * ctx->Visual->RedScale);
ctx->Current.ByteColor[1] = FloatToInt(CLAMP(c[1], 0.0F, 1.0F) * ctx->Visual->GreenScale);
ctx->Current.ByteColor[2] = FloatToInt(CLAMP(c[2], 0.0F, 1.0F) * ctx->Visual->BlueScale);
ctx->Current.ByteColor[3] = FloatToInt(ctx->Visual->AlphaScale);
/* update material */
ASSERT( ctx->Light.ColorMaterialEnabled );
ASSIGN_4V( color, c[0], c[1], c[2], 1.0F );
gl_set_material( ctx, ctx->Light.ColorMaterialBitmask, color );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* glColor() which modifies material(s).
* Caller: context->API.Color4f pointer.
*/
void gl_ColorMat4f( GLcontext *ctx,
GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha )
{
GLfloat color[4];
ctx->Current.ByteColor[0] = FloatToInt(CLAMP(red , 0.0F, 1.0F) * ctx->Visual->RedScale);
ctx->Current.ByteColor[1] = FloatToInt(CLAMP(green, 0.0F, 1.0F) * ctx->Visual->GreenScale);
ctx->Current.ByteColor[2] = FloatToInt(CLAMP(blue , 0.0F, 1.0F) * ctx->Visual->BlueScale);
ctx->Current.ByteColor[3] = FloatToInt(CLAMP(alpha, 0.0F, 1.0F) * ctx->Visual->AlphaScale);
/* update material */
ASSERT( ctx->Light.ColorMaterialEnabled );
ASSIGN_4V( color, red, green, blue, alpha );
gl_set_material( ctx, ctx->Light.ColorMaterialBitmask, color );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* glColor() which modifies material(s).
* Caller: context->API.Color4fv pointer.
*/
void gl_ColorMat4fv( GLcontext *ctx, const GLfloat *c )
{
GLfloat color[4];
ctx->Current.ByteColor[0] = FloatToInt(CLAMP(c[0], 0.0F, 1.0F) * ctx->Visual->RedScale);
ctx->Current.ByteColor[1] = FloatToInt(CLAMP(c[1], 0.0F, 1.0F) * ctx->Visual->GreenScale);
ctx->Current.ByteColor[2] = FloatToInt(CLAMP(c[2], 0.0F, 1.0F) * ctx->Visual->BlueScale);
ctx->Current.ByteColor[3] = FloatToInt(CLAMP(c[3], 0.0F, 1.0F) * ctx->Visual->AlphaScale);
/* update material */
ASSERT( ctx->Light.ColorMaterialEnabled );
ASSIGN_4V( color, c[0], c[1], c[2], c[3] );
gl_set_material( ctx, ctx->Light.ColorMaterialBitmask, color );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* glColor which modifies material(s).
* Caller: context->API.Color4ub pointer.
*/
void gl_ColorMat4ub( GLcontext *ctx,
GLubyte red, GLubyte green, GLubyte blue, GLubyte alpha )
{
GLfloat color[4];
if (ctx->Visual->EightBitColor) {
ASSIGN_4V( ctx->Current.ByteColor, red, green, blue, alpha );
}
else {
ctx->Current.ByteColor[0] = red * ctx->Visual->RedScale * (1.0F/255.0F);
ctx->Current.ByteColor[1] = green * ctx->Visual->GreenScale * (1.0F/255.0F);
ctx->Current.ByteColor[2] = blue * ctx->Visual->BlueScale * (1.0F/255.0F);
ctx->Current.ByteColor[3] = alpha * ctx->Visual->AlphaScale * (1.0F/255.0F);
}
/* update material */
ASSERT( ctx->Light.ColorMaterialEnabled );
color[0] = red * (1.0F/255.0F);
color[1] = green * (1.0F/255.0F);
color[2] = blue * (1.0F/255.0F);
color[3] = alpha * (1.0F/255.0F);
gl_set_material( ctx, ctx->Light.ColorMaterialBitmask, color );
ctx->VB->MonoColor = GL_FALSE;
}
/*
* glColor which modifies material(s).
* Caller: context->API.Color4ub pointer.
*/
void gl_ColorMat4ubv( GLcontext *ctx, const GLubyte *c )
{
gl_ColorMat4ub( ctx, c[0], c[1], c[2], c[3] );
}
/**********************************************************************/
/****** glEdgeFlag functions *****/
/**********************************************************************/
/*
* Caller: context->API.EdgeFlag pointer.
*/
void gl_EdgeFlag( GLcontext *ctx, GLboolean flag )
{
ctx->Current.EdgeFlag = flag;
}
/**********************************************************************/
/***** glVertex functions *****/
/**********************************************************************/
/*
* Used when in feedback mode.
* Caller: context->API.Vertex4f pointer.
*/
static void vertex4f_feedback( GLcontext *ctx,
GLfloat x, GLfloat y, GLfloat z, GLfloat w )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
/* vertex */
ASSIGN_4V( VB->Obj[count], x, y, z, w );
/* color */
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
/* index */
VB->Findex[count] = ctx->Current.Index;
/* normal */
COPY_3V( VB->Normal[count], ctx->Current.Normal );
/* texcoord */
COPY_4V( VB->TexCoord[count], ctx->Current.TexCoord );
/* edgeflag */
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
static void vertex3f_feedback( GLcontext *ctx, GLfloat x, GLfloat y, GLfloat z )
{
vertex4f_feedback(ctx, x, y, z, 1.0F);
}
static void vertex2f_feedback( GLcontext *ctx, GLfloat x, GLfloat y )
{
vertex4f_feedback(ctx, x, y, 0.0F, 1.0F);
}
static void vertex3fv_feedback( GLcontext *ctx, const GLfloat v[3] )
{
vertex4f_feedback(ctx, v[0], v[1], v[2], 1.0F);
}
/*
* Only one glVertex4 function since it's not too popular.
* Caller: context->API.Vertex4f pointer.
*/
static void vertex4( GLcontext *ctx,
GLfloat x, GLfloat y, GLfloat z, GLfloat w )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_4V( VB->Obj[count], x, y, z, w );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
COPY_4V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
VB->VertexSizeMask = VERTEX4_BIT;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, RGB color, normal, ST texture coords.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3f_normal_color_tex2( GLcontext *ctx,
GLfloat x, GLfloat y, GLfloat z )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, z );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
COPY_2V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, RGB color, normal, STRQ texture coords.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3f_normal_color_tex4( GLcontext *ctx,
GLfloat x, GLfloat y, GLfloat z )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, z );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
COPY_4V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, normal.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3f_normal( GLcontext *ctx, GLfloat x, GLfloat y, GLfloat z )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, z );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, ST texture coords.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3f_color_tex2( GLcontext *ctx,
GLfloat x, GLfloat y, GLfloat z )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, z );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_2V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, STRQ texture coords.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3f_color_tex4( GLcontext *ctx,
GLfloat x, GLfloat y, GLfloat z )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, z );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_4V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, RGB color.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3f_color( GLcontext *ctx, GLfloat x, GLfloat y, GLfloat z )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, z );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, color index.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3f_index( GLcontext *ctx, GLfloat x, GLfloat y, GLfloat z )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, z );
VB->Findex[count] = ctx->Current.Index;
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XY vertex, RGB color, normal, ST texture coords.
* Caller: context->API.Vertex2f pointer.
*/
static void vertex2f_normal_color_tex2( GLcontext *ctx, GLfloat x, GLfloat y )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, 0.0F );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
COPY_2V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XY vertex, RGB color, normal, STRQ texture coords.
* Caller: context->API.Vertex2f pointer.
*/
static void vertex2f_normal_color_tex4( GLcontext *ctx, GLfloat x, GLfloat y )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, 0.0F );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
COPY_4V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XY vertex, normal.
* Caller: context->API.Vertex2f pointer.
*/
static void vertex2f_normal( GLcontext *ctx, GLfloat x, GLfloat y )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, 0.0F );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XY vertex, ST texture coords.
* Caller: context->API.Vertex2f pointer.
*/
static void vertex2f_color_tex2( GLcontext *ctx, GLfloat x, GLfloat y )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, 0.0F );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_2V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XY vertex, STRQ texture coords.
* Caller: context->API.Vertex2f pointer.
*/
static void vertex2f_color_tex4( GLcontext *ctx, GLfloat x, GLfloat y )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, 0.0F );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_4V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XY vertex, RGB color.
* Caller: context->API.Vertex2f pointer.
*/
static void vertex2f_color( GLcontext *ctx, GLfloat x, GLfloat y )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, 0.0F );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XY vertex, color index.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex2f_index( GLcontext *ctx, GLfloat x, GLfloat y )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
ASSIGN_3V( VB->Obj[count], x, y, 0.0F );
VB->Findex[count] = ctx->Current.Index;
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, RGB color, normal, ST texture coords.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3fv_normal_color_tex2( GLcontext *ctx, const GLfloat v[3] )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
COPY_3V( VB->Obj[count], v );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
COPY_2V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, RGB color, normal, STRQ texture coords.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3fv_normal_color_tex4( GLcontext *ctx, const GLfloat v[3] )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
COPY_3V( VB->Obj[count], v );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
COPY_4V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, normal.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3fv_normal( GLcontext *ctx, const GLfloat v[3] )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
COPY_3V( VB->Obj[count], v );
COPY_3V( VB->Normal[count], ctx->Current.Normal );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, ST texture coords.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3fv_color_tex2( GLcontext *ctx, const GLfloat v[3] )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
COPY_3V( VB->Obj[count], v );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_2V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, STRQ texture coords.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3fv_color_tex4( GLcontext *ctx, const GLfloat v[3] )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
COPY_3V( VB->Obj[count], v );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
COPY_4V( VB->TexCoord[count], ctx->Current.TexCoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, RGB color.
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3fv_color( GLcontext *ctx, const GLfloat v[3] )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
COPY_3V( VB->Obj[count], v );
COPY_4UBV( VB->Fcolor[count], ctx->Current.ByteColor );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* XYZ vertex, Color index
* Caller: context->API.Vertex3f pointer.
*/
static void vertex3fv_index( GLcontext *ctx, const GLfloat v[3] )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count;
COPY_3V( VB->Obj[count], v );
VB->Findex[count] = ctx->Current.Index;
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/*
* Called when outside glBegin/glEnd, raises an error.
* Caller: context->API.Vertex4f pointer.
*/
void gl_vertex4f_nop( GLcontext *ctx,
GLfloat x, GLfloat y, GLfloat z, GLfloat w )
{
gl_error( ctx, GL_INVALID_OPERATION, "glVertex4" );
}
void gl_vertex3f_nop( GLcontext *ctx, GLfloat x, GLfloat y, GLfloat z )
{
gl_error( ctx, GL_INVALID_OPERATION, "glVertex3" );
}
void gl_vertex2f_nop( GLcontext *ctx, GLfloat x, GLfloat y )
{
gl_error( ctx, GL_INVALID_OPERATION, "glVertex2" );
}
void gl_vertex3fv_nop( GLcontext *ctx, const GLfloat v[3] )
{
gl_error( ctx, GL_INVALID_OPERATION, "glVertex3v" );
}
/**********************************************************************/
/****** glTexCoord functions *****/
/**********************************************************************/
/*
* Caller: context->API.TexCoord2f pointer.
*/
void gl_TexCoord2f( GLcontext *ctx, GLfloat s, GLfloat t )
{
ctx->Current.TexCoord[0] = s;
ctx->Current.TexCoord[1] = t;
}
/*
* Caller: context->API.TexCoord2f pointer.
* This version of glTexCoord2 is called if glTexCoord[34] was a predecessor.
*/
void gl_TexCoord2f4( GLcontext *ctx, GLfloat s, GLfloat t )
{
ctx->Current.TexCoord[0] = s;
ctx->Current.TexCoord[1] = t;
ctx->Current.TexCoord[2] = 0.0F;
ctx->Current.TexCoord[3] = 1.0F;
}
/*
* Caller: context->API.TexCoord4f pointer.
*/
void gl_TexCoord4f( GLcontext *ctx, GLfloat s, GLfloat t, GLfloat r, GLfloat q)
{
ctx->Current.TexCoord[0] = s;
ctx->Current.TexCoord[1] = t;
ctx->Current.TexCoord[2] = r;
ctx->Current.TexCoord[3] = q;
if (ctx->VB->TexCoordSize==2) {
/* Have to switch to 4-component texture mode now */
ctx->VB->TexCoordSize = 4;
gl_set_vertex_function( ctx );
ctx->Exec.TexCoord2f = ctx->API.TexCoord2f = gl_TexCoord2f4;
}
}
/*
* This function examines the current GL state and sets the
* ctx->Exec.Vertex[34]f pointers to point at the appropriate vertex
* processing functions.
*/
void gl_set_vertex_function( GLcontext *ctx )
{
if (ctx->RenderMode==GL_FEEDBACK) {
ctx->Exec.Vertex4f = vertex4f_feedback;
ctx->Exec.Vertex3f = vertex3f_feedback;
ctx->Exec.Vertex2f = vertex2f_feedback;
ctx->Exec.Vertex3fv = vertex3fv_feedback;
}
else {
ctx->Exec.Vertex4f = vertex4;
if (ctx->Visual->RGBAflag) {
if (ctx->NeedNormals) {
/* lighting enabled, need normal vectors */
if (ctx->Texture.Enabled) {
if (ctx->VB->TexCoordSize==2) {
ctx->Exec.Vertex2f = vertex2f_normal_color_tex2;
ctx->Exec.Vertex3f = vertex3f_normal_color_tex2;
ctx->Exec.Vertex3fv = vertex3fv_normal_color_tex2;
}
else {
ctx->Exec.Vertex2f = vertex2f_normal_color_tex4;
ctx->Exec.Vertex3f = vertex3f_normal_color_tex4;
ctx->Exec.Vertex3fv = vertex3fv_normal_color_tex4;
}
}
else {
ctx->Exec.Vertex2f = vertex2f_normal;
ctx->Exec.Vertex3f = vertex3f_normal;
ctx->Exec.Vertex3fv = vertex3fv_normal;
}
}
else {
/* not lighting, need vertex color */
if (ctx->Texture.Enabled) {
if (ctx->VB->TexCoordSize==2) {
ctx->Exec.Vertex2f = vertex2f_color_tex2;
ctx->Exec.Vertex3f = vertex3f_color_tex2;
ctx->Exec.Vertex3fv = vertex3fv_color_tex2;
}
else {
ctx->Exec.Vertex2f = vertex2f_color_tex4;
ctx->Exec.Vertex3f = vertex3f_color_tex4;
ctx->Exec.Vertex3fv = vertex3fv_color_tex4;
}
}
else {
ctx->Exec.Vertex2f = vertex2f_color;
ctx->Exec.Vertex3f = vertex3f_color;
ctx->Exec.Vertex3fv = vertex3fv_color;
}
}
}
else {
/* color index mode */
if (ctx->Light.Enabled) {
ctx->Exec.Vertex2f = vertex2f_normal;
ctx->Exec.Vertex3f = vertex3f_normal;
ctx->Exec.Vertex3fv = vertex3fv_normal;
}
else {
ctx->Exec.Vertex2f = vertex2f_index;
ctx->Exec.Vertex3f = vertex3f_index;
ctx->Exec.Vertex3fv = vertex3fv_index;
}
}
}
if (!ctx->CompileFlag) {
ctx->API.Vertex2f = ctx->Exec.Vertex2f;
ctx->API.Vertex3f = ctx->Exec.Vertex3f;
ctx->API.Vertex4f = ctx->Exec.Vertex4f;
ctx->API.Vertex3fv = ctx->Exec.Vertex3fv;
}
}
/*
* This function examines the current GL state and sets the
* ctx->Exec.Color[34]* pointers to point at the appropriate vertex
* processing functions.
*/
void gl_set_color_function( GLcontext *ctx )
{
ASSERT( !INSIDE_BEGIN_END(ctx) );
if (ctx->Light.ColorMaterialEnabled) {
ctx->Exec.Color3f = gl_ColorMat3f;
ctx->Exec.Color3fv = gl_ColorMat3fv;
ctx->Exec.Color4f = gl_ColorMat4f;
ctx->Exec.Color4fv = gl_ColorMat4fv;
ctx->Exec.Color4ub = gl_ColorMat4ub;
ctx->Exec.Color4ubv = gl_ColorMat4ubv;
}
else if (ctx->Visual->EightBitColor) {
ctx->Exec.Color3f = gl_Color3f8bit;
ctx->Exec.Color3fv = gl_Color3fv8bit;
ctx->Exec.Color4f = gl_Color4f8bit;
ctx->Exec.Color4fv = gl_Color4fv8bit;
ctx->Exec.Color4ub = gl_Color4ub8bit;
ctx->Exec.Color4ubv = gl_Color4ubv8bit;
}
else {
ctx->Exec.Color3f = gl_Color3f;
ctx->Exec.Color3fv = gl_Color3fv;
ctx->Exec.Color4f = gl_Color4f;
ctx->Exec.Color4fv = gl_Color4fv;
ctx->Exec.Color4ub = gl_Color4ub;
ctx->Exec.Color4ubv = gl_Color4ubv;
}
if (!ctx->CompileFlag) {
ctx->API.Color3f = ctx->Exec.Color3f;
ctx->API.Color3fv = ctx->Exec.Color3fv;
ctx->API.Color4f = ctx->Exec.Color4f;
ctx->API.Color4fv = ctx->Exec.Color4fv;
ctx->API.Color4ub = ctx->Exec.Color4ub;
ctx->API.Color4ubv = ctx->Exec.Color4ubv;
}
}
/**********************************************************************/
/***** Evaluator vertices *****/
/**********************************************************************/
/*
* Process a vertex produced by an evaluator.
* Caller: eval.c
* Input: vertex - the X,Y,Z,W vertex
* normal - normal vector
* color - 4 integer color components
* index - color index
* texcoord - texture coordinate
*/
void gl_eval_vertex( GLcontext *ctx,
const GLfloat vertex[4], const GLfloat normal[3],
const GLubyte color[4],
GLuint index,
const GLfloat texcoord[4] )
{
struct vertex_buffer *VB = ctx->VB;
GLuint count = VB->Count; /* copy to local var to encourage optimization */
VB->VertexSizeMask = VERTEX4_BIT;
VB->MonoNormal = GL_FALSE;
COPY_4V( VB->Obj[count], vertex );
COPY_3V( VB->Normal[count], normal );
COPY_4UBV( VB->Fcolor[count], color );
#ifdef GL_VERSION_1_1
if (ctx->Light.ColorMaterialEnabled
&& (ctx->Eval.Map1Color4 || ctx->Eval.Map2Color4)) {
GLfloat fcolor[4];
fcolor[0] = color[0] * ctx->Visual->InvRedScale;
fcolor[1] = color[1] * ctx->Visual->InvGreenScale;
fcolor[2] = color[2] * ctx->Visual->InvBlueScale;
fcolor[3] = color[3] * ctx->Visual->InvAlphaScale;
gl_set_material( ctx, ctx->Light.ColorMaterialBitmask, fcolor );
}
#endif
VB->Findex[count] = index;
COPY_4V( VB->TexCoord[count], texcoord );
VB->Edgeflag[count] = ctx->Current.EdgeFlag;
count++;
VB->Count = count;
if (count==VB_MAX) {
gl_transform_vb_part1( ctx, GL_FALSE );
}
}
/**********************************************************************/
/***** glBegin / glEnd *****/
/**********************************************************************/
#ifdef PROFILE
static GLdouble begin_time;
#endif
void gl_Begin( GLcontext *ctx, GLenum p )
{
struct vertex_buffer *VB = ctx->VB;
struct pixel_buffer *PB = ctx->PB;
#ifdef PROFILE
begin_time = gl_time();
#endif
if (INSIDE_BEGIN_END(ctx)) {
gl_error( ctx, GL_INVALID_OPERATION, "glBegin" );
return;
}
if (ctx->NewModelViewMatrix) {
gl_analyze_modelview_matrix(ctx);
}
if (ctx->NewProjectionMatrix) {
gl_analyze_projection_matrix(ctx);
}
if (ctx->NewState) {
gl_update_state(ctx);
}
else if (ctx->Exec.Vertex3f==gl_vertex3f_nop) {
gl_set_vertex_function(ctx);
}
if (ctx->Driver.Begin) {
(*ctx->Driver.Begin)( ctx, p );
}
ctx->Primitive = p;
VB->Start = VB->Count = 0;
VB->MonoColor = ctx->MonoPixels;
VB->MonoNormal = GL_TRUE;
if (VB->MonoColor) {
/* All pixels generated are likely to be the same color so have
* the device driver set the "monocolor" now.
*/
if (ctx->Visual->RGBAflag) {
GLubyte r = ctx->Current.ByteColor[0];
GLubyte g = ctx->Current.ByteColor[1];
GLubyte b = ctx->Current.ByteColor[2];
GLubyte a = ctx->Current.ByteColor[3];
(*ctx->Driver.Color)( ctx, r, g, b, a );
}
else {
(*ctx->Driver.Index)( ctx, ctx->Current.Index );
}
}
/* By default use front color/index. Two-sided lighting may override. */
VB->Color = VB->Fcolor;
VB->Index = VB->Findex;
switch (ctx->Primitive) {
case GL_POINTS:
ctx->LightTwoSide = GL_FALSE;
PB_INIT( PB, GL_POINT );
break;
case GL_LINES:
case GL_LINE_STRIP:
case GL_LINE_LOOP:
ctx->LightTwoSide = GL_FALSE;
ctx->StippleCounter = 0;
PB_INIT( PB, GL_LINE );
break;
case GL_TRIANGLES:
case GL_TRIANGLE_STRIP:
case GL_TRIANGLE_FAN:
case GL_QUADS:
case GL_QUAD_STRIP:
case GL_POLYGON:
ctx->LightTwoSide = ctx->Light.Enabled && ctx->Light.Model.TwoSide;
PB_INIT( PB, GL_POLYGON );
break;
default:
gl_error( ctx, GL_INVALID_ENUM, "glBegin" );
ctx->Primitive = GL_BITMAP;
}
}
void gl_End( GLcontext *ctx )
{
struct pixel_buffer *PB = ctx->PB;
struct vertex_buffer *VB = ctx->VB;
if (ctx->Primitive==GL_BITMAP) {
/* glEnd without glBegin */
gl_error( ctx, GL_INVALID_OPERATION, "glEnd" );
return;
}
if (VB->Count > VB->Start) {
gl_transform_vb_part1( ctx, GL_TRUE );
}
if (PB->count>0) {
gl_flush_pb(ctx);
}
if (ctx->Driver.End) {
(*ctx->Driver.End)(ctx);
}
PB->primitive = ctx->Primitive = GL_BITMAP; /* Default mode */
#ifdef PROFILE
ctx->BeginEndTime += gl_time() - begin_time;
ctx->BeginEndCount++;
#endif
}