points.c revision 848b8605
1848b8605Smrg/** 2848b8605Smrg * \file points.c 3848b8605Smrg * Point operations. 4848b8605Smrg */ 5848b8605Smrg 6848b8605Smrg/* 7848b8605Smrg * Mesa 3-D graphics library 8848b8605Smrg * 9848b8605Smrg * Copyright (C) 1999-2007 Brian Paul All Rights Reserved. 10848b8605Smrg * 11848b8605Smrg * Permission is hereby granted, free of charge, to any person obtaining a 12848b8605Smrg * copy of this software and associated documentation files (the "Software"), 13848b8605Smrg * to deal in the Software without restriction, including without limitation 14848b8605Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense, 15848b8605Smrg * and/or sell copies of the Software, and to permit persons to whom the 16848b8605Smrg * Software is furnished to do so, subject to the following conditions: 17848b8605Smrg * 18848b8605Smrg * The above copyright notice and this permission notice shall be included 19848b8605Smrg * in all copies or substantial portions of the Software. 20848b8605Smrg * 21848b8605Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS 22848b8605Smrg * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 23848b8605Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 24848b8605Smrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR 25848b8605Smrg * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 26848b8605Smrg * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 27848b8605Smrg * OTHER DEALINGS IN THE SOFTWARE. 28848b8605Smrg */ 29848b8605Smrg 30848b8605Smrg 31848b8605Smrg#include "glheader.h" 32848b8605Smrg#include "context.h" 33848b8605Smrg#include "macros.h" 34848b8605Smrg#include "points.h" 35848b8605Smrg#include "mtypes.h" 36848b8605Smrg 37848b8605Smrg 38848b8605Smrg/** 39848b8605Smrg * Set current point size. 40848b8605Smrg * \param size point diameter in pixels 41848b8605Smrg * \sa glPointSize(). 42848b8605Smrg */ 43848b8605Smrgvoid GLAPIENTRY 44848b8605Smrg_mesa_PointSize( GLfloat size ) 45848b8605Smrg{ 46848b8605Smrg GET_CURRENT_CONTEXT(ctx); 47848b8605Smrg 48848b8605Smrg if (size <= 0.0) { 49848b8605Smrg _mesa_error( ctx, GL_INVALID_VALUE, "glPointSize" ); 50848b8605Smrg return; 51848b8605Smrg } 52848b8605Smrg 53848b8605Smrg if (ctx->Point.Size == size) 54848b8605Smrg return; 55848b8605Smrg 56848b8605Smrg FLUSH_VERTICES(ctx, _NEW_POINT); 57848b8605Smrg ctx->Point.Size = size; 58848b8605Smrg 59848b8605Smrg if (ctx->Driver.PointSize) 60848b8605Smrg ctx->Driver.PointSize(ctx, size); 61848b8605Smrg} 62848b8605Smrg 63848b8605Smrg 64848b8605Smrgvoid GLAPIENTRY 65848b8605Smrg_mesa_PointParameteri( GLenum pname, GLint param ) 66848b8605Smrg{ 67848b8605Smrg GLfloat p[3]; 68848b8605Smrg p[0] = (GLfloat) param; 69848b8605Smrg p[1] = p[2] = 0.0F; 70848b8605Smrg _mesa_PointParameterfv(pname, p); 71848b8605Smrg} 72848b8605Smrg 73848b8605Smrg 74848b8605Smrgvoid GLAPIENTRY 75848b8605Smrg_mesa_PointParameteriv( GLenum pname, const GLint *params ) 76848b8605Smrg{ 77848b8605Smrg GLfloat p[3]; 78848b8605Smrg p[0] = (GLfloat) params[0]; 79848b8605Smrg if (pname == GL_DISTANCE_ATTENUATION_EXT) { 80848b8605Smrg p[1] = (GLfloat) params[1]; 81848b8605Smrg p[2] = (GLfloat) params[2]; 82848b8605Smrg } 83848b8605Smrg _mesa_PointParameterfv(pname, p); 84848b8605Smrg} 85848b8605Smrg 86848b8605Smrg 87848b8605Smrgvoid GLAPIENTRY 88848b8605Smrg_mesa_PointParameterf( GLenum pname, GLfloat param) 89848b8605Smrg{ 90848b8605Smrg GLfloat p[3]; 91848b8605Smrg p[0] = param; 92848b8605Smrg p[1] = p[2] = 0.0F; 93848b8605Smrg _mesa_PointParameterfv(pname, p); 94848b8605Smrg} 95848b8605Smrg 96848b8605Smrg 97848b8605Smrgvoid GLAPIENTRY 98848b8605Smrg_mesa_PointParameterfv( GLenum pname, const GLfloat *params) 99848b8605Smrg{ 100848b8605Smrg GET_CURRENT_CONTEXT(ctx); 101848b8605Smrg 102848b8605Smrg /* Drivers that support point sprites must also support point parameters. 103848b8605Smrg * If point parameters aren't supported, then this function shouldn't even 104848b8605Smrg * exist. 105848b8605Smrg */ 106848b8605Smrg ASSERT(!(ctx->Extensions.ARB_point_sprite 107848b8605Smrg || ctx->Extensions.NV_point_sprite) 108848b8605Smrg || ctx->Extensions.EXT_point_parameters); 109848b8605Smrg 110848b8605Smrg if (!ctx->Extensions.EXT_point_parameters) { 111848b8605Smrg _mesa_error(ctx, GL_INVALID_OPERATION, 112848b8605Smrg "unsupported function called (unsupported extension)"); 113848b8605Smrg return; 114848b8605Smrg } 115848b8605Smrg 116848b8605Smrg switch (pname) { 117848b8605Smrg case GL_DISTANCE_ATTENUATION_EXT: 118848b8605Smrg if (TEST_EQ_3V(ctx->Point.Params, params)) 119848b8605Smrg return; 120848b8605Smrg FLUSH_VERTICES(ctx, _NEW_POINT); 121848b8605Smrg COPY_3V(ctx->Point.Params, params); 122848b8605Smrg ctx->Point._Attenuated = (ctx->Point.Params[0] != 1.0 || 123848b8605Smrg ctx->Point.Params[1] != 0.0 || 124848b8605Smrg ctx->Point.Params[2] != 0.0); 125848b8605Smrg break; 126848b8605Smrg case GL_POINT_SIZE_MIN_EXT: 127848b8605Smrg if (params[0] < 0.0F) { 128848b8605Smrg _mesa_error( ctx, GL_INVALID_VALUE, 129848b8605Smrg "glPointParameterf[v]{EXT,ARB}(param)" ); 130848b8605Smrg return; 131848b8605Smrg } 132848b8605Smrg if (ctx->Point.MinSize == params[0]) 133848b8605Smrg return; 134848b8605Smrg FLUSH_VERTICES(ctx, _NEW_POINT); 135848b8605Smrg ctx->Point.MinSize = params[0]; 136848b8605Smrg break; 137848b8605Smrg case GL_POINT_SIZE_MAX_EXT: 138848b8605Smrg if (params[0] < 0.0F) { 139848b8605Smrg _mesa_error( ctx, GL_INVALID_VALUE, 140848b8605Smrg "glPointParameterf[v]{EXT,ARB}(param)" ); 141848b8605Smrg return; 142848b8605Smrg } 143848b8605Smrg if (ctx->Point.MaxSize == params[0]) 144848b8605Smrg return; 145848b8605Smrg FLUSH_VERTICES(ctx, _NEW_POINT); 146848b8605Smrg ctx->Point.MaxSize = params[0]; 147848b8605Smrg break; 148848b8605Smrg case GL_POINT_FADE_THRESHOLD_SIZE_EXT: 149848b8605Smrg if (params[0] < 0.0F) { 150848b8605Smrg _mesa_error( ctx, GL_INVALID_VALUE, 151848b8605Smrg "glPointParameterf[v]{EXT,ARB}(param)" ); 152848b8605Smrg return; 153848b8605Smrg } 154848b8605Smrg if (ctx->Point.Threshold == params[0]) 155848b8605Smrg return; 156848b8605Smrg FLUSH_VERTICES(ctx, _NEW_POINT); 157848b8605Smrg ctx->Point.Threshold = params[0]; 158848b8605Smrg break; 159848b8605Smrg case GL_POINT_SPRITE_R_MODE_NV: 160848b8605Smrg /* This is one area where ARB_point_sprite and NV_point_sprite 161848b8605Smrg * differ. In ARB_point_sprite the POINT_SPRITE_R_MODE is 162848b8605Smrg * always ZERO. NV_point_sprite adds the S and R modes. 163848b8605Smrg */ 164848b8605Smrg if (_mesa_is_desktop_gl(ctx) && ctx->Extensions.NV_point_sprite) { 165848b8605Smrg GLenum value = (GLenum) params[0]; 166848b8605Smrg if (value != GL_ZERO && value != GL_S && value != GL_R) { 167848b8605Smrg _mesa_error(ctx, GL_INVALID_VALUE, 168848b8605Smrg "glPointParameterf[v]{EXT,ARB}(param)"); 169848b8605Smrg return; 170848b8605Smrg } 171848b8605Smrg if (ctx->Point.SpriteRMode == value) 172848b8605Smrg return; 173848b8605Smrg FLUSH_VERTICES(ctx, _NEW_POINT); 174848b8605Smrg ctx->Point.SpriteRMode = value; 175848b8605Smrg } 176848b8605Smrg else { 177848b8605Smrg _mesa_error(ctx, GL_INVALID_ENUM, 178848b8605Smrg "glPointParameterf[v]{EXT,ARB}(pname)"); 179848b8605Smrg return; 180848b8605Smrg } 181848b8605Smrg break; 182848b8605Smrg case GL_POINT_SPRITE_COORD_ORIGIN: 183848b8605Smrg /* GL_POINT_SPRITE_COORD_ORIGIN was added to point sprites when the 184848b8605Smrg * extension was merged into OpenGL 2.0. 185848b8605Smrg */ 186848b8605Smrg if ((ctx->API == API_OPENGL_COMPAT && ctx->Version >= 20) 187848b8605Smrg || ctx->API == API_OPENGL_CORE) { 188848b8605Smrg GLenum value = (GLenum) params[0]; 189848b8605Smrg if (value != GL_LOWER_LEFT && value != GL_UPPER_LEFT) { 190848b8605Smrg _mesa_error(ctx, GL_INVALID_VALUE, 191848b8605Smrg "glPointParameterf[v]{EXT,ARB}(param)"); 192848b8605Smrg return; 193848b8605Smrg } 194848b8605Smrg if (ctx->Point.SpriteOrigin == value) 195848b8605Smrg return; 196848b8605Smrg FLUSH_VERTICES(ctx, _NEW_POINT); 197848b8605Smrg ctx->Point.SpriteOrigin = value; 198848b8605Smrg } 199848b8605Smrg else { 200848b8605Smrg _mesa_error(ctx, GL_INVALID_ENUM, 201848b8605Smrg "glPointParameterf[v]{EXT,ARB}(pname)"); 202848b8605Smrg return; 203848b8605Smrg } 204848b8605Smrg break; 205848b8605Smrg default: 206848b8605Smrg _mesa_error( ctx, GL_INVALID_ENUM, 207848b8605Smrg "glPointParameterf[v]{EXT,ARB}(pname)" ); 208848b8605Smrg return; 209848b8605Smrg } 210848b8605Smrg 211848b8605Smrg if (ctx->Driver.PointParameterfv) 212848b8605Smrg (*ctx->Driver.PointParameterfv)(ctx, pname, params); 213848b8605Smrg} 214848b8605Smrg 215848b8605Smrg 216848b8605Smrg 217848b8605Smrg/** 218848b8605Smrg * Initialize the context point state. 219848b8605Smrg * 220848b8605Smrg * \param ctx GL context. 221848b8605Smrg * 222848b8605Smrg * Initializes __struct gl_contextRec::Point and point related constants in 223848b8605Smrg * __struct gl_contextRec::Const. 224848b8605Smrg */ 225848b8605Smrgvoid 226848b8605Smrg_mesa_init_point(struct gl_context *ctx) 227848b8605Smrg{ 228848b8605Smrg GLuint i; 229848b8605Smrg 230848b8605Smrg ctx->Point.SmoothFlag = GL_FALSE; 231848b8605Smrg ctx->Point.Size = 1.0; 232848b8605Smrg ctx->Point.Params[0] = 1.0; 233848b8605Smrg ctx->Point.Params[1] = 0.0; 234848b8605Smrg ctx->Point.Params[2] = 0.0; 235848b8605Smrg ctx->Point._Attenuated = GL_FALSE; 236848b8605Smrg ctx->Point.MinSize = 0.0; 237848b8605Smrg ctx->Point.MaxSize 238848b8605Smrg = MAX2(ctx->Const.MaxPointSize, ctx->Const.MaxPointSizeAA); 239848b8605Smrg ctx->Point.Threshold = 1.0; 240848b8605Smrg 241848b8605Smrg /* Page 403 (page 423 of the PDF) of the OpenGL 3.0 spec says: 242848b8605Smrg * 243848b8605Smrg * "Non-sprite points (section 3.4) - Enable/Disable targets 244848b8605Smrg * POINT_SMOOTH and POINT_SPRITE, and all associated state. Point 245848b8605Smrg * rasterization is always performed as though POINT_SPRITE were 246848b8605Smrg * enabled." 247848b8605Smrg * 248848b8605Smrg * In a core context, the state will default to true, and the setters and 249848b8605Smrg * getters are disabled. 250848b8605Smrg */ 251848b8605Smrg ctx->Point.PointSprite = (ctx->API == API_OPENGL_CORE || 252848b8605Smrg ctx->API == API_OPENGLES2); 253848b8605Smrg 254848b8605Smrg ctx->Point.SpriteRMode = GL_ZERO; /* GL_NV_point_sprite (only!) */ 255848b8605Smrg ctx->Point.SpriteOrigin = GL_UPPER_LEFT; /* GL_ARB_point_sprite */ 256848b8605Smrg for (i = 0; i < Elements(ctx->Point.CoordReplace); i++) { 257848b8605Smrg ctx->Point.CoordReplace[i] = GL_FALSE; /* GL_ARB/NV_point_sprite */ 258848b8605Smrg } 259848b8605Smrg} 260