2007-06-22 13:08:49 -06:00
|
|
|
/**************************************************************************
|
|
|
|
*
|
|
|
|
* Copyright 2007 Tungsten Graphics, Inc., Cedar Park, Texas.
|
|
|
|
* All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
* copy of this software and associated documentation files (the
|
|
|
|
* "Software"), to deal in the Software without restriction, including
|
|
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
* distribute, sub license, and/or sell copies of the Software, and to
|
|
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
* the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice (including the
|
|
|
|
* next paragraph) shall be included in all copies or substantial portions
|
|
|
|
* of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
|
|
|
|
* IN NO EVENT SHALL TUNGSTEN GRAPHICS AND/OR ITS SUPPLIERS BE LIABLE FOR
|
|
|
|
* ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
|
|
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
|
|
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Authors:
|
|
|
|
* Keith Whitwell <keith@tungstengraphics.com>
|
|
|
|
* Brian Paul
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2008-05-03 09:26:25 -06:00
|
|
|
#include "main/macros.h"
|
2011-04-20 11:48:31 -07:00
|
|
|
#include "main/mtypes.h"
|
|
|
|
#include "main/samplerobj.h"
|
2012-02-12 16:30:28 -07:00
|
|
|
#include "main/texobj.h"
|
2008-05-03 09:26:25 -06:00
|
|
|
|
2007-06-22 13:08:49 -06:00
|
|
|
#include "st_context.h"
|
2008-08-14 15:38:09 -06:00
|
|
|
#include "st_cb_texture.h"
|
2011-02-21 16:54:23 -07:00
|
|
|
#include "st_format.h"
|
2007-06-22 13:08:49 -06:00
|
|
|
#include "st_atom.h"
|
2011-04-10 12:44:46 -06:00
|
|
|
#include "st_texture.h"
|
2007-06-22 13:08:49 -06:00
|
|
|
#include "pipe/p_context.h"
|
|
|
|
#include "pipe/p_defines.h"
|
2008-05-03 09:26:25 -06:00
|
|
|
|
2008-03-11 18:54:31 -06:00
|
|
|
#include "cso_cache/cso_context.h"
|
2007-06-22 13:08:49 -06:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert GLenum texcoord wrap tokens to pipe tokens.
|
|
|
|
*/
|
|
|
|
static GLuint
|
2009-04-06 14:00:03 -06:00
|
|
|
gl_wrap_xlate(GLenum wrap)
|
2007-06-22 13:08:49 -06:00
|
|
|
{
|
|
|
|
switch (wrap) {
|
|
|
|
case GL_REPEAT:
|
|
|
|
return PIPE_TEX_WRAP_REPEAT;
|
|
|
|
case GL_CLAMP:
|
|
|
|
return PIPE_TEX_WRAP_CLAMP;
|
|
|
|
case GL_CLAMP_TO_EDGE:
|
|
|
|
return PIPE_TEX_WRAP_CLAMP_TO_EDGE;
|
|
|
|
case GL_CLAMP_TO_BORDER:
|
|
|
|
return PIPE_TEX_WRAP_CLAMP_TO_BORDER;
|
|
|
|
case GL_MIRRORED_REPEAT:
|
|
|
|
return PIPE_TEX_WRAP_MIRROR_REPEAT;
|
|
|
|
case GL_MIRROR_CLAMP_EXT:
|
|
|
|
return PIPE_TEX_WRAP_MIRROR_CLAMP;
|
|
|
|
case GL_MIRROR_CLAMP_TO_EDGE_EXT:
|
|
|
|
return PIPE_TEX_WRAP_MIRROR_CLAMP_TO_EDGE;
|
|
|
|
case GL_MIRROR_CLAMP_TO_BORDER_EXT:
|
|
|
|
return PIPE_TEX_WRAP_MIRROR_CLAMP_TO_BORDER;
|
|
|
|
default:
|
2008-06-24 10:56:36 +09:00
|
|
|
assert(0);
|
2007-06-22 13:08:49 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static GLuint
|
2007-08-09 19:09:19 +01:00
|
|
|
gl_filter_to_mip_filter(GLenum filter)
|
2007-06-22 13:08:49 -06:00
|
|
|
{
|
|
|
|
switch (filter) {
|
|
|
|
case GL_NEAREST:
|
|
|
|
case GL_LINEAR:
|
2007-08-09 19:09:19 +01:00
|
|
|
return PIPE_TEX_MIPFILTER_NONE;
|
|
|
|
|
2007-06-22 13:08:49 -06:00
|
|
|
case GL_NEAREST_MIPMAP_NEAREST:
|
2007-08-09 19:09:19 +01:00
|
|
|
case GL_LINEAR_MIPMAP_NEAREST:
|
|
|
|
return PIPE_TEX_MIPFILTER_NEAREST;
|
|
|
|
|
2007-06-22 13:08:49 -06:00
|
|
|
case GL_NEAREST_MIPMAP_LINEAR:
|
2007-08-09 19:09:19 +01:00
|
|
|
case GL_LINEAR_MIPMAP_LINEAR:
|
|
|
|
return PIPE_TEX_MIPFILTER_LINEAR;
|
|
|
|
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
return PIPE_TEX_MIPFILTER_NONE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static GLuint
|
|
|
|
gl_filter_to_img_filter(GLenum filter)
|
|
|
|
{
|
|
|
|
switch (filter) {
|
|
|
|
case GL_NEAREST:
|
|
|
|
case GL_NEAREST_MIPMAP_NEAREST:
|
|
|
|
case GL_NEAREST_MIPMAP_LINEAR:
|
|
|
|
return PIPE_TEX_FILTER_NEAREST;
|
|
|
|
|
|
|
|
case GL_LINEAR:
|
2007-06-22 13:08:49 -06:00
|
|
|
case GL_LINEAR_MIPMAP_NEAREST:
|
|
|
|
case GL_LINEAR_MIPMAP_LINEAR:
|
2007-08-09 19:09:19 +01:00
|
|
|
return PIPE_TEX_FILTER_LINEAR;
|
|
|
|
|
2007-06-22 13:08:49 -06:00
|
|
|
default:
|
2007-08-09 19:09:19 +01:00
|
|
|
assert(0);
|
|
|
|
return PIPE_TEX_FILTER_NEAREST;
|
2007-06-22 13:08:49 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-14 09:24:50 -06:00
|
|
|
|
|
|
|
static void
|
|
|
|
convert_sampler(struct st_context *st,
|
|
|
|
struct pipe_sampler_state *sampler,
|
|
|
|
GLuint texUnit)
|
2011-05-14 18:04:08 +10:00
|
|
|
{
|
2011-06-14 09:24:50 -06:00
|
|
|
struct gl_texture_object *texobj;
|
|
|
|
struct gl_context *ctx = st->ctx;
|
|
|
|
struct gl_sampler_object *msamp;
|
|
|
|
|
|
|
|
texobj = ctx->Texture.Unit[texUnit]._Current;
|
|
|
|
if (!texobj) {
|
2012-02-12 16:30:28 -07:00
|
|
|
texobj = _mesa_get_fallback_texture(ctx, TEXTURE_2D_INDEX);
|
2011-06-14 09:24:50 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
msamp = _mesa_get_samplerobj(ctx, texUnit);
|
|
|
|
|
|
|
|
memset(sampler, 0, sizeof(*sampler));
|
|
|
|
sampler->wrap_s = gl_wrap_xlate(msamp->WrapS);
|
|
|
|
sampler->wrap_t = gl_wrap_xlate(msamp->WrapT);
|
|
|
|
sampler->wrap_r = gl_wrap_xlate(msamp->WrapR);
|
|
|
|
|
|
|
|
sampler->min_img_filter = gl_filter_to_img_filter(msamp->MinFilter);
|
|
|
|
sampler->min_mip_filter = gl_filter_to_mip_filter(msamp->MinFilter);
|
|
|
|
sampler->mag_img_filter = gl_filter_to_img_filter(msamp->MagFilter);
|
|
|
|
|
|
|
|
if (texobj->Target != GL_TEXTURE_RECTANGLE_ARB)
|
|
|
|
sampler->normalized_coords = 1;
|
|
|
|
|
|
|
|
sampler->lod_bias = ctx->Texture.Unit[texUnit].LodBias + msamp->LodBias;
|
|
|
|
|
|
|
|
sampler->min_lod = CLAMP(msamp->MinLod,
|
|
|
|
0.0f,
|
|
|
|
(GLfloat) texobj->MaxLevel - texobj->BaseLevel);
|
|
|
|
sampler->max_lod = MIN2((GLfloat) texobj->MaxLevel - texobj->BaseLevel,
|
|
|
|
msamp->MaxLod);
|
|
|
|
if (sampler->max_lod < sampler->min_lod) {
|
|
|
|
/* The GL spec doesn't seem to specify what to do in this case.
|
|
|
|
* Swap the values.
|
|
|
|
*/
|
|
|
|
float tmp = sampler->max_lod;
|
|
|
|
sampler->max_lod = sampler->min_lod;
|
|
|
|
sampler->min_lod = tmp;
|
|
|
|
assert(sampler->min_lod <= sampler->max_lod);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msamp->BorderColor.ui[0] ||
|
|
|
|
msamp->BorderColor.ui[1] ||
|
|
|
|
msamp->BorderColor.ui[2] ||
|
|
|
|
msamp->BorderColor.ui[3]) {
|
|
|
|
struct gl_texture_image *teximg;
|
|
|
|
|
|
|
|
teximg = texobj->Image[0][texobj->BaseLevel];
|
|
|
|
|
|
|
|
st_translate_color(msamp->BorderColor.f,
|
|
|
|
teximg ? teximg->_BaseFormat : GL_RGBA,
|
2011-09-27 10:08:34 +01:00
|
|
|
sampler->border_color.f);
|
2011-06-14 09:24:50 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
sampler->max_anisotropy = (msamp->MaxAnisotropy == 1.0 ?
|
|
|
|
0 : (GLuint) msamp->MaxAnisotropy);
|
|
|
|
|
|
|
|
/* only care about ARB_shadow, not SGI shadow */
|
|
|
|
if (msamp->CompareMode == GL_COMPARE_R_TO_TEXTURE) {
|
|
|
|
sampler->compare_mode = PIPE_TEX_COMPARE_R_TO_TEXTURE;
|
|
|
|
sampler->compare_func
|
|
|
|
= st_compare_func_to_pipe(msamp->CompareFunc);
|
|
|
|
}
|
|
|
|
|
|
|
|
sampler->seamless_cube_map =
|
|
|
|
ctx->Texture.CubeMapSeamless || msamp->CubeMapSeamless;
|
2011-05-14 18:04:08 +10:00
|
|
|
}
|
2007-06-22 13:08:49 -06:00
|
|
|
|
2011-06-14 09:24:50 -06:00
|
|
|
|
2012-08-06 08:35:20 -06:00
|
|
|
/**
|
|
|
|
* Update the gallium driver's sampler state for fragment, vertex or
|
|
|
|
* geometry shader stage.
|
|
|
|
*/
|
2011-05-14 18:04:08 +10:00
|
|
|
static void
|
2012-08-06 08:35:20 -06:00
|
|
|
update_shader_samplers(struct st_context *st,
|
|
|
|
unsigned shader_stage,
|
|
|
|
const struct gl_program *prog,
|
|
|
|
unsigned max_units,
|
|
|
|
struct pipe_sampler_state *samplers,
|
|
|
|
unsigned *num_samplers)
|
2007-06-22 13:08:49 -06:00
|
|
|
{
|
2012-08-06 08:35:20 -06:00
|
|
|
GLuint unit;
|
|
|
|
GLbitfield samplers_used;
|
|
|
|
const GLuint old_max = *num_samplers;
|
2011-05-14 18:04:08 +10:00
|
|
|
|
2012-08-06 08:35:20 -06:00
|
|
|
samplers_used = prog->SamplersUsed;
|
2011-05-14 18:04:08 +10:00
|
|
|
|
2012-08-06 08:35:20 -06:00
|
|
|
if (*num_samplers == 0 && samplers_used == 0x0)
|
2012-08-06 08:35:20 -06:00
|
|
|
return;
|
2007-06-22 13:08:49 -06:00
|
|
|
|
2012-08-06 08:35:20 -06:00
|
|
|
*num_samplers = 0;
|
2008-03-05 10:50:14 +01:00
|
|
|
|
2007-12-14 11:43:15 -07:00
|
|
|
/* loop over sampler units (aka tex image units) */
|
2012-08-06 08:35:20 -06:00
|
|
|
for (unit = 0; unit < max_units; unit++, samplers_used >>= 1) {
|
|
|
|
struct pipe_sampler_state *sampler = samplers + unit;
|
2007-06-22 13:08:49 -06:00
|
|
|
|
2012-02-08 13:58:14 +00:00
|
|
|
if (samplers_used & 1) {
|
2012-08-06 08:35:20 -06:00
|
|
|
const GLuint texUnit = prog->SamplerUnits[unit];
|
2011-05-14 18:04:08 +10:00
|
|
|
|
2012-02-08 13:58:14 +00:00
|
|
|
convert_sampler(st, sampler, texUnit);
|
2011-05-02 02:37:46 +02:00
|
|
|
|
2012-08-06 08:35:20 -06:00
|
|
|
*num_samplers = unit + 1;
|
2008-03-05 10:50:14 +01:00
|
|
|
|
2012-08-06 08:35:20 -06:00
|
|
|
cso_single_sampler(st->cso_context, shader_stage, unit, sampler);
|
2008-03-12 14:20:54 -06:00
|
|
|
}
|
2012-08-06 08:35:20 -06:00
|
|
|
else if (samplers_used != 0 || unit < old_max) {
|
|
|
|
cso_single_sampler(st->cso_context, shader_stage, unit, NULL);
|
|
|
|
}
|
|
|
|
else {
|
2012-08-06 08:35:20 -06:00
|
|
|
/* if we've reset all the old samplers and we have no more new ones */
|
2012-02-08 13:58:14 +00:00
|
|
|
break;
|
2008-03-12 14:20:54 -06:00
|
|
|
}
|
2007-06-22 13:08:49 -06:00
|
|
|
}
|
2008-03-05 10:50:14 +01:00
|
|
|
|
2012-08-06 08:35:20 -06:00
|
|
|
cso_single_sampler_done(st->cso_context, shader_stage);
|
2007-06-22 13:08:49 -06:00
|
|
|
}
|
|
|
|
|
2011-06-14 09:24:50 -06:00
|
|
|
|
2011-05-14 18:04:08 +10:00
|
|
|
static void
|
|
|
|
update_samplers(struct st_context *st)
|
|
|
|
{
|
2012-08-06 08:35:20 -06:00
|
|
|
const struct gl_context *ctx = st->ctx;
|
|
|
|
|
|
|
|
update_shader_samplers(st,
|
|
|
|
PIPE_SHADER_FRAGMENT,
|
|
|
|
&ctx->FragmentProgram._Current->Base,
|
|
|
|
ctx->Const.MaxTextureImageUnits,
|
|
|
|
st->state.fragment_samplers,
|
|
|
|
&st->state.num_fragment_samplers);
|
|
|
|
|
|
|
|
update_shader_samplers(st,
|
|
|
|
PIPE_SHADER_VERTEX,
|
|
|
|
&ctx->VertexProgram._Current->Base,
|
|
|
|
ctx->Const.MaxVertexTextureImageUnits,
|
|
|
|
st->state.vertex_samplers,
|
|
|
|
&st->state.num_vertex_samplers);
|
|
|
|
|
|
|
|
/*
|
|
|
|
update_shader_samplers(st,
|
|
|
|
PIPE_SHADER_GEOMETRY,
|
|
|
|
&ctx->GeometryProgram._Current->Base,
|
|
|
|
ctx->Const.MaxGeometryTextureImageUnits,
|
|
|
|
st->state.geometry_samplers,
|
|
|
|
&st->state.num_geometry_samplers);
|
|
|
|
*/
|
2011-05-14 18:04:08 +10:00
|
|
|
}
|
2007-06-22 13:08:49 -06:00
|
|
|
|
2011-06-14 09:24:50 -06:00
|
|
|
|
2007-06-22 13:08:49 -06:00
|
|
|
const struct st_tracked_state st_update_sampler = {
|
2008-05-02 10:08:03 +00:00
|
|
|
"st_update_sampler", /* name */
|
|
|
|
{ /* dirty */
|
|
|
|
_NEW_TEXTURE, /* mesa */
|
|
|
|
0, /* st */
|
2007-06-22 13:08:49 -06:00
|
|
|
},
|
2008-05-02 10:08:03 +00:00
|
|
|
update_samplers /* update */
|
2007-06-22 13:08:49 -06:00
|
|
|
};
|