mirror of
https://github.com/reactos/reactos.git
synced 2025-01-07 14:51:00 +00:00
5f2bebf7a5
With this commit, we now use a forked version of MESA which only supports OpenGL 1.1, like the windows implementation does. It exposes : - The same pixel formats - The same set of extensions - Nothing more All of this without taking 10% of your build time. If you need a more modern option, look at the MESA package from Rapps, which is (and must be) maintained outside of this code tree. CORE-7499
586 lines
16 KiB
C
586 lines
16 KiB
C
/* $Id: points.c,v 1.15 1998/02/03 23:46:00 brianp Exp $ */
|
|
|
|
/*
|
|
* Mesa 3-D graphics library
|
|
* Version: 2.5
|
|
* 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: points.c,v $
|
|
* Revision 1.15 1998/02/03 23:46:00 brianp
|
|
* fixed a few problems with condition expressions for Amiga StormC compiler
|
|
*
|
|
* Revision 1.14 1997/10/29 01:29:09 brianp
|
|
* added GL_EXT_point_parameters extension from Daniel Barrero
|
|
*
|
|
* Revision 1.13 1997/07/24 01:23:44 brianp
|
|
* changed precompiled header symbol from PCH to PC_HEADER
|
|
*
|
|
* Revision 1.12 1997/06/20 02:50:39 brianp
|
|
* changed color components from GLfixed to GLubyte
|
|
*
|
|
* Revision 1.11 1997/05/28 03:26:02 brianp
|
|
* added precompiled header (PCH) support
|
|
*
|
|
* Revision 1.10 1997/05/03 00:51:02 brianp
|
|
* removed calls to gl_texturing_enabled()
|
|
*
|
|
* Revision 1.9 1997/04/14 02:00:39 brianp
|
|
* #include "texstate.h" instead of "texture.h"
|
|
*
|
|
* Revision 1.8 1997/04/12 12:24:43 brianp
|
|
* replaced ctx->PointsFunc with ctx->Driver.PointsFunc
|
|
*
|
|
* Revision 1.7 1997/04/02 03:11:38 brianp
|
|
* replaced VB->Unclipped with VB->ClipMask
|
|
*
|
|
* Revision 1.6 1997/03/08 02:04:27 brianp
|
|
* better implementation of feedback function
|
|
*
|
|
* Revision 1.5 1997/02/09 18:43:52 brianp
|
|
* added GL_EXT_texture3D support
|
|
*
|
|
* Revision 1.4 1997/01/09 19:48:00 brianp
|
|
* now call gl_texturing_enabled()
|
|
*
|
|
* Revision 1.3 1996/11/08 02:21:21 brianp
|
|
* added null drawing function for GL_NO_RASTER
|
|
*
|
|
* Revision 1.2 1996/09/15 14:18:37 brianp
|
|
* now use GLframebuffer and GLvisual
|
|
*
|
|
* Revision 1.1 1996/09/13 01:38:16 brianp
|
|
* Initial revision
|
|
*
|
|
*/
|
|
|
|
|
|
#ifdef PC_HEADER
|
|
#include "all.h"
|
|
#else
|
|
#include "context.h"
|
|
#include "feedback.h"
|
|
#include "dlist.h"
|
|
#include "macros.h"
|
|
#include "pb.h"
|
|
#include "span.h"
|
|
#include "texstate.h"
|
|
#include "types.h"
|
|
#include "vb.h"
|
|
#include "mmath.h"
|
|
#endif
|
|
|
|
|
|
|
|
void gl_PointSize( GLcontext *ctx, GLfloat size )
|
|
{
|
|
if (size<=0.0) {
|
|
gl_error( ctx, GL_INVALID_VALUE, "glPointSize" );
|
|
return;
|
|
}
|
|
if (INSIDE_BEGIN_END(ctx)) {
|
|
gl_error( ctx, GL_INVALID_OPERATION, "glPointSize" );
|
|
return;
|
|
}
|
|
ctx->Point.Size = size;
|
|
ctx->NewState |= NEW_RASTER_OPS;
|
|
}
|
|
|
|
|
|
/**********************************************************************/
|
|
/***** Rasterization *****/
|
|
/**********************************************************************/
|
|
|
|
|
|
/*
|
|
* There are 3 pairs (RGBA, CI) of point rendering functions:
|
|
* 1. simple: size=1 and no special rasterization functions (fastest)
|
|
* 2. size1: size=1 and any rasterization functions
|
|
* 3. general: any size and rasterization functions (slowest)
|
|
*
|
|
* All point rendering functions take the same two arguments: first and
|
|
* last which specify that the points specified by VB[first] through
|
|
* VB[last] are to be rendered.
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
* Put points in feedback buffer.
|
|
*/
|
|
static void feedback_points( GLcontext *ctx, GLuint first, GLuint last )
|
|
{
|
|
struct vertex_buffer *VB = ctx->VB;
|
|
GLuint i;
|
|
GLfloat invRedScale = ctx->Visual->InvRedScale;
|
|
GLfloat invGreenScale = ctx->Visual->InvGreenScale;
|
|
GLfloat invBlueScale = ctx->Visual->InvBlueScale;
|
|
GLfloat invAlphaScale = ctx->Visual->InvAlphaScale;
|
|
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
GLfloat x, y, z, w, invq;
|
|
GLfloat color[4], texcoord[4];
|
|
|
|
x = VB->Win[i][0];
|
|
y = VB->Win[i][1];
|
|
z = VB->Win[i][2] / DEPTH_SCALE;
|
|
w = VB->Clip[i][3];
|
|
|
|
/* convert color from integer back to a float in [0,1] */
|
|
if (ctx->Light.ShadeModel==GL_SMOOTH) {
|
|
/* smooth shading - colors are in fixed point */
|
|
color[0] = FixedToFloat(VB->Color[i][0]) * invRedScale;
|
|
color[1] = FixedToFloat(VB->Color[i][1]) * invGreenScale;
|
|
color[2] = FixedToFloat(VB->Color[i][2]) * invBlueScale;
|
|
color[3] = FixedToFloat(VB->Color[i][3]) * invAlphaScale;
|
|
}
|
|
else {
|
|
/* flat shading - colors are integers */
|
|
color[0] = VB->Color[i][0] * invRedScale;
|
|
color[1] = VB->Color[i][1] * invGreenScale;
|
|
color[2] = VB->Color[i][2] * invBlueScale;
|
|
color[3] = VB->Color[i][3] * invAlphaScale;
|
|
}
|
|
invq = 1.0F / VB->TexCoord[i][3];
|
|
texcoord[0] = VB->TexCoord[i][0] * invq;
|
|
texcoord[1] = VB->TexCoord[i][1] * invq;
|
|
texcoord[2] = VB->TexCoord[i][2] * invq;
|
|
texcoord[3] = VB->TexCoord[i][3];
|
|
|
|
FEEDBACK_TOKEN( ctx, (GLfloat) (GLint) GL_POINT_TOKEN );
|
|
gl_feedback_vertex( ctx, x, y, z, w, color,
|
|
(GLfloat) VB->Index[i], texcoord );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Put points in selection buffer.
|
|
*/
|
|
static void select_points( GLcontext *ctx, GLuint first, GLuint last )
|
|
{
|
|
struct vertex_buffer *VB = ctx->VB;
|
|
GLuint i;
|
|
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
gl_update_hitflag( ctx, VB->Win[i][2] / DEPTH_SCALE );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* CI points with size == 1.0
|
|
*/
|
|
void size1_ci_points( GLcontext *ctx, GLuint first, GLuint last )
|
|
{
|
|
struct vertex_buffer *VB = ctx->VB;
|
|
struct pixel_buffer *PB = ctx->PB;
|
|
GLfloat *win;
|
|
GLint *pbx = PB->x, *pby = PB->y;
|
|
GLdepth *pbz = PB->z;
|
|
GLuint *pbi = PB->i;
|
|
GLuint pbcount = PB->count;
|
|
GLuint i;
|
|
|
|
win = &VB->Win[first][0];
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
pbx[pbcount] = (GLint) win[0];
|
|
pby[pbcount] = (GLint) win[1];
|
|
pbz[pbcount] = (GLint) (win[2] + ctx->PointZoffset);
|
|
pbi[pbcount] = VB->Index[i];
|
|
pbcount++;
|
|
}
|
|
win += 3;
|
|
}
|
|
PB->count = pbcount;
|
|
PB_CHECK_FLUSH(ctx, PB)
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* RGBA points with size == 1.0
|
|
*/
|
|
static void size1_rgba_points( GLcontext *ctx, GLuint first, GLuint last )
|
|
{
|
|
struct vertex_buffer *VB = ctx->VB;
|
|
struct pixel_buffer *PB = ctx->PB;
|
|
GLuint i;
|
|
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
GLint x, y, z;
|
|
GLint red, green, blue, alpha;
|
|
|
|
x = (GLint) VB->Win[i][0];
|
|
y = (GLint) VB->Win[i][1];
|
|
z = (GLint) (VB->Win[i][2] + ctx->PointZoffset);
|
|
|
|
red = VB->Color[i][0];
|
|
green = VB->Color[i][1];
|
|
blue = VB->Color[i][2];
|
|
alpha = VB->Color[i][3];
|
|
|
|
PB_WRITE_RGBA_PIXEL( PB, x, y, z, red, green, blue, alpha );
|
|
}
|
|
}
|
|
PB_CHECK_FLUSH(ctx,PB)
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* General CI points.
|
|
*/
|
|
static void general_ci_points( GLcontext *ctx, GLuint first, GLuint last )
|
|
{
|
|
struct vertex_buffer *VB = ctx->VB;
|
|
struct pixel_buffer *PB = ctx->PB;
|
|
GLuint i;
|
|
GLint isize;
|
|
|
|
isize = (GLint) (CLAMP(ctx->Point.Size,MIN_POINT_SIZE,MAX_POINT_SIZE) + 0.5F);
|
|
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
GLint x, y, z;
|
|
GLint x0, x1, y0, y1;
|
|
GLint ix, iy;
|
|
|
|
x = (GLint) VB->Win[i][0];
|
|
y = (GLint) VB->Win[i][1];
|
|
z = (GLint) (VB->Win[i][2] + ctx->PointZoffset);
|
|
|
|
if (isize&1) {
|
|
/* odd size */
|
|
x0 = x - isize/2;
|
|
x1 = x + isize/2;
|
|
y0 = y - isize/2;
|
|
y1 = y + isize/2;
|
|
}
|
|
else {
|
|
/* even size */
|
|
x0 = (GLint) (x + 0.5F) - isize/2;
|
|
x1 = x0 + isize-1;
|
|
y0 = (GLint) (y + 0.5F) - isize/2;
|
|
y1 = y0 + isize-1;
|
|
}
|
|
|
|
PB_SET_INDEX( ctx, PB, VB->Index[i] );
|
|
|
|
for (iy=y0;iy<=y1;iy++) {
|
|
for (ix=x0;ix<=x1;ix++) {
|
|
PB_WRITE_PIXEL( PB, ix, iy, z );
|
|
}
|
|
}
|
|
PB_CHECK_FLUSH(ctx,PB)
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* General RGBA points.
|
|
*/
|
|
static void general_rgba_points( GLcontext *ctx, GLuint first, GLuint last )
|
|
{
|
|
struct vertex_buffer *VB = ctx->VB;
|
|
struct pixel_buffer *PB = ctx->PB;
|
|
GLuint i;
|
|
GLint isize;
|
|
|
|
isize = (GLint) (CLAMP(ctx->Point.Size,MIN_POINT_SIZE,MAX_POINT_SIZE) + 0.5F);
|
|
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
GLint x, y, z;
|
|
GLint x0, x1, y0, y1;
|
|
GLint ix, iy;
|
|
|
|
x = (GLint) VB->Win[i][0];
|
|
y = (GLint) VB->Win[i][1];
|
|
z = (GLint) (VB->Win[i][2] + ctx->PointZoffset);
|
|
|
|
if (isize&1) {
|
|
/* odd size */
|
|
x0 = x - isize/2;
|
|
x1 = x + isize/2;
|
|
y0 = y - isize/2;
|
|
y1 = y + isize/2;
|
|
}
|
|
else {
|
|
/* even size */
|
|
x0 = (GLint) (x + 0.5F) - isize/2;
|
|
x1 = x0 + isize-1;
|
|
y0 = (GLint) (y + 0.5F) - isize/2;
|
|
y1 = y0 + isize-1;
|
|
}
|
|
|
|
PB_SET_COLOR( ctx, PB,
|
|
VB->Color[i][0],
|
|
VB->Color[i][1],
|
|
VB->Color[i][2],
|
|
VB->Color[i][3] );
|
|
|
|
for (iy=y0;iy<=y1;iy++) {
|
|
for (ix=x0;ix<=x1;ix++) {
|
|
PB_WRITE_PIXEL( PB, ix, iy, z );
|
|
}
|
|
}
|
|
PB_CHECK_FLUSH(ctx,PB)
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
* Textured RGBA points.
|
|
*/
|
|
static void textured_rgba_points( GLcontext *ctx, GLuint first, GLuint last )
|
|
{
|
|
struct vertex_buffer *VB = ctx->VB;
|
|
struct pixel_buffer *PB = ctx->PB;
|
|
GLuint i;
|
|
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
GLint x, y, z;
|
|
GLint x0, x1, y0, y1;
|
|
GLint ix, iy;
|
|
GLint isize;
|
|
GLint red, green, blue, alpha;
|
|
GLfloat s, t, u;
|
|
|
|
x = (GLint) VB->Win[i][0];
|
|
y = (GLint) VB->Win[i][1];
|
|
z = (GLint) (VB->Win[i][2] + ctx->PointZoffset);
|
|
|
|
isize = (GLint)
|
|
(CLAMP(ctx->Point.Size,MIN_POINT_SIZE,MAX_POINT_SIZE) + 0.5F);
|
|
if (isize<1) {
|
|
isize = 1;
|
|
}
|
|
|
|
if (isize&1) {
|
|
/* odd size */
|
|
x0 = x - isize/2;
|
|
x1 = x + isize/2;
|
|
y0 = y - isize/2;
|
|
y1 = y + isize/2;
|
|
}
|
|
else {
|
|
/* even size */
|
|
x0 = (GLint) (x + 0.5F) - isize/2;
|
|
x1 = x0 + isize-1;
|
|
y0 = (GLint) (y + 0.5F) - isize/2;
|
|
y1 = y0 + isize-1;
|
|
}
|
|
|
|
red = VB->Color[i][0];
|
|
green = VB->Color[i][1];
|
|
blue = VB->Color[i][2];
|
|
alpha = VB->Color[i][3];
|
|
s = VB->TexCoord[i][0] / VB->TexCoord[i][3];
|
|
t = VB->TexCoord[i][1] / VB->TexCoord[i][3];
|
|
u = VB->TexCoord[i][2] / VB->TexCoord[i][3];
|
|
|
|
/* don't think this is needed
|
|
PB_SET_COLOR( red, green, blue, alpha );
|
|
*/
|
|
|
|
for (iy=y0;iy<=y1;iy++) {
|
|
for (ix=x0;ix<=x1;ix++) {
|
|
PB_WRITE_TEX_PIXEL( PB, ix, iy, z, red, green, blue, alpha, s, t, u );
|
|
}
|
|
}
|
|
PB_CHECK_FLUSH(ctx,PB)
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Antialiased points with or without texture mapping.
|
|
*/
|
|
static void antialiased_rgba_points( GLcontext *ctx,
|
|
GLuint first, GLuint last )
|
|
{
|
|
struct vertex_buffer *VB = ctx->VB;
|
|
struct pixel_buffer *PB = ctx->PB;
|
|
GLuint i;
|
|
GLfloat radius, rmin, rmax, rmin2, rmax2, cscale;
|
|
|
|
radius = CLAMP( ctx->Point.Size, MIN_POINT_SIZE, MAX_POINT_SIZE ) * 0.5F;
|
|
rmin = radius - 0.7071F; /* 0.7071 = sqrt(2)/2 */
|
|
rmax = radius + 0.7071F;
|
|
rmin2 = rmin*rmin;
|
|
rmax2 = rmax*rmax;
|
|
cscale = 256.0F / (rmax2-rmin2);
|
|
|
|
if (ctx->Texture.Enabled) {
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
GLint xmin, ymin, xmax, ymax;
|
|
GLint x, y, z;
|
|
GLint red, green, blue, alpha;
|
|
GLfloat s, t, u;
|
|
|
|
xmin = (GLint) (VB->Win[i][0] - radius);
|
|
xmax = (GLint) (VB->Win[i][0] + radius);
|
|
ymin = (GLint) (VB->Win[i][1] - radius);
|
|
ymax = (GLint) (VB->Win[i][1] + radius);
|
|
z = (GLint) (VB->Win[i][2] + ctx->PointZoffset);
|
|
|
|
red = VB->Color[i][0];
|
|
green = VB->Color[i][1];
|
|
blue = VB->Color[i][2];
|
|
s = VB->TexCoord[i][0] / VB->TexCoord[i][3];
|
|
t = VB->TexCoord[i][1] / VB->TexCoord[i][3];
|
|
u = VB->TexCoord[i][2] / VB->TexCoord[i][3];
|
|
|
|
for (y=ymin;y<=ymax;y++) {
|
|
for (x=xmin;x<=xmax;x++) {
|
|
GLfloat dx = x/*+0.5F*/ - VB->Win[i][0];
|
|
GLfloat dy = y/*+0.5F*/ - VB->Win[i][1];
|
|
GLfloat dist2 = dx*dx + dy*dy;
|
|
if (dist2<rmax2) {
|
|
alpha = VB->Color[i][3];
|
|
if (dist2>=rmin2) {
|
|
GLint coverage = (GLint) (256.0F-(dist2-rmin2)*cscale);
|
|
/* coverage is in [0,256] */
|
|
alpha = (alpha * coverage) >> 8;
|
|
}
|
|
PB_WRITE_TEX_PIXEL( PB, x,y,z, red, green, blue, alpha, s, t, u );
|
|
}
|
|
}
|
|
}
|
|
PB_CHECK_FLUSH(ctx,PB)
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
/* Not texture mapped */
|
|
for (i=first;i<=last;i++) {
|
|
if (VB->ClipMask[i]==0) {
|
|
GLint xmin, ymin, xmax, ymax;
|
|
GLint x, y, z;
|
|
GLint red, green, blue, alpha;
|
|
|
|
xmin = (GLint) (VB->Win[i][0] - radius);
|
|
xmax = (GLint) (VB->Win[i][0] + radius);
|
|
ymin = (GLint) (VB->Win[i][1] - radius);
|
|
ymax = (GLint) (VB->Win[i][1] + radius);
|
|
z = (GLint) (VB->Win[i][2] + ctx->PointZoffset);
|
|
|
|
red = VB->Color[i][0];
|
|
green = VB->Color[i][1];
|
|
blue = VB->Color[i][2];
|
|
|
|
for (y=ymin;y<=ymax;y++) {
|
|
for (x=xmin;x<=xmax;x++) {
|
|
GLfloat dx = x/*+0.5F*/ - VB->Win[i][0];
|
|
GLfloat dy = y/*+0.5F*/ - VB->Win[i][1];
|
|
GLfloat dist2 = dx*dx + dy*dy;
|
|
if (dist2<rmax2) {
|
|
alpha = VB->Color[i][3];
|
|
if (dist2>=rmin2) {
|
|
GLint coverage = (GLint) (256.0F-(dist2-rmin2)*cscale);
|
|
/* coverage is in [0,256] */
|
|
alpha = (alpha * coverage) >> 8;
|
|
}
|
|
PB_WRITE_RGBA_PIXEL( PB, x, y, z, red, green, blue, alpha );
|
|
}
|
|
}
|
|
}
|
|
PB_CHECK_FLUSH(ctx,PB)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Null rasterizer for measuring transformation speed.
|
|
*/
|
|
static void null_points( GLcontext *ctx, GLuint first, GLuint last )
|
|
{
|
|
}
|
|
|
|
/* Definition of the functions for GL_EXT_point_parameters */
|
|
|
|
/*
|
|
* Examine the current context to determine which point drawing function
|
|
* should be used.
|
|
*/
|
|
void gl_set_point_function( GLcontext *ctx )
|
|
{
|
|
GLboolean rgbmode = ctx->Visual->RGBAflag;
|
|
|
|
if (ctx->RenderMode==GL_RENDER) {
|
|
if (ctx->NoRaster) {
|
|
ctx->Driver.PointsFunc = null_points;
|
|
return;
|
|
}
|
|
if (ctx->Driver.PointsFunc) {
|
|
/* Device driver will draw points. */
|
|
ctx->Driver.PointsFunc = ctx->Driver.PointsFunc;
|
|
}
|
|
else {
|
|
if (ctx->Point.SmoothFlag && rgbmode) {
|
|
ctx->Driver.PointsFunc = antialiased_rgba_points;
|
|
}
|
|
else if (ctx->Texture.Enabled) {
|
|
ctx->Driver.PointsFunc = textured_rgba_points;
|
|
}
|
|
else if (ctx->Point.Size==1.0) {
|
|
/* size=1, any raster ops */
|
|
if (rgbmode)
|
|
ctx->Driver.PointsFunc = size1_rgba_points;
|
|
else
|
|
ctx->Driver.PointsFunc = size1_ci_points;
|
|
}
|
|
else {
|
|
/* every other kind of point rendering */
|
|
if (rgbmode)
|
|
ctx->Driver.PointsFunc = general_rgba_points;
|
|
else
|
|
ctx->Driver.PointsFunc = general_ci_points;
|
|
}
|
|
}
|
|
}
|
|
else if (ctx->RenderMode==GL_FEEDBACK) {
|
|
ctx->Driver.PointsFunc = feedback_points;
|
|
}
|
|
else {
|
|
/* GL_SELECT mode */
|
|
ctx->Driver.PointsFunc = select_points;
|
|
}
|
|
|
|
}
|
|
|