xref: /aosp_15_r20/external/mesa3d/src/panfrost/compiler/bir.c (revision 6104692788411f58d303aa86923a9ff6ecaded22)
1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker  * Copyright (C) 2020 Collabora Ltd.
3*61046927SAndroid Build Coastguard Worker  *
4*61046927SAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a
5*61046927SAndroid Build Coastguard Worker  * copy of this software and associated documentation files (the "Software"),
6*61046927SAndroid Build Coastguard Worker  * to deal in the Software without restriction, including without limitation
7*61046927SAndroid Build Coastguard Worker  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*61046927SAndroid Build Coastguard Worker  * and/or sell copies of the Software, and to permit persons to whom the
9*61046927SAndroid Build Coastguard Worker  * Software is furnished to do so, subject to the following conditions:
10*61046927SAndroid Build Coastguard Worker  *
11*61046927SAndroid Build Coastguard Worker  * The above copyright notice and this permission notice (including the next
12*61046927SAndroid Build Coastguard Worker  * paragraph) shall be included in all copies or substantial portions of the
13*61046927SAndroid Build Coastguard Worker  * Software.
14*61046927SAndroid Build Coastguard Worker  *
15*61046927SAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*61046927SAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*61046927SAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18*61046927SAndroid Build Coastguard Worker  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*61046927SAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20*61046927SAndroid Build Coastguard Worker  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21*61046927SAndroid Build Coastguard Worker  * SOFTWARE.
22*61046927SAndroid Build Coastguard Worker  *
23*61046927SAndroid Build Coastguard Worker  * Authors (Collabora):
24*61046927SAndroid Build Coastguard Worker  *      Alyssa Rosenzweig <[email protected]>
25*61046927SAndroid Build Coastguard Worker  */
26*61046927SAndroid Build Coastguard Worker 
27*61046927SAndroid Build Coastguard Worker #include "bi_builder.h"
28*61046927SAndroid Build Coastguard Worker #include "compiler.h"
29*61046927SAndroid Build Coastguard Worker 
30*61046927SAndroid Build Coastguard Worker bool
bi_has_arg(const bi_instr * ins,bi_index arg)31*61046927SAndroid Build Coastguard Worker bi_has_arg(const bi_instr *ins, bi_index arg)
32*61046927SAndroid Build Coastguard Worker {
33*61046927SAndroid Build Coastguard Worker    if (!ins)
34*61046927SAndroid Build Coastguard Worker       return false;
35*61046927SAndroid Build Coastguard Worker 
36*61046927SAndroid Build Coastguard Worker    bi_foreach_src(ins, s) {
37*61046927SAndroid Build Coastguard Worker       if (bi_is_equiv(ins->src[s], arg))
38*61046927SAndroid Build Coastguard Worker          return true;
39*61046927SAndroid Build Coastguard Worker    }
40*61046927SAndroid Build Coastguard Worker 
41*61046927SAndroid Build Coastguard Worker    return false;
42*61046927SAndroid Build Coastguard Worker }
43*61046927SAndroid Build Coastguard Worker 
44*61046927SAndroid Build Coastguard Worker /* Precondition: valid 16-bit or 32-bit register format. Returns whether it is
45*61046927SAndroid Build Coastguard Worker  * 32-bit. Note auto reads to 32-bit registers even if the memory format is
46*61046927SAndroid Build Coastguard Worker  * 16-bit, so is considered as such here */
47*61046927SAndroid Build Coastguard Worker 
48*61046927SAndroid Build Coastguard Worker bool
bi_is_regfmt_16(enum bi_register_format fmt)49*61046927SAndroid Build Coastguard Worker bi_is_regfmt_16(enum bi_register_format fmt)
50*61046927SAndroid Build Coastguard Worker {
51*61046927SAndroid Build Coastguard Worker    switch (fmt) {
52*61046927SAndroid Build Coastguard Worker    case BI_REGISTER_FORMAT_F16:
53*61046927SAndroid Build Coastguard Worker    case BI_REGISTER_FORMAT_S16:
54*61046927SAndroid Build Coastguard Worker    case BI_REGISTER_FORMAT_U16:
55*61046927SAndroid Build Coastguard Worker       return true;
56*61046927SAndroid Build Coastguard Worker    case BI_REGISTER_FORMAT_F32:
57*61046927SAndroid Build Coastguard Worker    case BI_REGISTER_FORMAT_S32:
58*61046927SAndroid Build Coastguard Worker    case BI_REGISTER_FORMAT_U32:
59*61046927SAndroid Build Coastguard Worker    case BI_REGISTER_FORMAT_AUTO:
60*61046927SAndroid Build Coastguard Worker       return false;
61*61046927SAndroid Build Coastguard Worker    default:
62*61046927SAndroid Build Coastguard Worker       unreachable("Invalid register format");
63*61046927SAndroid Build Coastguard Worker    }
64*61046927SAndroid Build Coastguard Worker }
65*61046927SAndroid Build Coastguard Worker 
66*61046927SAndroid Build Coastguard Worker static unsigned
bi_count_staging_registers(const bi_instr * ins)67*61046927SAndroid Build Coastguard Worker bi_count_staging_registers(const bi_instr *ins)
68*61046927SAndroid Build Coastguard Worker {
69*61046927SAndroid Build Coastguard Worker    enum bi_sr_count count = bi_opcode_props[ins->op].sr_count;
70*61046927SAndroid Build Coastguard Worker    unsigned vecsize = ins->vecsize + 1; /* XXX: off-by-one */
71*61046927SAndroid Build Coastguard Worker 
72*61046927SAndroid Build Coastguard Worker    switch (count) {
73*61046927SAndroid Build Coastguard Worker    case BI_SR_COUNT_0 ... BI_SR_COUNT_4:
74*61046927SAndroid Build Coastguard Worker       return count;
75*61046927SAndroid Build Coastguard Worker    case BI_SR_COUNT_FORMAT:
76*61046927SAndroid Build Coastguard Worker       return bi_is_regfmt_16(ins->register_format) ? DIV_ROUND_UP(vecsize, 2)
77*61046927SAndroid Build Coastguard Worker                                                    : vecsize;
78*61046927SAndroid Build Coastguard Worker    case BI_SR_COUNT_VECSIZE:
79*61046927SAndroid Build Coastguard Worker       return vecsize;
80*61046927SAndroid Build Coastguard Worker    case BI_SR_COUNT_SR_COUNT:
81*61046927SAndroid Build Coastguard Worker       return ins->sr_count;
82*61046927SAndroid Build Coastguard Worker    }
83*61046927SAndroid Build Coastguard Worker 
84*61046927SAndroid Build Coastguard Worker    unreachable("Invalid sr_count");
85*61046927SAndroid Build Coastguard Worker }
86*61046927SAndroid Build Coastguard Worker 
87*61046927SAndroid Build Coastguard Worker unsigned
bi_count_read_registers(const bi_instr * ins,unsigned s)88*61046927SAndroid Build Coastguard Worker bi_count_read_registers(const bi_instr *ins, unsigned s)
89*61046927SAndroid Build Coastguard Worker {
90*61046927SAndroid Build Coastguard Worker    /* ATOM reads 1 but writes 2. Exception for ACMPXCHG */
91*61046927SAndroid Build Coastguard Worker    if (s == 0 && ins->op == BI_OPCODE_ATOM_RETURN_I32)
92*61046927SAndroid Build Coastguard Worker       return (ins->atom_opc == BI_ATOM_OPC_ACMPXCHG) ? 2 : 1;
93*61046927SAndroid Build Coastguard Worker    else if (s == 0 && bi_opcode_props[ins->op].sr_read)
94*61046927SAndroid Build Coastguard Worker       return bi_count_staging_registers(ins);
95*61046927SAndroid Build Coastguard Worker    else if (s == 4 && ins->op == BI_OPCODE_BLEND)
96*61046927SAndroid Build Coastguard Worker       return ins->sr_count_2; /* Dual source blending */
97*61046927SAndroid Build Coastguard Worker    else if (s == 0 && ins->op == BI_OPCODE_SPLIT_I32)
98*61046927SAndroid Build Coastguard Worker       return ins->nr_dests;
99*61046927SAndroid Build Coastguard Worker    else
100*61046927SAndroid Build Coastguard Worker       return 1;
101*61046927SAndroid Build Coastguard Worker }
102*61046927SAndroid Build Coastguard Worker 
103*61046927SAndroid Build Coastguard Worker unsigned
bi_count_write_registers(const bi_instr * ins,unsigned d)104*61046927SAndroid Build Coastguard Worker bi_count_write_registers(const bi_instr *ins, unsigned d)
105*61046927SAndroid Build Coastguard Worker {
106*61046927SAndroid Build Coastguard Worker    if (d == 0 && bi_opcode_props[ins->op].sr_write) {
107*61046927SAndroid Build Coastguard Worker       switch (ins->op) {
108*61046927SAndroid Build Coastguard Worker       case BI_OPCODE_TEXC:
109*61046927SAndroid Build Coastguard Worker       case BI_OPCODE_TEXC_DUAL:
110*61046927SAndroid Build Coastguard Worker          if (ins->sr_count_2)
111*61046927SAndroid Build Coastguard Worker             return ins->sr_count;
112*61046927SAndroid Build Coastguard Worker          else
113*61046927SAndroid Build Coastguard Worker             return bi_is_regfmt_16(ins->register_format) ? 2 : 4;
114*61046927SAndroid Build Coastguard Worker 
115*61046927SAndroid Build Coastguard Worker       case BI_OPCODE_TEX_SINGLE:
116*61046927SAndroid Build Coastguard Worker       case BI_OPCODE_TEX_FETCH:
117*61046927SAndroid Build Coastguard Worker       case BI_OPCODE_TEX_GATHER: {
118*61046927SAndroid Build Coastguard Worker          unsigned chans = util_bitcount(ins->write_mask);
119*61046927SAndroid Build Coastguard Worker 
120*61046927SAndroid Build Coastguard Worker          return bi_is_regfmt_16(ins->register_format) ? DIV_ROUND_UP(chans, 2)
121*61046927SAndroid Build Coastguard Worker                                                       : chans;
122*61046927SAndroid Build Coastguard Worker       }
123*61046927SAndroid Build Coastguard Worker 
124*61046927SAndroid Build Coastguard Worker       case BI_OPCODE_ACMPXCHG_I32:
125*61046927SAndroid Build Coastguard Worker          /* Reads 2 but writes 1 */
126*61046927SAndroid Build Coastguard Worker          return 1;
127*61046927SAndroid Build Coastguard Worker 
128*61046927SAndroid Build Coastguard Worker       case BI_OPCODE_ATOM1_RETURN_I32:
129*61046927SAndroid Build Coastguard Worker          /* Allow omitting the destination for plain ATOM1 */
130*61046927SAndroid Build Coastguard Worker          return bi_is_null(ins->dest[0]) ? 0 : ins->sr_count;
131*61046927SAndroid Build Coastguard Worker       default:
132*61046927SAndroid Build Coastguard Worker          return bi_count_staging_registers(ins);
133*61046927SAndroid Build Coastguard Worker       }
134*61046927SAndroid Build Coastguard Worker    } else if (ins->op == BI_OPCODE_SEG_ADD_I64) {
135*61046927SAndroid Build Coastguard Worker       return 2;
136*61046927SAndroid Build Coastguard Worker    } else if (ins->op == BI_OPCODE_TEXC_DUAL && d == 1) {
137*61046927SAndroid Build Coastguard Worker       return ins->sr_count_2;
138*61046927SAndroid Build Coastguard Worker    } else if (ins->op == BI_OPCODE_COLLECT_I32 && d == 0) {
139*61046927SAndroid Build Coastguard Worker       return ins->nr_srcs;
140*61046927SAndroid Build Coastguard Worker    }
141*61046927SAndroid Build Coastguard Worker 
142*61046927SAndroid Build Coastguard Worker    return 1;
143*61046927SAndroid Build Coastguard Worker }
144*61046927SAndroid Build Coastguard Worker 
145*61046927SAndroid Build Coastguard Worker unsigned
bi_writemask(const bi_instr * ins,unsigned d)146*61046927SAndroid Build Coastguard Worker bi_writemask(const bi_instr *ins, unsigned d)
147*61046927SAndroid Build Coastguard Worker {
148*61046927SAndroid Build Coastguard Worker    unsigned mask = BITFIELD_MASK(bi_count_write_registers(ins, d));
149*61046927SAndroid Build Coastguard Worker    unsigned shift = ins->dest[d].offset;
150*61046927SAndroid Build Coastguard Worker    return (mask << shift);
151*61046927SAndroid Build Coastguard Worker }
152*61046927SAndroid Build Coastguard Worker 
153*61046927SAndroid Build Coastguard Worker bi_clause *
bi_next_clause(bi_context * ctx,bi_block * block,bi_clause * clause)154*61046927SAndroid Build Coastguard Worker bi_next_clause(bi_context *ctx, bi_block *block, bi_clause *clause)
155*61046927SAndroid Build Coastguard Worker {
156*61046927SAndroid Build Coastguard Worker    if (!block && !clause)
157*61046927SAndroid Build Coastguard Worker       return NULL;
158*61046927SAndroid Build Coastguard Worker 
159*61046927SAndroid Build Coastguard Worker    /* Try the first clause in this block if we're starting from scratch */
160*61046927SAndroid Build Coastguard Worker    if (!clause && !list_is_empty(&block->clauses))
161*61046927SAndroid Build Coastguard Worker       return list_first_entry(&block->clauses, bi_clause, link);
162*61046927SAndroid Build Coastguard Worker 
163*61046927SAndroid Build Coastguard Worker    /* Try the next clause in this block */
164*61046927SAndroid Build Coastguard Worker    if (clause && clause->link.next != &block->clauses)
165*61046927SAndroid Build Coastguard Worker       return list_first_entry(&(clause->link), bi_clause, link);
166*61046927SAndroid Build Coastguard Worker 
167*61046927SAndroid Build Coastguard Worker    /* Try the next block, or the one after that if it's empty, etc .*/
168*61046927SAndroid Build Coastguard Worker    bi_block *next_block = bi_next_block(block);
169*61046927SAndroid Build Coastguard Worker 
170*61046927SAndroid Build Coastguard Worker    bi_foreach_block_from(ctx, next_block, block) {
171*61046927SAndroid Build Coastguard Worker       if (!list_is_empty(&block->clauses))
172*61046927SAndroid Build Coastguard Worker          return list_first_entry(&block->clauses, bi_clause, link);
173*61046927SAndroid Build Coastguard Worker    }
174*61046927SAndroid Build Coastguard Worker 
175*61046927SAndroid Build Coastguard Worker    return NULL;
176*61046927SAndroid Build Coastguard Worker }
177*61046927SAndroid Build Coastguard Worker 
178*61046927SAndroid Build Coastguard Worker /* Does an instruction have a side effect not captured by its register
179*61046927SAndroid Build Coastguard Worker  * destination? Applies to certain message-passing instructions, +DISCARD, and
180*61046927SAndroid Build Coastguard Worker  * branching only, used in dead code elimation. Branches are characterized by
181*61046927SAndroid Build Coastguard Worker  * `last` which applies to them and some atomics, +BARRIER, +BLEND which
182*61046927SAndroid Build Coastguard Worker  * implies no loss of generality */
183*61046927SAndroid Build Coastguard Worker 
184*61046927SAndroid Build Coastguard Worker bool
bi_side_effects(const bi_instr * I)185*61046927SAndroid Build Coastguard Worker bi_side_effects(const bi_instr *I)
186*61046927SAndroid Build Coastguard Worker {
187*61046927SAndroid Build Coastguard Worker    if (bi_opcode_props[I->op].last)
188*61046927SAndroid Build Coastguard Worker       return true;
189*61046927SAndroid Build Coastguard Worker 
190*61046927SAndroid Build Coastguard Worker    switch (I->op) {
191*61046927SAndroid Build Coastguard Worker    case BI_OPCODE_DISCARD_F32:
192*61046927SAndroid Build Coastguard Worker    case BI_OPCODE_DISCARD_B32:
193*61046927SAndroid Build Coastguard Worker       return true;
194*61046927SAndroid Build Coastguard Worker    default:
195*61046927SAndroid Build Coastguard Worker       break;
196*61046927SAndroid Build Coastguard Worker    }
197*61046927SAndroid Build Coastguard Worker 
198*61046927SAndroid Build Coastguard Worker    switch (bi_opcode_props[I->op].message) {
199*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_NONE:
200*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_VARYING:
201*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_ATTRIBUTE:
202*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_TEX:
203*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_VARTEX:
204*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_LOAD:
205*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_64BIT:
206*61046927SAndroid Build Coastguard Worker       return false;
207*61046927SAndroid Build Coastguard Worker 
208*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_STORE:
209*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_ATOMIC:
210*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_BARRIER:
211*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_BLEND:
212*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_Z_STENCIL:
213*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_ATEST:
214*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_JOB:
215*61046927SAndroid Build Coastguard Worker       return true;
216*61046927SAndroid Build Coastguard Worker 
217*61046927SAndroid Build Coastguard Worker    case BIFROST_MESSAGE_TILE:
218*61046927SAndroid Build Coastguard Worker       return (I->op != BI_OPCODE_LD_TILE);
219*61046927SAndroid Build Coastguard Worker    }
220*61046927SAndroid Build Coastguard Worker 
221*61046927SAndroid Build Coastguard Worker    unreachable("Invalid message type");
222*61046927SAndroid Build Coastguard Worker }
223*61046927SAndroid Build Coastguard Worker 
224*61046927SAndroid Build Coastguard Worker /* Branch reconvergence is required when the execution mask may change
225*61046927SAndroid Build Coastguard Worker  * between adjacent instructions (clauses). This occurs for conditional
226*61046927SAndroid Build Coastguard Worker  * branches and for the last instruction (clause) in a block whose
227*61046927SAndroid Build Coastguard Worker  * fallthrough successor has multiple predecessors.
228*61046927SAndroid Build Coastguard Worker  */
229*61046927SAndroid Build Coastguard Worker 
230*61046927SAndroid Build Coastguard Worker bool
bi_reconverge_branches(bi_block * block)231*61046927SAndroid Build Coastguard Worker bi_reconverge_branches(bi_block *block)
232*61046927SAndroid Build Coastguard Worker {
233*61046927SAndroid Build Coastguard Worker    if (bi_num_successors(block) == 1)
234*61046927SAndroid Build Coastguard Worker       return bi_num_predecessors(block->successors[0]) > 1;
235*61046927SAndroid Build Coastguard Worker    else
236*61046927SAndroid Build Coastguard Worker       return true;
237*61046927SAndroid Build Coastguard Worker }
238*61046927SAndroid Build Coastguard Worker 
239*61046927SAndroid Build Coastguard Worker /*
240*61046927SAndroid Build Coastguard Worker  * When MUX.i32 or MUX.v2i16 is used to multiplex entire sources, they can be
241*61046927SAndroid Build Coastguard Worker  * replaced by CSEL as follows:
242*61046927SAndroid Build Coastguard Worker  *
243*61046927SAndroid Build Coastguard Worker  *      MUX.neg(x, y, b) -> CSEL.s.lt(b, 0, x, y)
244*61046927SAndroid Build Coastguard Worker  *      MUX.int_zero(x, y, b) -> CSEL.i.eq(b, 0, x, y)
245*61046927SAndroid Build Coastguard Worker  *      MUX.fp_zero(x, y, b) -> CSEL.f.eq(b, 0, x, y)
246*61046927SAndroid Build Coastguard Worker  *
247*61046927SAndroid Build Coastguard Worker  * MUX.bit cannot be transformed like this.
248*61046927SAndroid Build Coastguard Worker  *
249*61046927SAndroid Build Coastguard Worker  * Note that MUX.v2i16 has partial support for swizzles, which CSEL.v2i16 lacks.
250*61046927SAndroid Build Coastguard Worker  * So we must check the swizzles too.
251*61046927SAndroid Build Coastguard Worker  */
252*61046927SAndroid Build Coastguard Worker bool
bi_can_replace_with_csel(bi_instr * I)253*61046927SAndroid Build Coastguard Worker bi_can_replace_with_csel(bi_instr *I)
254*61046927SAndroid Build Coastguard Worker {
255*61046927SAndroid Build Coastguard Worker    return ((I->op == BI_OPCODE_MUX_I32) || (I->op == BI_OPCODE_MUX_V2I16)) &&
256*61046927SAndroid Build Coastguard Worker           (I->mux != BI_MUX_BIT) && (I->src[0].swizzle == BI_SWIZZLE_H01) &&
257*61046927SAndroid Build Coastguard Worker           (I->src[1].swizzle == BI_SWIZZLE_H01) &&
258*61046927SAndroid Build Coastguard Worker           (I->src[2].swizzle == BI_SWIZZLE_H01);
259*61046927SAndroid Build Coastguard Worker }
260*61046927SAndroid Build Coastguard Worker 
261*61046927SAndroid Build Coastguard Worker static enum bi_opcode
bi_csel_for_mux(bool must_sign,bool b32,enum bi_mux mux)262*61046927SAndroid Build Coastguard Worker bi_csel_for_mux(bool must_sign, bool b32, enum bi_mux mux)
263*61046927SAndroid Build Coastguard Worker {
264*61046927SAndroid Build Coastguard Worker    switch (mux) {
265*61046927SAndroid Build Coastguard Worker    case BI_MUX_INT_ZERO:
266*61046927SAndroid Build Coastguard Worker       if (must_sign)
267*61046927SAndroid Build Coastguard Worker          return b32 ? BI_OPCODE_CSEL_U32 : BI_OPCODE_CSEL_V2U16;
268*61046927SAndroid Build Coastguard Worker       else
269*61046927SAndroid Build Coastguard Worker          return b32 ? BI_OPCODE_CSEL_I32 : BI_OPCODE_CSEL_V2I16;
270*61046927SAndroid Build Coastguard Worker    case BI_MUX_NEG:
271*61046927SAndroid Build Coastguard Worker       return b32 ? BI_OPCODE_CSEL_S32 : BI_OPCODE_CSEL_V2S16;
272*61046927SAndroid Build Coastguard Worker    case BI_MUX_FP_ZERO:
273*61046927SAndroid Build Coastguard Worker       return b32 ? BI_OPCODE_CSEL_F32 : BI_OPCODE_CSEL_V2F16;
274*61046927SAndroid Build Coastguard Worker    default:
275*61046927SAndroid Build Coastguard Worker       unreachable("No CSEL for MUX.bit");
276*61046927SAndroid Build Coastguard Worker    }
277*61046927SAndroid Build Coastguard Worker }
278*61046927SAndroid Build Coastguard Worker 
279*61046927SAndroid Build Coastguard Worker bi_instr *
bi_csel_from_mux(bi_builder * b,const bi_instr * I,bool must_sign)280*61046927SAndroid Build Coastguard Worker bi_csel_from_mux(bi_builder *b, const bi_instr *I, bool must_sign)
281*61046927SAndroid Build Coastguard Worker {
282*61046927SAndroid Build Coastguard Worker    assert(I->op == BI_OPCODE_MUX_I32 || I->op == BI_OPCODE_MUX_V2I16);
283*61046927SAndroid Build Coastguard Worker 
284*61046927SAndroid Build Coastguard Worker    /* Build a new CSEL */
285*61046927SAndroid Build Coastguard Worker    enum bi_cmpf cmpf = (I->mux == BI_MUX_NEG) ? BI_CMPF_LT : BI_CMPF_EQ;
286*61046927SAndroid Build Coastguard Worker    bi_instr *csel = bi_csel_u32_to(b, I->dest[0], I->src[2], bi_zero(),
287*61046927SAndroid Build Coastguard Worker                                    I->src[0], I->src[1], cmpf);
288*61046927SAndroid Build Coastguard Worker 
289*61046927SAndroid Build Coastguard Worker    /* Fixup the opcode and use it */
290*61046927SAndroid Build Coastguard Worker    csel->op = bi_csel_for_mux(must_sign, I->op == BI_OPCODE_MUX_I32, I->mux);
291*61046927SAndroid Build Coastguard Worker    return csel;
292*61046927SAndroid Build Coastguard Worker }
293