[Piglit] [PATCH] mesa/state_tracker: Fix draw-pixel-with-texture piglit test.

Emil Velikov emil.l.velikov at gmail.com
Sat Mar 28 16:41:18 PDT 2015


Hi Matthew

I'm suspecting that this patch was meant for mesa-dev ?

-Emil

On 28/03/15 17:07, Matthew Dawson wrote:
> When glDrawPixels was used with an external texture, the pixels passed in
> were sampled instead of the texture.  Change gallium to instead move the user
> texture to a new sampler below the glDrawPixels samplers and use the texture
> coordinates from the raster position.
> 
> This uses a uniform for the texture coordinates instead passing it through
> the vertex shader as the texture coordinates are constant for the entire
> operation.  While working the vertex shader would be possible, implementing
> that solution would break several assumptions throughout the glDrawPixels
> implementation as well as helper functions used by other code paths, increasing
> the chance for breakage.
> 
> Tested on llvmpipe, r600, and radeonsi.
> 
> V2: Complete everything missing from V1.
> 
> v3:
>  - Fix style issues.
>  - Add comments about how the texture references are fixed up to be valid.
> ---
>  src/mesa/state_tracker/st_cb_drawpixels.c  | 39 ++++++++++++++++++++++++++---
>  src/mesa/state_tracker/st_glsl_to_tgsi.cpp | 40 ++++++++++++++++++++++++++++++
>  2 files changed, 76 insertions(+), 3 deletions(-)
> 
> diff --git a/src/mesa/state_tracker/st_cb_drawpixels.c b/src/mesa/state_tracker/st_cb_drawpixels.c
> index 3edf31b..47abd16 100644
> --- a/src/mesa/state_tracker/st_cb_drawpixels.c
> +++ b/src/mesa/state_tracker/st_cb_drawpixels.c
> @@ -44,6 +44,7 @@
>  #include "main/texstore.h"
>  #include "main/glformats.h"
>  #include "program/program.h"
> +#include "program/prog_parameter.h"
>  #include "program/prog_print.h"
>  #include "program/prog_instruction.h"
>  
> @@ -676,6 +677,8 @@ draw_textured_quad(struct gl_context *ctx, GLint x, GLint y, GLfloat z,
>     GLfloat x0, y0, x1, y1;
>     GLsizei maxSize;
>     boolean normalized = sv[0]->texture->target != PIPE_TEXTURE_RECT;
> +   GLuint num_user_samplers = st->state.num_samplers[PIPE_SHADER_FRAGMENT];
> +   unsigned int i;
>  
>     /* limit checks */
>     /* XXX if DrawPixels image is larger than max texture size, break
> @@ -765,6 +768,10 @@ draw_textured_quad(struct gl_context *ctx, GLint x, GLint y, GLfloat z,
>        if (num_sampler_view > 1) {
>           cso_single_sampler(cso, PIPE_SHADER_FRAGMENT, 1, &sampler);
>        }
> +      for (i = 0; i < num_user_samplers; ++i) {
> +         cso_single_sampler(cso, PIPE_SHADER_FRAGMENT, i+num_sampler_view,
> +                            &st->state.samplers[PIPE_SHADER_FRAGMENT][i]);
> +      }
>        cso_single_sampler_done(cso, PIPE_SHADER_FRAGMENT);
>     }
>  
> @@ -786,7 +793,14 @@ draw_textured_quad(struct gl_context *ctx, GLint x, GLint y, GLfloat z,
>     cso_set_stream_outputs(st->cso_context, 0, NULL, NULL);
>  
>     /* texture state: */
> -   cso_set_sampler_views(cso, PIPE_SHADER_FRAGMENT, num_sampler_view, sv);
> +   {
> +      struct pipe_sampler_view *lsv[PIPE_MAX_SAMPLERS];
> +      memcpy(lsv, sv, num_sampler_view*sizeof(struct pipe_sampler_view*));
> +      memcpy(lsv+num_sampler_view, st->state.sampler_views[PIPE_SHADER_FRAGMENT],
> +             num_user_samplers*sizeof(struct pipe_sampler_view*));
> +      cso_set_sampler_views(cso, PIPE_SHADER_FRAGMENT,
> +                            num_sampler_view+num_user_samplers, lsv);
> +   }
>  
>     /* Compute Gallium window coords (y=0=top) with pixel zoom.
>      * Recall that these coords are transformed by the current
> @@ -1160,8 +1174,27 @@ st_DrawPixels(struct gl_context *ctx, GLint x, GLint y,
>        }
>     }
>  
> -   /* update fragment program constants */
> -   st_upload_constants(st, fpv->parameters, PIPE_SHADER_FRAGMENT);
> +   /* updated texture coordinates and fragment program constants. */
> +   {
> +      int i;
> +      struct gl_program_parameter_list *parameters = fpv->parameters;
> +      /* Update the texture coordinates from all the used texture units, using
> +       * the values associated with the raster position */
> +      for (i = 0; i < parameters->NumParameters; ++i) {
> +         const char *name = parameters->Parameters[i].Name;
> +         /* Any texture coordinate will use the parameter name texcoord_, where
> +          * _ is the texture units id added to the ascii character 'A'.  This
> +          * checks if a parameter matches this patterns, and updates the
> +          * parameter's value with the appropriate raster position texture
> +          * coordinate. */
> +         if (strncmp("texcoord", name, 8) == 0 && strlen(name) == 9) {
> +            memcpy(parameters->ParameterValues[i],
> +                   st->ctx->Current.RasterTexCoords[name[8] - 'A'],
> +                   sizeof(GL_FLOAT) * 4);
> +         }
> +      }
> +      st_upload_constants(st, parameters, PIPE_SHADER_FRAGMENT);
> +   }
>  
>     /* draw with textured quad */
>     {
> diff --git a/src/mesa/state_tracker/st_glsl_to_tgsi.cpp b/src/mesa/state_tracker/st_glsl_to_tgsi.cpp
> index e97ab83..b2f6e6e 100644
> --- a/src/mesa/state_tracker/st_glsl_to_tgsi.cpp
> +++ b/src/mesa/state_tracker/st_glsl_to_tgsi.cpp
> @@ -4256,6 +4256,7 @@ get_pixel_transfer_visitor(struct st_fragment_program *fp,
>     st_src_reg coord, src0;
>     st_dst_reg dst0;
>     glsl_to_tgsi_instruction *inst;
> +   unsigned int count_samplers_used = 0;
>  
>     /* Copy attributes of the glsl_to_tgsi_visitor in the original shader. */
>     v->ctx = original->ctx;
> @@ -4286,6 +4287,7 @@ get_pixel_transfer_visitor(struct st_fragment_program *fp,
>     prog->InputsRead |= VARYING_BIT_TEX0;
>     prog->SamplersUsed |= (1 << 0); /* mark sampler 0 as used */
>     v->samplers_used |= (1 << 0);
> +   count_samplers_used++;
>  
>     if (scale_and_bias) {
>        static const gl_state_index scale_state[STATE_LENGTH] =
> @@ -4333,6 +4335,7 @@ get_pixel_transfer_visitor(struct st_fragment_program *fp,
>  
>        prog->SamplersUsed |= (1 << 1); /* mark sampler 1 as used */
>        v->samplers_used |= (1 << 1);
> +      count_samplers_used++;
>  
>        /* MOV colorTemp, temp; */
>        inst = v->emit(NULL, TGSI_OPCODE_MOV, dst0, temp);
> @@ -4361,6 +4364,43 @@ get_pixel_transfer_visitor(struct st_fragment_program *fp,
>        newinst = v->emit(NULL, inst->op, inst->dst[0], src_regs[0], src_regs[1], src_regs[2]);
>        newinst->tex_target = inst->tex_target;
>        newinst->sampler_array_size = inst->sampler_array_size;
> +
> +      /* If the original fragment shader (either user supplied, or generated from
> +       * the fixed function pipeline) references a texture, fix up the instruction
> +       * here to not conflict with the DrawPixels shader. */
> +      if (newinst->tex_target != 0) {
> +         int new_sampler_index;
> +         char name[10];
> +         GLint param;
> +
> +         /* Grab the original sampler, and move it up by the number of samplers
> +          * used by the DrawPixels shader. */
> +         newinst->sampler = inst->sampler;
> +         new_sampler_index = newinst->sampler.index + count_samplers_used;
> +
> +         /* This name records what texture coordinates are need for this parameter.
> +         * This is used later to update the texture coordinates as appropriate. */
> +         memcpy(name, "texcoord_", 10);
> +         name[8] = 'A' + newinst->sampler.index; // Replaces the underscore from above, while keeping the null byte.
> +
> +         if ((prog->SamplersUsed & (1 << new_sampler_index)) == 0) {
> +            /* If this is the first instruction referencing it, create a uniform
> +             * to store its texture coordinate, which will be a constant for the
> +             * whole drawn area */
> +            param = _mesa_add_parameter(params, PROGRAM_UNIFORM, name, 4, GL_FLOAT, (const gl_constant_value*)st->ctx->Current.RasterTexCoords[newinst->sampler.index], 0);
> +         } else {
> +            /* Otherwise get its parameter index to use below */
> +            param = _mesa_lookup_parameter_index(params, -1, name);
> +         }
> +         /* Update the new instruction with the new sampler index and texture
> +          * coordinate parameter. */
> +         newinst->src[0] = st_src_reg(PROGRAM_UNIFORM, param, GLSL_TYPE_FLOAT);
> +         newinst->sampler.index = new_sampler_index;
> +
> +         /* And mark the sampler as used */
> +         prog->SamplersUsed |= (1 << new_sampler_index);
> +         v->samplers_used |= (1 << new_sampler_index);
> +      }
>     }
>  
>     /* Make modifications to fragment program info. */
> 



More information about the Piglit mailing list