texgen.c revision 3464ebd5
1c1f859d4Smrg/*
2c1f859d4Smrg * Mesa 3-D graphics library
34a49301eSmrg * Version:  7.5
4c1f859d4Smrg *
5c1f859d4Smrg * Copyright (C) 1999-2008  Brian Paul   All Rights Reserved.
64a49301eSmrg * Copyright (C) 2009  VMware, Inc.  All Rights Reserved.
7c1f859d4Smrg *
8c1f859d4Smrg * Permission is hereby granted, free of charge, to any person obtaining a
9c1f859d4Smrg * copy of this software and associated documentation files (the "Software"),
10c1f859d4Smrg * to deal in the Software without restriction, including without limitation
11c1f859d4Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense,
12c1f859d4Smrg * and/or sell copies of the Software, and to permit persons to whom the
13c1f859d4Smrg * Software is furnished to do so, subject to the following conditions:
14c1f859d4Smrg *
15c1f859d4Smrg * The above copyright notice and this permission notice shall be included
16c1f859d4Smrg * in all copies or substantial portions of the Software.
17c1f859d4Smrg *
18c1f859d4Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19c1f859d4Smrg * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20c1f859d4Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
21c1f859d4Smrg * BRIAN PAUL BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
22c1f859d4Smrg * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
23c1f859d4Smrg * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24c1f859d4Smrg */
25c1f859d4Smrg
26c1f859d4Smrg/**
27c1f859d4Smrg * \file texgen.c
28c1f859d4Smrg *
29c1f859d4Smrg * glTexGen-related functions
30c1f859d4Smrg */
31c1f859d4Smrg
32c1f859d4Smrg
33c1f859d4Smrg#include "main/glheader.h"
34c1f859d4Smrg#include "main/context.h"
35c1f859d4Smrg#include "main/enums.h"
36c1f859d4Smrg#include "main/macros.h"
373464ebd5Sriastradh#include "main/mfeatures.h"
38c1f859d4Smrg#include "main/texgen.h"
394a49301eSmrg#include "main/texstate.h"
404a49301eSmrg#include "math/m_matrix.h"
41cdc920a0Smrg#include "main/dispatch.h"
42c1f859d4Smrg
43c1f859d4Smrg
444a49301eSmrg#if FEATURE_texgen
454a49301eSmrg
464a49301eSmrg
474a49301eSmrg/**
484a49301eSmrg * Return texgen state for given coordinate
494a49301eSmrg */
504a49301eSmrgstatic struct gl_texgen *
514a49301eSmrgget_texgen(struct gl_texture_unit *texUnit, GLenum coord)
524a49301eSmrg{
534a49301eSmrg   switch (coord) {
544a49301eSmrg   case GL_S:
554a49301eSmrg      return &texUnit->GenS;
564a49301eSmrg   case GL_T:
574a49301eSmrg      return &texUnit->GenT;
584a49301eSmrg   case GL_R:
594a49301eSmrg      return &texUnit->GenR;
604a49301eSmrg   case GL_Q:
614a49301eSmrg      return &texUnit->GenQ;
624a49301eSmrg   default:
634a49301eSmrg      return NULL;
644a49301eSmrg   }
654a49301eSmrg}
664a49301eSmrg
67c1f859d4Smrg
68c1f859d4Smrgvoid GLAPIENTRY
69c1f859d4Smrg_mesa_TexGenfv( GLenum coord, GLenum pname, const GLfloat *params )
70c1f859d4Smrg{
71c1f859d4Smrg   struct gl_texture_unit *texUnit;
724a49301eSmrg   struct gl_texgen *texgen;
734a49301eSmrg   GET_CURRENT_CONTEXT(ctx);
74c1f859d4Smrg   ASSERT_OUTSIDE_BEGIN_END(ctx);
75c1f859d4Smrg
76c1f859d4Smrg   if (MESA_VERBOSE&(VERBOSE_API|VERBOSE_TEXTURE))
77c1f859d4Smrg      _mesa_debug(ctx, "glTexGen %s %s %.1f(%s)...\n",
78c1f859d4Smrg                  _mesa_lookup_enum_by_nr(coord),
79c1f859d4Smrg                  _mesa_lookup_enum_by_nr(pname),
80c1f859d4Smrg                  *params,
81c1f859d4Smrg		  _mesa_lookup_enum_by_nr((GLenum) (GLint) *params));
82c1f859d4Smrg
83c1f859d4Smrg   if (ctx->Texture.CurrentUnit >= ctx->Const.MaxTextureCoordUnits) {
84c1f859d4Smrg      _mesa_error(ctx, GL_INVALID_OPERATION, "glTexGen(current unit)");
85c1f859d4Smrg      return;
86c1f859d4Smrg   }
87c1f859d4Smrg
884a49301eSmrg   texUnit = _mesa_get_current_tex_unit(ctx);
89c1f859d4Smrg
904a49301eSmrg   texgen = get_texgen(texUnit, coord);
914a49301eSmrg   if (!texgen) {
924a49301eSmrg      _mesa_error(ctx, GL_INVALID_ENUM, "glTexGen(coord)");
934a49301eSmrg      return;
944a49301eSmrg   }
954a49301eSmrg
964a49301eSmrg   switch (pname) {
974a49301eSmrg   case GL_TEXTURE_GEN_MODE:
984a49301eSmrg      {
994a49301eSmrg         GLenum mode = (GLenum) (GLint) params[0];
1004a49301eSmrg         GLbitfield bit = 0x0;
1014a49301eSmrg         if (texgen->Mode == mode)
1024a49301eSmrg            return;
1034a49301eSmrg         switch (mode) {
1044a49301eSmrg         case GL_OBJECT_LINEAR:
1054a49301eSmrg            bit = TEXGEN_OBJ_LINEAR;
1064a49301eSmrg            break;
1074a49301eSmrg         case GL_EYE_LINEAR:
1084a49301eSmrg            bit = TEXGEN_EYE_LINEAR;
1094a49301eSmrg            break;
1104a49301eSmrg         case GL_SPHERE_MAP:
1114a49301eSmrg            if (coord == GL_S || coord == GL_T)
1124a49301eSmrg               bit = TEXGEN_SPHERE_MAP;
1134a49301eSmrg            break;
1144a49301eSmrg         case GL_REFLECTION_MAP_NV:
1154a49301eSmrg            if (coord != GL_Q)
1164a49301eSmrg               bit = TEXGEN_REFLECTION_MAP_NV;
1174a49301eSmrg            break;
1184a49301eSmrg         case GL_NORMAL_MAP_NV:
1194a49301eSmrg            if (coord != GL_Q)
1204a49301eSmrg               bit = TEXGEN_NORMAL_MAP_NV;
1214a49301eSmrg            break;
1224a49301eSmrg         default:
1234a49301eSmrg            ; /* nop */
1244a49301eSmrg         }
1254a49301eSmrg         if (!bit) {
1264a49301eSmrg            _mesa_error( ctx, GL_INVALID_ENUM, "glTexGenfv(param)" );
1274a49301eSmrg            return;
1284a49301eSmrg         }
1294a49301eSmrg         FLUSH_VERTICES(ctx, _NEW_TEXTURE);
1304a49301eSmrg         texgen->Mode = mode;
1314a49301eSmrg         texgen->_ModeBit = bit;
1324a49301eSmrg      }
1334a49301eSmrg      break;
1344a49301eSmrg
1354a49301eSmrg   case GL_OBJECT_PLANE:
1364a49301eSmrg      {
1374a49301eSmrg         if (TEST_EQ_4V(texgen->ObjectPlane, params))
1384a49301eSmrg            return;
1394a49301eSmrg         FLUSH_VERTICES(ctx, _NEW_TEXTURE);
1404a49301eSmrg         COPY_4FV(texgen->ObjectPlane, params);
1414a49301eSmrg      }
1424a49301eSmrg      break;
1434a49301eSmrg
1444a49301eSmrg   case GL_EYE_PLANE:
1454a49301eSmrg      {
1464a49301eSmrg         GLfloat tmp[4];
1474a49301eSmrg         /* Transform plane equation by the inverse modelview matrix */
1484a49301eSmrg         if (_math_matrix_is_dirty(ctx->ModelviewMatrixStack.Top)) {
1494a49301eSmrg            _math_matrix_analyse(ctx->ModelviewMatrixStack.Top);
1504a49301eSmrg         }
1514a49301eSmrg         _mesa_transform_vector(tmp, params,
1524a49301eSmrg                                ctx->ModelviewMatrixStack.Top->inv);
1534a49301eSmrg         if (TEST_EQ_4V(texgen->EyePlane, tmp))
1544a49301eSmrg            return;
1554a49301eSmrg         FLUSH_VERTICES(ctx, _NEW_TEXTURE);
1564a49301eSmrg         COPY_4FV(texgen->EyePlane, tmp);
1574a49301eSmrg      }
1584a49301eSmrg      break;
1594a49301eSmrg
1604a49301eSmrg   default:
1614a49301eSmrg      _mesa_error( ctx, GL_INVALID_ENUM, "glTexGenfv(pname)" );
1624a49301eSmrg      return;
163c1f859d4Smrg   }
164c1f859d4Smrg
165c1f859d4Smrg   if (ctx->Driver.TexGen)
166c1f859d4Smrg      ctx->Driver.TexGen( ctx, coord, pname, params );
167c1f859d4Smrg}
168c1f859d4Smrg
169c1f859d4Smrg
1704a49301eSmrgstatic void GLAPIENTRY
171c1f859d4Smrg_mesa_TexGeniv(GLenum coord, GLenum pname, const GLint *params )
172c1f859d4Smrg{
173c1f859d4Smrg   GLfloat p[4];
174c1f859d4Smrg   p[0] = (GLfloat) params[0];
175c1f859d4Smrg   if (pname == GL_TEXTURE_GEN_MODE) {
176c1f859d4Smrg      p[1] = p[2] = p[3] = 0.0F;
177c1f859d4Smrg   }
178c1f859d4Smrg   else {
179c1f859d4Smrg      p[1] = (GLfloat) params[1];
180c1f859d4Smrg      p[2] = (GLfloat) params[2];
181c1f859d4Smrg      p[3] = (GLfloat) params[3];
182c1f859d4Smrg   }
183c1f859d4Smrg   _mesa_TexGenfv(coord, pname, p);
184c1f859d4Smrg}
185c1f859d4Smrg
186c1f859d4Smrg
1874a49301eSmrgstatic void GLAPIENTRY
188c1f859d4Smrg_mesa_TexGend(GLenum coord, GLenum pname, GLdouble param )
189c1f859d4Smrg{
1904a49301eSmrg   GLfloat p[4];
1914a49301eSmrg   p[0] = (GLfloat) param;
1924a49301eSmrg   p[1] = p[2] = p[3] = 0.0F;
1934a49301eSmrg   _mesa_TexGenfv( coord, pname, p );
194c1f859d4Smrg}
195c1f859d4Smrg
1963464ebd5Sriastradh#if FEATURE_ES1
1973464ebd5Sriastradh
1983464ebd5Sriastradhvoid GLAPIENTRY
1993464ebd5Sriastradh_es_GetTexGenfv(GLenum coord, GLenum pname, GLfloat *params)
2003464ebd5Sriastradh{
2013464ebd5Sriastradh   ASSERT(coord == GL_TEXTURE_GEN_STR_OES);
2023464ebd5Sriastradh   _mesa_GetTexGenfv(GL_S, pname, params);
2033464ebd5Sriastradh}
2043464ebd5Sriastradh
2053464ebd5Sriastradh
2063464ebd5Sriastradhvoid GLAPIENTRY
2073464ebd5Sriastradh_es_TexGenf(GLenum coord, GLenum pname, GLfloat param)
2083464ebd5Sriastradh{
2093464ebd5Sriastradh   ASSERT(coord == GL_TEXTURE_GEN_STR_OES);
2103464ebd5Sriastradh   /* set S, T, and R at the same time */
2113464ebd5Sriastradh   _mesa_TexGenf(GL_S, pname, param);
2123464ebd5Sriastradh   _mesa_TexGenf(GL_T, pname, param);
2133464ebd5Sriastradh   _mesa_TexGenf(GL_R, pname, param);
2143464ebd5Sriastradh}
2153464ebd5Sriastradh
2163464ebd5Sriastradh
2173464ebd5Sriastradhvoid GLAPIENTRY
2183464ebd5Sriastradh_es_TexGenfv(GLenum coord, GLenum pname, const GLfloat *params)
2193464ebd5Sriastradh{
2203464ebd5Sriastradh   ASSERT(coord == GL_TEXTURE_GEN_STR_OES);
2213464ebd5Sriastradh   /* set S, T, and R at the same time */
2223464ebd5Sriastradh   _mesa_TexGenfv(GL_S, pname, params);
2233464ebd5Sriastradh   _mesa_TexGenfv(GL_T, pname, params);
2243464ebd5Sriastradh   _mesa_TexGenfv(GL_R, pname, params);
2253464ebd5Sriastradh}
2263464ebd5Sriastradh
2273464ebd5Sriastradh#endif
228c1f859d4Smrg
2294a49301eSmrgstatic void GLAPIENTRY
230c1f859d4Smrg_mesa_TexGendv(GLenum coord, GLenum pname, const GLdouble *params )
231c1f859d4Smrg{
232c1f859d4Smrg   GLfloat p[4];
233c1f859d4Smrg   p[0] = (GLfloat) params[0];
234c1f859d4Smrg   if (pname == GL_TEXTURE_GEN_MODE) {
235c1f859d4Smrg      p[1] = p[2] = p[3] = 0.0F;
236c1f859d4Smrg   }
237c1f859d4Smrg   else {
238c1f859d4Smrg      p[1] = (GLfloat) params[1];
239c1f859d4Smrg      p[2] = (GLfloat) params[2];
240c1f859d4Smrg      p[3] = (GLfloat) params[3];
241c1f859d4Smrg   }
242c1f859d4Smrg   _mesa_TexGenfv( coord, pname, p );
243c1f859d4Smrg}
244c1f859d4Smrg
245c1f859d4Smrg
246cdc920a0Smrgvoid GLAPIENTRY
247c1f859d4Smrg_mesa_TexGenf( GLenum coord, GLenum pname, GLfloat param )
248c1f859d4Smrg{
2494a49301eSmrg   GLfloat p[4];
2504a49301eSmrg   p[0] = param;
2514a49301eSmrg   p[1] = p[2] = p[3] = 0.0F;
2524a49301eSmrg   _mesa_TexGenfv(coord, pname, p);
253c1f859d4Smrg}
254c1f859d4Smrg
255c1f859d4Smrg
256c1f859d4Smrgvoid GLAPIENTRY
257c1f859d4Smrg_mesa_TexGeni( GLenum coord, GLenum pname, GLint param )
258c1f859d4Smrg{
2594a49301eSmrg   GLint p[4];
2604a49301eSmrg   p[0] = param;
2614a49301eSmrg   p[1] = p[2] = p[3] = 0;
2624a49301eSmrg   _mesa_TexGeniv( coord, pname, p );
263c1f859d4Smrg}
264c1f859d4Smrg
265c1f859d4Smrg
266c1f859d4Smrg
2674a49301eSmrgstatic void GLAPIENTRY
268c1f859d4Smrg_mesa_GetTexGendv( GLenum coord, GLenum pname, GLdouble *params )
269c1f859d4Smrg{
2704a49301eSmrg   struct gl_texture_unit *texUnit;
2714a49301eSmrg   struct gl_texgen *texgen;
272c1f859d4Smrg   GET_CURRENT_CONTEXT(ctx);
273c1f859d4Smrg   ASSERT_OUTSIDE_BEGIN_END(ctx);
274c1f859d4Smrg
275c1f859d4Smrg   if (ctx->Texture.CurrentUnit >= ctx->Const.MaxTextureCoordUnits) {
276c1f859d4Smrg      _mesa_error(ctx, GL_INVALID_OPERATION, "glGetTexGendv(current unit)");
277c1f859d4Smrg      return;
278c1f859d4Smrg   }
279c1f859d4Smrg
2804a49301eSmrg   texUnit = _mesa_get_current_tex_unit(ctx);
281c1f859d4Smrg
2824a49301eSmrg   texgen = get_texgen(texUnit, coord);
2834a49301eSmrg   if (!texgen) {
2844a49301eSmrg      _mesa_error(ctx, GL_INVALID_ENUM, "glGetTexGendv(coord)");
2854a49301eSmrg      return;
2864a49301eSmrg   }
2874a49301eSmrg
2884a49301eSmrg   switch (pname) {
2894a49301eSmrg   case GL_TEXTURE_GEN_MODE:
2904a49301eSmrg      params[0] = ENUM_TO_DOUBLE(texgen->Mode);
2914a49301eSmrg      break;
2924a49301eSmrg   case GL_OBJECT_PLANE:
2934a49301eSmrg      COPY_4V(params, texgen->ObjectPlane);
2944a49301eSmrg      break;
2954a49301eSmrg   case GL_EYE_PLANE:
2964a49301eSmrg      COPY_4V(params, texgen->EyePlane);
2974a49301eSmrg      break;
2984a49301eSmrg   default:
2994a49301eSmrg      _mesa_error( ctx, GL_INVALID_ENUM, "glGetTexGendv(pname)" );
300c1f859d4Smrg   }
301c1f859d4Smrg}
302c1f859d4Smrg
303c1f859d4Smrg
304c1f859d4Smrg
305cdc920a0Smrgvoid GLAPIENTRY
306c1f859d4Smrg_mesa_GetTexGenfv( GLenum coord, GLenum pname, GLfloat *params )
307c1f859d4Smrg{
3084a49301eSmrg   struct gl_texture_unit *texUnit;
3094a49301eSmrg   struct gl_texgen *texgen;
310c1f859d4Smrg   GET_CURRENT_CONTEXT(ctx);
311c1f859d4Smrg   ASSERT_OUTSIDE_BEGIN_END(ctx);
312c1f859d4Smrg
313c1f859d4Smrg   if (ctx->Texture.CurrentUnit >= ctx->Const.MaxTextureCoordUnits) {
314c1f859d4Smrg      _mesa_error(ctx, GL_INVALID_OPERATION, "glGetTexGenfv(current unit)");
315c1f859d4Smrg      return;
316c1f859d4Smrg   }
317c1f859d4Smrg
3184a49301eSmrg   texUnit = _mesa_get_current_tex_unit(ctx);
319c1f859d4Smrg
3204a49301eSmrg   texgen = get_texgen(texUnit, coord);
3214a49301eSmrg   if (!texgen) {
3224a49301eSmrg      _mesa_error(ctx, GL_INVALID_ENUM, "glGetTexGenfv(coord)");
3234a49301eSmrg      return;
3244a49301eSmrg   }
3254a49301eSmrg
3264a49301eSmrg   switch (pname) {
3274a49301eSmrg   case GL_TEXTURE_GEN_MODE:
3284a49301eSmrg      params[0] = ENUM_TO_FLOAT(texgen->Mode);
3294a49301eSmrg      break;
3304a49301eSmrg   case GL_OBJECT_PLANE:
3314a49301eSmrg      COPY_4V(params, texgen->ObjectPlane);
3324a49301eSmrg      break;
3334a49301eSmrg   case GL_EYE_PLANE:
3344a49301eSmrg      COPY_4V(params, texgen->EyePlane);
3354a49301eSmrg      break;
3364a49301eSmrg   default:
3374a49301eSmrg      _mesa_error( ctx, GL_INVALID_ENUM, "glGetTexGenfv(pname)" );
338c1f859d4Smrg   }
339c1f859d4Smrg}
340c1f859d4Smrg
341c1f859d4Smrg
342c1f859d4Smrg
3434a49301eSmrgstatic void GLAPIENTRY
344c1f859d4Smrg_mesa_GetTexGeniv( GLenum coord, GLenum pname, GLint *params )
345c1f859d4Smrg{
3464a49301eSmrg   struct gl_texture_unit *texUnit;
3474a49301eSmrg   struct gl_texgen *texgen;
348c1f859d4Smrg   GET_CURRENT_CONTEXT(ctx);
349c1f859d4Smrg   ASSERT_OUTSIDE_BEGIN_END(ctx);
350c1f859d4Smrg
351c1f859d4Smrg   if (ctx->Texture.CurrentUnit >= ctx->Const.MaxTextureCoordUnits) {
352c1f859d4Smrg      _mesa_error(ctx, GL_INVALID_OPERATION, "glGetTexGeniv(current unit)");
353c1f859d4Smrg      return;
354c1f859d4Smrg   }
355c1f859d4Smrg
3564a49301eSmrg   texUnit = _mesa_get_current_tex_unit(ctx);
357c1f859d4Smrg
3584a49301eSmrg   texgen = get_texgen(texUnit, coord);
3594a49301eSmrg   if (!texgen) {
3604a49301eSmrg      _mesa_error(ctx, GL_INVALID_ENUM, "glGetTexGeniv(coord)");
3614a49301eSmrg      return;
362c1f859d4Smrg   }
3634a49301eSmrg
3644a49301eSmrg   switch (pname) {
3654a49301eSmrg   case GL_TEXTURE_GEN_MODE:
3664a49301eSmrg      params[0] = texgen->Mode;
3674a49301eSmrg      break;
3684a49301eSmrg   case GL_OBJECT_PLANE:
3694a49301eSmrg      params[0] = (GLint) texgen->ObjectPlane[0];
3704a49301eSmrg      params[1] = (GLint) texgen->ObjectPlane[1];
3714a49301eSmrg      params[2] = (GLint) texgen->ObjectPlane[2];
3724a49301eSmrg      params[3] = (GLint) texgen->ObjectPlane[3];
3734a49301eSmrg      break;
3744a49301eSmrg   case GL_EYE_PLANE:
3754a49301eSmrg      params[0] = (GLint) texgen->EyePlane[0];
3764a49301eSmrg      params[1] = (GLint) texgen->EyePlane[1];
3774a49301eSmrg      params[2] = (GLint) texgen->EyePlane[2];
3784a49301eSmrg      params[3] = (GLint) texgen->EyePlane[3];
3794a49301eSmrg      break;
3804a49301eSmrg   default:
3814a49301eSmrg      _mesa_error( ctx, GL_INVALID_ENUM, "glGetTexGeniv(pname)" );
3824a49301eSmrg   }
3834a49301eSmrg}
3844a49301eSmrg
3854a49301eSmrg
3864a49301eSmrgvoid
3874a49301eSmrg_mesa_init_texgen_dispatch(struct _glapi_table *disp)
3884a49301eSmrg{
3894a49301eSmrg   SET_GetTexGendv(disp, _mesa_GetTexGendv);
3904a49301eSmrg   SET_GetTexGenfv(disp, _mesa_GetTexGenfv);
3914a49301eSmrg   SET_GetTexGeniv(disp, _mesa_GetTexGeniv);
3924a49301eSmrg   SET_TexGend(disp, _mesa_TexGend);
3934a49301eSmrg   SET_TexGendv(disp, _mesa_TexGendv);
3944a49301eSmrg   SET_TexGenf(disp, _mesa_TexGenf);
3954a49301eSmrg   SET_TexGenfv(disp, _mesa_TexGenfv);
3964a49301eSmrg   SET_TexGeni(disp, _mesa_TexGeni);
3974a49301eSmrg   SET_TexGeniv(disp, _mesa_TexGeniv);
398c1f859d4Smrg}
399c1f859d4Smrg
400c1f859d4Smrg
4014a49301eSmrg#endif /* FEATURE_texgen */
402