i386.c (ix86_expand_fp_movcc): Do not attempt to construct SSE based conditional...
authorJan Hubicka <jh@suse.cz>
Fri, 16 Mar 2001 13:51:49 +0000 (14:51 +0100)
committerJan Hubicka <hubicka@gcc.gnu.org>
Fri, 16 Mar 2001 13:51:49 +0000 (13:51 +0000)
* i386.c (ix86_expand_fp_movcc): Do not attempt to construct
SSE based conditional moves on LTGT/UNEQ conditions;
Canonicalize EQ to NE.
* i386.md (sse_mov?fcc): Disallow EQ and NE in IEEE mode.
(sse_mov?fcc_ne): New.

* i386-protos.h (sign_extended_value, zero_extended_value): Declare.

From-SVN: r40554

gcc/ChangeLog
gcc/config/i386/i386-protos.h
gcc/config/i386/i386.c
gcc/config/i386/i386.md

index 7d3d804e43c727d83f5bccd9dbf9608f65607965..27699c80f23d8ff81c64c7f3e6085eb6384b187c 100644 (file)
@@ -1,3 +1,13 @@
+Fri Mar 16 14:47:57 CET 2001  Jan Hubicka  <jh@suse.cz>
+
+       * i386.c (ix86_expand_fp_movcc): Do not attempt to construct
+       SSE based conditional moves on LTGT/UNEQ conditions;
+       Canonicalize EQ to NE.
+       * i386.md (sse_mov?fcc): Disallow EQ and NE in IEEE mode.
+       (sse_mov?fcc_ne): New.
+
+       * i386-protos.h (sign_extended_value, zero_extended_value): Declare.
+
 2001-03-16  Alexandre Oliva  <aoliva@redhat.com>
 
        * config/sh/sh.c (sh_expand_prologue): Insns that set up the PIC
index 60b2998100e88d74bd1d9cc1fdc457cd6d089848..e4639a34b61b1cf055ab53ff31179d7194ebd345 100644 (file)
@@ -128,6 +128,9 @@ extern int ix86_sched_reorder PARAMS ((FILE *, int, rtx *, int, int));
 extern int ix86_variable_issue PARAMS ((FILE *, int, rtx, int));
 extern enum machine_mode ix86_fp_compare_mode PARAMS ((enum rtx_code));
 
+extern int x86_64_sign_extended_value PARAMS ((rtx));
+extern int x86_64_zero_extended_value PARAMS ((rtx));
+
 extern rtx ix86_force_to_memory PARAMS ((enum machine_mode, rtx));
 extern void ix86_free_from_memory PARAMS ((enum machine_mode));
 extern void ix86_split_fp_branch PARAMS ((rtx, rtx, rtx, rtx, rtx, rtx));
index 16dfe454b3976f59ea11432c211ff49ee033d21e..20b7eca7eb570a83e479650ebd181ff4d2492c81 100644 (file)
@@ -1844,7 +1844,7 @@ x86_64_sign_extended_value (value)
        else
          {
            HOST_WIDE_INT val = trunc_int_for_mode (INTVAL (value), DImode);
-           return (HOST_WIDE_INT)(int)val == val;
+           return trunc_int_for_mode (val, SImode) == val;
          }
        break;
 
@@ -6312,6 +6312,9 @@ ix86_expand_fp_movcc (operands)
   if (((TARGET_SSE && GET_MODE (operands[0]) == SFmode)
        || (TARGET_SSE2 && GET_MODE (operands[0]) == DFmode))
       && GET_MODE (ix86_compare_op0) == GET_MODE (operands[0])
+      /* The SSE comparisons does not support the LTGT/UNEQ pair.  */
+      && (!TARGET_IEEE_FP
+         || (GET_CODE (operands[1]) != LTGT && GET_CODE (operands[1]) != UNEQ))
       /* We may be called from the post-reload splitter.  */
       && (!REG_P (operands[0])
          || SSE_REG_P (operands[0])
@@ -6371,8 +6374,10 @@ ix86_expand_fp_movcc (operands)
                                        ix86_compare_op1);
        }
       /* Similary try to manage result to be first operand of conditional
-        move. */
-      if (rtx_equal_p (operands[0], operands[3]))
+        move. We also don't support the NE comparison on SSE, so try to
+        avoid it.  */
+      if (rtx_equal_p (operands[0], operands[3])
+         || GET_CODE (operands[1]) == NE)
        {
          rtx tmp = operands[2];
          operands[2] = operands[3];
index c700f2317b025c6756e52677d9fb6ebef0d1fd82..1f36c5fd78eb725c8085f80e5501618fcb99ecbf 100644 (file)
 ;; based moves emulation or to usual cmove sequence.  Little bit unfortunate
 ;; fact is that compares supported by the cmp??ss instructions are exactly
 ;; swapped of those supported by cmove sequence.
+;; The EQ/NE comparisons also needs bit care, since they are not directly
+;; supported by i387 comparisons and we do need to emit two conditional moves
+;; in tandem.
 
 (define_insn "sse_movsfcc"
   [(set (match_operand:SF 0 "register_operand" "=&x#rf,x#rf,?f#xr,?f#xr,?f#xr,?f#xr,?r#xf,?r#xf,?r#xf,?r#xf")
                      (match_operand:SF 3 "nonimmediate_operand" "x#fr,x#fr,0#fx,f#fx,0#fx,f#fx,0#fx,rm#rx,0#rx,rm#rx")))
    (clobber (match_scratch:SF 6 "=2,&4,X,X,X,X,X,X,X,X"))
    (clobber (reg:CC 17))]
+  "TARGET_SSE
+   && (GET_CODE (operands[2]) != MEM || GET_CODE (operands[3]) != MEM)
+   && (!TARGET_IEEE_FP
+       || (GET_CODE (operands[1]) != EQ && GET_CODE (operands[1]) != NE))"
+  "#")
+
+(define_insn "sse_movsfcc_eq"
+  [(set (match_operand:SF 0 "register_operand" "=&x#rf,x#rf,?f#xr,?f#xr,?r#xf,?r#xf")
+       (if_then_else:SF (eq (match_operand:SF 3 "nonimmediate_operand" "%0#fx,x#fx,f#x,xm#f,f#x,xm#f")
+                            (match_operand:SF 4 "nonimmediate_operand" "xm#f,xm#f,f#x,x#f,f#x,x#f"))
+                     (match_operand:SF 1 "nonimmediate_operand" "x#fr,0#fr,0#fx,0#fx,0#rx,0#rx")
+                     (match_operand:SF 2 "nonimmediate_operand" "x#fr,x#fr,f#fx,f#fx,rm#rx,rm#rx")))
+   (clobber (match_scratch:SF 5 "=1,&4,X,X,X,X"))
+   (clobber (reg:CC 17))]
   "TARGET_SSE
    && (GET_CODE (operands[2]) != MEM || GET_CODE (operands[3]) != MEM)"
   "#")
    (clobber (match_scratch:DF 6 "=2,&4,X,X,X,X,X,X,X,X"))
    (clobber (reg:CC 17))]
   "TARGET_SSE2
+   && (GET_CODE (operands[2]) != MEM || GET_CODE (operands[3]) != MEM)
+   && (!TARGET_IEEE_FP
+       || (GET_CODE (operands[1]) != EQ && GET_CODE (operands[1]) != NE))"
+  "#")
+
+(define_insn "sse_movdfcc_eq"
+  [(set (match_operand:DF 0 "register_operand" "=&x#rf,x#rf,?f#xr,?f#xr,?r#xf,?r#xf")
+       (if_then_else:DF (eq (match_operand:DF 3 "nonimmediate_operand" "%0#fx,x#fx,f#x,xm#f,f#x,xm#f")
+                            (match_operand:DF 4 "nonimmediate_operand" "xm#f,xm#f,f#x,x#f,f#x,x#f"))
+                     (match_operand:DF 1 "nonimmediate_operand" "x#fr,0#fr,0#fx,0#fx,0#rx,0#rx")
+                     (match_operand:DF 2 "nonimmediate_operand" "x#fr,x#fr,f#fx,f#fx,rm#rx,rm#rx")))
+   (clobber (match_scratch:DF 5 "=1,&3,X,X,X,X"))
+   (clobber (reg:CC 17))]
+  "TARGET_SSE
    && (GET_CODE (operands[2]) != MEM || GET_CODE (operands[3]) != MEM)"
   "#")