aboutsummaryrefslogtreecommitdiff
path: root/gcc
diff options
context:
space:
mode:
authorJakub Jelinek <jakub@redhat.com>2007-11-23 14:39:44 +0100
committerJakub Jelinek <jakub@gcc.gnu.org>2007-11-23 14:39:44 +0100
commitcfdaefecf6ad134e393cd67fe54f4f5cdd21c5a0 (patch)
treee85ad82f029ca77413c4971bdede1ccb117867cc /gcc
parentbb1f5840b580ba1672d97cf0c8eccbcb50c3d3c0 (diff)
downloadgcc-cfdaefecf6ad134e393cd67fe54f4f5cdd21c5a0.zip
gcc-cfdaefecf6ad134e393cd67fe54f4f5cdd21c5a0.tar.gz
gcc-cfdaefecf6ad134e393cd67fe54f4f5cdd21c5a0.tar.bz2
re PR c++/34198 (-Wconversion gives apparent erroneous warning with g++ 4.3-20071109)
PR c++/34198 * c-common.c (conversion_warning): For INTEGER_TYPE to INTEGER_TYPE conversions call get_narrower on expr to avoid spurious warnings from binop shortening or when the implicit conversion can't change the value. * gcc.dg/Wconversion-5.c: New test. * g++.dg/Wconversion3.C: New test. From-SVN: r130377
Diffstat (limited to 'gcc')
-rw-r--r--gcc/ChangeLog8
-rw-r--r--gcc/c-common.c35
-rw-r--r--gcc/testsuite/ChangeLog6
-rw-r--r--gcc/testsuite/g++.dg/warn/Wconversion3.C35
-rw-r--r--gcc/testsuite/gcc.dg/Wconversion-5.c35
5 files changed, 114 insertions, 5 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog
index e6f3640..722ec42 100644
--- a/gcc/ChangeLog
+++ b/gcc/ChangeLog
@@ -1,3 +1,11 @@
+2007-11-23 Jakub Jelinek <jakub@redhat.com>
+
+ PR c++/34198
+ * c-common.c (conversion_warning): For INTEGER_TYPE to
+ INTEGER_TYPE conversions call get_narrower on expr to avoid
+ spurious warnings from binop shortening or when the implicit
+ conversion can't change the value.
+
2007-11-22 Joseph Myers <joseph@codesourcery.com>
PR c/14050
diff --git a/gcc/c-common.c b/gcc/c-common.c
index edc9b2c..6872e3a 100644
--- a/gcc/c-common.c
+++ b/gcc/c-common.c
@@ -1280,6 +1280,14 @@ conversion_warning (tree type, tree expr)
else if (TREE_CODE (TREE_TYPE (expr)) == INTEGER_TYPE
&& TREE_CODE (type) == INTEGER_TYPE)
{
+ /* Don't warn about unsigned char y = 0xff, x = (int) y; */
+ int uns;
+ tree orig_expr = expr;
+ expr = get_narrower (expr, &uns);
+
+ if (expr == orig_expr)
+ uns = TYPE_UNSIGNED (TREE_TYPE (expr));
+
/* Warn for integer types converted to smaller integer types. */
if (formal_prec < TYPE_PRECISION (TREE_TYPE (expr)))
give_warning = true;
@@ -1287,14 +1295,31 @@ conversion_warning (tree type, tree expr)
/* When they are the same width but different signedness,
then the value may change. */
else if ((formal_prec == TYPE_PRECISION (TREE_TYPE (expr))
- && TYPE_UNSIGNED (TREE_TYPE (expr)) != TYPE_UNSIGNED (type))
+ && uns != TYPE_UNSIGNED (type))
/* Even when converted to a bigger type, if the type is
unsigned but expr is signed, then negative values
will be changed. */
- || (TYPE_UNSIGNED (type) && !TYPE_UNSIGNED (TREE_TYPE (expr))))
- warning (OPT_Wsign_conversion,
- "conversion to %qT from %qT may change the sign of the result",
- type, TREE_TYPE (expr));
+ || (TYPE_UNSIGNED (type) && !uns))
+ {
+ if (uns != TYPE_UNSIGNED (TREE_TYPE (expr)))
+ {
+ /* For signed char s1, s2 = (int) (unsigned char) s1;
+ get_narrower returns s1, but uns = 1. Find the
+ narrowest type with uns == TYPE_UNSIGNED (type). */
+ tree unsexpr = orig_expr;
+
+ while (TREE_CODE (unsexpr) == NOP_EXPR
+ && unsexpr != expr
+ && TYPE_UNSIGNED (TREE_TYPE (TREE_OPERAND (unsexpr,
+ 0)))
+ == uns)
+ unsexpr = TREE_OPERAND (unsexpr, 0);
+ expr = unsexpr;
+ }
+ warning (OPT_Wsign_conversion,
+ "conversion to %qT from %qT may change the sign of the result",
+ type, TREE_TYPE (expr));
+ }
}
/* Warn for integer types converted to real types if and only if
diff --git a/gcc/testsuite/ChangeLog b/gcc/testsuite/ChangeLog
index d9aee04..7f0c52b 100644
--- a/gcc/testsuite/ChangeLog
+++ b/gcc/testsuite/ChangeLog
@@ -1,3 +1,9 @@
+2007-11-23 Jakub Jelinek <jakub@redhat.com>
+
+ PR c++/34198
+ * gcc.dg/Wconversion-5.c: New test.
+ * g++.dg/Wconversion3.C: New test.
+
2007-11-23 Richard Guenther <rguenther@suse.de>
* gcc.dg/tree-ssa/alias-17.c: New testcase.
diff --git a/gcc/testsuite/g++.dg/warn/Wconversion3.C b/gcc/testsuite/g++.dg/warn/Wconversion3.C
new file mode 100644
index 0000000..24202b7
--- /dev/null
+++ b/gcc/testsuite/g++.dg/warn/Wconversion3.C
@@ -0,0 +1,35 @@
+// PR c++/34198
+// { dg-do compile }
+// { dg-options "-O2 -Wconversion -Wsign-conversion" }
+
+signed char sc;
+unsigned char uc;
+short int ss;
+unsigned short int us;
+int si;
+unsigned int ui;
+
+void test1 (void)
+{
+ int a = uc & 0xff;
+ int b = sc & 0x7f;
+ int c = 0xff & uc;
+ int d = 0x7f & sc;
+ int e = uc & sc;
+ unsigned char f = (int) uc;
+ signed char g = (int) sc;
+ unsigned char h = (unsigned int) (short int) uc;
+ signed char i = (int) (unsigned short int) sc; // { dg-warning "may alter its value" }
+ unsigned char j = (unsigned int) (short int) us; // { dg-warning "may alter its value" }
+ signed char k = (int) (unsigned short int) ss; // { dg-warning "may alter its value" }
+}
+
+void test2 (void)
+{
+ signed char a = (unsigned char) sc; // { dg-warning "may change the sign" }
+ unsigned char b = (signed char) uc; // { dg-warning "may change the sign" }
+ signed char c = (int) (unsigned char) sc; // { dg-warning "may change the sign" }
+ unsigned char d = (int) (signed char) uc; // { dg-warning "may change the sign" }
+ int e = (unsigned int) si; // { dg-warning "may change the sign" }
+ unsigned int f = (int) ui; // { dg-warning "may change the sign" }
+}
diff --git a/gcc/testsuite/gcc.dg/Wconversion-5.c b/gcc/testsuite/gcc.dg/Wconversion-5.c
new file mode 100644
index 0000000..a09caae
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/Wconversion-5.c
@@ -0,0 +1,35 @@
+/* PR c++/34198 */
+/* { dg-do compile } */
+/* { dg-options "-O2 -Wconversion" } */
+
+signed char sc;
+unsigned char uc;
+short int ss;
+unsigned short int us;
+int si;
+unsigned int ui;
+
+void test1 (void)
+{
+ int a = uc & 0xff;
+ int b = sc & 0x7f;
+ int c = 0xff & uc;
+ int d = 0x7f & sc;
+ int e = uc & sc;
+ unsigned char f = (int) uc;
+ signed char g = (int) sc;
+ unsigned char h = (unsigned int) (short int) uc;
+ signed char i = (int) (unsigned short int) sc; /* { dg-warning "may alter its value" } */
+ unsigned char j = (unsigned int) (short int) us; /* { dg-warning "may alter its value" } */
+ signed char k = (int) (unsigned short int) ss; /* { dg-warning "may alter its value" } */
+}
+
+void test2 (void)
+{
+ signed char a = (unsigned char) sc; /* { dg-warning "may change the sign" } */
+ unsigned char b = (signed char) uc; /* { dg-warning "may change the sign" } */
+ signed char c = (int) (unsigned char) sc; /* { dg-warning "may change the sign" } */
+ unsigned char d = (int) (signed char) uc; /* { dg-warning "may change the sign" } */
+ int e = (unsigned int) si; /* { dg-warning "may change the sign" } */
+ unsigned int f = (int) ui; /* { dg-warning "may change the sign" } */
+}