• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**************************************************************************
2  *
3  * Copyright 2009 VMware, Inc.
4  * All Rights Reserved.
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining a
7  * copy of this software and associated documentation files (the
8  * "Software"), to deal in the Software without restriction, including
9  * without limitation the rights to use, copy, modify, merge, publish,
10  * distribute, sub license, and/or sell copies of the Software, and to
11  * permit persons to whom the Software is furnished to do so, subject to
12  * the following conditions:
13  *
14  * The above copyright notice and this permission notice (including the
15  * next paragraph) shall be included in all copies or substantial portions
16  * of the Software.
17  *
18  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
21  * IN NO EVENT SHALL VMWARE AND/OR ITS SUPPLIERS BE LIABLE FOR
22  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23  * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24  * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
25  *
26  **************************************************************************/
27 
28 
29 #include "pipe/p_defines.h"
30 
31 #include "util/format/u_format.h"
32 #include "util/u_memory.h"
33 #include "util/u_string.h"
34 #include "util/u_math.h"
35 
36 #include "lp_bld_type.h"
37 #include "lp_bld_const.h"
38 #include "lp_bld_conv.h"
39 #include "lp_bld_swizzle.h"
40 #include "lp_bld_gather.h"
41 #include "lp_bld_debug.h"
42 #include "lp_bld_format.h"
43 #include "lp_bld_arit.h"
44 #include "lp_bld_pack.h"
45 #include "lp_bld_flow.h"
46 #include "lp_bld_printf.h"
47 #include "lp_bld_intr.h"
48 
49 static void
convert_to_soa(struct gallivm_state * gallivm,LLVMValueRef src_aos[LP_MAX_VECTOR_WIDTH/32],LLVMValueRef dst_soa[4],const struct lp_type soa_type)50 convert_to_soa(struct gallivm_state *gallivm,
51                LLVMValueRef src_aos[LP_MAX_VECTOR_WIDTH / 32],
52                LLVMValueRef dst_soa[4],
53                const struct lp_type soa_type)
54 {
55    unsigned j, k;
56    struct lp_type aos_channel_type = soa_type;
57 
58    LLVMValueRef aos_channels[4];
59    unsigned pixels_per_channel = soa_type.length / 4;
60 
61    debug_assert((soa_type.length % 4) == 0);
62 
63    aos_channel_type.length >>= 1;
64 
65    for (j = 0; j < 4; ++j) {
66       LLVMValueRef channel[LP_MAX_VECTOR_LENGTH] = { 0 };
67 
68       assert(pixels_per_channel <= LP_MAX_VECTOR_LENGTH);
69 
70       for (k = 0; k < pixels_per_channel; ++k) {
71          channel[k] = src_aos[j + 4 * k];
72       }
73 
74       aos_channels[j] = lp_build_concat(gallivm, channel, aos_channel_type, pixels_per_channel);
75    }
76 
77    lp_build_transpose_aos(gallivm, soa_type, aos_channels, dst_soa);
78 }
79 
80 
81 void
lp_build_format_swizzle_soa(const struct util_format_description * format_desc,struct lp_build_context * bld,const LLVMValueRef unswizzled[4],LLVMValueRef swizzled_out[4])82 lp_build_format_swizzle_soa(const struct util_format_description *format_desc,
83                             struct lp_build_context *bld,
84                             const LLVMValueRef unswizzled[4],
85                             LLVMValueRef swizzled_out[4])
86 {
87    if (format_desc->colorspace == UTIL_FORMAT_COLORSPACE_ZS) {
88       enum pipe_swizzle swizzle;
89       LLVMValueRef depth_or_stencil;
90 
91       if (util_format_has_stencil(format_desc) &&
92           !util_format_has_depth(format_desc)) {
93          assert(!bld->type.floating);
94          swizzle = format_desc->swizzle[1];
95       }
96       else {
97          assert(bld->type.floating);
98          swizzle = format_desc->swizzle[0];
99       }
100       /*
101        * Return zzz1 or sss1 for depth-stencil formats here.
102        * Correct swizzling will be handled by apply_sampler_swizzle() later.
103        */
104       depth_or_stencil = lp_build_swizzle_soa_channel(bld, unswizzled, swizzle);
105 
106       swizzled_out[2] = swizzled_out[1] = swizzled_out[0] = depth_or_stencil;
107       swizzled_out[3] = bld->one;
108    }
109    else {
110       unsigned chan;
111       for (chan = 0; chan < 4; ++chan) {
112          enum pipe_swizzle swizzle = format_desc->swizzle[chan];
113          swizzled_out[chan] = lp_build_swizzle_soa_channel(bld, unswizzled, swizzle);
114       }
115    }
116 }
117 
118 
119 
120 static LLVMValueRef
lp_build_extract_soa_chan(struct lp_build_context * bld,unsigned blockbits,boolean srgb_chan,struct util_format_channel_description chan_desc,LLVMValueRef packed)121 lp_build_extract_soa_chan(struct lp_build_context *bld,
122                           unsigned blockbits,
123                           boolean srgb_chan,
124                           struct util_format_channel_description chan_desc,
125                           LLVMValueRef packed)
126 {
127    struct gallivm_state *gallivm = bld->gallivm;
128    LLVMBuilderRef builder = gallivm->builder;
129    struct lp_type type = bld->type;
130    LLVMValueRef input = packed;
131    const unsigned width = chan_desc.size;
132    const unsigned start = chan_desc.shift;
133    const unsigned stop = start + width;
134 
135    /* Decode the input vector component */
136 
137    switch(chan_desc.type) {
138    case UTIL_FORMAT_TYPE_VOID:
139       input = bld->undef;
140       break;
141 
142    case UTIL_FORMAT_TYPE_UNSIGNED:
143       /*
144        * Align the LSB
145        */
146       if (start) {
147          input = LLVMBuildLShr(builder, input,
148                                lp_build_const_int_vec(gallivm, type, start), "");
149       }
150 
151       /*
152        * Zero the MSBs
153        */
154       if (stop < blockbits) {
155          unsigned mask = ((unsigned long long)1 << width) - 1;
156          input = LLVMBuildAnd(builder, input,
157                               lp_build_const_int_vec(gallivm, type, mask), "");
158       }
159 
160       /*
161        * Type conversion
162        */
163       if (type.floating) {
164          if (srgb_chan) {
165             struct lp_type conv_type = lp_uint_type(type);
166             input = lp_build_srgb_to_linear(gallivm, conv_type, width, input);
167          }
168          else {
169             if(chan_desc.normalized)
170                input = lp_build_unsigned_norm_to_float(gallivm, width, type, input);
171             else
172                input = LLVMBuildUIToFP(builder, input, bld->vec_type, "");
173          }
174       }
175       else if (chan_desc.pure_integer) {
176          /* Nothing to do */
177       } else {
178           /* FIXME */
179           assert(0);
180       }
181       break;
182 
183    case UTIL_FORMAT_TYPE_SIGNED:
184       /*
185        * Align the sign bit first.
186        */
187       if (stop < type.width) {
188          unsigned bits = type.width - stop;
189          LLVMValueRef bits_val = lp_build_const_int_vec(gallivm, type, bits);
190          input = LLVMBuildShl(builder, input, bits_val, "");
191       }
192 
193       /*
194        * Align the LSB (with an arithmetic shift to preserve the sign)
195        */
196       if (chan_desc.size < type.width) {
197          unsigned bits = type.width - chan_desc.size;
198          LLVMValueRef bits_val = lp_build_const_int_vec(gallivm, type, bits);
199          input = LLVMBuildAShr(builder, input, bits_val, "");
200       }
201 
202       /*
203        * Type conversion
204        */
205       if (type.floating) {
206          input = LLVMBuildSIToFP(builder, input, bld->vec_type, "");
207          if (chan_desc.normalized) {
208             double scale = 1.0 / ((1 << (chan_desc.size - 1)) - 1);
209             LLVMValueRef scale_val = lp_build_const_vec(gallivm, type, scale);
210             input = LLVMBuildFMul(builder, input, scale_val, "");
211             /*
212              * The formula above will produce value below -1.0 for most negative values.
213              * compliance requires clamping it.
214              * GTF-GL45.gtf33.GL3Tests.vertex_type_2_10_10_10_rev.vertex_type_2_10_10_10_rev_conversion.
215              */
216             input = lp_build_max(bld, input,
217                                  lp_build_const_vec(gallivm, type, -1.0f));
218          }
219       }
220       else if (chan_desc.pure_integer) {
221          /* Nothing to do */
222       } else {
223           /* FIXME */
224           assert(0);
225       }
226       break;
227 
228    case UTIL_FORMAT_TYPE_FLOAT:
229       if (type.floating) {
230          if (chan_desc.size == 16) {
231             struct lp_type f16i_type = type;
232             f16i_type.width /= 2;
233             f16i_type.floating = 0;
234             if (start) {
235                input = LLVMBuildLShr(builder, input,
236                                      lp_build_const_int_vec(gallivm, type, start), "");
237             }
238             input = LLVMBuildTrunc(builder, input,
239                                    lp_build_vec_type(gallivm, f16i_type), "");
240             input = lp_build_half_to_float(gallivm, input);
241          } else {
242             assert(start == 0);
243             assert(stop == 32);
244             assert(type.width == 32);
245          }
246          input = LLVMBuildBitCast(builder, input, bld->vec_type, "");
247       }
248       else {
249          /* FIXME */
250          assert(0);
251          input = bld->undef;
252       }
253       break;
254 
255    case UTIL_FORMAT_TYPE_FIXED:
256       if (type.floating) {
257          double scale = 1.0 / ((1 << (chan_desc.size/2)) - 1);
258          LLVMValueRef scale_val = lp_build_const_vec(gallivm, type, scale);
259          input = LLVMBuildSIToFP(builder, input, bld->vec_type, "");
260          input = LLVMBuildFMul(builder, input, scale_val, "");
261       }
262       else {
263          /* FIXME */
264          assert(0);
265          input = bld->undef;
266       }
267       break;
268 
269    default:
270       assert(0);
271       input = bld->undef;
272       break;
273    }
274 
275    return input;
276 }
277 
278 
279 /**
280  * Unpack several pixels in SoA.
281  *
282  * It takes a vector of packed pixels:
283  *
284  *   packed = {P0, P1, P2, P3, ..., Pn}
285  *
286  * And will produce four vectors:
287  *
288  *   red    = {R0, R1, R2, R3, ..., Rn}
289  *   green  = {G0, G1, G2, G3, ..., Gn}
290  *   blue   = {B0, B1, B2, B3, ..., Bn}
291  *   alpha  = {A0, A1, A2, A3, ..., An}
292  *
293  * It requires that a packed pixel fits into an element of the output
294  * channels. The common case is when converting pixel with a depth of 32 bit or
295  * less into floats.
296  *
297  * \param format_desc  the format of the 'packed' incoming pixel vector
298  * \param type  the desired type for rgba_out (type.length = n, above)
299  * \param packed  the incoming vector of packed pixels
300  * \param rgba_out  returns the SoA R,G,B,A vectors
301  */
302 void
lp_build_unpack_rgba_soa(struct gallivm_state * gallivm,const struct util_format_description * format_desc,struct lp_type type,LLVMValueRef packed,LLVMValueRef rgba_out[4])303 lp_build_unpack_rgba_soa(struct gallivm_state *gallivm,
304                          const struct util_format_description *format_desc,
305                          struct lp_type type,
306                          LLVMValueRef packed,
307                          LLVMValueRef rgba_out[4])
308 {
309    struct lp_build_context bld;
310    LLVMValueRef inputs[4];
311    unsigned chan;
312 
313    assert(format_desc->layout == UTIL_FORMAT_LAYOUT_PLAIN);
314    assert(format_desc->block.width == 1);
315    assert(format_desc->block.height == 1);
316    assert(format_desc->block.bits <= type.width);
317    /* FIXME: Support more output types */
318    assert(type.width == 32);
319 
320    lp_build_context_init(&bld, gallivm, type);
321 
322    /* Decode the input vector components */
323    for (chan = 0; chan < format_desc->nr_channels; ++chan) {
324       struct util_format_channel_description chan_desc = format_desc->channel[chan];
325       boolean srgb_chan = FALSE;
326 
327       if (format_desc->colorspace == UTIL_FORMAT_COLORSPACE_SRGB &&
328           format_desc->swizzle[3] != chan) {
329          srgb_chan = TRUE;
330       }
331 
332       inputs[chan] = lp_build_extract_soa_chan(&bld,
333                                                format_desc->block.bits,
334                                                srgb_chan,
335                                                chan_desc,
336                                                packed);
337    }
338 
339    lp_build_format_swizzle_soa(format_desc, &bld, inputs, rgba_out);
340 }
341 
342 
343 /**
344  * Convert a vector of rgba8 values into 32bit wide SoA vectors.
345  *
346  * \param dst_type  The desired return type. For pure integer formats
347  *                  this should be a 32bit wide int or uint vector type,
348  *                  otherwise a float vector type.
349  *
350  * \param packed    The rgba8 values to pack.
351  *
352  * \param rgba      The 4 SoA return vectors.
353  */
354 void
lp_build_rgba8_to_fi32_soa(struct gallivm_state * gallivm,struct lp_type dst_type,LLVMValueRef packed,LLVMValueRef * rgba)355 lp_build_rgba8_to_fi32_soa(struct gallivm_state *gallivm,
356                            struct lp_type dst_type,
357                            LLVMValueRef packed,
358                            LLVMValueRef *rgba)
359 {
360    LLVMBuilderRef builder = gallivm->builder;
361    LLVMValueRef mask = lp_build_const_int_vec(gallivm, dst_type, 0xff);
362    unsigned chan;
363 
364    /* XXX technically shouldn't use that for uint dst_type */
365    packed = LLVMBuildBitCast(builder, packed,
366                              lp_build_int_vec_type(gallivm, dst_type), "");
367 
368    /* Decode the input vector components */
369    for (chan = 0; chan < 4; ++chan) {
370 #if UTIL_ARCH_LITTLE_ENDIAN
371       unsigned start = chan*8;
372 #else
373       unsigned start = (3-chan)*8;
374 #endif
375       unsigned stop = start + 8;
376       LLVMValueRef input;
377 
378       input = packed;
379 
380       if (start)
381          input = LLVMBuildLShr(builder, input,
382                                lp_build_const_int_vec(gallivm, dst_type, start), "");
383 
384       if (stop < 32)
385          input = LLVMBuildAnd(builder, input, mask, "");
386 
387       if (dst_type.floating)
388          input = lp_build_unsigned_norm_to_float(gallivm, 8, dst_type, input);
389 
390       rgba[chan] = input;
391    }
392 }
393 
394 
395 
396 /**
397  * Fetch a texels from a texture, returning them in SoA layout.
398  *
399  * \param type  the desired return type for 'rgba'.  The vector length
400  *              is the number of texels to fetch
401  * \param aligned if the offset is guaranteed to be aligned to element width
402  *
403  * \param base_ptr  points to the base of the texture mip tree.
404  * \param offset    offset to start of the texture image block.  For non-
405  *                  compressed formats, this simply is an offset to the texel.
406  *                  For compressed formats, it is an offset to the start of the
407  *                  compressed data block.
408  *
409  * \param i, j  the sub-block pixel coordinates.  For non-compressed formats
410  *              these will always be (0,0).  For compressed formats, i will
411  *              be in [0, block_width-1] and j will be in [0, block_height-1].
412  * \param cache  optional value pointing to a lp_build_format_cache structure
413  */
414 void
lp_build_fetch_rgba_soa(struct gallivm_state * gallivm,const struct util_format_description * format_desc,struct lp_type type,boolean aligned,LLVMValueRef base_ptr,LLVMValueRef offset,LLVMValueRef i,LLVMValueRef j,LLVMValueRef cache,LLVMValueRef rgba_out[4])415 lp_build_fetch_rgba_soa(struct gallivm_state *gallivm,
416                         const struct util_format_description *format_desc,
417                         struct lp_type type,
418                         boolean aligned,
419                         LLVMValueRef base_ptr,
420                         LLVMValueRef offset,
421                         LLVMValueRef i,
422                         LLVMValueRef j,
423                         LLVMValueRef cache,
424                         LLVMValueRef rgba_out[4])
425 {
426    LLVMBuilderRef builder = gallivm->builder;
427    enum pipe_format format = format_desc->format;
428    struct lp_type fetch_type;
429 
430    if (format_desc->layout == UTIL_FORMAT_LAYOUT_PLAIN &&
431        (format_desc->colorspace == UTIL_FORMAT_COLORSPACE_RGB ||
432         format_desc->colorspace == UTIL_FORMAT_COLORSPACE_SRGB ||
433         format_desc->colorspace == UTIL_FORMAT_COLORSPACE_ZS) &&
434        format_desc->block.width == 1 &&
435        format_desc->block.height == 1 &&
436        format_desc->block.bits <= type.width &&
437        (format_desc->channel[0].type != UTIL_FORMAT_TYPE_FLOAT ||
438         format_desc->channel[0].size == 32 ||
439         format_desc->channel[0].size == 16))
440    {
441       /*
442        * The packed pixel fits into an element of the destination format. Put
443        * the packed pixels into a vector and extract each component for all
444        * vector elements in parallel.
445        */
446 
447       LLVMValueRef packed;
448 
449       /*
450        * gather the texels from the texture
451        * Ex: packed = {XYZW, XYZW, XYZW, XYZW}
452        */
453       assert(format_desc->block.bits <= type.width);
454       fetch_type = lp_type_uint(type.width);
455       packed = lp_build_gather(gallivm,
456                                type.length,
457                                format_desc->block.bits,
458                                fetch_type,
459                                aligned,
460                                base_ptr, offset, FALSE);
461 
462       /*
463        * convert texels to float rgba
464        */
465       lp_build_unpack_rgba_soa(gallivm,
466                                format_desc,
467                                type,
468                                packed, rgba_out);
469       return;
470    }
471 
472 
473    if (format_desc->layout == UTIL_FORMAT_LAYOUT_PLAIN &&
474        (format_desc->colorspace == UTIL_FORMAT_COLORSPACE_RGB) &&
475        format_desc->block.width == 1 &&
476        format_desc->block.height == 1 &&
477        format_desc->block.bits > type.width &&
478        ((format_desc->block.bits <= type.width * type.length &&
479          format_desc->channel[0].size <= type.width) ||
480         (format_desc->channel[0].size == 64 &&
481          format_desc->channel[0].type == UTIL_FORMAT_TYPE_FLOAT &&
482          type.floating)))
483    {
484       /*
485        * Similar to above, but the packed pixel is larger than what fits
486        * into an element of the destination format. The packed pixels will be
487        * shuffled into SoA vectors appropriately, and then the extraction will
488        * be done in parallel as much as possible.
489        * Good for 16xn (n > 2) and 32xn (n > 1) formats, care is taken so
490        * the gathered vectors can be shuffled easily (even with avx).
491        * 64xn float -> 32xn float is handled too but it's a bit special as
492        * it does the conversion pre-shuffle.
493        */
494 
495       LLVMValueRef packed[4], dst[4], output[4], shuffles[LP_MAX_VECTOR_WIDTH/32];
496       struct lp_type fetch_type, gather_type = type;
497       unsigned num_gather, fetch_width, i, j;
498       struct lp_build_context bld;
499       boolean fp64 = format_desc->channel[0].size == 64;
500 
501       lp_build_context_init(&bld, gallivm, type);
502 
503       assert(type.width == 32);
504       assert(format_desc->block.bits > type.width);
505 
506       /*
507        * First, figure out fetch order.
508        */
509       fetch_width = util_next_power_of_two(format_desc->block.bits);
510       /*
511        * fp64 are treated like fp32 except we fetch twice wide values
512        * (as we shuffle after trunc). The shuffles for that work out
513        * mostly fine (slightly suboptimal for 4-wide, perfect for AVX)
514        * albeit we miss the potential opportunity for hw gather (as it
515        * only handles native size).
516        */
517       num_gather = fetch_width / type.width;
518       gather_type.width *= num_gather;
519       if (fp64) {
520          num_gather /= 2;
521       }
522       gather_type.length /= num_gather;
523 
524       for (i = 0; i < num_gather; i++) {
525          LLVMValueRef offsetr, shuf_vec;
526          if(num_gather == 4) {
527             for (j = 0; j < gather_type.length; j++) {
528                unsigned idx = i + 4*j;
529                shuffles[j] = lp_build_const_int32(gallivm, idx);
530             }
531             shuf_vec = LLVMConstVector(shuffles, gather_type.length);
532             offsetr = LLVMBuildShuffleVector(builder, offset, offset, shuf_vec, "");
533 
534          }
535          else if (num_gather == 2) {
536             assert(num_gather == 2);
537             for (j = 0; j < gather_type.length; j++) {
538                unsigned idx = i*2 + (j%2) + (j/2)*4;
539                shuffles[j] = lp_build_const_int32(gallivm, idx);
540             }
541             shuf_vec = LLVMConstVector(shuffles, gather_type.length);
542             offsetr = LLVMBuildShuffleVector(builder, offset, offset, shuf_vec, "");
543          }
544          else {
545             assert(num_gather == 1);
546             offsetr = offset;
547          }
548          if (gather_type.length == 1) {
549             LLVMValueRef zero = lp_build_const_int32(gallivm, 0);
550             offsetr = LLVMBuildExtractElement(builder, offsetr, zero, "");
551          }
552 
553          /*
554           * Determine whether to use float or int loads. This is mostly
555           * to outsmart the (stupid) llvm int/float shuffle logic, we
556           * don't really care much if the data is floats or ints...
557           * But llvm will refuse to use single float shuffle with int data
558           * and instead use 3 int shuffles instead, the code looks atrocious.
559           * (Note bitcasts often won't help, as llvm is too smart to be
560           * fooled by that.)
561           * Nobody cares about simd float<->int domain transition penalties,
562           * which usually don't even exist for shuffles anyway.
563           * With 4x32bit (and 3x32bit) fetch, we use float vec (the data is
564           * going into transpose, which is unpacks, so doesn't really matter
565           * much).
566           * With 2x32bit or 4x16bit fetch, we use float vec, since those
567           * go into the weird channel separation shuffle. With floats,
568           * this is (with 128bit vectors):
569           * - 2 movq, 2 movhpd, 2 shufps
570           * With ints it would be:
571           * - 4 movq, 2 punpcklqdq, 4 pshufd, 2 blendw
572           * I've seen texture functions increase in code size by 15% just due
573           * to that (there's lots of such fetches in them...)
574           * (We could chose a different gather order to improve this somewhat
575           * for the int path, but it would basically just drop the blends,
576           * so the float path with this order really is optimal.)
577           * Albeit it is tricky sometimes llvm doesn't ignore the float->int
578           * casts so must avoid them until we're done with the float shuffle...
579           * 3x16bit formats (the same is also true for 3x8) are pretty bad but
580           * there's nothing we can do about them (we could overallocate by
581           * those couple bytes and use unaligned but pot sized load).
582           * Note that this is very much x86 specific. I don't know if this
583           * affect other archs at all.
584           */
585          if (num_gather > 1) {
586             /*
587              * We always want some float type here (with x86)
588              * due to shuffles being float ones afterwards (albeit for
589              * the num_gather == 4 case int should work fine too
590              * (unless there's some problems with avx but not avx2).
591              */
592             if (format_desc->channel[0].size == 64) {
593                fetch_type = lp_type_float_vec(64, gather_type.width);
594             } else {
595                fetch_type = lp_type_int_vec(32, gather_type.width);
596             }
597          }
598          else {
599             /* type doesn't matter much */
600             if (format_desc->channel[0].type == UTIL_FORMAT_TYPE_FLOAT &&
601                 (format_desc->channel[0].size == 32 ||
602                  format_desc->channel[0].size == 64)) {
603             fetch_type = lp_type_float(gather_type.width);
604             } else {
605                fetch_type = lp_type_uint(gather_type.width);
606             }
607          }
608 
609          /* Now finally gather the values */
610          packed[i] = lp_build_gather(gallivm, gather_type.length,
611                                      format_desc->block.bits,
612                                      fetch_type, aligned,
613                                      base_ptr, offsetr, FALSE);
614          if (fp64) {
615             struct lp_type conv_type = type;
616             conv_type.width *= 2;
617             packed[i] = LLVMBuildBitCast(builder, packed[i],
618                                          lp_build_vec_type(gallivm, conv_type), "");
619             packed[i] = LLVMBuildFPTrunc(builder, packed[i], bld.vec_type, "");
620          }
621       }
622 
623       /* shuffle the gathered values to SoA */
624       if (num_gather == 2) {
625          for (i = 0; i < num_gather; i++) {
626             for (j = 0; j < type.length; j++) {
627                unsigned idx = (j%2)*2 + (j/4)*4 + i;
628                if ((j/2)%2)
629                   idx += type.length;
630                shuffles[j] = lp_build_const_int32(gallivm, idx);
631             }
632             dst[i] = LLVMBuildShuffleVector(builder, packed[0], packed[1],
633                                             LLVMConstVector(shuffles, type.length), "");
634          }
635       }
636       else if (num_gather == 4) {
637          lp_build_transpose_aos(gallivm, lp_int_type(type), packed, dst);
638       }
639       else {
640          assert(num_gather == 1);
641          dst[0] = packed[0];
642       }
643 
644       /*
645        * And finally unpack exactly as above, except that
646        * chan shift is adjusted and the right vector selected.
647        */
648       if (!fp64) {
649          for (i = 0; i < num_gather; i++) {
650             dst[i] = LLVMBuildBitCast(builder, dst[i], bld.int_vec_type, "");
651          }
652          for (i = 0; i < format_desc->nr_channels; i++) {
653             struct util_format_channel_description chan_desc = format_desc->channel[i];
654             unsigned blockbits = type.width;
655             unsigned vec_nr;
656 
657 #if UTIL_ARCH_BIG_ENDIAN
658             vec_nr = (format_desc->block.bits - (chan_desc.shift + chan_desc.size)) / type.width;
659 #else
660             vec_nr = chan_desc.shift / type.width;
661 #endif
662             chan_desc.shift %= type.width;
663 
664             output[i] = lp_build_extract_soa_chan(&bld,
665                                                   blockbits,
666                                                   FALSE,
667                                                   chan_desc,
668                                                   dst[vec_nr]);
669          }
670       }
671       else {
672          for (i = 0; i < format_desc->nr_channels; i++)  {
673             output[i] = dst[i];
674          }
675       }
676 
677       lp_build_format_swizzle_soa(format_desc, &bld, output, rgba_out);
678       return;
679    }
680 
681    if (format == PIPE_FORMAT_R11G11B10_FLOAT ||
682        format == PIPE_FORMAT_R9G9B9E5_FLOAT) {
683       /*
684        * similar conceptually to above but requiring special
685        * AoS packed -> SoA float conversion code.
686        */
687       LLVMValueRef packed;
688       struct lp_type fetch_type = lp_type_uint(type.width);
689 
690       assert(type.floating);
691       assert(type.width == 32);
692 
693       packed = lp_build_gather(gallivm, type.length,
694                                format_desc->block.bits,
695                                fetch_type, aligned,
696                                base_ptr, offset, FALSE);
697       if (format == PIPE_FORMAT_R11G11B10_FLOAT) {
698          lp_build_r11g11b10_to_float(gallivm, packed, rgba_out);
699       }
700       else {
701          lp_build_rgb9e5_to_float(gallivm, packed, rgba_out);
702       }
703       return;
704    }
705 
706    if (format_desc->colorspace == UTIL_FORMAT_COLORSPACE_ZS &&
707        format_desc->block.bits == 64) {
708       /*
709        * special case the format is 64 bits but we only require
710        * 32bit (or 8bit) from each block.
711        */
712       LLVMValueRef packed;
713       struct lp_type fetch_type = lp_type_uint(type.width);
714 
715       if (format == PIPE_FORMAT_X32_S8X24_UINT) {
716          /*
717           * for stencil simply fix up offsets - could in fact change
718           * base_ptr instead even outside the shader.
719           */
720          unsigned mask = (1 << 8) - 1;
721          LLVMValueRef s_offset = lp_build_const_int_vec(gallivm, type, 4);
722          offset = LLVMBuildAdd(builder, offset, s_offset, "");
723          packed = lp_build_gather(gallivm, type.length, 32, fetch_type,
724                                   aligned, base_ptr, offset, FALSE);
725          packed = LLVMBuildAnd(builder, packed,
726                                lp_build_const_int_vec(gallivm, type, mask), "");
727       }
728       else {
729          assert (format == PIPE_FORMAT_Z32_FLOAT_S8X24_UINT);
730          packed = lp_build_gather(gallivm, type.length, 32, fetch_type,
731                                   aligned, base_ptr, offset, TRUE);
732          packed = LLVMBuildBitCast(builder, packed,
733                                    lp_build_vec_type(gallivm, type), "");
734       }
735       /* for consistency with lp_build_unpack_rgba_soa() return sss1 or zzz1 */
736       rgba_out[0] = rgba_out[1] = rgba_out[2] = packed;
737       rgba_out[3] = lp_build_const_vec(gallivm, type, 1.0f);
738       return;
739    }
740 
741    /*
742     * Try calling lp_build_fetch_rgba_aos for all pixels.
743     * Should only really hit subsampled, compressed
744     * (for s3tc srgb and rgtc too).
745     * (This is invalid for plain 8unorm formats because we're lazy with
746     * the swizzle since some results would arrive swizzled, some not.)
747     */
748 
749    if ((format_desc->layout != UTIL_FORMAT_LAYOUT_PLAIN) &&
750        (util_format_fits_8unorm(format_desc) ||
751         format_desc->layout == UTIL_FORMAT_LAYOUT_RGTC ||
752         format_desc->layout == UTIL_FORMAT_LAYOUT_S3TC) &&
753        type.floating && type.width == 32 &&
754        (type.length == 1 || (type.length % 4 == 0))) {
755       struct lp_type tmp_type;
756       struct lp_build_context bld;
757       LLVMValueRef packed, rgba[4];
758       const struct util_format_description *flinear_desc;
759       const struct util_format_description *frgba8_desc;
760       unsigned chan;
761       bool is_signed = (format_desc->format == PIPE_FORMAT_RGTC1_SNORM ||
762                         format_desc->format == PIPE_FORMAT_RGTC2_SNORM ||
763                         format_desc->format == PIPE_FORMAT_LATC1_SNORM ||
764                         format_desc->format == PIPE_FORMAT_LATC2_SNORM);
765 
766       lp_build_context_init(&bld, gallivm, type);
767 
768       /*
769        * Make sure the conversion in aos really only does convert to rgba8
770        * and not anything more (so use linear format, adjust type).
771        */
772       flinear_desc = util_format_description(util_format_linear(format));
773       memset(&tmp_type, 0, sizeof tmp_type);
774       tmp_type.width = 8;
775       tmp_type.length = type.length * 4;
776       tmp_type.norm = TRUE;
777       tmp_type.sign = is_signed;
778 
779       packed = lp_build_fetch_rgba_aos(gallivm, flinear_desc, tmp_type,
780                                        aligned, base_ptr, offset, i, j, cache);
781       packed = LLVMBuildBitCast(builder, packed, bld.int_vec_type, "");
782 
783       /*
784        * The values are now packed so they match ordinary (srgb) RGBA8 format,
785        * hence need to use matching format for unpack.
786        */
787       frgba8_desc = util_format_description(is_signed ? PIPE_FORMAT_R8G8B8A8_SNORM : PIPE_FORMAT_R8G8B8A8_UNORM);
788       if (format_desc->colorspace == UTIL_FORMAT_COLORSPACE_SRGB) {
789          assert(format_desc->layout == UTIL_FORMAT_LAYOUT_S3TC);
790          frgba8_desc = util_format_description(PIPE_FORMAT_R8G8B8A8_SRGB);
791       }
792       lp_build_unpack_rgba_soa(gallivm,
793                                frgba8_desc,
794                                type,
795                                packed, rgba);
796 
797       /*
798        * We converted 4 channels. Make sure llvm can drop unneeded ones
799        * (luckily the rgba order is fixed, only LA needs special case).
800        */
801       for (chan = 0; chan < 4; chan++) {
802          enum pipe_swizzle swizzle = format_desc->swizzle[chan];
803          if (chan == 3 && util_format_is_luminance_alpha(format)) {
804             swizzle = PIPE_SWIZZLE_W;
805          }
806          rgba_out[chan] = lp_build_swizzle_soa_channel(&bld, rgba, swizzle);
807       }
808       return;
809    }
810 
811 
812    /*
813     * Fallback to calling lp_build_fetch_rgba_aos for each pixel.
814     *
815     * This is not the most efficient way of fetching pixels, as we
816     * miss some opportunities to do vectorization, but this is
817     * convenient for formats or scenarios for which there was no
818     * opportunity or incentive to optimize.
819     *
820     * We do NOT want to end up here, this typically is quite terrible,
821     * in particular if the formats have less than 4 channels.
822     *
823     * Right now, this should only be hit for:
824     * - ETC formats
825     *   (those miss fast fetch functions hence they are terrible anyway)
826     */
827 
828    {
829       unsigned k;
830       struct lp_type tmp_type;
831       LLVMValueRef aos_fetch[LP_MAX_VECTOR_WIDTH / 32];
832 
833       if (gallivm_debug & GALLIVM_DEBUG_PERF) {
834          debug_printf("%s: AoS fetch fallback for %s\n",
835                       __FUNCTION__, format_desc->short_name);
836       }
837 
838       tmp_type = type;
839       tmp_type.length = 4;
840 
841       if (type.length == 1) {
842          LLVMValueRef fetch = lp_build_fetch_rgba_aos(gallivm, format_desc, tmp_type,
843                                                       aligned, base_ptr, offset,
844                                                       i, j, cache);
845 
846          for (k = 0; k < 4; k++)
847             rgba_out[k] = LLVMBuildExtractElement(gallivm->builder, fetch, lp_build_const_int32(gallivm, k), "");
848          return;
849       }
850 
851       /*
852        * Note that vector transpose can be worse compared to insert/extract
853        * for aos->soa conversion (for formats with 1 or 2 channels). However,
854        * we should try to avoid getting here for just about all formats, so
855        * don't bother.
856        */
857 
858       /* loop over number of pixels */
859       for(k = 0; k < type.length; ++k) {
860          LLVMValueRef index = lp_build_const_int32(gallivm, k);
861          LLVMValueRef offset_elem;
862          LLVMValueRef i_elem, j_elem;
863 
864          offset_elem = LLVMBuildExtractElement(builder, offset,
865                                                index, "");
866 
867          i_elem = LLVMBuildExtractElement(builder, i, index, "");
868          j_elem = LLVMBuildExtractElement(builder, j, index, "");
869 
870          /* Get a single float[4]={R,G,B,A} pixel */
871          aos_fetch[k] = lp_build_fetch_rgba_aos(gallivm, format_desc, tmp_type,
872                                                 aligned, base_ptr, offset_elem,
873                                                 i_elem, j_elem, cache);
874 
875       }
876       convert_to_soa(gallivm, aos_fetch, rgba_out, type);
877    }
878 }
879 
880 static void
lp_build_insert_soa_chan(struct lp_build_context * bld,unsigned blockbits,struct util_format_channel_description chan_desc,LLVMValueRef * output,LLVMValueRef rgba)881 lp_build_insert_soa_chan(struct lp_build_context *bld,
882                          unsigned blockbits,
883                          struct util_format_channel_description chan_desc,
884                          LLVMValueRef *output,
885                          LLVMValueRef rgba)
886 {
887     struct gallivm_state *gallivm = bld->gallivm;
888     LLVMBuilderRef builder = gallivm->builder;
889     struct lp_type type = bld->type;
890     const unsigned width = chan_desc.size;
891     const unsigned start = chan_desc.shift;
892     const uint32_t chan_mask = (1ULL << width) - 1;
893     ASSERTED const unsigned stop = start + width;
894     LLVMValueRef chan = NULL;
895     switch(chan_desc.type) {
896     case UTIL_FORMAT_TYPE_UNSIGNED:
897 
898        if (chan_desc.pure_integer) {
899           chan = LLVMBuildBitCast(builder, rgba, bld->int_vec_type, "");
900           LLVMValueRef mask_val = lp_build_const_int_vec(gallivm, type, chan_mask);
901           LLVMValueRef mask = LLVMBuildICmp(builder, LLVMIntUGT, chan, mask_val, "");
902           chan = LLVMBuildSelect(builder, mask, mask_val, chan, "");
903        }
904        else if (type.floating) {
905           if (chan_desc.normalized) {
906              rgba = lp_build_clamp(bld, rgba, bld->zero, bld->one);
907              chan = lp_build_clamped_float_to_unsigned_norm(gallivm, type, width, rgba);
908           } else
909              chan = LLVMBuildFPToSI(builder, rgba, bld->vec_type, "");
910        }
911        if (start)
912           chan = LLVMBuildShl(builder, chan,
913                               lp_build_const_int_vec(gallivm, type, start), "");
914        if (!*output)
915           *output = chan;
916        else
917           *output = LLVMBuildOr(builder, *output, chan, "");
918        break;
919     case UTIL_FORMAT_TYPE_SIGNED:
920        if (chan_desc.pure_integer) {
921           chan = LLVMBuildBitCast(builder, rgba, bld->int_vec_type, "");
922           /* clamp to SINT range for < 32-bit values */
923           if (width < 32) {
924              struct lp_build_context int_bld;
925              lp_build_context_init(&int_bld, gallivm, lp_int_type(bld->type));
926              chan = lp_build_clamp(&int_bld, chan,
927                                    lp_build_const_int_vec(gallivm, type, -(1ULL << (width - 1))),
928                                    lp_build_const_int_vec(gallivm, type, (1ULL << (width - 1)) - 1));
929              chan = LLVMBuildAnd(builder, chan, lp_build_const_int_vec(gallivm, type, chan_mask), "");
930           }
931        } else if (type.floating) {
932           if (chan_desc.normalized) {
933              char intrin[32];
934              double scale = ((1 << (chan_desc.size - 1)) - 1);
935              LLVMValueRef scale_val = lp_build_const_vec(gallivm, type, scale);
936              rgba = lp_build_clamp(bld, rgba, lp_build_negate(bld, bld->one), bld->one);
937              rgba = LLVMBuildFMul(builder, rgba, scale_val, "");
938              lp_format_intrinsic(intrin, sizeof intrin, "llvm.rint", bld->vec_type);
939              rgba = lp_build_intrinsic_unary(builder, intrin, bld->vec_type, rgba);
940           }
941           chan = LLVMBuildFPToSI(builder, rgba, bld->int_vec_type, "");
942           chan = LLVMBuildAnd(builder, chan, lp_build_const_int_vec(gallivm, type, chan_mask), "");
943        }
944        if (start)
945           chan = LLVMBuildShl(builder, chan,
946                               lp_build_const_int_vec(gallivm, type, start), "");
947        if (!*output)
948           *output = chan;
949        else
950           *output = LLVMBuildOr(builder, *output, chan, "");
951        break;
952     case UTIL_FORMAT_TYPE_FLOAT:
953        if (type.floating) {
954           if (chan_desc.size == 16) {
955              chan = lp_build_float_to_half(gallivm, rgba);
956              chan = LLVMBuildBitCast(builder, chan,
957 				     lp_build_vec_type(gallivm, lp_type_int_vec(16, 16 * type.length)), "");
958              chan = LLVMBuildZExt(builder, chan, bld->int_vec_type, "");
959              if (start)
960                 chan = LLVMBuildShl(builder, chan,
961                                     lp_build_const_int_vec(gallivm, type, start), "");
962              if (!*output)
963                 *output = chan;
964              else
965                 *output = LLVMBuildOr(builder, *output, chan, "");
966           } else {
967              assert(start == 0);
968              assert(stop == 32);
969              assert(type.width == 32);
970              *output = LLVMBuildBitCast(builder, rgba, bld->int_vec_type, "");
971           }
972        } else
973           assert(0);
974        break;
975     default:
976        assert(0);
977        *output = bld->undef;
978     }
979 }
980 
981 static void
lp_build_pack_rgba_soa(struct gallivm_state * gallivm,const struct util_format_description * format_desc,struct lp_type type,const LLVMValueRef rgba_in[4],LLVMValueRef * packed)982 lp_build_pack_rgba_soa(struct gallivm_state *gallivm,
983                        const struct util_format_description *format_desc,
984                        struct lp_type type,
985                        const LLVMValueRef rgba_in[4],
986                        LLVMValueRef *packed)
987 {
988    unsigned chan;
989    struct lp_build_context bld;
990    assert(format_desc->layout == UTIL_FORMAT_LAYOUT_PLAIN);
991    assert(format_desc->block.width == 1);
992    assert(format_desc->block.height == 1);
993    assert(format_desc->block.bits <= type.width);
994    /* FIXME: Support more output types */
995    assert(type.width == 32);
996 
997    lp_build_context_init(&bld, gallivm, type);
998    for (chan = 0; chan < format_desc->nr_channels; ++chan) {
999       struct util_format_channel_description chan_desc = format_desc->channel[chan];
1000 
1001       lp_build_insert_soa_chan(&bld, format_desc->block.bits,
1002                                chan_desc,
1003                                packed,
1004                                rgba_in[chan]);
1005    }
1006 }
1007 
1008 void
lp_build_store_rgba_soa(struct gallivm_state * gallivm,const struct util_format_description * format_desc,struct lp_type type,LLVMValueRef exec_mask,LLVMValueRef base_ptr,LLVMValueRef offset,LLVMValueRef out_of_bounds,const LLVMValueRef rgba_in[4])1009 lp_build_store_rgba_soa(struct gallivm_state *gallivm,
1010                         const struct util_format_description *format_desc,
1011                         struct lp_type type,
1012                         LLVMValueRef exec_mask,
1013                         LLVMValueRef base_ptr,
1014                         LLVMValueRef offset,
1015                         LLVMValueRef out_of_bounds,
1016                         const LLVMValueRef rgba_in[4])
1017 {
1018    enum pipe_format format = format_desc->format;
1019    LLVMValueRef packed[4];
1020    unsigned num_stores = 0;
1021 
1022    memset(packed, 0, sizeof(LLVMValueRef) * 4);
1023    if (format_desc->layout == UTIL_FORMAT_LAYOUT_PLAIN &&
1024        format_desc->colorspace == UTIL_FORMAT_COLORSPACE_RGB &&
1025        format_desc->block.width == 1 &&
1026        format_desc->block.height == 1 &&
1027        format_desc->block.bits <= type.width &&
1028        (format_desc->channel[0].type != UTIL_FORMAT_TYPE_FLOAT ||
1029         format_desc->channel[0].size == 32 ||
1030         format_desc->channel[0].size == 16))
1031    {
1032       lp_build_pack_rgba_soa(gallivm, format_desc, type, rgba_in, &packed[0]);
1033 
1034       num_stores = 1;
1035    } else if (format_desc->layout == UTIL_FORMAT_LAYOUT_PLAIN &&
1036        (format_desc->colorspace == UTIL_FORMAT_COLORSPACE_RGB) &&
1037        format_desc->block.width == 1 &&
1038        format_desc->block.height == 1 &&
1039        format_desc->block.bits > type.width &&
1040        ((format_desc->block.bits <= type.width * type.length &&
1041          format_desc->channel[0].size <= type.width) ||
1042         (format_desc->channel[0].size == 64 &&
1043          format_desc->channel[0].type == UTIL_FORMAT_TYPE_FLOAT &&
1044          type.floating)))
1045    {
1046       /*
1047        * Similar to above, but the packed pixel is larger than what fits
1048        * into an element of the destination format. The packed pixels will be
1049        * shuffled into SoA vectors appropriately, and then the extraction will
1050        * be done in parallel as much as possible.
1051        * Good for 16xn (n > 2) and 32xn (n > 1) formats, care is taken so
1052        * the gathered vectors can be shuffled easily (even with avx).
1053        * 64xn float -> 32xn float is handled too but it's a bit special as
1054        * it does the conversion pre-shuffle.
1055        */
1056       struct lp_build_context bld;
1057 
1058       lp_build_context_init(&bld, gallivm, type);
1059       assert(type.width == 32);
1060       assert(format_desc->block.bits > type.width);
1061 
1062       unsigned store_width = util_next_power_of_two(format_desc->block.bits);
1063       num_stores = store_width / type.width;
1064       for (unsigned i = 0; i < format_desc->nr_channels; i++) {
1065             struct util_format_channel_description chan_desc = format_desc->channel[i];
1066             unsigned blockbits = type.width;
1067             unsigned vec_nr;
1068 
1069             vec_nr = chan_desc.shift / type.width;
1070             chan_desc.shift %= type.width;
1071 
1072             lp_build_insert_soa_chan(&bld, blockbits,
1073                                      chan_desc,
1074                                      &packed[vec_nr],
1075                                      rgba_in[i]);
1076       }
1077 
1078       assert(num_stores == 4 || num_stores == 2);
1079       /* we can transpose and store at the same time */
1080    } else if (format == PIPE_FORMAT_R11G11B10_FLOAT) {
1081       packed[0] = lp_build_float_to_r11g11b10(gallivm, rgba_in);
1082       num_stores = 1;
1083    } else
1084       assert(0);
1085 
1086    assert(exec_mask);
1087 
1088    LLVMTypeRef int32_ptr_type = LLVMPointerType(LLVMInt32TypeInContext(gallivm->context), 0);
1089    LLVMTypeRef int16_ptr_type = LLVMPointerType(LLVMInt16TypeInContext(gallivm->context), 0);
1090    LLVMTypeRef int8_ptr_type = LLVMPointerType(LLVMInt8TypeInContext(gallivm->context), 0);
1091 
1092    LLVMValueRef should_store_mask = LLVMBuildAnd(gallivm->builder, exec_mask, LLVMBuildNot(gallivm->builder, out_of_bounds, ""), "store_mask");
1093    should_store_mask = LLVMBuildICmp(gallivm->builder, LLVMIntNE, should_store_mask, lp_build_const_int_vec(gallivm, type, 0), "");
1094    for (unsigned i = 0; i < num_stores; i++) {
1095       struct lp_build_loop_state loop_state;
1096 
1097       LLVMValueRef store_offset = LLVMBuildAdd(gallivm->builder, offset, lp_build_const_int_vec(gallivm, type, i * 4), "");
1098       store_offset = LLVMBuildGEP(gallivm->builder, base_ptr, &store_offset, 1, "");
1099 
1100       lp_build_loop_begin(&loop_state, gallivm, lp_build_const_int32(gallivm, 0));
1101 
1102       struct lp_build_if_state ifthen;
1103       LLVMValueRef cond = LLVMBuildExtractElement(gallivm->builder, should_store_mask, loop_state.counter, "");
1104       lp_build_if(&ifthen, gallivm, cond);
1105 
1106       LLVMValueRef data = LLVMBuildExtractElement(gallivm->builder, packed[i], loop_state.counter, "");
1107       LLVMValueRef this_offset = LLVMBuildExtractElement(gallivm->builder, store_offset, loop_state.counter, "");
1108 
1109       if (format_desc->block.bits == 8) {
1110          this_offset = LLVMBuildBitCast(gallivm->builder, this_offset, int8_ptr_type, "");
1111          data = LLVMBuildTrunc(gallivm->builder, data, LLVMInt8TypeInContext(gallivm->context), "");
1112       } else if (format_desc->block.bits == 16) {
1113          this_offset = LLVMBuildBitCast(gallivm->builder, this_offset, int16_ptr_type, "");
1114          data = LLVMBuildTrunc(gallivm->builder, data, LLVMInt16TypeInContext(gallivm->context), "");
1115       } else
1116          this_offset = LLVMBuildBitCast(gallivm->builder, this_offset, int32_ptr_type, "");
1117       LLVMBuildStore(gallivm->builder, data, this_offset);
1118       lp_build_endif(&ifthen);
1119       lp_build_loop_end_cond(&loop_state, lp_build_const_int32(gallivm, type.length),
1120                              NULL, LLVMIntUGE);
1121    }
1122 }
1123