[Mesa-dev] [PATCH v4 01/28] mesa: Add an implementation of a master convert function.

Jason Ekstrand jason at jlekstrand.net
Thu Jan 8 10:38:40 PST 2015


On Wed, Jan 7, 2015 at 11:21 PM, Iago Toral Quiroga <itoral at igalia.com>
wrote:

> From: Jason Ekstrand <jason.ekstrand at intel.com>
>
> v2 by Iago Toral <itoral at igalia.com>:
>
> - When testing if we can directly pack we should use the src format to
> check
>   if we are packing from an RGBA format. The original code used the dst
> format
>   for the ubyte case by mistake.
> - Fixed incorrect number of bits for dst, it was computed using the src
> format
>   instead of the dst format.
> - If the dst format is an array format, check if it is signed. We were only
>   checking this for the case where it was not an array format, but we need
>   to know this in both scenarios.
> - Fixed incorrect swizzle transform for the cases where we convert between
>   array formats.
> - Compute is_signed and bits only once and for the dst format. We were
>   computing these for the src format too but they were overwritten by the
>   dst values immediately after.
> - Be more careful when selecting the integer path. Specifically, check that
>   both src and dst are integer types. Checking only one of them should
> suffice
>   since OpenGL does not allow conversions between normalized and integer
> types,
>   but putting extra care here makes sense and also makes the actual
> requirements
>   for this path more clear.
> - The format argument for pack functions is the destination format we are
>   packing to, not the source format (which has to be RGBA).
> - Expose RGBA8888_* to other files. These will come in handy when in need
> to
>   test if a given array format is RGBA or in need to pass RGBA formats to
>   mesa_format_convert.
>
> v3 by Samuel Iglesias <siglesias at igalia.com>:
>
> - Add an RGBA8888_INT definition.
>
> v4 by Iago Toral <itoral at igalia.com> after review by Jason Ekstrand:
>
> - Added documentation for _mesa_format_convert.
> - Added additional explanatory comments for integer conversions.
> - Ensure that we use _messa_swizzle_and_convert for all signed source
> formats.
> - Squashed: do not directly (un)pack to RGBA UINT if the source is not
> unsigned.
>
> v5 by Iago Toral <itoral at igalia.com>:
>
> - Adapted to the new implementation of mesa_array_format as a plain
> uint32_t
>   bitfield.
>
> Reviewed-by: Jason Ekstrand <jason.ekstrand at intel.com>
> ---
>  src/mesa/main/format_utils.c | 408
> +++++++++++++++++++++++++++++++++++++++++++
>  src/mesa/main/format_utils.h |  10 ++
>  2 files changed, 418 insertions(+)
>
> diff --git a/src/mesa/main/format_utils.c b/src/mesa/main/format_utils.c
> index a6b8883..6467307 100644
> --- a/src/mesa/main/format_utils.c
> +++ b/src/mesa/main/format_utils.c
> @@ -24,6 +24,414 @@
>
>  #include "format_utils.h"
>  #include "glformats.h"
> +#include "format_pack.h"
> +#include "format_unpack.h"
> +
> +const mesa_array_format RGBA8888_FLOAT =
> +   MESA_ARRAY_FORMAT(4, 1, 1, 1, 4, 0, 1, 2, 3);
> +
> +const mesa_array_format RGBA8888_UBYTE =
> +   MESA_ARRAY_FORMAT(1, 0, 0, 1, 4, 0, 1, 2, 3);
> +
> +const mesa_array_format RGBA8888_UINT =
> +   MESA_ARRAY_FORMAT(4, 0, 0, 0, 4, 0, 1, 2, 3);
> +
> +const mesa_array_format RGBA8888_INT =
> +   MESA_ARRAY_FORMAT(4, 1, 0, 0, 4, 0, 1, 2, 3);
>

I'm sorry I didn't notice this before, but 8888_FLOAT doesn't make sense.
We need to either drop the 8888 from each of these or maybe call them
RGBA32_FLOAT, RGBA32_UINT, RGBA8_UBYTE, etc.  That said, let's do that as a
fixup patch later and not hold up the series for it.


> +
> +static void
> +invert_swizzle(uint8_t dst[4], const uint8_t src[4])
> +{
> +   int i, j;
> +
> +   dst[0] = MESA_FORMAT_SWIZZLE_NONE;
> +   dst[1] = MESA_FORMAT_SWIZZLE_NONE;
> +   dst[2] = MESA_FORMAT_SWIZZLE_NONE;
> +   dst[3] = MESA_FORMAT_SWIZZLE_NONE;
> +
> +   for (i = 0; i < 4; ++i)
> +      for (j = 0; j < 4; ++j)
> +         if (src[j] == i && dst[i] == MESA_FORMAT_SWIZZLE_NONE)
> +            dst[i] = j;
> +}
> +
> +static GLenum
> +gl_type_for_array_format_datatype(enum mesa_array_format_datatype type)
> +{
> +   switch (type) {
> +   case MESA_ARRAY_FORMAT_TYPE_UBYTE:
> +      return GL_UNSIGNED_BYTE;
> +   case MESA_ARRAY_FORMAT_TYPE_USHORT:
> +      return GL_UNSIGNED_SHORT;
> +   case MESA_ARRAY_FORMAT_TYPE_UINT:
> +      return GL_UNSIGNED_INT;
> +   case MESA_ARRAY_FORMAT_TYPE_BYTE:
> +      return GL_BYTE;
> +   case MESA_ARRAY_FORMAT_TYPE_SHORT:
> +      return GL_SHORT;
> +   case MESA_ARRAY_FORMAT_TYPE_INT:
> +      return GL_INT;
> +   case MESA_ARRAY_FORMAT_TYPE_HALF:
> +      return GL_HALF_FLOAT;
> +   case MESA_ARRAY_FORMAT_TYPE_FLOAT:
> +      return GL_FLOAT;
> +   default:
> +      assert(!"Invalid datatype");
> +      return GL_NONE;
> +   }
> +}
> +
> +/**
> + * This can be used to convert between most color formats.
> + *
> + * Limitations:
> + * - This function doesn't handle GL_COLOR_INDEX or YCBCR formats.
> + * - This function doesn't handle byte-swapping or transferOps, these
> should
> + *   be handled by the caller.
> + *
> + * \param void_dst  The address where converted color data will be stored.
> + *                  The caller must ensure that the buffer is large enough
> + *                  to hold the converted pixel data.
> + * \param dst_format  The destination color format. It can be a
> mesa_format
> + *                    or a mesa_array_format represented as an uint32_t.
> + * \param dst_stride  The stride of the destination format in bytes.
> + * \param void_src  The address of the source color data to convert.
> + * \param src_format  The source color format. It can be a mesa_format
> + *                    or a mesa_array_format represented as an uint32_t.
> + * \param src_stride  The stride of the source format in bytes.
> + * \param width  The width, in pixels, of the source image to convert.
> + * \param height  The height, in pixels, of the source image to convert.
> + */
> +void
> +_mesa_format_convert(void *void_dst, uint32_t dst_format, size_t
> dst_stride,
> +                     void *void_src, uint32_t src_format, size_t
> src_stride,
> +                     size_t width, size_t height)
> +{
> +   uint8_t *dst = (uint8_t *)void_dst;
> +   uint8_t *src = (uint8_t *)void_src;
> +   mesa_array_format src_array_format, dst_array_format;
> +   bool src_format_is_mesa_array_format, dst_format_is_mesa_array_format;
> +   uint8_t src2dst[4], src2rgba[4], rgba2dst[4], dst2rgba[4];
> +   GLenum src_gl_type, dst_gl_type, common_gl_type;
> +   bool normalized, dst_integer, src_integer, is_signed;
> +   int src_num_channels = 0, dst_num_channels = 0;
> +   uint8_t (*tmp_ubyte)[4];
> +   float (*tmp_float)[4];
> +   uint32_t (*tmp_uint)[4];
> +   int i, bits;
> +   size_t row;
> +
> +   if (_mesa_format_is_mesa_array_format(src_format)) {
> +      src_format_is_mesa_array_format = true;
> +      src_array_format = src_format;
> +   } else {
> +      assert(_mesa_is_format_color_format(src_format));
> +      src_format_is_mesa_array_format = false;
> +      src_array_format = _mesa_format_to_array_format(src_format);
> +   }
> +
> +   if (_mesa_format_is_mesa_array_format(dst_format)) {
> +      dst_format_is_mesa_array_format = true;
> +      dst_array_format = dst_format;
> +   } else {
> +      assert(_mesa_is_format_color_format(dst_format));
> +      dst_format_is_mesa_array_format = false;
> +      dst_array_format = _mesa_format_to_array_format(dst_format);
> +   }
> +
> +   /* Handle the cases where we can directly unpack */
> +   if (!src_format_is_mesa_array_format) {
> +      if (dst_array_format == RGBA8888_FLOAT) {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_unpack_rgba_row(src_format, width,
> +                                  src, (float (*)[4])dst);
> +            src += src_stride;
> +            dst += dst_stride;
> +         }
> +         return;
> +      } else if (dst_array_format == RGBA8888_UBYTE) {
> +         assert(!_mesa_is_format_integer_color(src_format));
> +         for (row = 0; row < height; ++row) {
> +            _mesa_unpack_ubyte_rgba_row(src_format, width,
> +                                        src, (uint8_t (*)[4])dst);
> +            src += src_stride;
> +            dst += dst_stride;
> +         }
> +         return;
> +      } else if (dst_array_format == RGBA8888_UINT &&
> +                 _mesa_is_format_unsigned(src_format)) {
> +         assert(_mesa_is_format_integer_color(src_format));
> +         for (row = 0; row < height; ++row) {
> +            _mesa_unpack_uint_rgba_row(src_format, width,
> +                                       src, (uint32_t (*)[4])dst);
> +            src += src_stride;
> +            dst += dst_stride;
> +         }
> +         return;
> +      }
> +   }
> +
> +   /* Handle the cases where we can directly pack */
> +   if (!dst_format_is_mesa_array_format) {
> +      if (src_array_format == RGBA8888_FLOAT) {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_pack_float_rgba_row(dst_format, width,
> +                                      (const float (*)[4])src, dst);
> +            src += src_stride;
> +            dst += dst_stride;
> +         }
> +         return;
> +      } else if (src_array_format == RGBA8888_UBYTE) {
> +         assert(!_mesa_is_format_integer_color(dst_format));
> +         for (row = 0; row < height; ++row) {
> +            _mesa_pack_ubyte_rgba_row(dst_format, width,
> +                                      (const uint8_t (*)[4])src, dst);
> +            src += src_stride;
> +            dst += dst_stride;
> +         }
> +         return;
> +      } else if (src_array_format == RGBA8888_UINT &&
> +                 _mesa_is_format_unsigned(dst_format)) {
> +         assert(_mesa_is_format_integer_color(dst_format));
> +         for (row = 0; row < height; ++row) {
> +            _mesa_pack_uint_rgba_row(dst_format, width,
> +                                     (const uint32_t (*)[4])src, dst);
> +            src += src_stride;
> +            dst += dst_stride;
> +         }
> +         return;
> +      }
> +   }
> +
> +   /* Handle conversions between array formats */
> +   normalized = false;
> +   if (src_array_format) {
> +      enum mesa_array_format_datatype datatype =
> +         _mesa_array_format_get_datatype(src_array_format);
> +      src_gl_type = gl_type_for_array_format_datatype(datatype);
> +
> +      src_num_channels =
> _mesa_array_format_get_num_channels(src_array_format);
> +
> +      _mesa_array_format_get_swizzle(src_array_format, src2rgba);
> +
> +      normalized = _mesa_array_format_is_normalized(src_array_format);
> +   }
> +
> +   if (dst_array_format) {
> +      enum mesa_array_format_datatype datatype =
> +         _mesa_array_format_get_datatype(dst_array_format);
> +      dst_gl_type = gl_type_for_array_format_datatype(datatype);
> +
> +      dst_num_channels =
> _mesa_array_format_get_num_channels(dst_array_format);
> +
> +      _mesa_array_format_get_swizzle(dst_array_format, dst2rgba);
> +      invert_swizzle(rgba2dst, dst2rgba);
> +
> +      normalized |= _mesa_array_format_is_normalized(dst_array_format);
> +   }
> +
> +   if (src_array_format && dst_array_format) {
> +      assert(_mesa_array_format_is_normalized(src_array_format) ==
> +             _mesa_array_format_is_normalized(dst_array_format));
> +
> +      for (i = 0; i < 4; i++) {
> +         if (rgba2dst[i] > MESA_FORMAT_SWIZZLE_W) {
> +            src2dst[i] = rgba2dst[i];
> +         } else {
> +            src2dst[i] = src2rgba[rgba2dst[i]];
> +         }
> +      }
> +
> +      for (row = 0; row < height; ++row) {
> +         _mesa_swizzle_and_convert(dst, dst_gl_type, dst_num_channels,
> +                                   src, src_gl_type, src_num_channels,
> +                                   src2dst, normalized, width);
> +         src += src_stride;
> +         dst += dst_stride;
> +      }
> +      return;
> +   }
> +
> +   /* At this point, we're fresh out of fast-paths and we need to convert
> +    * to float, uint32, or, if we're lucky, uint8.
> +    */
> +   dst_integer = false;
> +   src_integer = false;
> +
> +   if (src_array_format) {
> +      if (!_mesa_array_format_is_float(src_array_format) &&
> +          !_mesa_array_format_is_normalized(src_array_format))
> +         src_integer = true;
> +   } else {
> +      switch (_mesa_get_format_datatype(src_format)) {
> +      case GL_UNSIGNED_INT:
> +      case GL_INT:
> +         src_integer = true;
> +         break;
> +      }
> +   }
> +
> +   /* If the destination format is signed but the source is unsigned,
> then we
> +    * don't loose any data by converting to a signed intermediate format
> above
> +    * and beyond the precision that we loose in the conversion itself. If
> the
> +    * destination is unsigned then, by using an unsigned intermediate
> format,
> +    * we make the conversion function that converts from the source to the
> +    * intermediate format take care of truncating at zero. The exception
> here
> +    * is if the intermediate format is float, in which case the first
> +    * conversion will leave it signed and the second conversion will
> truncate
> +    * at zero.
> +    */
> +   is_signed = false;
> +   if (dst_array_format) {
> +      if (!_mesa_array_format_is_float(dst_array_format) &&
> +          !_mesa_array_format_is_normalized(dst_array_format))
> +         dst_integer = true;
> +      is_signed = _mesa_array_format_is_signed(dst_array_format);
> +      bits = 8 * _mesa_array_format_get_type_size(dst_array_format);
> +   } else {
> +      switch (_mesa_get_format_datatype(dst_format)) {
> +      case GL_UNSIGNED_NORMALIZED:
> +         is_signed = false;
> +         break;
> +      case GL_SIGNED_NORMALIZED:
> +         is_signed = true;
> +         break;
> +      case GL_FLOAT:
> +         is_signed = true;
> +         break;
> +      case GL_UNSIGNED_INT:
> +         is_signed = false;
> +         dst_integer = true;
> +         break;
> +      case GL_INT:
> +         is_signed = true;
> +         dst_integer = true;
> +         break;
> +      }
> +      bits = _mesa_get_format_max_bits(dst_format);
> +   }
> +
> +   assert(src_integer == dst_integer);
> +
> +   if (src_integer && dst_integer) {
> +      tmp_uint = malloc(width * height * sizeof(*tmp_uint));
> +
> +      /* The [un]packing functions for unsigned datatypes treat the 32-bit
> +       * integer array as signed for signed formats and as unsigned for
> +       * unsigned formats. This is a bit of a problem if we ever convert
> from
> +       * a signed to an unsigned format because the unsigned packing
> function
> +       * doesn't know that the input is signed and will treat it as
> unsigned
> +       * and not do the trunctation. The thing that saves us here is that
> all
> +       * of the packed formats are unsigned, so we can just always use
> +       * _mesa_swizzle_and_convert for signed formats, which is aware of
> the
> +       * truncation problem.
> +       */
> +      common_gl_type = is_signed ? GL_INT : GL_UNSIGNED_INT;
> +      if (src_array_format) {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_swizzle_and_convert(tmp_uint + row * width,
> common_gl_type, 4,
> +                                      src, src_gl_type, src_num_channels,
> +                                      src2rgba, normalized, width);
> +            src += src_stride;
> +         }
> +      } else {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_unpack_uint_rgba_row(src_format, width,
> +                                       src, tmp_uint + row * width);
> +            src += src_stride;
> +         }
> +      }
> +
> +      /* At this point, we have already done the truncation if the source
> is
> +       * signed but the destination is unsigned, so no need to force the
> +       * _mesa_swizzle_and_convert path.
> +       */
> +      if (dst_format_is_mesa_array_format) {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_swizzle_and_convert(dst, dst_gl_type, dst_num_channels,
> +                                      tmp_uint + row * width,
> common_gl_type, 4,
> +                                      rgba2dst, normalized, width);
> +            dst += dst_stride;
> +         }
> +      } else {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_pack_uint_rgba_row(dst_format, width,
> +                                     (const uint32_t (*)[4])tmp_uint +
> row * width, dst);
> +            dst += dst_stride;
> +         }
> +      }
> +
> +      free(tmp_uint);
> +   } else if (is_signed || bits > 8) {
> +      tmp_float = malloc(width * height * sizeof(*tmp_float));
> +
> +      if (src_format_is_mesa_array_format) {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_swizzle_and_convert(tmp_float + row * width, GL_FLOAT,
> 4,
> +                                      src, src_gl_type, src_num_channels,
> +                                      src2rgba, normalized, width);
> +            src += src_stride;
> +         }
> +      } else {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_unpack_rgba_row(src_format, width,
> +                                  src, tmp_float + row * width);
> +            src += src_stride;
> +         }
> +      }
> +
> +      if (dst_format_is_mesa_array_format) {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_swizzle_and_convert(dst, dst_gl_type, dst_num_channels,
> +                                      tmp_float + row * width, GL_FLOAT,
> 4,
> +                                      rgba2dst, normalized, width);
> +            dst += dst_stride;
> +         }
> +      } else {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_pack_float_rgba_row(dst_format, width,
> +                                      (const float (*)[4])tmp_float + row
> * width, dst);
> +            dst += dst_stride;
> +         }
> +      }
> +
> +      free(tmp_float);
> +   } else {
> +      tmp_ubyte = malloc(width * height * sizeof(*tmp_ubyte));
> +
> +      if (src_format_is_mesa_array_format) {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_swizzle_and_convert(tmp_ubyte + row * width,
> GL_UNSIGNED_BYTE, 4,
> +                                      src, src_gl_type, src_num_channels,
> +                                      src2rgba, normalized, width);
> +            src += src_stride;
> +         }
> +      } else {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_unpack_ubyte_rgba_row(src_format, width,
> +                                        src, tmp_ubyte + row * width);
> +            src += src_stride;
> +         }
> +      }
> +
> +      if (dst_format_is_mesa_array_format) {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_swizzle_and_convert(dst, dst_gl_type, dst_num_channels,
> +                                      tmp_ubyte + row * width,
> GL_UNSIGNED_BYTE, 4,
> +                                      rgba2dst, normalized, width);
> +            dst += dst_stride;
> +         }
> +      } else {
> +         for (row = 0; row < height; ++row) {
> +            _mesa_pack_ubyte_rgba_row(dst_format, width,
> +                                      (const uint8_t (*)[4])tmp_ubyte +
> row * width, dst);
> +            dst += dst_stride;
> +         }
> +      }
> +
> +      free(tmp_ubyte);
> +   }
> +}
>
>  static const uint8_t map_identity[7] = { 0, 1, 2, 3, 4, 5, 6 };
>  static const uint8_t map_3210[7] = { 3, 2, 1, 0, 4, 5, 6 };
> diff --git a/src/mesa/main/format_utils.h b/src/mesa/main/format_utils.h
> index 4e63200..f006e18 100644
> --- a/src/mesa/main/format_utils.h
> +++ b/src/mesa/main/format_utils.h
> @@ -34,6 +34,11 @@
>  #include "imports.h"
>  #include "macros.h"
>
> +extern const mesa_array_format RGBA8888_FLOAT;
> +extern const mesa_array_format RGBA8888_UBYTE;
> +extern const mesa_array_format RGBA8888_UINT;
> +extern const mesa_array_format RGBA8888_INT;
> +
>  /* Only guaranteed to work for BITS <= 32 */
>  #define MAX_UINT(BITS) ((BITS) == 32 ? UINT32_MAX : ((1u << (BITS)) - 1))
>  #define MAX_INT(BITS) ((int)MAX_UINT((BITS) - 1))
> @@ -207,4 +212,9 @@ _mesa_swizzle_and_convert(void *dst, GLenum dst_type,
> int num_dst_channels,
>                            const void *src, GLenum src_type, int
> num_src_channels,
>                            const uint8_t swizzle[4], bool normalized, int
> count);
>
> +void
> +_mesa_format_convert(void *void_dst, uint32_t dst_format, size_t
> dst_stride,
> +                     void *void_src, uint32_t src_format, size_t
> src_stride,
> +                     size_t width, size_t height);
> +
>  #endif
> --
> 1.9.1
>
> _______________________________________________
> mesa-dev mailing list
> mesa-dev at lists.freedesktop.org
> http://lists.freedesktop.org/mailman/listinfo/mesa-dev
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.freedesktop.org/archives/mesa-dev/attachments/20150108/5bacf661/attachment-0001.html>


More information about the mesa-dev mailing list