[Mesa-dev] [PATCH v2] st/mesa: fix the layer of VDPAU surface samplers

Christian König deathsimple at vodafone.de
Fri Nov 4 13:35:43 UTC 2016


Am 04.11.2016 um 13:28 schrieb Nicolai Hähnle:
> From: Nicolai Hähnle <nicolai.haehnle at amd.com>
>
> A (latent) bug in VDPAU interop was exposed by commit
> e5cc84dd43be066c1dd418e32f5ad258e31a150a.
>
> Before that commit, the st_vdpau code created samplers with
> first_layer == last_layer == 1 that the general texture handling code
> would immediately delete and re-create, because the layer does not match
> the information in the GL texture object.
>
> This was correct behavior at least in the DMABUF case, because the imported
> resource is supposed to have the correct offset already applied.  In the
> non-DMABUF case, this was just plain wrong but apparently nobody noticed.
>
> After that commit, the state tracker assumes that an existing sampler is
> correct at all times.  Existing samplers are supposed to be deleted when
> they may become invalid, and they will be created on-demand.  This meant
> that the sampler with first_layer == last_layer == 1 stuck around, leading
> to rendering artefacts (on radeonsi), command stream failures (on r600), and
> assertions (in debug builds everywhere).
>
> This patch fixes the problem by simply not creating a sampler at all in
> st_vdpau_map_surface.  We rely on the generic texture code to do the right
> thing, adding the layer_override to make the non-DMABUF case work.
>
> v2: add the layer_override
>
> Bugzilla: https://bugs.freedesktop.org/show_bug.cgi?id=98512
> Cc: 13.0 <mesa-stable at lists.freedesktop.org>
> Cc: Christian König <deathsimple at vodafone.de>
> Cc: Ilia Mirkin <imirkin at alum.mit.edu>

Reviewed-by: Christian König <christian.koenig at amd.com>.

> --
> This fixes the problem on radeon, but it would be good to get some testing
> on Nouveau as well. According to Christian, testing nouveau with Kodi should
> do the trick, probably mpv --hwdec=vdpau --vo=opengl should also work. Errors
> may only be noticeable as bad (de-)interlacing.
> ---
>   src/mesa/state_tracker/st_sampler_view.c | 15 +++++++++++----
>   src/mesa/state_tracker/st_texture.h      |  9 +++++++++
>   src/mesa/state_tracker/st_vdpau.c        | 20 +++++++-------------
>   3 files changed, 27 insertions(+), 17 deletions(-)
>
> diff --git a/src/mesa/state_tracker/st_sampler_view.c b/src/mesa/state_tracker/st_sampler_view.c
> index 9fe0bfe..2b2fa8b 100644
> --- a/src/mesa/state_tracker/st_sampler_view.c
> +++ b/src/mesa/state_tracker/st_sampler_view.c
> @@ -423,22 +423,26 @@ st_create_texture_sampler_view_from_stobj(struct st_context *st,
>         size = MIN2(stObj->pt->width0 - base, (unsigned)stObj->base.BufferSize);
>         if (!size)
>            return NULL;
>   
>         templ.u.buf.offset = base;
>         templ.u.buf.size = size;
>      } else {
>         templ.u.tex.first_level = stObj->base.MinLevel + stObj->base.BaseLevel;
>         templ.u.tex.last_level = last_level(stObj);
>         assert(templ.u.tex.first_level <= templ.u.tex.last_level);
> -      templ.u.tex.first_layer = stObj->base.MinLayer;
> -      templ.u.tex.last_layer = last_layer(stObj);
> +      if (stObj->layer_override) {
> +         templ.u.tex.first_layer = templ.u.tex.last_layer = stObj->layer_override;
> +      } else {
> +         templ.u.tex.first_layer = stObj->base.MinLayer;
> +         templ.u.tex.last_layer = last_layer(stObj);
> +      }
>         assert(templ.u.tex.first_layer <= templ.u.tex.last_layer);
>         templ.target = gl_target_to_pipe(stObj->base.Target);
>      }
>   
>      templ.swizzle_r = GET_SWZ(swizzle, 0);
>      templ.swizzle_g = GET_SWZ(swizzle, 1);
>      templ.swizzle_b = GET_SWZ(swizzle, 2);
>      templ.swizzle_a = GET_SWZ(swizzle, 3);
>   
>      return st->pipe->create_sampler_view(st->pipe, stObj->pt, &templ);
> @@ -471,22 +475,25 @@ st_get_texture_sampler_view_from_stobj(struct st_context *st,
>            unsigned base = stObj->base.BufferOffset;
>            unsigned size = MIN2(stObj->pt->width0 - base,
>                                 (unsigned) stObj->base.BufferSize);
>            assert(view->u.buf.offset == base);
>            assert(view->u.buf.size == size);
>         }
>         else {
>            assert(stObj->base.MinLevel + stObj->base.BaseLevel ==
>                   view->u.tex.first_level);
>            assert(last_level(stObj) == view->u.tex.last_level);
> -         assert(stObj->base.MinLayer == view->u.tex.first_layer);
> -         assert(last_layer(stObj) == view->u.tex.last_layer);
> +         assert(stObj->layer_override || stObj->base.MinLayer == view->u.tex.first_layer);
> +         assert(stObj->layer_override || last_layer(stObj) == view->u.tex.last_layer);
> +         assert(!stObj->layer_override ||
> +                (stObj->layer_override == view->u.tex.first_layer &&
> +                 stObj->layer_override == view->u.tex.last_layer));
>         }
>      }
>      else {
>         /* create new sampler view */
>         enum pipe_format format = get_sampler_view_format(st, stObj, samp);
>   
>         *sv = st_create_texture_sampler_view_from_stobj(st, stObj,
>                                                         format, glsl_version);
>   
>      }
> diff --git a/src/mesa/state_tracker/st_texture.h b/src/mesa/state_tracker/st_texture.h
> index 730843a..0ce7989 100644
> --- a/src/mesa/state_tracker/st_texture.h
> +++ b/src/mesa/state_tracker/st_texture.h
> @@ -101,20 +101,29 @@ struct st_texture_object
>       * cannot be reallocated and the format can only be changed with a sampler
>       * view or a surface.
>       */
>      GLboolean surface_based;
>   
>      /* If surface_based is true, this format should be used for all sampler
>       * views and surfaces instead of pt->format.
>       */
>      enum pipe_format surface_format;
>   
> +   /* When non-zero, samplers should use this layer instead of the one
> +    * specified by the GL state.
> +    *
> +    * This is used for VDPAU interop, where imported pipe_resources may be
> +    * array textures (containing layers with different fields) even though the
> +    * GL state describes one non-array texture per field.
> +    */
> +   uint layer_override;
> +
>      /** The glsl version of the shader seen during the previous validation */
>      unsigned prev_glsl_version;
>      /** The value of the sampler's sRGBDecode state at the previous validation */
>      GLenum prev_sRGBDecode;
>   };
>   
>   
>   static inline struct st_texture_image *
>   st_texture_image(struct gl_texture_image *img)
>   {
> diff --git a/src/mesa/state_tracker/st_vdpau.c b/src/mesa/state_tracker/st_vdpau.c
> index 7912057..0273815 100644
> --- a/src/mesa/state_tracker/st_vdpau.c
> +++ b/src/mesa/state_tracker/st_vdpau.c
> @@ -182,34 +182,36 @@ static void
>   st_vdpau_map_surface(struct gl_context *ctx, GLenum target, GLenum access,
>                        GLboolean output, struct gl_texture_object *texObj,
>                        struct gl_texture_image *texImage,
>                        const void *vdpSurface, GLuint index)
>   {
>      struct st_context *st = st_context(ctx);
>      struct st_texture_object *stObj = st_texture_object(texObj);
>      struct st_texture_image *stImage = st_texture_image(texImage);
>   
>      struct pipe_resource *res;
> -   struct pipe_sampler_view templ, **sampler_view;
>      mesa_format texFormat;
> +   uint layer_override = 0;
>   
>      if (output) {
>         res = st_vdpau_output_surface_dma_buf(ctx, vdpSurface);
>   
>         if (!res)
>            res = st_vdpau_output_surface_gallium(ctx, vdpSurface);
>   
>      } else {
>         res = st_vdpau_video_surface_dma_buf(ctx, vdpSurface, index);
>   
> -      if (!res)
> +      if (!res) {
>            res = st_vdpau_video_surface_gallium(ctx, vdpSurface, index);
> +         layer_override = index & 1;
> +      }
>      }
>   
>      if (!res) {
>         _mesa_error(ctx, GL_INVALID_OPERATION, "VDPAUMapSurfacesNV");
>         return;
>      }
>   
>      /* do we have different screen objects ? */
>      if (res->screen != st->pipe->screen) {
>         _mesa_error(ctx, GL_INVALID_OPERATION, "VDPAUMapSurfacesNV");
> @@ -226,51 +228,43 @@ st_vdpau_map_surface(struct gl_context *ctx, GLenum target, GLenum access,
>      texFormat = st_pipe_format_to_mesa_format(res->format);
>   
>      _mesa_init_teximage_fields(ctx, texImage,
>                                 res->width0, res->height0, 1, 0, GL_RGBA,
>                                 texFormat);
>   
>      pipe_resource_reference(&stObj->pt, res);
>      st_texture_release_all_sampler_views(st, stObj);
>      pipe_resource_reference(&stImage->pt, res);
>   
> -   u_sampler_view_default_template(&templ, res, res->format);
> -   templ.u.tex.first_layer = index & 1;
> -   templ.u.tex.last_layer = index & 1;
> -   templ.swizzle_r = GET_SWZ(stObj->base._Swizzle, 0);
> -   templ.swizzle_g = GET_SWZ(stObj->base._Swizzle, 1);
> -   templ.swizzle_b = GET_SWZ(stObj->base._Swizzle, 2);
> -   templ.swizzle_a = GET_SWZ(stObj->base._Swizzle, 3);
> -
> -   sampler_view = st_texture_get_sampler_view(st, stObj);
> -   *sampler_view = st->pipe->create_sampler_view(st->pipe, res, &templ);
> -
>      stObj->surface_format = res->format;
> +   stObj->layer_override = layer_override;
>   
>      _mesa_dirty_texobj(ctx, texObj);
>      pipe_resource_reference(&res, NULL);
>   }
>   
>   static void
>   st_vdpau_unmap_surface(struct gl_context *ctx, GLenum target, GLenum access,
>                          GLboolean output, struct gl_texture_object *texObj,
>                          struct gl_texture_image *texImage,
>                          const void *vdpSurface, GLuint index)
>   {
>      struct st_context *st = st_context(ctx);
>      struct st_texture_object *stObj = st_texture_object(texObj);
>      struct st_texture_image *stImage = st_texture_image(texImage);
>   
>      pipe_resource_reference(&stObj->pt, NULL);
>      st_texture_release_all_sampler_views(st, stObj);
>      pipe_resource_reference(&stImage->pt, NULL);
>   
> +   stObj->layer_override = 0;
> +
>      _mesa_dirty_texobj(ctx, texObj);
>   
>      st_flush(st, NULL, 0);
>   }
>   
>   #endif
>   
>   void
>   st_init_vdpau_functions(struct dd_function_table *functions)
>   {




More information about the mesa-dev mailing list