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