On December 4, 2014 7:37:32 PM CET, Marek Polacek <pola...@redhat.com> wrote:
>On Thu, Dec 04, 2014 at 05:34:35PM +0100, Richard Biener wrote:
>> On December 4, 2014 4:45:25 PM CET, Marek Polacek
><pola...@redhat.com> wrote:
>> >The PR shows a case in which fold introduces undefined behavior in a
>> >valid program, because what it does here is
>> >-(long int) (ul + ULONG_MAX) - 1 ->
>> >~(long int) (ul + ULONG_MAX) ->
>> >-(long int) ul
>> >But the latter transformation is wrong if ul is unsigned long and
>> >equals
>> >LONG_MAX + 1UL, because that introduces -LONG_MIN, and ubsan/-ftrapv
>> >errors on that, even though the program is valid.
>> >(Converting LONG_MAX + 1UL to long is implementation-defined.)
>> 
>> Implementation defined behavior is OK.
> 
>Sure.
>
>> Rather than disabling the transform I'd rather make it defined using
>unsigned types.
>
>Such as the following then?  That is, instead of -(long int) ul do
>(long int) -ul.

Yes.

Richard.

>Bootstrapped/regtested on ppc64-linux.
>
>2014-12-04  Marek Polacek  <pola...@redhat.com>
>
>       PR middle-end/56917
>       * fold-const.c (fold_unary_loc): Perform the negation in A's type
>       when transforming ~ (A - 1) or ~ (A + -1) to -A.
>
>       * c-c++-common/ubsan/pr56917.c: New test.
>
>diff --git gcc/fold-const.c gcc/fold-const.c
>index 17eb5bb..4ab443d 100644
>--- gcc/fold-const.c
>+++ gcc/fold-const.c
>@@ -8141,9 +8141,14 @@ fold_unary_loc (location_t loc, enum tree_code
>code, tree type, tree op0)
>                   && integer_onep (TREE_OPERAND (arg0, 1)))
>                  || (TREE_CODE (arg0) == PLUS_EXPR
>                      && integer_all_onesp (TREE_OPERAND (arg0, 1)))))
>-      return fold_build1_loc (loc, NEGATE_EXPR, type,
>-                          fold_convert_loc (loc, type,
>-                                            TREE_OPERAND (arg0, 0)));
>+      {
>+        /* Perform the negation in ARG0's type and only then convert
>+           to TYPE as to avoid introducing undefined behavior.  */
>+        tree t = fold_build1_loc (loc, NEGATE_EXPR,
>+                                  TREE_TYPE (TREE_OPERAND (arg0, 0)),
>+                                  TREE_OPERAND (arg0, 0));
>+        return fold_convert_loc (loc, type, t);
>+      }
>      /* Convert ~(X ^ Y) to ~X ^ Y or X ^ ~Y if ~X or ~Y simplify.  */
>       else if (TREE_CODE (arg0) == BIT_XOR_EXPR
>              && (tem = fold_unary_loc (loc, BIT_NOT_EXPR, type,
>diff --git gcc/testsuite/c-c++-common/ubsan/pr56917.c
>gcc/testsuite/c-c++-common/ubsan/pr56917.c
>index e69de29..cfbae97 100644
>--- gcc/testsuite/c-c++-common/ubsan/pr56917.c
>+++ gcc/testsuite/c-c++-common/ubsan/pr56917.c
>@@ -0,0 +1,34 @@
>+/* PR middle-end/56917 */
>+/* { dg-do run } */
>+/* { dg-options "-fsanitize=undefined -fno-sanitize-recover=undefined"
>} */
>+
>+#define INT_MIN (-__INT_MAX__ - 1)
>+#define LONG_MIN (-__LONG_MAX__ - 1L)
>+#define LLONG_MIN (-__LONG_LONG_MAX__ - 1LL)
>+
>+int __attribute__ ((noinline,noclone))
>+fn1 (unsigned int u)
>+{
>+  return (-(int) (u - 1U)) - 1;
>+}
>+
>+long __attribute__ ((noinline,noclone))
>+fn2 (unsigned long int ul)
>+{
>+  return (-(long) (ul - 1UL)) - 1L;
>+}
>+
>+long long __attribute__ ((noinline,noclone))
>+fn3 (unsigned long long int ull)
>+{
>+  return (-(long long) (ull - 1ULL)) - 1LL;
>+}
>+
>+int
>+main (void)
>+{
>+  if (fn1 (__INT_MAX__ + 1U) != INT_MIN
>+      || fn2 (__LONG_MAX__ + 1UL) != LONG_MIN
>+      || fn3 (__LONG_LONG_MAX__ + 1ULL) != LLONG_MIN)
>+    __builtin_abort ();
>+}
>
>       Marek


Reply via email to