Fix previous commit
[official-gcc.git] / gcc / optabs-tree.c
bloba5ecbf075b5d0cce6f1dfbcc51399fae87f5e1be
1 /* Tree-based target query functions relating to optabs
2 Copyright (C) 1987-2019 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_UNPACK_HI_EXPR:
174 return (TYPE_UNSIGNED (type)
175 ? vec_unpacku_hi_optab : vec_unpacks_hi_optab);
177 case VEC_UNPACK_LO_EXPR:
178 return (TYPE_UNSIGNED (type)
179 ? vec_unpacku_lo_optab : vec_unpacks_lo_optab);
181 case VEC_UNPACK_FLOAT_HI_EXPR:
182 /* The signedness is determined from input operand. */
183 return (TYPE_UNSIGNED (type)
184 ? vec_unpacku_float_hi_optab : vec_unpacks_float_hi_optab);
186 case VEC_UNPACK_FLOAT_LO_EXPR:
187 /* The signedness is determined from input operand. */
188 return (TYPE_UNSIGNED (type)
189 ? vec_unpacku_float_lo_optab : vec_unpacks_float_lo_optab);
191 case VEC_UNPACK_FIX_TRUNC_HI_EXPR:
192 /* The signedness is determined from output operand. */
193 return (TYPE_UNSIGNED (type)
194 ? vec_unpack_ufix_trunc_hi_optab
195 : vec_unpack_sfix_trunc_hi_optab);
197 case VEC_UNPACK_FIX_TRUNC_LO_EXPR:
198 /* The signedness is determined from output operand. */
199 return (TYPE_UNSIGNED (type)
200 ? vec_unpack_ufix_trunc_lo_optab
201 : vec_unpack_sfix_trunc_lo_optab);
203 case VEC_PACK_TRUNC_EXPR:
204 return vec_pack_trunc_optab;
206 case VEC_PACK_SAT_EXPR:
207 return TYPE_UNSIGNED (type) ? vec_pack_usat_optab : vec_pack_ssat_optab;
209 case VEC_PACK_FIX_TRUNC_EXPR:
210 /* The signedness is determined from output operand. */
211 return (TYPE_UNSIGNED (type)
212 ? vec_pack_ufix_trunc_optab : vec_pack_sfix_trunc_optab);
214 case VEC_PACK_FLOAT_EXPR:
215 /* The signedness is determined from input operand. */
216 return (TYPE_UNSIGNED (type)
217 ? vec_packu_float_optab : vec_packs_float_optab);
219 case VEC_DUPLICATE_EXPR:
220 return vec_duplicate_optab;
222 case VEC_SERIES_EXPR:
223 return vec_series_optab;
225 default:
226 break;
229 trapv = INTEGRAL_TYPE_P (type) && TYPE_OVERFLOW_TRAPS (type);
230 switch (code)
232 case POINTER_PLUS_EXPR:
233 case PLUS_EXPR:
234 if (TYPE_SATURATING (type))
235 return TYPE_UNSIGNED (type) ? usadd_optab : ssadd_optab;
236 return trapv ? addv_optab : add_optab;
238 case POINTER_DIFF_EXPR:
239 case MINUS_EXPR:
240 if (TYPE_SATURATING (type))
241 return TYPE_UNSIGNED (type) ? ussub_optab : sssub_optab;
242 return trapv ? subv_optab : sub_optab;
244 case MULT_EXPR:
245 if (TYPE_SATURATING (type))
246 return TYPE_UNSIGNED (type) ? usmul_optab : ssmul_optab;
247 return trapv ? smulv_optab : smul_optab;
249 case NEGATE_EXPR:
250 if (TYPE_SATURATING (type))
251 return TYPE_UNSIGNED (type) ? usneg_optab : ssneg_optab;
252 return trapv ? negv_optab : neg_optab;
254 case ABS_EXPR:
255 return trapv ? absv_optab : abs_optab;
257 case ABSU_EXPR:
258 return abs_optab;
259 default:
260 return unknown_optab;
264 /* Function supportable_convert_operation
266 Check whether an operation represented by the code CODE is a
267 convert operation that is supported by the target platform in
268 vector form (i.e., when operating on arguments of type VECTYPE_IN
269 producing a result of type VECTYPE_OUT).
271 Convert operations we currently support directly are FIX_TRUNC and FLOAT.
272 This function checks if these operations are supported
273 by the target platform either directly (via vector tree-codes), or via
274 target builtins.
276 Output:
277 - CODE1 is code of vector operation to be used when
278 vectorizing the operation, if available.
279 - DECL is decl of target builtin functions to be used
280 when vectorizing the operation, if available. In this case,
281 CODE1 is CALL_EXPR. */
283 bool
284 supportable_convert_operation (enum tree_code code,
285 tree vectype_out, tree vectype_in,
286 tree *decl, enum tree_code *code1)
288 machine_mode m1,m2;
289 bool truncp;
291 m1 = TYPE_MODE (vectype_out);
292 m2 = TYPE_MODE (vectype_in);
294 /* First check if we can done conversion directly. */
295 if ((code == FIX_TRUNC_EXPR
296 && can_fix_p (m1,m2,TYPE_UNSIGNED (vectype_out), &truncp)
297 != CODE_FOR_nothing)
298 || (code == FLOAT_EXPR
299 && can_float_p (m1,m2,TYPE_UNSIGNED (vectype_in))
300 != CODE_FOR_nothing))
302 *code1 = code;
303 return true;
306 /* Now check for builtin. */
307 if (targetm.vectorize.builtin_conversion
308 && targetm.vectorize.builtin_conversion (code, vectype_out, vectype_in))
310 *code1 = CALL_EXPR;
311 *decl = targetm.vectorize.builtin_conversion (code, vectype_out,
312 vectype_in);
313 return true;
315 return false;
318 /* Return TRUE if appropriate vector insn is available
319 for vector comparison expr with vector type VALUE_TYPE
320 and resulting mask with MASK_TYPE. */
322 bool
323 expand_vec_cmp_expr_p (tree value_type, tree mask_type, enum tree_code code)
325 if (get_vec_cmp_icode (TYPE_MODE (value_type), TYPE_MODE (mask_type),
326 TYPE_UNSIGNED (value_type)) != CODE_FOR_nothing)
327 return true;
328 if ((code == EQ_EXPR || code == NE_EXPR)
329 && (get_vec_cmp_eq_icode (TYPE_MODE (value_type), TYPE_MODE (mask_type))
330 != CODE_FOR_nothing))
331 return true;
332 return false;
335 /* Return true iff vcond_optab/vcondu_optab can handle a vector
336 comparison for code CODE, comparing operands of type CMP_OP_TYPE and
337 producing a result of type VALUE_TYPE. */
339 static bool
340 vcond_icode_p (tree value_type, tree cmp_op_type, enum tree_code code)
342 return can_vcond_compare_p (get_rtx_code (code, TYPE_UNSIGNED (cmp_op_type)),
343 TYPE_MODE (value_type), TYPE_MODE (cmp_op_type));
346 /* Return true iff vcondeq_optab can handle a vector comparison for code CODE,
347 comparing operands of type CMP_OP_TYPE and producing a result of type
348 VALUE_TYPE. */
350 static bool
351 vcond_eq_icode_p (tree value_type, tree cmp_op_type, enum tree_code code)
353 if (code != EQ_EXPR && code != NE_EXPR)
354 return false;
356 return get_vcond_eq_icode (TYPE_MODE (value_type), TYPE_MODE (cmp_op_type))
357 != CODE_FOR_nothing;
360 /* Return TRUE iff, appropriate vector insns are available
361 for vector cond expr with vector type VALUE_TYPE and a comparison
362 with operand vector types in CMP_OP_TYPE. */
364 bool
365 expand_vec_cond_expr_p (tree value_type, tree cmp_op_type, enum tree_code code)
367 machine_mode value_mode = TYPE_MODE (value_type);
368 machine_mode cmp_op_mode = TYPE_MODE (cmp_op_type);
369 if (VECTOR_BOOLEAN_TYPE_P (cmp_op_type)
370 && get_vcond_mask_icode (TYPE_MODE (value_type),
371 TYPE_MODE (cmp_op_type)) != CODE_FOR_nothing)
372 return true;
374 if (maybe_ne (GET_MODE_SIZE (value_mode), GET_MODE_SIZE (cmp_op_mode))
375 || maybe_ne (GET_MODE_NUNITS (value_mode), GET_MODE_NUNITS (cmp_op_mode)))
376 return false;
378 if (TREE_CODE_CLASS (code) != tcc_comparison)
379 /* This may happen, for example, if code == SSA_NAME, in which case we
380 cannot be certain whether a vector insn is available. */
381 return false;
383 return vcond_icode_p (value_type, cmp_op_type, code)
384 || vcond_eq_icode_p (value_type, cmp_op_type, code);
387 /* Use the current target and options to initialize
388 TREE_OPTIMIZATION_OPTABS (OPTNODE). */
390 void
391 init_tree_optimization_optabs (tree optnode)
393 /* Quick exit if we have already computed optabs for this target. */
394 if (TREE_OPTIMIZATION_BASE_OPTABS (optnode) == this_target_optabs)
395 return;
397 /* Forget any previous information and set up for the current target. */
398 TREE_OPTIMIZATION_BASE_OPTABS (optnode) = this_target_optabs;
399 struct target_optabs *tmp_optabs = (struct target_optabs *)
400 TREE_OPTIMIZATION_OPTABS (optnode);
401 if (tmp_optabs)
402 memset (tmp_optabs, 0, sizeof (struct target_optabs));
403 else
404 tmp_optabs = ggc_cleared_alloc<target_optabs> ();
406 /* Generate a new set of optabs into tmp_optabs. */
407 init_all_optabs (tmp_optabs);
409 /* If the optabs changed, record it. */
410 if (memcmp (tmp_optabs, this_target_optabs, sizeof (struct target_optabs)))
411 TREE_OPTIMIZATION_OPTABS (optnode) = tmp_optabs;
412 else
414 TREE_OPTIMIZATION_OPTABS (optnode) = NULL;
415 ggc_free (tmp_optabs);
419 /* Return TRUE if the target has support for vector right shift of an
420 operand of type TYPE. If OT_TYPE is OPTAB_DEFAULT, check for existence
421 of a shift by either a scalar or a vector. Otherwise, check only
422 for a shift that matches OT_TYPE. */
424 bool
425 target_supports_op_p (tree type, enum tree_code code,
426 enum optab_subtype ot_subtype)
428 optab ot = optab_for_tree_code (code, type, ot_subtype);
429 return (ot != unknown_optab
430 && optab_handler (ot, TYPE_MODE (type)) != CODE_FOR_nothing);