<div dir="ltr"><div class="gmail_extra"><div class="gmail_quote">On Tue, Jun 27, 2017 at 4:29 PM, Chad Versace <span dir="ltr"><<a href="mailto:chadversary@chromium.org" target="_blank">chadversary@chromium.org</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div class="HOEnZb"><div class="h5">On Fri 23 Jun 2017, <a href="mailto:alexandros.frantzis@collabora.com">alexandros.frantzis@collabora.<wbr>com</a> wrote:<br>
> From: Alexandros Frantzis <<a href="mailto:alexandros.frantzis@collabora.com">alexandros.frantzis@<wbr>collabora.com</a>><br>
><br>
> Introduce utilities to describe, find and compare Vulkan formats based<br>
> on their color component masks, taking into account system endianness.<br>
> ---<br>
>  src/vulkan/Makefile.sources      |   2 +<br>
>  src/vulkan/util/vk_format_<wbr>util.c | 173 ++++++++++++++++++++++++++++++<wbr>+++++++++<br>
>  src/vulkan/util/vk_format_<wbr>util.h | 100 ++++++++++++++++++++++<br>
>  3 files changed, 275 insertions(+)<br>
>  create mode 100644 src/vulkan/util/vk_format_<wbr>util.c<br>
>  create mode 100644 src/vulkan/util/vk_format_<wbr>util.h<br>
><br>
> diff --git a/src/vulkan/Makefile.sources b/src/vulkan/Makefile.sources<br>
> index 2cf7218e92..f4f1f14d5a 100644<br>
> --- a/src/vulkan/Makefile.sources<br>
> +++ b/src/vulkan/Makefile.sources<br>
> @@ -17,6 +17,8 @@ VULKAN_WSI_X11_FILES := \<br>
><br>
>  VULKAN_UTIL_FILES := \<br>
>       util/vk_alloc.h \<br>
> +     util/vk_format_util.c \<br>
> +     util/vk_format_util.h \<br>
>       util/vk_util.c \<br>
>       util/vk_util.h<br>
><br>
> diff --git a/src/vulkan/util/vk_format_<wbr>util.c b/src/vulkan/util/vk_format_<wbr>util.c<br>
> new file mode 100644<br>
> index 0000000000..aa6d403e84<br>
> --- /dev/null<br>
> +++ b/src/vulkan/util/vk_format_<wbr>util.c<br>
> @@ -0,0 +1,173 @@<br>
> +/*<br>
> + * Copyright © 2017 Collabora Ltd.<br>
> + *<br>
> + * Permission is hereby granted, free of charge, to any person obtaining a<br>
> + * copy of this software and associated documentation files (the "Software"),<br>
> + * to deal in the Software without restriction, including without limitation<br>
> + * the rights to use, copy, modify, merge, publish, distribute, sublicense,<br>
> + * and/or sell copies of the Software, and to permit persons to whom the<br>
> + * Software is furnished to do so, subject to the following conditions:<br>
> + *<br>
> + * The above copyright notice and this permission notice (including the next<br>
> + * paragraph) shall be included in all copies or substantial portions of the<br>
> + * Software.<br>
> + *<br>
> + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR<br>
> + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,<br>
> + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL<br>
> + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER<br>
> + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING<br>
> + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS<br>
> + * IN THE SOFTWARE.<br>
> + */<br>
> +<br>
> +#include "vk_format_util.h"<br>
> +<br>
> +#include "util/u_vector.h"<br>
> +#include "util/u_math.h"<br>
> +<br>
> +#include <assert.h><br>
> +<br>
> +static inline uint32_t<br>
> +bswap24(uint32_t n)<br>
> +{<br>
> +   return (n & 0xff000000) |<br>
> +          ((n >> 16) & 0x000000ff) |<br>
> +          (n & 0x0000ff00) |<br>
> +          ((n << 16) & 0x00ff0000);<br>
> +}<br>
> +<br>
> +static inline uint32_t<br>
> +bswap(uint32_t n, int bits)<br>
> +{<br>
> +   switch (bits)<br>
> +   {<br>
<br>
</div></div>Mesa code-style puts the '{' on the same line as 'switch'.<br>
<span class=""><br>
> +   case 8:<br>
> +      return n;<br>
> +   case 16:<br>
> +      return util_bswap16(n);<br>
> +   case 24:<br>
> +      return bswap24(n);<br>
> +   case 32:<br>
> +      return util_bswap32(n);<br>
> +   default:<br>
> +      assert(!"Invalid bswap bits");<br>
> +      return n;<br>
<br>
</span>If someone passes an invalid param here, there's little point in<br>
continuing, even in release builds, because continuing will produce only<br>
undefined behavior. So let's optimize the release build by removing the<br>
default case, like this:<br>
<br>
    bswap(...)<br>
    {<br>
        switch (bits) {<br>
        case 8:<br>
            ...<br>
        case 16:<br>
            ...<br>
        case 32:<br>
            ...<br>
        }<br>
<br>
        unreachable("invalid bswap bits");<br>
<span class="">    }<br>
<br>
> +   }<br>
> +}<br>
> +<br>
> +struct format_spec {<br>
> +   VkFormat format;<br>
> +   struct vk_format_util_spec spec;<br>
> +};<br>
> +<br>
> +#define VF(FMT, BPP, R, G, B, A, E, S) \<br>
> +   {VK_FORMAT_##FMT, {BPP, R, G, B, A, VK_FORMAT_UTIL_ENDIANNESS_##E, VK_FORMAT_UTIL_##S}}<br>
> +<br>
> +// SRGB formats are presented to the user before the UNORM ones.<br>
<br>
</span>The comment should explain why the sRGB formats precede the others.<br>
<br>
Also, Mesa code-style uses /* comments, not // comments.<br>
<div><div class="h5"><br>
> +static const struct format_spec format_specs[] = {<br>
> +   VF(R8_SRGB, 8, 0x000000ff, 0x00000000, 0x00000000, 0x00000000, BIG, SWAPPABLE),<br>
> +   VF(R8G8_SRGB, 16, 0x0000ff00, 0x000000ff, 0x00000000, 0x00000000, BIG, SWAPPABLE),<br>
> +   VF(R8G8B8_SRGB, 24, 0x00ff0000, 0x0000ff00, 0x000000ff, 0x00000000, BIG, SWAPPABLE),<br>
> +   VF(B8G8R8_SRGB, 24, 0x000000ff, 0x0000ff00, 0x00ff0000, 0x00000000, BIG, SWAPPABLE),<br>
> +   VF(R8G8B8A8_SRGB, 32, 0xff000000, 0x00ff0000, 0x0000ff00, 0x000000ff, BIG, SWAPPABLE),<br>
> +   VF(B8G8R8A8_SRGB, 32, 0x0000ff00, 0x00ff0000, 0xff000000, 0x000000ff, BIG, SWAPPABLE),<br>
> +   VF(A8B8G8R8_SRGB_PACK32, 32, 0x000000ff, 0x0000ff00, 0x00ff0000, 0xff000000, NATIVE, SWAPPABLE),<br>
> +<br>
> +   VF(R8_UNORM, 8, 0x000000ff, 0x00000000, 0x00000000, 0x00000000, BIG, SWAPPABLE),<br>
> +   VF(R8G8_UNORM, 16, 0x0000ff00, 0x000000ff, 0x00000000, 0x00000000, BIG, SWAPPABLE),<br>
> +   VF(R8G8B8_UNORM, 24, 0x00ff0000, 0x0000ff00, 0x000000ff, 0x00000000, BIG, SWAPPABLE),<br>
> +   VF(B8G8R8_UNORM, 24, 0x000000ff, 0x0000ff00, 0x00ff0000, 0x00000000, BIG, SWAPPABLE),<br>
> +   VF(R8G8B8A8_UNORM, 32, 0xff000000, 0x00ff0000, 0x0000ff00, 0x000000ff, BIG, SWAPPABLE),<br>
> +   VF(B8G8R8A8_UNORM, 32, 0x0000ff00, 0x00ff0000, 0xff000000, 0x000000ff, BIG, SWAPPABLE),<br>
> +   VF(R4G4_UNORM_PACK8, 8, 0x000000f0, 0x0000000f, 0x00000000, 0x00000000, NATIVE, NON_SWAPPABLE),<br>
> +   VF(R4G4B4A4_UNORM_PACK16, 16, 0x0000f000, 0x00000f00, 0x000000f0, 0x0000000f, NATIVE, NON_SWAPPABLE),<br>
> +   VF(B4G4R4A4_UNORM_PACK16, 16, 0x000000f0, 0x00000f00, 0x0000f000, 0x0000000f, NATIVE, NON_SWAPPABLE),<br>
> +   VF(R5G6B5_UNORM_PACK16, 16, 0x0000f800, 0x000007e0, 0x0000001f, 0x00000000, NATIVE, NON_SWAPPABLE),<br>
> +   VF(B5G6R5_UNORM_PACK16, 16, 0x0000001f, 0x000007e0, 0x0000f800, 0x00000000, NATIVE, NON_SWAPPABLE),<br>
> +   VF(R5G5B5A1_UNORM_PACK16, 16, 0x0000f800, 0x000007c0, 0x0000003e, 0x00000001, NATIVE, NON_SWAPPABLE),<br>
> +   VF(B5G5R5A1_UNORM_PACK16, 16, 0x0000003e, 0x000007c0, 0x0000f800, 0x00000001, NATIVE, NON_SWAPPABLE),<br>
> +   VF(A1R5G5B5_UNORM_PACK16, 16, 0x00007c00, 0x000003e0, 0x0000001f, 0x00008000, NATIVE, NON_SWAPPABLE),<br>
> +   VF(A8B8G8R8_UNORM_PACK32, 32, 0x000000ff, 0x0000ff00, 0x00ff0000, 0xff000000, NATIVE, SWAPPABLE),<br>
> +   VF(A2R10G10B10_UNORM_PACK32, 32, 0x3ff00000, 0x000ffc00, 0x000003ff, 0xc0000000, NATIVE, NON_SWAPPABLE),<br>
> +   VF(A2B10G10R10_UNORM_PACK32, 32, 0x000003ff, 0x000ffc00, 0x3ff00000, 0xc0000000, NATIVE, NON_SWAPPABLE),<br>
> +};<br>
<br>
</div></div>The concepts of "swappable" and "non-swappable", as applied to formats<br>
here, is not a concept present in the Vulkan spec.<br>
<br>
I'm not sure what swappable/non-swappable encodes in this table, and why<br>
vk_compare_format_specs() is trying to do with swappability. I tried<br>
reverse-engineering the patch's intent, and formed the following<br>
observations and questions:<br>
<br>
    1. The table partitions the formats into 3 classes:<br>
<br>
        a. Big-endian and swappable. All non-packed formats belong to<br>
           this class.<br>
<br>
        b. Native-endian and non-swappable. All packed formats, except<br>
           VK_FORMAT_A8B8G8R8_SRGB_PACK32 and<br>
           VK_FORMAT_A8B8G8R8_UNORM_<wbr>PACK32, belong to this class.<br>
<br>
        c. Native-endian and swappable. Exactly<br>
           VK_FORMAT_A8B8G8R8_SRGB_PACK32 and<br>
           VK_FORMAT_A8B8G8R8_UNORM_<wbr>PACK32 belong to this class.<br>
<br>
    2. Why are no formats marked as little-endian?<br>
<br>
    3. Why are the A8B8G8R8 formats special? Because each channel is<br>
       byte-aligned?<br>
<br>
I suspect that the table is actually trying to capture the<br>
distinction between "packed" and "non-packed" formats, as defined<br>
by the Vulkan spec, in section 32.2.1 Format Definition. (Some people<br>
say "array" formats instead of "non-packed" formats). (The Vulkan spec<br>
also defines the concept of "block" formats, but that's irrelevant to<br>
this patch).<br>
<br>
I don't believe the endianness scheme in this patch is not able to capture the<br>
bit-layout of VK_FORMAT_R16G16B16A16_UNORM, but the Vulkan spec's notion<br>
of "packed/non-packed" is. According to table "Byte mappings for<br>
non-packed/compressed color formats" in the Vulkan 1.0.51 spec,<br>
VK_FORMAT_R16G16B16A16_UNORM is a non-packed format where the red channel<br>
is bytes 0-1, green is bytes 2-3, blue is bytes 4-5, etc, but the bytes<br>
within each channel are ordered by host endianness.<br></blockquote><div><br></div><div>Drive-by: Yes, I agree that this doesn't match Vulkan particularly well.  It's a very X11 or drm_fourcc way of described formats and is nothing like anything else we have in mesa.  That doesn't mean it's strictly wrong, just that it's awkward to people who are used to some of the other mechanisms for describing formats.<br></div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
I don't understand how swappable/non-swappable apply to<br>
VK_FORMAT_R16G16B16A16_UNORM.<br>
<span class=""><br>
> +<br>
> +#undef VF<br>
> +<br>
> +static inline vk_format_util_endianness<br>
> +native_endianness()<br>
<br>
</span>In C, the empty parameter list is not what you want. The compiler<br>
interprets the empty parameter parameter list as an old-style variadic<br>
parameter list.  You want native_endianness(void) instead.<br>
<br>
Anyway, this function should be replaced by the macros in<br>
src/util/u_endian.h.<br>
<span class=""><br>
> +{<br>
> +   const unsigned ui = 1;<br>
> +   return *((const char *) &ui) == 1 ? VK_FORMAT_UTIL_ENDIANNESS_<wbr>LITTLE :<br>
> +                                       VK_FORMAT_UTIL_ENDIANNESS_BIG;<br>
> +}<br>
> +<br>
> +void<br>
> +vk_get_format_spec(VkFormat format,<br>
> +                   struct vk_format_util_spec *spec)<br>
> +{<br>
> +   static const struct vk_format_util_spec empty_spec =<br>
> +      {0, 0x0, 0x0, 0x0, 0x0, VK_FORMAT_UTIL_ENDIANNESS_<wbr>NATIVE, VK_FORMAT_UTIL_NON_SWAPPABLE};<br>
<br>
</span>The verbose struct initializer should be replaced with `empty_spec = {0};`<br>
The C99 spec guarantees that `= {0}` produces a zero-initialized struct<br>
here.<br>
<div><div class="h5"><br>
> +<br>
> +   for (size_t i = 0; i < ARRAY_SIZE(format_specs); i++) {<br>
> +      if (format_specs[i].format == format) {<br>
> +         *spec = format_specs[i].spec;<br>
> +         return;<br>
> +      }<br>
> +   }<br>
> +<br>
> +   *spec = empty_spec;<br>
> +}<br>
> +<br>
> +bool<br>
> +vk_compare_format_specs(const struct vk_format_util_spec* spec1,<br>
> +                        const struct vk_format_util_spec* spec2)<br>
> +{<br>
> +   const vk_format_util_endianness endianness1 =<br>
> +      spec1->endianness == VK_FORMAT_UTIL_ENDIANNESS_<wbr>NATIVE ?<br>
> +      native_endianness() : spec1->endianness;<br>
> +<br>
> +   const vk_format_util_endianness endianness2 =<br>
> +      spec2->endianness == VK_FORMAT_UTIL_ENDIANNESS_<wbr>NATIVE ?<br>
> +      native_endianness() : spec2->endianness;<br>
> +<br>
> +   if (endianness1 == endianness2) {<br>
> +      return spec1->bits_per_pixel == spec2->bits_per_pixel &&<br>
> +             spec1->red_mask == spec2->red_mask &&<br>
> +             spec1->green_mask == spec2->green_mask &&<br>
> +             spec1->blue_mask == spec2->blue_mask &&<br>
> +             spec1->alpha_mask == spec2->alpha_mask;<br>
> +   }<br>
> +   else {<br>
<br>
</div></div>I don't understand why the 'else' branch exists. But that's likely due<br>
to my confusion regarding swappable/non-swappable, explained above.<br>
<br>
Also, Mesa code-style is `} else {`.<br>
<div><div class="h5"><br>
> +      if (spec1->swappable == VK_FORMAT_UTIL_SWAPPABLE) {<br>
> +         return spec1->bits_per_pixel == spec2->bits_per_pixel &&<br>
> +                bswap(spec1->red_mask, spec1->bits_per_pixel) == spec2->red_mask &&<br>
> +                bswap(spec1->green_mask, spec1->bits_per_pixel) == spec2->green_mask &&<br>
> +                bswap(spec1->blue_mask, spec1->bits_per_pixel) == spec2->blue_mask &&<br>
> +                bswap(spec1->alpha_mask, spec1->bits_per_pixel) == spec2->alpha_mask;<br>
> +      }<br>
> +      else if (spec2->swappable == VK_FORMAT_UTIL_SWAPPABLE) {<br>
> +         return spec1->bits_per_pixel == spec2->bits_per_pixel &&<br>
> +                spec1->red_mask == bswap(spec2->red_mask, spec2->bits_per_pixel) &&<br>
> +                spec1->green_mask == bswap(spec2->green_mask, spec2->bits_per_pixel) &&<br>
> +                spec1->blue_mask == bswap(spec2->blue_mask, spec2->bits_per_pixel) &&<br>
> +                spec1->alpha_mask == bswap(spec2->alpha_mask, spec2->bits_per_pixel);<br>
> +      }<br>
> +   }<br>
> +<br>
> +   return false;<br>
> +}<br>
> +<br>
> +void<br>
> +vk_find_matching_formats(<wbr>const struct vk_format_util_spec *spec,<br>
> +                         struct u_vector *formats)<br>
> +{<br>
> +   for (size_t i = 0; i < ARRAY_SIZE(format_specs); i++) {<br>
> +      if (vk_compare_format_specs(spec, &format_specs[i].spec)) {<br>
> +         VkFormat *f = u_vector_add(formats);<br>
> +         if (f)<br>
> +            *f = format_specs[i].format;<br>
> +      }<br>
> +   }<br>
> +}<br>
> diff --git a/src/vulkan/util/vk_format_<wbr>util.h b/src/vulkan/util/vk_format_<wbr>util.h<br>
> new file mode 100644<br>
> index 0000000000..a1f9d87abe<br>
> --- /dev/null<br>
> +++ b/src/vulkan/util/vk_format_<wbr>util.h<br>
> @@ -0,0 +1,100 @@<br>
> +/*<br>
> + * Copyright © 2017 Collabora Ltd.<br>
> + *<br>
> + * Permission is hereby granted, free of charge, to any person obtaining a<br>
> + * copy of this software and associated documentation files (the "Software"),<br>
> + * to deal in the Software without restriction, including without limitation<br>
> + * the rights to use, copy, modify, merge, publish, distribute, sublicense,<br>
> + * and/or sell copies of the Software, and to permit persons to whom the<br>
> + * Software is furnished to do so, subject to the following conditions:<br>
> + *<br>
> + * The above copyright notice and this permission notice (including the next<br>
> + * paragraph) shall be included in all copies or substantial portions of the<br>
> + * Software.<br>
> + *<br>
> + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR<br>
> + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,<br>
> + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL<br>
> + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER<br>
> + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING<br>
> + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS<br>
> + * IN THE SOFTWARE.<br>
> + */<br>
> +<br>
> +#ifndef VK_FORMAT_UTIL_H<br>
> +#define VK_FORMAT_UTIL_H<br>
> +<br>
> +#include <vulkan/vulkan.h><br>
> +#include <stdbool.h><br>
> +<br>
> +struct u_vector;<br>
> +<br>
> +typedef enum vk_format_util_endianness {<br>
> +   VK_FORMAT_UTIL_ENDIANNESS_<wbr>NATIVE,<br>
> +   VK_FORMAT_UTIL_ENDIANNESS_<wbr>LITTLE,<br>
> +   VK_FORMAT_UTIL_ENDIANNESS_BIG,<br>
> +} vk_format_util_endianness;<br>
> +<br>
> +typedef enum vk_format_util_swappable {<br>
> +   VK_FORMAT_UTIL_NON_SWAPPABLE,<br>
> +   VK_FORMAT_UTIL_SWAPPABLE,<br>
> +} vk_format_util_swappable;<br>
> +<br>
> +/*<br>
> + * Structure describing a pixel format.<br>
> + *<br>
> + * The masks express the bit positions of the components in an native integer<br>
> + * storing a pixel of this format.<br>
> + *<br>
> + * Examples:<br>
> + *<br>
> + *   * A pixel format which is always stored as a 0xR8G8B8A8 32-bit integer:<br>
> + *<br>
> + *     {32, 0xff000000, 0x00ff0000, 0x0000ff00, 0x000000ff, NATIVE, SWAPPABLE}<br>
> + *<br>
> + *   * A pixel format which is always stored as R8,G8,B8,A8 in memory order,<br>
> + *     with R at the lowest address:<br>
> + *<br>
> + *     {32, 0xff000000, 0x00ff0000, 0x0000ff00, 0x000000ff, BIG, SWAPPABLE}<br>
> + *     or<br>
> + *     {32, 0x000000ff, 0x0000ff00, 0x00ff0000, 0xff000000, LITTLE, SWAPPABLE}<br>
> + *<br>
> + *   * A pixel format which is always stored as a 0xR5G6B5 16-bit integer,<br>
> + *     and cannot change endianness by byte-swapping:<br>
> + *<br>
> + *     {16, 0x0000f800, 0x000007e0, 0x0000001f, 0x00000000, NATIVE, NON_SWAPPABLE},<br>
> + *<br>
> + */<br>
> +struct vk_format_util_spec {<br>
> +   int bits_per_pixel;<br>
> +   uint32_t red_mask;<br>
> +   uint32_t green_mask;<br>
> +   uint32_t blue_mask;<br>
> +   uint32_t alpha_mask;<br>
> +   /* The endianness for which the masks are correct */<br>
> +   vk_format_util_endianness endianness;<br>
> +   /* Whether the masks can be byte-swapped to switch endianness */<br>
> +   vk_format_util_swappable swappable;<br>
> +};<br>
> +<br>
> +<br>
> +/* Fill a vk_format_util_spec structure to describe a Vulkan format. */<br>
> +void<br>
> +vk_get_format_spec(VkFormat format,<br>
> +                   struct vk_format_util_spec *spec);<br>
> +<br>
> +/* Compare two pixel formats described by vk_format_util_spec structures. */<br>
> +bool<br>
> +vk_compare_format_specs(const struct vk_format_util_spec *spec1,<br>
> +                        const struct vk_format_util_spec *spec2);<br>
> +<br>
> +/*<br>
> + * Find Vulkan pixel formats that match a vk_format_util_spec.<br>
> + *<br>
> + * Note that, at the moment, only UNORM and SRGB formats are considered.<br>
> + */<br>
> +void<br>
> +vk_find_matching_formats(<wbr>const struct vk_format_util_spec *spec,<br>
> +                         struct u_vector *formats);<br>
> +<br>
> +#endif<br>
> --<br>
> 2.11.0<br>
><br>
</div></div>> ______________________________<wbr>_________________<br>
> mesa-dev mailing list<br>
> <a href="mailto:mesa-dev@lists.freedesktop.org">mesa-dev@lists.freedesktop.org</a><br>
> <a href="https://lists.freedesktop.org/mailman/listinfo/mesa-dev" rel="noreferrer" target="_blank">https://lists.freedesktop.org/<wbr>mailman/listinfo/mesa-dev</a><br>
______________________________<wbr>_________________<br>
mesa-dev mailing list<br>
<a href="mailto:mesa-dev@lists.freedesktop.org">mesa-dev@lists.freedesktop.org</a><br>
<a href="https://lists.freedesktop.org/mailman/listinfo/mesa-dev" rel="noreferrer" target="_blank">https://lists.freedesktop.org/<wbr>mailman/listinfo/mesa-dev</a><br>
</blockquote></div><br></div></div>