aix: align double complex
[official-gcc.git] / gcc / optabs-tree.c
blob95ffe397c23e80c105afea52e9d47216bf52f55a
1 /* Tree-based target query functions relating to optabs
2 Copyright (C) 1987-2021 Free Software Foundation, Inc.
4 This file is part of GCC.
6 GCC is free software; you can redistribute it and/or modify it under
7 the terms of the GNU General Public License as published by the Free
8 Software Foundation; either version 3, or (at your option) any later
9 version.
11 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
12 WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 for more details.
16 You should have received a copy of the GNU General Public License
17 along with GCC; see the file COPYING3. If not see
18 <http://www.gnu.org/licenses/>. */
21 #include "config.h"
22 #include "system.h"
23 #include "coretypes.h"
24 #include "target.h"
25 #include "insn-codes.h"
26 #include "rtl.h"
27 #include "tree.h"
28 #include "memmodel.h"
29 #include "optabs.h"
30 #include "optabs-tree.h"
31 #include "stor-layout.h"
33 /* Return the optab used for computing the operation given by the tree code,
34 CODE and the tree EXP. This function is not always usable (for example, it
35 cannot give complete results for multiplication or division) but probably
36 ought to be relied on more widely throughout the expander. */
37 optab
38 optab_for_tree_code (enum tree_code code, const_tree type,
39 enum optab_subtype subtype)
41 bool trapv;
42 switch (code)
44 case BIT_AND_EXPR:
45 return and_optab;
47 case BIT_IOR_EXPR:
48 return ior_optab;
50 case BIT_NOT_EXPR:
51 return one_cmpl_optab;
53 case BIT_XOR_EXPR:
54 return xor_optab;
56 case MULT_HIGHPART_EXPR:
57 return TYPE_UNSIGNED (type) ? umul_highpart_optab : smul_highpart_optab;
59 case TRUNC_MOD_EXPR:
60 case CEIL_MOD_EXPR:
61 case FLOOR_MOD_EXPR:
62 case ROUND_MOD_EXPR:
63 return TYPE_UNSIGNED (type) ? umod_optab : smod_optab;
65 case RDIV_EXPR:
66 case TRUNC_DIV_EXPR:
67 case CEIL_DIV_EXPR:
68 case FLOOR_DIV_EXPR:
69 case ROUND_DIV_EXPR:
70 case EXACT_DIV_EXPR:
71 if (TYPE_SATURATING (type))
72 return TYPE_UNSIGNED (type) ? usdiv_optab : ssdiv_optab;
73 return TYPE_UNSIGNED (type) ? udiv_optab : sdiv_optab;
75 case LSHIFT_EXPR:
76 if (TREE_CODE (type) == VECTOR_TYPE)
78 if (subtype == optab_vector)
79 return TYPE_SATURATING (type) ? unknown_optab : vashl_optab;
81 gcc_assert (subtype == optab_scalar);
83 if (TYPE_SATURATING (type))
84 return TYPE_UNSIGNED (type) ? usashl_optab : ssashl_optab;
85 return ashl_optab;
87 case RSHIFT_EXPR:
88 if (TREE_CODE (type) == VECTOR_TYPE)
90 if (subtype == optab_vector)
91 return TYPE_UNSIGNED (type) ? vlshr_optab : vashr_optab;
93 gcc_assert (subtype == optab_scalar);
95 return TYPE_UNSIGNED (type) ? lshr_optab : ashr_optab;
97 case LROTATE_EXPR:
98 if (TREE_CODE (type) == VECTOR_TYPE)
100 if (subtype == optab_vector)
101 return vrotl_optab;
103 gcc_assert (subtype == optab_scalar);
105 return rotl_optab;
107 case RROTATE_EXPR:
108 if (TREE_CODE (type) == VECTOR_TYPE)
110 if (subtype == optab_vector)
111 return vrotr_optab;
113 gcc_assert (subtype == optab_scalar);
115 return rotr_optab;
117 case MAX_EXPR:
118 return TYPE_UNSIGNED (type) ? umax_optab : smax_optab;
120 case MIN_EXPR:
121 return TYPE_UNSIGNED (type) ? umin_optab : smin_optab;
123 case REALIGN_LOAD_EXPR:
124 return vec_realign_load_optab;
126 case WIDEN_SUM_EXPR:
127 return TYPE_UNSIGNED (type) ? usum_widen_optab : ssum_widen_optab;
129 case DOT_PROD_EXPR:
130 return TYPE_UNSIGNED (type) ? udot_prod_optab : sdot_prod_optab;
132 case SAD_EXPR:
133 return TYPE_UNSIGNED (type) ? usad_optab : ssad_optab;
135 case WIDEN_MULT_PLUS_EXPR:
136 return (TYPE_UNSIGNED (type)
137 ? (TYPE_SATURATING (type)
138 ? usmadd_widen_optab : umadd_widen_optab)
139 : (TYPE_SATURATING (type)
140 ? ssmadd_widen_optab : smadd_widen_optab));
142 case WIDEN_MULT_MINUS_EXPR:
143 return (TYPE_UNSIGNED (type)
144 ? (TYPE_SATURATING (type)
145 ? usmsub_widen_optab : umsub_widen_optab)
146 : (TYPE_SATURATING (type)
147 ? ssmsub_widen_optab : smsub_widen_optab));
149 case VEC_WIDEN_MULT_HI_EXPR:
150 return (TYPE_UNSIGNED (type)
151 ? vec_widen_umult_hi_optab : vec_widen_smult_hi_optab);
153 case VEC_WIDEN_MULT_LO_EXPR:
154 return (TYPE_UNSIGNED (type)
155 ? vec_widen_umult_lo_optab : vec_widen_smult_lo_optab);
157 case VEC_WIDEN_MULT_EVEN_EXPR:
158 return (TYPE_UNSIGNED (type)
159 ? vec_widen_umult_even_optab : vec_widen_smult_even_optab);
161 case VEC_WIDEN_MULT_ODD_EXPR:
162 return (TYPE_UNSIGNED (type)
163 ? vec_widen_umult_odd_optab : vec_widen_smult_odd_optab);
165 case VEC_WIDEN_LSHIFT_HI_EXPR:
166 return (TYPE_UNSIGNED (type)
167 ? vec_widen_ushiftl_hi_optab : vec_widen_sshiftl_hi_optab);
169 case VEC_WIDEN_LSHIFT_LO_EXPR:
170 return (TYPE_UNSIGNED (type)
171 ? vec_widen_ushiftl_lo_optab : vec_widen_sshiftl_lo_optab);
173 case VEC_WIDEN_PLUS_LO_EXPR:
174 return (TYPE_UNSIGNED (type)
175 ? vec_widen_uaddl_lo_optab : vec_widen_saddl_lo_optab);
177 case VEC_WIDEN_PLUS_HI_EXPR:
178 return (TYPE_UNSIGNED (type)
179 ? vec_widen_uaddl_hi_optab : vec_widen_saddl_hi_optab);
181 case VEC_WIDEN_MINUS_LO_EXPR:
182 return (TYPE_UNSIGNED (type)
183 ? vec_widen_usubl_lo_optab : vec_widen_ssubl_lo_optab);
185 case VEC_WIDEN_MINUS_HI_EXPR:
186 return (TYPE_UNSIGNED (type)
187 ? vec_widen_usubl_hi_optab : vec_widen_ssubl_hi_optab);
189 case VEC_UNPACK_HI_EXPR:
190 return (TYPE_UNSIGNED (type)
191 ? vec_unpacku_hi_optab : vec_unpacks_hi_optab);
193 case VEC_UNPACK_LO_EXPR:
194 return (TYPE_UNSIGNED (type)
195 ? vec_unpacku_lo_optab : vec_unpacks_lo_optab);
197 case VEC_UNPACK_FLOAT_HI_EXPR:
198 /* The signedness is determined from input operand. */
199 return (TYPE_UNSIGNED (type)
200 ? vec_unpacku_float_hi_optab : vec_unpacks_float_hi_optab);
202 case VEC_UNPACK_FLOAT_LO_EXPR:
203 /* The signedness is determined from input operand. */
204 return (TYPE_UNSIGNED (type)
205 ? vec_unpacku_float_lo_optab : vec_unpacks_float_lo_optab);
207 case VEC_UNPACK_FIX_TRUNC_HI_EXPR:
208 /* The signedness is determined from output operand. */
209 return (TYPE_UNSIGNED (type)
210 ? vec_unpack_ufix_trunc_hi_optab
211 : vec_unpack_sfix_trunc_hi_optab);
213 case VEC_UNPACK_FIX_TRUNC_LO_EXPR:
214 /* The signedness is determined from output operand. */
215 return (TYPE_UNSIGNED (type)
216 ? vec_unpack_ufix_trunc_lo_optab
217 : vec_unpack_sfix_trunc_lo_optab);
219 case VEC_PACK_TRUNC_EXPR:
220 return vec_pack_trunc_optab;
222 case VEC_PACK_SAT_EXPR:
223 return TYPE_UNSIGNED (type) ? vec_pack_usat_optab : vec_pack_ssat_optab;
225 case VEC_PACK_FIX_TRUNC_EXPR:
226 /* The signedness is determined from output operand. */
227 return (TYPE_UNSIGNED (type)
228 ? vec_pack_ufix_trunc_optab : vec_pack_sfix_trunc_optab);
230 case VEC_PACK_FLOAT_EXPR:
231 /* The signedness is determined from input operand. */
232 return (TYPE_UNSIGNED (type)
233 ? vec_packu_float_optab : vec_packs_float_optab);
235 case VEC_DUPLICATE_EXPR:
236 return vec_duplicate_optab;
238 case VEC_SERIES_EXPR:
239 return vec_series_optab;
241 default:
242 break;
245 trapv = INTEGRAL_TYPE_P (type) && TYPE_OVERFLOW_TRAPS (type);
246 switch (code)
248 case POINTER_PLUS_EXPR:
249 case PLUS_EXPR:
250 if (TYPE_SATURATING (type))
251 return TYPE_UNSIGNED (type) ? usadd_optab : ssadd_optab;
252 return trapv ? addv_optab : add_optab;
254 case POINTER_DIFF_EXPR:
255 case MINUS_EXPR:
256 if (TYPE_SATURATING (type))
257 return TYPE_UNSIGNED (type) ? ussub_optab : sssub_optab;
258 return trapv ? subv_optab : sub_optab;
260 case MULT_EXPR:
261 if (TYPE_SATURATING (type))
262 return TYPE_UNSIGNED (type) ? usmul_optab : ssmul_optab;
263 return trapv ? smulv_optab : smul_optab;
265 case NEGATE_EXPR:
266 if (TYPE_SATURATING (type))
267 return TYPE_UNSIGNED (type) ? usneg_optab : ssneg_optab;
268 return trapv ? negv_optab : neg_optab;
270 case ABS_EXPR:
271 return trapv ? absv_optab : abs_optab;
273 case ABSU_EXPR:
274 return abs_optab;
275 default:
276 return unknown_optab;
280 /* Check whether an operation represented by CODE is a 'half' widening operation
281 in which the input vector type has half the number of bits of the output
282 vector type e.g. V8QI->V8HI.
284 This is handled by widening the inputs using NOP_EXPRs then using a
285 non-widening stmt e.g. MINUS_EXPR. RTL fusing converts these to the widening
286 hardware instructions if supported.
288 The more typical case (handled in supportable_widening_operation) is where
289 the input vector type has the same number of bits as the output vector type.
290 In this case half the elements of the input vectors must be processed at a
291 time into respective vector outputs with elements twice as wide i.e. a
292 'hi'/'lo' pair using codes such as VEC_WIDEN_MINUS_HI/LO.
294 Supported widening operations:
295 WIDEN_MINUS_EXPR
296 WIDEN_PLUS_EXPR
297 WIDEN_MULT_EXPR
298 WIDEN_LSHIFT_EXPR
300 Output:
301 - CODE1 - The non-widened code, which will be used after the inputs are
302 converted to the wide type. */
303 bool
304 supportable_half_widening_operation (enum tree_code code, tree vectype_out,
305 tree vectype_in, enum tree_code *code1)
307 machine_mode m1,m2;
308 enum tree_code dummy_code;
309 optab op;
311 gcc_assert (VECTOR_TYPE_P (vectype_out) && VECTOR_TYPE_P (vectype_in));
313 m1 = TYPE_MODE (vectype_out);
314 m2 = TYPE_MODE (vectype_in);
316 if (!VECTOR_MODE_P (m1) || !VECTOR_MODE_P (m2))
317 return false;
319 if (maybe_ne (TYPE_VECTOR_SUBPARTS (vectype_in),
320 TYPE_VECTOR_SUBPARTS (vectype_out)))
321 return false;
323 switch (code)
325 case WIDEN_LSHIFT_EXPR:
326 *code1 = LSHIFT_EXPR;
327 break;
328 case WIDEN_MINUS_EXPR:
329 *code1 = MINUS_EXPR;
330 break;
331 case WIDEN_PLUS_EXPR:
332 *code1 = PLUS_EXPR;
333 break;
334 case WIDEN_MULT_EXPR:
335 *code1 = MULT_EXPR;
336 break;
337 default:
338 return false;
341 if (!supportable_convert_operation (NOP_EXPR, vectype_out, vectype_in,
342 &dummy_code))
343 return false;
345 op = optab_for_tree_code (*code1, vectype_out, optab_vector);
346 return (optab_handler (op, TYPE_MODE (vectype_out)) != CODE_FOR_nothing);
349 /* Function supportable_convert_operation
351 Check whether an operation represented by the code CODE is a
352 convert operation that is supported by the target platform in
353 vector form (i.e., when operating on arguments of type VECTYPE_IN
354 producing a result of type VECTYPE_OUT).
356 Convert operations we currently support directly are FIX_TRUNC and FLOAT.
357 This function checks if these operations are supported
358 by the target platform directly (via vector tree-codes).
360 Output:
361 - CODE1 is code of vector operation to be used when
362 vectorizing the operation, if available. */
364 bool
365 supportable_convert_operation (enum tree_code code,
366 tree vectype_out, tree vectype_in,
367 enum tree_code *code1)
369 machine_mode m1,m2;
370 bool truncp;
372 gcc_assert (VECTOR_TYPE_P (vectype_out) && VECTOR_TYPE_P (vectype_in));
374 m1 = TYPE_MODE (vectype_out);
375 m2 = TYPE_MODE (vectype_in);
377 if (!VECTOR_MODE_P (m1) || !VECTOR_MODE_P (m2))
378 return false;
380 /* First check if we can done conversion directly. */
381 if ((code == FIX_TRUNC_EXPR
382 && can_fix_p (m1,m2,TYPE_UNSIGNED (vectype_out), &truncp)
383 != CODE_FOR_nothing)
384 || (code == FLOAT_EXPR
385 && can_float_p (m1,m2,TYPE_UNSIGNED (vectype_in))
386 != CODE_FOR_nothing))
388 *code1 = code;
389 return true;
392 if (GET_MODE_UNIT_PRECISION (m1) > GET_MODE_UNIT_PRECISION (m2)
393 && can_extend_p (m1, m2, TYPE_UNSIGNED (vectype_in)))
395 *code1 = code;
396 return true;
399 if (GET_MODE_UNIT_PRECISION (m1) < GET_MODE_UNIT_PRECISION (m2)
400 && convert_optab_handler (trunc_optab, m1, m2) != CODE_FOR_nothing)
402 *code1 = code;
403 return true;
406 return false;
409 /* Return true iff vec_cmp_optab/vec_cmpu_optab can handle a vector comparison
410 for code CODE, comparing operands of type VALUE_TYPE and producing a result
411 of type MASK_TYPE. */
413 static bool
414 vec_cmp_icode_p (tree value_type, tree mask_type, enum tree_code code)
416 enum rtx_code rcode = get_rtx_code_1 (code, TYPE_UNSIGNED (value_type));
417 if (rcode == UNKNOWN)
418 return false;
420 return can_vec_cmp_compare_p (rcode, TYPE_MODE (value_type),
421 TYPE_MODE (mask_type));
424 /* Return true iff vec_cmpeq_optab can handle a vector comparison for code
425 CODE, comparing operands of type VALUE_TYPE and producing a result of type
426 MASK_TYPE. */
428 static bool
429 vec_cmp_eq_icode_p (tree value_type, tree mask_type, enum tree_code code)
431 if (code != EQ_EXPR && code != NE_EXPR)
432 return false;
434 return get_vec_cmp_eq_icode (TYPE_MODE (value_type), TYPE_MODE (mask_type))
435 != CODE_FOR_nothing;
438 /* Return TRUE if appropriate vector insn is available
439 for vector comparison expr with vector type VALUE_TYPE
440 and resulting mask with MASK_TYPE. */
442 bool
443 expand_vec_cmp_expr_p (tree value_type, tree mask_type, enum tree_code code)
445 return vec_cmp_icode_p (value_type, mask_type, code)
446 || vec_cmp_eq_icode_p (value_type, mask_type, code);
449 /* Return true iff vcond_optab/vcondu_optab can handle a vector
450 comparison for code CODE, comparing operands of type CMP_OP_TYPE and
451 producing a result of type VALUE_TYPE. */
453 static bool
454 vcond_icode_p (tree value_type, tree cmp_op_type, enum tree_code code)
456 enum rtx_code rcode = get_rtx_code_1 (code, TYPE_UNSIGNED (cmp_op_type));
457 if (rcode == UNKNOWN)
458 return false;
460 return can_vcond_compare_p (rcode, TYPE_MODE (value_type),
461 TYPE_MODE (cmp_op_type));
464 /* Return true iff vcondeq_optab can handle a vector comparison for code CODE,
465 comparing operands of type CMP_OP_TYPE and producing a result of type
466 VALUE_TYPE. */
468 static bool
469 vcond_eq_icode_p (tree value_type, tree cmp_op_type, enum tree_code code)
471 if (code != EQ_EXPR && code != NE_EXPR)
472 return false;
474 return get_vcond_eq_icode (TYPE_MODE (value_type), TYPE_MODE (cmp_op_type))
475 != CODE_FOR_nothing;
478 /* Return TRUE iff, appropriate vector insns are available
479 for vector cond expr with vector type VALUE_TYPE and a comparison
480 with operand vector types in CMP_OP_TYPE. */
482 bool
483 expand_vec_cond_expr_p (tree value_type, tree cmp_op_type, enum tree_code code)
485 machine_mode value_mode = TYPE_MODE (value_type);
486 machine_mode cmp_op_mode = TYPE_MODE (cmp_op_type);
487 if (VECTOR_BOOLEAN_TYPE_P (cmp_op_type)
488 && get_vcond_mask_icode (TYPE_MODE (value_type),
489 TYPE_MODE (cmp_op_type)) != CODE_FOR_nothing)
490 return true;
492 if (maybe_ne (GET_MODE_NUNITS (value_mode), GET_MODE_NUNITS (cmp_op_mode)))
493 return false;
495 if (TREE_CODE_CLASS (code) != tcc_comparison)
496 /* This may happen, for example, if code == SSA_NAME, in which case we
497 cannot be certain whether a vector insn is available. */
498 return false;
500 return vcond_icode_p (value_type, cmp_op_type, code)
501 || vcond_eq_icode_p (value_type, cmp_op_type, code);
504 /* Use the current target and options to initialize
505 TREE_OPTIMIZATION_OPTABS (OPTNODE). */
507 void
508 init_tree_optimization_optabs (tree optnode)
510 /* Quick exit if we have already computed optabs for this target. */
511 if (TREE_OPTIMIZATION_BASE_OPTABS (optnode) == this_target_optabs)
512 return;
514 /* Forget any previous information and set up for the current target. */
515 TREE_OPTIMIZATION_BASE_OPTABS (optnode) = this_target_optabs;
516 struct target_optabs *tmp_optabs = (struct target_optabs *)
517 TREE_OPTIMIZATION_OPTABS (optnode);
518 if (tmp_optabs)
519 memset (tmp_optabs, 0, sizeof (struct target_optabs));
520 else
521 tmp_optabs = ggc_cleared_alloc<target_optabs> ();
523 /* Generate a new set of optabs into tmp_optabs. */
524 init_all_optabs (tmp_optabs);
526 /* If the optabs changed, record it. */
527 if (memcmp (tmp_optabs, this_target_optabs, sizeof (struct target_optabs)))
528 TREE_OPTIMIZATION_OPTABS (optnode) = tmp_optabs;
529 else
531 TREE_OPTIMIZATION_OPTABS (optnode) = NULL;
532 ggc_free (tmp_optabs);
536 /* Return TRUE if the target has support for vector right shift of an
537 operand of type TYPE. If OT_TYPE is OPTAB_DEFAULT, check for existence
538 of a shift by either a scalar or a vector. Otherwise, check only
539 for a shift that matches OT_TYPE. */
541 bool
542 target_supports_op_p (tree type, enum tree_code code,
543 enum optab_subtype ot_subtype)
545 optab ot = optab_for_tree_code (code, type, ot_subtype);
546 return (ot != unknown_optab
547 && optab_handler (ot, TYPE_MODE (type)) != CODE_FOR_nothing);