summaryrefslogtreecommitdiff
path: root/gcc
diff options
context:
space:
mode:
authorsayle <sayle@138bc75d-0d04-0410-961f-82ee72b054a4>2002-05-10 22:24:13 +0000
committersayle <sayle@138bc75d-0d04-0410-961f-82ee72b054a4>2002-05-10 22:24:13 +0000
commit843dd7a3db214e25bf208900bff2936f525a4cda (patch)
tree8a995b794b7a305a3fea68fbd495596f5a3b2c15 /gcc
parent44d233416485f12f4bf2299b2e1bd1ae74a4f800 (diff)
downloadgcc-843dd7a3db214e25bf208900bff2936f525a4cda.tar.gz
* fold-const.c (build_range_check): Optimize (c>=1) && (c<=127)
into the equivalent (signed char)c > 0. * gcc.c-torture/execute/20020510-1.c: New test case. git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@53373 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc')
-rw-r--r--gcc/ChangeLog5
-rw-r--r--gcc/fold-const.c62
-rw-r--r--gcc/testsuite/ChangeLog4
-rw-r--r--gcc/testsuite/gcc.c-torture/execute/20020510-1.c85
4 files changed, 141 insertions, 15 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog
index bc20fd32eba..192b2be851d 100644
--- a/gcc/ChangeLog
+++ b/gcc/ChangeLog
@@ -1,3 +1,8 @@
+2002-05-10 Roger Sayle <roger@eyesopen.com>
+
+ * fold-const.c (build_range_check): Optimize (c>=1) && (c<=127)
+ into the equivalent (signed char)c > 0.
+
2002-05-10 Janis Johnson <janis187@us.ibm.com>
* loop.c: (PREFETCH_EXTREME_DIFFERENCE, PREFETCH_BEFORE_LOOP): New.
diff --git a/gcc/fold-const.c b/gcc/fold-const.c
index 6b4982aad75..2a9f6c6ee7c 100644
--- a/gcc/fold-const.c
+++ b/gcc/fold-const.c
@@ -3091,41 +3091,73 @@ build_range_check (type, exp, in_p, low, high)
tree low, high;
{
tree etype = TREE_TYPE (exp);
- tree utype, value;
+ tree value;
if (! in_p
&& (0 != (value = build_range_check (type, exp, 1, low, high))))
return invert_truthvalue (value);
- else if (low == 0 && high == 0)
+ if (low == 0 && high == 0)
return convert (type, integer_one_node);
- else if (low == 0)
+ if (low == 0)
return fold (build (LE_EXPR, type, exp, high));
- else if (high == 0)
+ if (high == 0)
return fold (build (GE_EXPR, type, exp, low));
- else if (operand_equal_p (low, high, 0))
+ if (operand_equal_p (low, high, 0))
return fold (build (EQ_EXPR, type, exp, low));
- else if (TREE_UNSIGNED (etype) && integer_zerop (low))
- return build_range_check (type, exp, 1, 0, high);
+ if (integer_zerop (low))
+ {
+ if (! TREE_UNSIGNED (etype))
+ {
+ etype = (*lang_hooks.types.unsigned_type) (etype);
+ high = convert (etype, high);
+ exp = convert (etype, exp);
+ }
+ return build_range_check (type, exp, 1, 0, high);
+ }
- else if (integer_zerop (low))
+ /* Optimize (c>=1) && (c<=127) into (signed char)c > 0. */
+ if (integer_onep (low) && TREE_CODE (high) == INTEGER_CST)
{
- utype = (*lang_hooks.types.unsigned_type) (etype);
- return build_range_check (type, convert (utype, exp), 1, 0,
- convert (utype, high));
+ unsigned HOST_WIDE_INT lo;
+ HOST_WIDE_INT hi;
+ int prec;
+
+ prec = TYPE_PRECISION (etype);
+ if (prec <= HOST_BITS_PER_WIDE_INT)
+ {
+ hi = 0;
+ lo = ((unsigned HOST_WIDE_INT) 1 << (prec - 1)) - 1;
+ }
+ else
+ {
+ hi = ((HOST_WIDE_INT) 1 << (prec - HOST_BITS_PER_WIDE_INT - 1)) - 1;
+ lo = (unsigned HOST_WIDE_INT) -1;
+ }
+
+ if (TREE_INT_CST_HIGH (high) == hi && TREE_INT_CST_LOW (high) == lo)
+ {
+ if (TREE_UNSIGNED (etype))
+ {
+ etype = (*lang_hooks.types.signed_type) (etype);
+ exp = convert (etype, exp);
+ }
+ return fold (build (GT_EXPR, type, exp,
+ convert (etype, integer_zero_node)));
+ }
}
- else if (0 != (value = const_binop (MINUS_EXPR, high, low, 0))
- && ! TREE_OVERFLOW (value))
+ if (0 != (value = const_binop (MINUS_EXPR, high, low, 0))
+ && ! TREE_OVERFLOW (value))
return build_range_check (type,
fold (build (MINUS_EXPR, etype, exp, low)),
1, convert (etype, integer_zero_node), value);
- else
- return 0;
+
+ return 0;
}
/* Given two ranges, see if we can merge them into one. Return 1 if we
diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog
index c66c70f16b2..d0f2142e087 100644
--- a/gcc/testsuite/ChangeLog
+++ b/gcc/testsuite/ChangeLog
@@ -1,3 +1,7 @@
+2002-05-10 Roger Sayle <roger@eyesopen.com>
+
+ * gcc.c-torture/execute/20020510-1.c: New test case.
+
2002-05-10 David S. Miller <davem@redhat.com>
* gcc.c-torture/execute/conversion.c: Test long double too.
diff --git a/gcc/testsuite/gcc.c-torture/execute/20020510-1.c b/gcc/testsuite/gcc.c-torture/execute/20020510-1.c
new file mode 100644
index 00000000000..90fb2775557
--- /dev/null
+++ b/gcc/testsuite/gcc.c-torture/execute/20020510-1.c
@@ -0,0 +1,85 @@
+/* Copyright (C) 2002 Free Software Foundation.
+
+ Test that optimizing ((c>=1) && (c<=127)) into (signed char)c < 0
+ doesn't cause any problems for the compiler and behaves correctly.
+
+ Written by Roger Sayle, 8th May 2002. */
+
+#include <limits.h>
+
+extern void abort (void);
+
+void
+testc (unsigned char c, int ok)
+{
+ if ((c>=1) && (c<=SCHAR_MAX))
+ {
+ if (!ok) abort ();
+ }
+ else
+ if (ok) abort ();
+}
+
+void
+tests (unsigned short s, int ok)
+{
+ if ((s>=1) && (s<=SHRT_MAX))
+ {
+ if (!ok) abort ();
+ }
+ else
+ if (ok) abort ();
+}
+
+void
+testi (unsigned int i, int ok)
+{
+ if ((i>=1) && (i<=INT_MAX))
+ {
+ if (!ok) abort ();
+ }
+ else
+ if (ok) abort ();
+}
+
+void
+testl (unsigned long l, int ok)
+{
+ if ((l>=1) && (l<=LONG_MAX))
+ {
+ if (!ok) abort ();
+ }
+ else
+ if (ok) abort ();
+}
+
+int
+main ()
+{
+ testc (0, 0);
+ testc (1, 1);
+ testc (SCHAR_MAX, 1);
+ testc (SCHAR_MAX+1, 0);
+ testc (UCHAR_MAX, 0);
+
+ tests (0, 0);
+ tests (1, 1);
+ tests (SHRT_MAX, 1);
+ tests (SHRT_MAX+1, 0);
+ tests (USHRT_MAX, 0);
+
+ testi (0, 0);
+ testi (1, 1);
+ testi (INT_MAX, 1);
+ testi (INT_MAX+1U, 0);
+ testi (UINT_MAX, 0);
+
+ testl (0, 0);
+ testl (1, 1);
+ testl (LONG_MAX, 1);
+ testl (LONG_MAX+1UL, 0);
+ testl (ULONG_MAX, 0);
+
+ return 0;
+}
+