1 /* Convert tree expression to rtl instructions, for GNU compiler.
2 Copyright (C) 1988, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999,
3 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010
4 Free Software Foundation, Inc.
6 This file is part of GCC.
8 GCC is free software; you can redistribute it and/or modify it under
9 the terms of the GNU General Public License as published by the Free
10 Software Foundation; either version 3, or (at your option) any later
13 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
14 WARRANTY; without even the implied warranty of MERCHANTABILITY or
15 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
18 You should have received a copy of the GNU General Public License
19 along with GCC; see the file COPYING3. If not see
20 <http://www.gnu.org/licenses/>. */
24 #include "coretypes.h"
30 #include "insn-config.h"
31 #include "insn-attr.h"
32 /* Include expr.h after insn-config.h so we get HAVE_conditional_move. */
35 #include "langhooks.h"
37 #include "basic-block.h"
40 static bool prefer_and_bit_test (enum machine_mode
, int);
41 static void do_jump_by_parts_greater (tree
, tree
, int, rtx
, rtx
, int);
42 static void do_jump_by_parts_equality (tree
, tree
, rtx
, rtx
, int);
43 static void do_compare_and_jump (tree
, tree
, enum rtx_code
, enum rtx_code
, rtx
,
46 /* Invert probability if there is any. -1 stands for unknown. */
51 return prob
== -1 ? -1 : REG_BR_PROB_BASE
- prob
;
54 /* At the start of a function, record that we have no previously-pushed
55 arguments waiting to be popped. */
58 init_pending_stack_adjust (void)
60 pending_stack_adjust
= 0;
63 /* Discard any pending stack adjustment. This avoid relying on the
64 RTL optimizers to remove useless adjustments when we know the
65 stack pointer value is dead. */
67 discard_pending_stack_adjust (void)
69 stack_pointer_delta
-= pending_stack_adjust
;
70 pending_stack_adjust
= 0;
73 /* When exiting from function, if safe, clear out any pending stack adjust
74 so the adjustment won't get done.
76 Note, if the current function calls alloca, then it must have a
77 frame pointer regardless of the value of flag_omit_frame_pointer. */
80 clear_pending_stack_adjust (void)
83 && (! flag_omit_frame_pointer
|| cfun
->calls_alloca
)
85 discard_pending_stack_adjust ();
88 /* Pop any previously-pushed arguments that have not been popped yet. */
91 do_pending_stack_adjust (void)
93 if (inhibit_defer_pop
== 0)
95 if (pending_stack_adjust
!= 0)
96 adjust_stack (GEN_INT (pending_stack_adjust
));
97 pending_stack_adjust
= 0;
101 /* Expand conditional expressions. */
103 /* Generate code to evaluate EXP and jump to LABEL if the value is zero.
104 LABEL is an rtx of code CODE_LABEL, in this function and all the
108 jumpifnot (tree exp
, rtx label
, int prob
)
110 do_jump (exp
, label
, NULL_RTX
, inv (prob
));
114 jumpifnot_1 (enum tree_code code
, tree op0
, tree op1
, rtx label
, int prob
)
116 do_jump_1 (code
, op0
, op1
, label
, NULL_RTX
, inv (prob
));
119 /* Generate code to evaluate EXP and jump to LABEL if the value is nonzero. */
122 jumpif (tree exp
, rtx label
, int prob
)
124 do_jump (exp
, NULL_RTX
, label
, prob
);
128 jumpif_1 (enum tree_code code
, tree op0
, tree op1
, rtx label
, int prob
)
130 do_jump_1 (code
, op0
, op1
, NULL_RTX
, label
, prob
);
133 /* Used internally by prefer_and_bit_test. */
135 static GTY(()) rtx and_reg
;
136 static GTY(()) rtx and_test
;
137 static GTY(()) rtx shift_test
;
139 /* Compare the relative costs of "(X & (1 << BITNUM))" and "(X >> BITNUM) & 1",
140 where X is an arbitrary register of mode MODE. Return true if the former
144 prefer_and_bit_test (enum machine_mode mode
, int bitnum
)
150 /* Set up rtxes for the two variations. Use NULL as a placeholder
151 for the BITNUM-based constants. */
152 and_reg
= gen_rtx_REG (mode
, FIRST_PSEUDO_REGISTER
);
153 and_test
= gen_rtx_AND (mode
, and_reg
, NULL
);
154 shift_test
= gen_rtx_AND (mode
, gen_rtx_ASHIFTRT (mode
, and_reg
, NULL
),
159 /* Change the mode of the previously-created rtxes. */
160 PUT_MODE (and_reg
, mode
);
161 PUT_MODE (and_test
, mode
);
162 PUT_MODE (shift_test
, mode
);
163 PUT_MODE (XEXP (shift_test
, 0), mode
);
166 /* Fill in the integers. */
168 = immed_double_int_const (double_int_zero
.set_bit (bitnum
), mode
);
169 XEXP (XEXP (shift_test
, 0), 1) = GEN_INT (bitnum
);
171 speed_p
= optimize_insn_for_speed_p ();
172 return (rtx_cost (and_test
, IF_THEN_ELSE
, 0, speed_p
)
173 <= rtx_cost (shift_test
, IF_THEN_ELSE
, 0, speed_p
));
176 /* Subroutine of do_jump, dealing with exploded comparisons of the type
177 OP0 CODE OP1 . IF_FALSE_LABEL and IF_TRUE_LABEL like in do_jump.
178 PROB is probability of jump to if_true_label, or -1 if unknown. */
181 do_jump_1 (enum tree_code code
, tree op0
, tree op1
,
182 rtx if_false_label
, rtx if_true_label
, int prob
)
184 enum machine_mode mode
;
185 rtx drop_through_label
= 0;
191 tree inner_type
= TREE_TYPE (op0
);
193 gcc_assert (GET_MODE_CLASS (TYPE_MODE (inner_type
))
194 != MODE_COMPLEX_FLOAT
);
195 gcc_assert (GET_MODE_CLASS (TYPE_MODE (inner_type
))
196 != MODE_COMPLEX_INT
);
198 if (integer_zerop (op1
))
199 do_jump (op0
, if_true_label
, if_false_label
, inv (prob
));
200 else if (GET_MODE_CLASS (TYPE_MODE (inner_type
)) == MODE_INT
201 && !can_compare_p (EQ
, TYPE_MODE (inner_type
), ccp_jump
))
202 do_jump_by_parts_equality (op0
, op1
, if_false_label
, if_true_label
,
205 do_compare_and_jump (op0
, op1
, EQ
, EQ
, if_false_label
, if_true_label
,
212 tree inner_type
= TREE_TYPE (op0
);
214 gcc_assert (GET_MODE_CLASS (TYPE_MODE (inner_type
))
215 != MODE_COMPLEX_FLOAT
);
216 gcc_assert (GET_MODE_CLASS (TYPE_MODE (inner_type
))
217 != MODE_COMPLEX_INT
);
219 if (integer_zerop (op1
))
220 do_jump (op0
, if_false_label
, if_true_label
, prob
);
221 else if (GET_MODE_CLASS (TYPE_MODE (inner_type
)) == MODE_INT
222 && !can_compare_p (NE
, TYPE_MODE (inner_type
), ccp_jump
))
223 do_jump_by_parts_equality (op0
, op1
, if_true_label
, if_false_label
,
226 do_compare_and_jump (op0
, op1
, NE
, NE
, if_false_label
, if_true_label
,
232 mode
= TYPE_MODE (TREE_TYPE (op0
));
233 if (GET_MODE_CLASS (mode
) == MODE_INT
234 && ! can_compare_p (LT
, mode
, ccp_jump
))
235 do_jump_by_parts_greater (op0
, op1
, 1, if_false_label
, if_true_label
,
238 do_compare_and_jump (op0
, op1
, LT
, LTU
, if_false_label
, if_true_label
,
243 mode
= TYPE_MODE (TREE_TYPE (op0
));
244 if (GET_MODE_CLASS (mode
) == MODE_INT
245 && ! can_compare_p (LE
, mode
, ccp_jump
))
246 do_jump_by_parts_greater (op0
, op1
, 0, if_true_label
, if_false_label
,
249 do_compare_and_jump (op0
, op1
, LE
, LEU
, if_false_label
, if_true_label
,
254 mode
= TYPE_MODE (TREE_TYPE (op0
));
255 if (GET_MODE_CLASS (mode
) == MODE_INT
256 && ! can_compare_p (GT
, mode
, ccp_jump
))
257 do_jump_by_parts_greater (op0
, op1
, 0, if_false_label
, if_true_label
,
260 do_compare_and_jump (op0
, op1
, GT
, GTU
, if_false_label
, if_true_label
,
265 mode
= TYPE_MODE (TREE_TYPE (op0
));
266 if (GET_MODE_CLASS (mode
) == MODE_INT
267 && ! can_compare_p (GE
, mode
, ccp_jump
))
268 do_jump_by_parts_greater (op0
, op1
, 1, if_true_label
, if_false_label
,
271 do_compare_and_jump (op0
, op1
, GE
, GEU
, if_false_label
, if_true_label
,
276 do_compare_and_jump (op0
, op1
, ORDERED
, ORDERED
,
277 if_false_label
, if_true_label
, prob
);
281 do_compare_and_jump (op0
, op1
, UNORDERED
, UNORDERED
,
282 if_false_label
, if_true_label
, prob
);
286 do_compare_and_jump (op0
, op1
, UNLT
, UNLT
, if_false_label
, if_true_label
,
291 do_compare_and_jump (op0
, op1
, UNLE
, UNLE
, if_false_label
, if_true_label
,
296 do_compare_and_jump (op0
, op1
, UNGT
, UNGT
, if_false_label
, if_true_label
,
301 do_compare_and_jump (op0
, op1
, UNGE
, UNGE
, if_false_label
, if_true_label
,
306 do_compare_and_jump (op0
, op1
, UNEQ
, UNEQ
, if_false_label
, if_true_label
,
311 do_compare_and_jump (op0
, op1
, LTGT
, LTGT
, if_false_label
, if_true_label
,
315 case TRUTH_ANDIF_EXPR
:
316 if (if_false_label
== NULL_RTX
)
318 drop_through_label
= gen_label_rtx ();
319 do_jump (op0
, drop_through_label
, NULL_RTX
, prob
);
320 do_jump (op1
, NULL_RTX
, if_true_label
, prob
);
324 do_jump (op0
, if_false_label
, NULL_RTX
, prob
);
325 do_jump (op1
, if_false_label
, if_true_label
, prob
);
329 case TRUTH_ORIF_EXPR
:
330 if (if_true_label
== NULL_RTX
)
332 drop_through_label
= gen_label_rtx ();
333 do_jump (op0
, NULL_RTX
, drop_through_label
, prob
);
334 do_jump (op1
, if_false_label
, NULL_RTX
, prob
);
338 do_jump (op0
, NULL_RTX
, if_true_label
, prob
);
339 do_jump (op1
, if_false_label
, if_true_label
, prob
);
347 if (drop_through_label
)
349 do_pending_stack_adjust ();
350 emit_label (drop_through_label
);
354 /* Generate code to evaluate EXP and jump to IF_FALSE_LABEL if
355 the result is zero, or IF_TRUE_LABEL if the result is one.
356 Either of IF_FALSE_LABEL and IF_TRUE_LABEL may be zero,
357 meaning fall through in that case.
359 do_jump always does any pending stack adjust except when it does not
360 actually perform a jump. An example where there is no jump
361 is when EXP is `(foo (), 0)' and IF_FALSE_LABEL is null.
363 PROB is probability of jump to if_true_label, or -1 if unknown. */
366 do_jump (tree exp
, rtx if_false_label
, rtx if_true_label
, int prob
)
368 enum tree_code code
= TREE_CODE (exp
);
372 enum machine_mode mode
;
373 rtx drop_through_label
= 0;
381 temp
= integer_zerop (exp
) ? if_false_label
: if_true_label
;
387 /* This is not true with #pragma weak */
389 /* The address of something can never be zero. */
391 emit_jump (if_true_label
);
396 if (TREE_CODE (TREE_OPERAND (exp
, 0)) == COMPONENT_REF
397 || TREE_CODE (TREE_OPERAND (exp
, 0)) == BIT_FIELD_REF
398 || TREE_CODE (TREE_OPERAND (exp
, 0)) == ARRAY_REF
399 || TREE_CODE (TREE_OPERAND (exp
, 0)) == ARRAY_RANGE_REF
)
402 /* If we are narrowing the operand, we have to do the compare in the
404 if ((TYPE_PRECISION (TREE_TYPE (exp
))
405 < TYPE_PRECISION (TREE_TYPE (TREE_OPERAND (exp
, 0)))))
407 case NON_LVALUE_EXPR
:
412 /* These cannot change zero->nonzero or vice versa. */
413 do_jump (TREE_OPERAND (exp
, 0), if_false_label
, if_true_label
, prob
);
417 do_jump (TREE_OPERAND (exp
, 0), if_true_label
, if_false_label
,
423 rtx label1
= gen_label_rtx ();
424 if (!if_true_label
|| !if_false_label
)
426 drop_through_label
= gen_label_rtx ();
428 if_true_label
= drop_through_label
;
430 if_false_label
= drop_through_label
;
433 do_pending_stack_adjust ();
434 do_jump (TREE_OPERAND (exp
, 0), label1
, NULL_RTX
, -1);
435 do_jump (TREE_OPERAND (exp
, 1), if_false_label
, if_true_label
, prob
);
437 do_jump (TREE_OPERAND (exp
, 2), if_false_label
, if_true_label
, prob
);
442 /* Lowered by gimplify.c. */
446 /* Nonzero iff operands of minus differ. */
464 case TRUTH_ANDIF_EXPR
:
465 case TRUTH_ORIF_EXPR
:
467 do_jump_1 (code
, TREE_OPERAND (exp
, 0), TREE_OPERAND (exp
, 1),
468 if_false_label
, if_true_label
, prob
);
472 /* fold_single_bit_test() converts (X & (1 << C)) into (X >> C) & 1.
473 See if the former is preferred for jump tests and restore it
475 if (integer_onep (TREE_OPERAND (exp
, 1)))
477 tree exp0
= TREE_OPERAND (exp
, 0);
478 rtx set_label
, clr_label
;
479 int setclr_prob
= prob
;
481 /* Strip narrowing integral type conversions. */
482 while (CONVERT_EXPR_P (exp0
)
483 && TREE_OPERAND (exp0
, 0) != error_mark_node
484 && TYPE_PRECISION (TREE_TYPE (exp0
))
485 <= TYPE_PRECISION (TREE_TYPE (TREE_OPERAND (exp0
, 0))))
486 exp0
= TREE_OPERAND (exp0
, 0);
488 /* "exp0 ^ 1" inverts the sense of the single bit test. */
489 if (TREE_CODE (exp0
) == BIT_XOR_EXPR
490 && integer_onep (TREE_OPERAND (exp0
, 1)))
492 exp0
= TREE_OPERAND (exp0
, 0);
493 clr_label
= if_true_label
;
494 set_label
= if_false_label
;
495 setclr_prob
= inv (prob
);
499 clr_label
= if_false_label
;
500 set_label
= if_true_label
;
503 if (TREE_CODE (exp0
) == RSHIFT_EXPR
)
505 tree arg
= TREE_OPERAND (exp0
, 0);
506 tree shift
= TREE_OPERAND (exp0
, 1);
507 tree argtype
= TREE_TYPE (arg
);
508 if (TREE_CODE (shift
) == INTEGER_CST
509 && compare_tree_int (shift
, 0) >= 0
510 && compare_tree_int (shift
, HOST_BITS_PER_WIDE_INT
) < 0
511 && prefer_and_bit_test (TYPE_MODE (argtype
),
512 TREE_INT_CST_LOW (shift
)))
514 unsigned HOST_WIDE_INT mask
515 = (unsigned HOST_WIDE_INT
) 1 << TREE_INT_CST_LOW (shift
);
516 do_jump (build2 (BIT_AND_EXPR
, argtype
, arg
,
517 build_int_cstu (argtype
, mask
)),
518 clr_label
, set_label
, setclr_prob
);
524 /* If we are AND'ing with a small constant, do this comparison in the
525 smallest type that fits. If the machine doesn't have comparisons
526 that small, it will be converted back to the wider comparison.
527 This helps if we are testing the sign bit of a narrower object.
528 combine can't do this for us because it can't know whether a
529 ZERO_EXTRACT or a compare in a smaller mode exists, but we do. */
531 if (! SLOW_BYTE_ACCESS
532 && TREE_CODE (TREE_OPERAND (exp
, 1)) == INTEGER_CST
533 && TYPE_PRECISION (TREE_TYPE (exp
)) <= HOST_BITS_PER_WIDE_INT
534 && (i
= tree_floor_log2 (TREE_OPERAND (exp
, 1))) >= 0
535 && (mode
= mode_for_size (i
+ 1, MODE_INT
, 0)) != BLKmode
536 && (type
= lang_hooks
.types
.type_for_mode (mode
, 1)) != 0
537 && TYPE_PRECISION (type
) < TYPE_PRECISION (TREE_TYPE (exp
))
538 && have_insn_for (COMPARE
, TYPE_MODE (type
)))
540 do_jump (fold_convert (type
, exp
), if_false_label
, if_true_label
,
545 if (TYPE_PRECISION (TREE_TYPE (exp
)) > 1
546 || TREE_CODE (TREE_OPERAND (exp
, 1)) == INTEGER_CST
)
549 /* Boolean comparisons can be compiled as TRUTH_AND_EXPR. */
552 /* High branch cost, expand as the bitwise AND of the conditions.
553 Do the same if the RHS has side effects, because we're effectively
554 turning a TRUTH_AND_EXPR into a TRUTH_ANDIF_EXPR. */
555 if (BRANCH_COST (optimize_insn_for_speed_p (),
557 || TREE_SIDE_EFFECTS (TREE_OPERAND (exp
, 1)))
559 code
= TRUTH_ANDIF_EXPR
;
564 /* High branch cost, expand as the bitwise OR of the conditions.
565 Do the same if the RHS has side effects, because we're effectively
566 turning a TRUTH_OR_EXPR into a TRUTH_ORIF_EXPR. */
567 if (BRANCH_COST (optimize_insn_for_speed_p (), false) >= 4
568 || TREE_SIDE_EFFECTS (TREE_OPERAND (exp
, 1)))
570 code
= TRUTH_ORIF_EXPR
;
573 /* Fall through and generate the normal code. */
576 temp
= expand_normal (exp
);
577 do_pending_stack_adjust ();
578 /* The RTL optimizers prefer comparisons against pseudos. */
579 if (GET_CODE (temp
) == SUBREG
)
581 /* Compare promoted variables in their promoted mode. */
582 if (SUBREG_PROMOTED_VAR_P (temp
)
583 && REG_P (XEXP (temp
, 0)))
584 temp
= XEXP (temp
, 0);
586 temp
= copy_to_reg (temp
);
588 do_compare_rtx_and_jump (temp
, CONST0_RTX (GET_MODE (temp
)),
589 NE
, TYPE_UNSIGNED (TREE_TYPE (exp
)),
590 GET_MODE (temp
), NULL_RTX
,
591 if_false_label
, if_true_label
, prob
);
594 if (drop_through_label
)
596 do_pending_stack_adjust ();
597 emit_label (drop_through_label
);
601 /* Compare OP0 with OP1, word at a time, in mode MODE.
602 UNSIGNEDP says to do unsigned comparison.
603 Jump to IF_TRUE_LABEL if OP0 is greater, IF_FALSE_LABEL otherwise. */
606 do_jump_by_parts_greater_rtx (enum machine_mode mode
, int unsignedp
, rtx op0
,
607 rtx op1
, rtx if_false_label
, rtx if_true_label
,
610 int nwords
= (GET_MODE_SIZE (mode
) / UNITS_PER_WORD
);
611 rtx drop_through_label
= 0;
612 bool drop_through_if_true
= false, drop_through_if_false
= false;
613 enum rtx_code code
= GT
;
616 if (! if_true_label
|| ! if_false_label
)
617 drop_through_label
= gen_label_rtx ();
620 if_true_label
= drop_through_label
;
621 drop_through_if_true
= true;
623 if (! if_false_label
)
625 if_false_label
= drop_through_label
;
626 drop_through_if_false
= true;
629 /* Deal with the special case 0 > x: only one comparison is necessary and
630 we reverse it to avoid jumping to the drop-through label. */
631 if (op0
== const0_rtx
&& drop_through_if_true
&& !drop_through_if_false
)
634 if_true_label
= if_false_label
;
635 if_false_label
= drop_through_label
;
636 drop_through_if_true
= false;
637 drop_through_if_false
= true;
640 /* Compare a word at a time, high order first. */
641 for (i
= 0; i
< nwords
; i
++)
643 rtx op0_word
, op1_word
;
645 if (WORDS_BIG_ENDIAN
)
647 op0_word
= operand_subword_force (op0
, i
, mode
);
648 op1_word
= operand_subword_force (op1
, i
, mode
);
652 op0_word
= operand_subword_force (op0
, nwords
- 1 - i
, mode
);
653 op1_word
= operand_subword_force (op1
, nwords
- 1 - i
, mode
);
656 /* All but high-order word must be compared as unsigned. */
657 do_compare_rtx_and_jump (op0_word
, op1_word
, code
, (unsignedp
|| i
> 0),
658 word_mode
, NULL_RTX
, NULL_RTX
, if_true_label
,
661 /* Emit only one comparison for 0. Do not emit the last cond jump. */
662 if (op0
== const0_rtx
|| i
== nwords
- 1)
665 /* Consider lower words only if these are equal. */
666 do_compare_rtx_and_jump (op0_word
, op1_word
, NE
, unsignedp
, word_mode
,
667 NULL_RTX
, NULL_RTX
, if_false_label
, inv (prob
));
670 if (!drop_through_if_false
)
671 emit_jump (if_false_label
);
672 if (drop_through_label
)
673 emit_label (drop_through_label
);
676 /* Given a comparison expression EXP for values too wide to be compared
677 with one insn, test the comparison and jump to the appropriate label.
678 The code of EXP is ignored; we always test GT if SWAP is 0,
679 and LT if SWAP is 1. */
682 do_jump_by_parts_greater (tree treeop0
, tree treeop1
, int swap
,
683 rtx if_false_label
, rtx if_true_label
, int prob
)
685 rtx op0
= expand_normal (swap
? treeop1
: treeop0
);
686 rtx op1
= expand_normal (swap
? treeop0
: treeop1
);
687 enum machine_mode mode
= TYPE_MODE (TREE_TYPE (treeop0
));
688 int unsignedp
= TYPE_UNSIGNED (TREE_TYPE (treeop0
));
690 do_jump_by_parts_greater_rtx (mode
, unsignedp
, op0
, op1
, if_false_label
,
691 if_true_label
, prob
);
694 /* Jump according to whether OP0 is 0. We assume that OP0 has an integer
695 mode, MODE, that is too wide for the available compare insns. Either
696 Either (but not both) of IF_TRUE_LABEL and IF_FALSE_LABEL may be NULL_RTX
697 to indicate drop through. */
700 do_jump_by_parts_zero_rtx (enum machine_mode mode
, rtx op0
,
701 rtx if_false_label
, rtx if_true_label
, int prob
)
703 int nwords
= GET_MODE_SIZE (mode
) / UNITS_PER_WORD
;
706 rtx drop_through_label
= 0;
708 /* The fastest way of doing this comparison on almost any machine is to
709 "or" all the words and compare the result. If all have to be loaded
710 from memory and this is a very wide item, it's possible this may
711 be slower, but that's highly unlikely. */
713 part
= gen_reg_rtx (word_mode
);
714 emit_move_insn (part
, operand_subword_force (op0
, 0, mode
));
715 for (i
= 1; i
< nwords
&& part
!= 0; i
++)
716 part
= expand_binop (word_mode
, ior_optab
, part
,
717 operand_subword_force (op0
, i
, mode
),
718 part
, 1, OPTAB_WIDEN
);
722 do_compare_rtx_and_jump (part
, const0_rtx
, EQ
, 1, word_mode
,
723 NULL_RTX
, if_false_label
, if_true_label
, prob
);
727 /* If we couldn't do the "or" simply, do this with a series of compares. */
728 if (! if_false_label
)
729 drop_through_label
= if_false_label
= gen_label_rtx ();
731 for (i
= 0; i
< nwords
; i
++)
732 do_compare_rtx_and_jump (operand_subword_force (op0
, i
, mode
),
733 const0_rtx
, EQ
, 1, word_mode
, NULL_RTX
,
734 if_false_label
, NULL_RTX
, prob
);
737 emit_jump (if_true_label
);
739 if (drop_through_label
)
740 emit_label (drop_through_label
);
743 /* Test for the equality of two RTX expressions OP0 and OP1 in mode MODE,
744 where MODE is an integer mode too wide to be compared with one insn.
745 Either (but not both) of IF_TRUE_LABEL and IF_FALSE_LABEL may be NULL_RTX
746 to indicate drop through. */
749 do_jump_by_parts_equality_rtx (enum machine_mode mode
, rtx op0
, rtx op1
,
750 rtx if_false_label
, rtx if_true_label
, int prob
)
752 int nwords
= (GET_MODE_SIZE (mode
) / UNITS_PER_WORD
);
753 rtx drop_through_label
= 0;
756 if (op1
== const0_rtx
)
758 do_jump_by_parts_zero_rtx (mode
, op0
, if_false_label
, if_true_label
,
762 else if (op0
== const0_rtx
)
764 do_jump_by_parts_zero_rtx (mode
, op1
, if_false_label
, if_true_label
,
769 if (! if_false_label
)
770 drop_through_label
= if_false_label
= gen_label_rtx ();
772 for (i
= 0; i
< nwords
; i
++)
773 do_compare_rtx_and_jump (operand_subword_force (op0
, i
, mode
),
774 operand_subword_force (op1
, i
, mode
),
775 EQ
, 0, word_mode
, NULL_RTX
,
776 if_false_label
, NULL_RTX
, prob
);
779 emit_jump (if_true_label
);
780 if (drop_through_label
)
781 emit_label (drop_through_label
);
784 /* Given an EQ_EXPR expression EXP for values too wide to be compared
785 with one insn, test the comparison and jump to the appropriate label. */
788 do_jump_by_parts_equality (tree treeop0
, tree treeop1
, rtx if_false_label
,
789 rtx if_true_label
, int prob
)
791 rtx op0
= expand_normal (treeop0
);
792 rtx op1
= expand_normal (treeop1
);
793 enum machine_mode mode
= TYPE_MODE (TREE_TYPE (treeop0
));
794 do_jump_by_parts_equality_rtx (mode
, op0
, op1
, if_false_label
,
795 if_true_label
, prob
);
798 /* Split a comparison into two others, the second of which has the other
799 "orderedness". The first is always ORDERED or UNORDERED if MODE
800 does not honor NaNs (which means that it can be skipped in that case;
801 see do_compare_rtx_and_jump).
803 The two conditions are written in *CODE1 and *CODE2. Return true if
804 the conditions must be ANDed, false if they must be ORed. */
807 split_comparison (enum rtx_code code
, enum machine_mode mode
,
808 enum rtx_code
*code1
, enum rtx_code
*code2
)
857 /* Do not turn a trapping comparison into a non-trapping one. */
858 if (HONOR_SNANS (mode
))
876 /* Like do_compare_and_jump but expects the values to compare as two rtx's.
877 The decision as to signed or unsigned comparison must be made by the caller.
879 If MODE is BLKmode, SIZE is an RTX giving the size of the objects being
883 do_compare_rtx_and_jump (rtx op0
, rtx op1
, enum rtx_code code
, int unsignedp
,
884 enum machine_mode mode
, rtx size
, rtx if_false_label
,
885 rtx if_true_label
, int prob
)
888 rtx dummy_label
= NULL_RTX
;
890 /* Reverse the comparison if that is safe and we want to jump if it is
891 false. Also convert to the reverse comparison if the target can
894 || ! can_compare_p (code
, mode
, ccp_jump
))
895 && (! FLOAT_MODE_P (mode
)
896 || code
== ORDERED
|| code
== UNORDERED
897 || (! HONOR_NANS (mode
) && (code
== LTGT
|| code
== UNEQ
))
898 || (! HONOR_SNANS (mode
) && (code
== EQ
|| code
== NE
))))
901 if (FLOAT_MODE_P (mode
))
902 rcode
= reverse_condition_maybe_unordered (code
);
904 rcode
= reverse_condition (code
);
906 /* Canonicalize to UNORDERED for the libcall. */
907 if (can_compare_p (rcode
, mode
, ccp_jump
)
908 || (code
== ORDERED
&& ! can_compare_p (ORDERED
, mode
, ccp_jump
)))
911 if_true_label
= if_false_label
;
912 if_false_label
= tem
;
918 /* If one operand is constant, make it the second one. Only do this
919 if the other operand is not constant as well. */
921 if (swap_commutative_operands_p (op0
, op1
))
926 code
= swap_condition (code
);
929 do_pending_stack_adjust ();
931 code
= unsignedp
? unsigned_condition (code
) : code
;
932 if (0 != (tem
= simplify_relational_operation (code
, mode
, VOIDmode
,
935 if (CONSTANT_P (tem
))
937 rtx label
= (tem
== const0_rtx
|| tem
== CONST0_RTX (mode
))
938 ? if_false_label
: if_true_label
;
944 code
= GET_CODE (tem
);
945 mode
= GET_MODE (tem
);
948 unsignedp
= (code
== GTU
|| code
== LTU
|| code
== GEU
|| code
== LEU
);
952 dummy_label
= if_true_label
= gen_label_rtx ();
954 if (GET_MODE_CLASS (mode
) == MODE_INT
955 && ! can_compare_p (code
, mode
, ccp_jump
))
960 do_jump_by_parts_greater_rtx (mode
, 1, op1
, op0
,
961 if_false_label
, if_true_label
, prob
);
965 do_jump_by_parts_greater_rtx (mode
, 1, op0
, op1
,
966 if_true_label
, if_false_label
,
971 do_jump_by_parts_greater_rtx (mode
, 1, op0
, op1
,
972 if_false_label
, if_true_label
, prob
);
976 do_jump_by_parts_greater_rtx (mode
, 1, op1
, op0
,
977 if_true_label
, if_false_label
,
982 do_jump_by_parts_greater_rtx (mode
, 0, op1
, op0
,
983 if_false_label
, if_true_label
, prob
);
987 do_jump_by_parts_greater_rtx (mode
, 0, op0
, op1
,
988 if_true_label
, if_false_label
,
993 do_jump_by_parts_greater_rtx (mode
, 0, op0
, op1
,
994 if_false_label
, if_true_label
, prob
);
998 do_jump_by_parts_greater_rtx (mode
, 0, op1
, op0
,
999 if_true_label
, if_false_label
,
1004 do_jump_by_parts_equality_rtx (mode
, op0
, op1
, if_false_label
,
1005 if_true_label
, prob
);
1009 do_jump_by_parts_equality_rtx (mode
, op0
, op1
, if_true_label
,
1010 if_false_label
, inv (prob
));
1019 if (SCALAR_FLOAT_MODE_P (mode
)
1020 && ! can_compare_p (code
, mode
, ccp_jump
)
1021 && can_compare_p (swap_condition (code
), mode
, ccp_jump
))
1024 code
= swap_condition (code
);
1029 else if (SCALAR_FLOAT_MODE_P (mode
)
1030 && ! can_compare_p (code
, mode
, ccp_jump
)
1031 /* Never split ORDERED and UNORDERED.
1032 These must be implemented. */
1033 && (code
!= ORDERED
&& code
!= UNORDERED
)
1034 /* Split a floating-point comparison if
1035 we can jump on other conditions... */
1036 && (have_insn_for (COMPARE
, mode
)
1037 /* ... or if there is no libcall for it. */
1038 || code_to_optab (code
) == unknown_optab
))
1040 enum rtx_code first_code
;
1041 bool and_them
= split_comparison (code
, mode
, &first_code
, &code
);
1043 /* If there are no NaNs, the first comparison should always fall
1045 if (!HONOR_NANS (mode
))
1046 gcc_assert (first_code
== (and_them
? ORDERED
: UNORDERED
));
1053 /* If we only jump if true, just bypass the second jump. */
1054 if (! if_false_label
)
1057 dummy_label
= gen_label_rtx ();
1058 dest_label
= dummy_label
;
1061 dest_label
= if_false_label
;
1062 do_compare_rtx_and_jump (op0
, op1
, first_code
, unsignedp
, mode
,
1063 size
, dest_label
, NULL_RTX
, prob
);
1066 do_compare_rtx_and_jump (op0
, op1
, first_code
, unsignedp
, mode
,
1067 size
, NULL_RTX
, if_true_label
, prob
);
1071 emit_cmp_and_jump_insns (op0
, op1
, code
, size
, mode
, unsignedp
,
1072 if_true_label
, prob
);
1076 emit_jump (if_false_label
);
1078 emit_label (dummy_label
);
1081 /* Generate code for a comparison expression EXP (including code to compute
1082 the values to be compared) and a conditional jump to IF_FALSE_LABEL and/or
1083 IF_TRUE_LABEL. One of the labels can be NULL_RTX, in which case the
1084 generated code will drop through.
1085 SIGNED_CODE should be the rtx operation for this comparison for
1086 signed data; UNSIGNED_CODE, likewise for use if data is unsigned.
1088 We force a stack adjustment unless there are currently
1089 things pushed on the stack that aren't yet used. */
1092 do_compare_and_jump (tree treeop0
, tree treeop1
, enum rtx_code signed_code
,
1093 enum rtx_code unsigned_code
, rtx if_false_label
,
1094 rtx if_true_label
, int prob
)
1098 enum machine_mode mode
;
1102 /* Don't crash if the comparison was erroneous. */
1103 op0
= expand_normal (treeop0
);
1104 if (TREE_CODE (treeop0
) == ERROR_MARK
)
1107 op1
= expand_normal (treeop1
);
1108 if (TREE_CODE (treeop1
) == ERROR_MARK
)
1111 type
= TREE_TYPE (treeop0
);
1112 mode
= TYPE_MODE (type
);
1113 if (TREE_CODE (treeop0
) == INTEGER_CST
1114 && (TREE_CODE (treeop1
) != INTEGER_CST
1115 || (GET_MODE_BITSIZE (mode
)
1116 > GET_MODE_BITSIZE (TYPE_MODE (TREE_TYPE (treeop1
))))))
1118 /* op0 might have been replaced by promoted constant, in which
1119 case the type of second argument should be used. */
1120 type
= TREE_TYPE (treeop1
);
1121 mode
= TYPE_MODE (type
);
1123 unsignedp
= TYPE_UNSIGNED (type
);
1124 code
= unsignedp
? unsigned_code
: signed_code
;
1126 #ifdef HAVE_canonicalize_funcptr_for_compare
1127 /* If function pointers need to be "canonicalized" before they can
1128 be reliably compared, then canonicalize them.
1129 Only do this if *both* sides of the comparison are function pointers.
1130 If one side isn't, we want a noncanonicalized comparison. See PR
1131 middle-end/17564. */
1132 if (HAVE_canonicalize_funcptr_for_compare
1133 && TREE_CODE (TREE_TYPE (treeop0
)) == POINTER_TYPE
1134 && TREE_CODE (TREE_TYPE (TREE_TYPE (treeop0
)))
1136 && TREE_CODE (TREE_TYPE (treeop1
)) == POINTER_TYPE
1137 && TREE_CODE (TREE_TYPE (TREE_TYPE (treeop1
)))
1140 rtx new_op0
= gen_reg_rtx (mode
);
1141 rtx new_op1
= gen_reg_rtx (mode
);
1143 emit_insn (gen_canonicalize_funcptr_for_compare (new_op0
, op0
));
1146 emit_insn (gen_canonicalize_funcptr_for_compare (new_op1
, op1
));
1151 do_compare_rtx_and_jump (op0
, op1
, code
, unsignedp
, mode
,
1153 ? expr_size (treeop0
) : NULL_RTX
),
1154 if_false_label
, if_true_label
, prob
);
1157 #include "gt-dojump.h"