[Mesa-dev] [PATCHv2 18/20] i965/fs: Translate image load, store and atomic NIR intrinsics.

Jason Ekstrand jason at jlekstrand.net
Thu Jul 30 16:07:03 PDT 2015


Reviewed-by: Jason Ekstrand <jason.ekstrand at intel.com>

On Thu, Jul 23, 2015 at 10:31 AM, Francisco Jerez <currojerez at riseup.net> wrote:
> v2: Move array coordinate workaround into the surface builder.
> ---
>  src/mesa/drivers/dri/i965/brw_fs_nir.cpp | 106 +++++++++++++++++++++++++++++++
>  1 file changed, 106 insertions(+)
>
> diff --git a/src/mesa/drivers/dri/i965/brw_fs_nir.cpp b/src/mesa/drivers/dri/i965/brw_fs_nir.cpp
> index 805f782..318b600 100644
> --- a/src/mesa/drivers/dri/i965/brw_fs_nir.cpp
> +++ b/src/mesa/drivers/dri/i965/brw_fs_nir.cpp
> @@ -24,6 +24,7 @@
>  #include "glsl/ir.h"
>  #include "glsl/ir_optimization.h"
>  #include "glsl/nir/glsl_to_nir.h"
> +#include "main/shaderimage.h"
>  #include "program/prog_to_nir.h"
>  #include "brw_fs.h"
>  #include "brw_fs_surface_builder.h"
> @@ -1246,6 +1247,55 @@ fs_visitor::emit_percomp(const fs_builder &bld, const fs_inst &inst,
>     }
>  }
>
> +/**
> + * Get the matching channel register datatype for an image intrinsic of the
> + * specified GLSL image type.
> + */
> +static brw_reg_type
> +get_image_base_type(const glsl_type *type)
> +{
> +   switch ((glsl_base_type)type->sampler_type) {
> +   case GLSL_TYPE_UINT:
> +      return BRW_REGISTER_TYPE_UD;
> +   case GLSL_TYPE_INT:
> +      return BRW_REGISTER_TYPE_D;
> +   case GLSL_TYPE_FLOAT:
> +      return BRW_REGISTER_TYPE_F;
> +   default:
> +      unreachable("Not reached.");
> +   }
> +}
> +
> +/**
> + * Get the appropriate atomic op for an image atomic intrinsic.
> + */
> +static unsigned
> +get_image_atomic_op(nir_intrinsic_op op, const glsl_type *type)
> +{
> +   switch (op) {
> +   case nir_intrinsic_image_atomic_add:
> +      return BRW_AOP_ADD;
> +   case nir_intrinsic_image_atomic_min:
> +      return (get_image_base_type(type) == BRW_REGISTER_TYPE_D ?
> +              BRW_AOP_IMIN : BRW_AOP_UMIN);
> +   case nir_intrinsic_image_atomic_max:
> +      return (get_image_base_type(type) == BRW_REGISTER_TYPE_D ?
> +              BRW_AOP_IMAX : BRW_AOP_UMAX);
> +   case nir_intrinsic_image_atomic_and:
> +      return BRW_AOP_AND;
> +   case nir_intrinsic_image_atomic_or:
> +      return BRW_AOP_OR;
> +   case nir_intrinsic_image_atomic_xor:
> +      return BRW_AOP_XOR;
> +   case nir_intrinsic_image_atomic_exchange:
> +      return BRW_AOP_MOV;
> +   case nir_intrinsic_image_atomic_comp_swap:
> +      return BRW_AOP_CMPWR;
> +   default:
> +      unreachable("Not reachable.");
> +   }
> +}
> +
>  void
>  fs_visitor::nir_emit_intrinsic(const fs_builder &bld, nir_intrinsic_instr *instr)
>  {
> @@ -1320,6 +1370,62 @@ fs_visitor::nir_emit_intrinsic(const fs_builder &bld, nir_intrinsic_instr *instr
>        break;
>     }
>
> +   case nir_intrinsic_image_load:
> +   case nir_intrinsic_image_store:
> +   case nir_intrinsic_image_atomic_add:
> +   case nir_intrinsic_image_atomic_min:
> +   case nir_intrinsic_image_atomic_max:
> +   case nir_intrinsic_image_atomic_and:
> +   case nir_intrinsic_image_atomic_or:
> +   case nir_intrinsic_image_atomic_xor:
> +   case nir_intrinsic_image_atomic_exchange:
> +   case nir_intrinsic_image_atomic_comp_swap: {
> +      using namespace image_access;
> +
> +      /* Get the referenced image variable and type. */
> +      const nir_variable *var = instr->variables[0]->var;
> +      const glsl_type *type = var->type->without_array();
> +      const brw_reg_type base_type = get_image_base_type(type);
> +
> +      /* Get some metadata from the image intrinsic. */
> +      const nir_intrinsic_info *info = &nir_intrinsic_infos[instr->intrinsic];
> +      const unsigned arr_dims = type->sampler_array ? 1 : 0;
> +      const unsigned surf_dims = type->coordinate_components() - arr_dims;
> +      const mesa_format format =
> +         (var->data.image.write_only ? MESA_FORMAT_NONE :
> +          _mesa_get_shader_image_format(var->data.image.format));
> +
> +      /* Get the arguments of the image intrinsic. */
> +      const fs_reg image = get_nir_image_deref(instr->variables[0]);
> +      const fs_reg addr = retype(get_nir_src(instr->src[0]),
> +                                 BRW_REGISTER_TYPE_UD);
> +      const fs_reg src0 = (info->num_srcs >= 3 ?
> +                           retype(get_nir_src(instr->src[2]), base_type) :
> +                           fs_reg());
> +      const fs_reg src1 = (info->num_srcs >= 4 ?
> +                           retype(get_nir_src(instr->src[3]), base_type) :
> +                           fs_reg());
> +      fs_reg tmp;
> +
> +      /* Emit an image load, store or atomic op. */
> +      if (instr->intrinsic == nir_intrinsic_image_load)
> +         tmp = emit_image_load(bld, image, addr, surf_dims, arr_dims, format);
> +
> +      else if (instr->intrinsic == nir_intrinsic_image_store)
> +         emit_image_store(bld, image, addr, src0, surf_dims, arr_dims, format);
> +
> +      else
> +         tmp = emit_image_atomic(bld, image, addr, src0, src1,
> +                                 surf_dims, arr_dims, info->dest_components,
> +                                 get_image_atomic_op(instr->intrinsic, type));
> +
> +      /* Assign the result. */
> +      for (unsigned c = 0; c < info->dest_components; ++c)
> +         bld.MOV(offset(retype(dest, base_type), bld, c),
> +                 offset(tmp, bld, c));
> +      break;
> +   }
> +
>     case nir_intrinsic_load_front_face:
>        bld.MOV(retype(dest, BRW_REGISTER_TYPE_D),
>                *emit_frontfacing_interpolation());
> --
> 2.4.3
>


More information about the mesa-dev mailing list