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