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
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
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/>. */
23 #include "coretypes.h"
25 #include "insn-codes.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. */
38 optab_for_tree_code (enum tree_code code
, const_tree type
,
39 enum optab_subtype subtype
)
51 return one_cmpl_optab
;
56 case MULT_HIGHPART_EXPR
:
57 return TYPE_UNSIGNED (type
) ? umul_highpart_optab
: smul_highpart_optab
;
63 return TYPE_UNSIGNED (type
) ? umod_optab
: smod_optab
;
71 if (TYPE_SATURATING (type
))
72 return TYPE_UNSIGNED (type
) ? usdiv_optab
: ssdiv_optab
;
73 return TYPE_UNSIGNED (type
) ? udiv_optab
: sdiv_optab
;
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
;
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
;
98 if (TREE_CODE (type
) == VECTOR_TYPE
)
100 if (subtype
== optab_vector
)
103 gcc_assert (subtype
== optab_scalar
);
108 if (TREE_CODE (type
) == VECTOR_TYPE
)
110 if (subtype
== optab_vector
)
113 gcc_assert (subtype
== optab_scalar
);
118 return TYPE_UNSIGNED (type
) ? umax_optab
: smax_optab
;
121 return TYPE_UNSIGNED (type
) ? umin_optab
: smin_optab
;
123 case REALIGN_LOAD_EXPR
:
124 return vec_realign_load_optab
;
127 return TYPE_UNSIGNED (type
) ? usum_widen_optab
: ssum_widen_optab
;
130 return TYPE_UNSIGNED (type
) ? udot_prod_optab
: sdot_prod_optab
;
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
;
245 trapv
= INTEGRAL_TYPE_P (type
) && TYPE_OVERFLOW_TRAPS (type
);
248 case POINTER_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
:
256 if (TYPE_SATURATING (type
))
257 return TYPE_UNSIGNED (type
) ? ussub_optab
: sssub_optab
;
258 return trapv
? subv_optab
: sub_optab
;
261 if (TYPE_SATURATING (type
))
262 return TYPE_UNSIGNED (type
) ? usmul_optab
: ssmul_optab
;
263 return trapv
? smulv_optab
: smul_optab
;
266 if (TYPE_SATURATING (type
))
267 return TYPE_UNSIGNED (type
) ? usneg_optab
: ssneg_optab
;
268 return trapv
? negv_optab
: neg_optab
;
271 return trapv
? absv_optab
: abs_optab
;
276 return unknown_optab
;
280 /* Function supportable_convert_operation
282 Check whether an operation represented by the code CODE is a
283 convert operation that is supported by the target platform in
284 vector form (i.e., when operating on arguments of type VECTYPE_IN
285 producing a result of type VECTYPE_OUT).
287 Convert operations we currently support directly are FIX_TRUNC and FLOAT.
288 This function checks if these operations are supported
289 by the target platform directly (via vector tree-codes).
292 - CODE1 is code of vector operation to be used when
293 vectorizing the operation, if available. */
296 supportable_convert_operation (enum tree_code code
,
297 tree vectype_out
, tree vectype_in
,
298 enum tree_code
*code1
)
303 gcc_assert (VECTOR_TYPE_P (vectype_out
) && VECTOR_TYPE_P (vectype_in
));
305 m1
= TYPE_MODE (vectype_out
);
306 m2
= TYPE_MODE (vectype_in
);
308 if (!VECTOR_MODE_P (m1
) || !VECTOR_MODE_P (m2
))
311 /* First check if we can done conversion directly. */
312 if ((code
== FIX_TRUNC_EXPR
313 && can_fix_p (m1
,m2
,TYPE_UNSIGNED (vectype_out
), &truncp
)
315 || (code
== FLOAT_EXPR
316 && can_float_p (m1
,m2
,TYPE_UNSIGNED (vectype_in
))
317 != CODE_FOR_nothing
))
323 if (GET_MODE_UNIT_PRECISION (m1
) > GET_MODE_UNIT_PRECISION (m2
)
324 && can_extend_p (m1
, m2
, TYPE_UNSIGNED (vectype_in
)))
330 if (GET_MODE_UNIT_PRECISION (m1
) < GET_MODE_UNIT_PRECISION (m2
)
331 && convert_optab_handler (trunc_optab
, m1
, m2
) != CODE_FOR_nothing
)
340 /* Return true iff vec_cmp_optab/vec_cmpu_optab can handle a vector comparison
341 for code CODE, comparing operands of type VALUE_TYPE and producing a result
342 of type MASK_TYPE. */
345 vec_cmp_icode_p (tree value_type
, tree mask_type
, enum tree_code code
)
347 enum rtx_code rcode
= get_rtx_code_1 (code
, TYPE_UNSIGNED (value_type
));
348 if (rcode
== UNKNOWN
)
351 return can_vec_cmp_compare_p (rcode
, TYPE_MODE (value_type
),
352 TYPE_MODE (mask_type
));
355 /* Return true iff vec_cmpeq_optab can handle a vector comparison for code
356 CODE, comparing operands of type VALUE_TYPE and producing a result of type
360 vec_cmp_eq_icode_p (tree value_type
, tree mask_type
, enum tree_code code
)
362 if (code
!= EQ_EXPR
&& code
!= NE_EXPR
)
365 return get_vec_cmp_eq_icode (TYPE_MODE (value_type
), TYPE_MODE (mask_type
))
369 /* Return TRUE if appropriate vector insn is available
370 for vector comparison expr with vector type VALUE_TYPE
371 and resulting mask with MASK_TYPE. */
374 expand_vec_cmp_expr_p (tree value_type
, tree mask_type
, enum tree_code code
)
376 return vec_cmp_icode_p (value_type
, mask_type
, code
)
377 || vec_cmp_eq_icode_p (value_type
, mask_type
, code
);
380 /* Return true iff vcond_optab/vcondu_optab can handle a vector
381 comparison for code CODE, comparing operands of type CMP_OP_TYPE and
382 producing a result of type VALUE_TYPE. */
385 vcond_icode_p (tree value_type
, tree cmp_op_type
, enum tree_code code
)
387 enum rtx_code rcode
= get_rtx_code_1 (code
, TYPE_UNSIGNED (cmp_op_type
));
388 if (rcode
== UNKNOWN
)
391 return can_vcond_compare_p (rcode
, TYPE_MODE (value_type
),
392 TYPE_MODE (cmp_op_type
));
395 /* Return true iff vcondeq_optab can handle a vector comparison for code CODE,
396 comparing operands of type CMP_OP_TYPE and producing a result of type
400 vcond_eq_icode_p (tree value_type
, tree cmp_op_type
, enum tree_code code
)
402 if (code
!= EQ_EXPR
&& code
!= NE_EXPR
)
405 return get_vcond_eq_icode (TYPE_MODE (value_type
), TYPE_MODE (cmp_op_type
))
409 /* Return TRUE iff, appropriate vector insns are available
410 for vector cond expr with vector type VALUE_TYPE and a comparison
411 with operand vector types in CMP_OP_TYPE. */
414 expand_vec_cond_expr_p (tree value_type
, tree cmp_op_type
, enum tree_code code
)
416 machine_mode value_mode
= TYPE_MODE (value_type
);
417 machine_mode cmp_op_mode
= TYPE_MODE (cmp_op_type
);
418 if (VECTOR_BOOLEAN_TYPE_P (cmp_op_type
)
419 && get_vcond_mask_icode (TYPE_MODE (value_type
),
420 TYPE_MODE (cmp_op_type
)) != CODE_FOR_nothing
)
423 if (maybe_ne (GET_MODE_NUNITS (value_mode
), GET_MODE_NUNITS (cmp_op_mode
)))
426 if (TREE_CODE_CLASS (code
) != tcc_comparison
)
427 /* This may happen, for example, if code == SSA_NAME, in which case we
428 cannot be certain whether a vector insn is available. */
431 return vcond_icode_p (value_type
, cmp_op_type
, code
)
432 || vcond_eq_icode_p (value_type
, cmp_op_type
, code
);
435 /* Use the current target and options to initialize
436 TREE_OPTIMIZATION_OPTABS (OPTNODE). */
439 init_tree_optimization_optabs (tree optnode
)
441 /* Quick exit if we have already computed optabs for this target. */
442 if (TREE_OPTIMIZATION_BASE_OPTABS (optnode
) == this_target_optabs
)
445 /* Forget any previous information and set up for the current target. */
446 TREE_OPTIMIZATION_BASE_OPTABS (optnode
) = this_target_optabs
;
447 struct target_optabs
*tmp_optabs
= (struct target_optabs
*)
448 TREE_OPTIMIZATION_OPTABS (optnode
);
450 memset (tmp_optabs
, 0, sizeof (struct target_optabs
));
452 tmp_optabs
= ggc_cleared_alloc
<target_optabs
> ();
454 /* Generate a new set of optabs into tmp_optabs. */
455 init_all_optabs (tmp_optabs
);
457 /* If the optabs changed, record it. */
458 if (memcmp (tmp_optabs
, this_target_optabs
, sizeof (struct target_optabs
)))
459 TREE_OPTIMIZATION_OPTABS (optnode
) = tmp_optabs
;
462 TREE_OPTIMIZATION_OPTABS (optnode
) = NULL
;
463 ggc_free (tmp_optabs
);
467 /* Return TRUE if the target has support for vector right shift of an
468 operand of type TYPE. If OT_TYPE is OPTAB_DEFAULT, check for existence
469 of a shift by either a scalar or a vector. Otherwise, check only
470 for a shift that matches OT_TYPE. */
473 target_supports_op_p (tree type
, enum tree_code code
,
474 enum optab_subtype ot_subtype
)
476 optab ot
= optab_for_tree_code (code
, type
, ot_subtype
);
477 return (ot
!= unknown_optab
478 && optab_handler (ot
, TYPE_MODE (type
)) != CODE_FOR_nothing
);