| /* { dg-do run { xfail { mmix-*-* } } } */ |
| /* We don't (and don't want to) perform this optimisation on soft-float targets, |
| where each addition is a library call. / |
| /* { dg-require-effective-target hard_float } */ |
| /* -fassociative-math requires -fno-trapping-math and -fno-signed-zeros. */ |
| /* { dg-options "-O2 -funroll-loops -fassociative-math -fno-trapping-math -fno-signed-zeros -fvariable-expansion-in-unroller -fdump-rtl-loop2_unroll" } */ |
| |
| extern void abort (void); |
| extern void exit (int); |
| |
| float __attribute__((noinline)) |
| foo (float d, int n) |
| { |
| unsigned i; |
| float accum = d; |
| |
| for (i = 0; i < n; i++) |
| accum += d; |
| |
| return accum; |
| } |
| |
| int |
| main () |
| { |
| /* When compiling standard compliant we expect foo to return -0.0. But the |
| variable expansion during unrolling optimization (for this testcase enabled |
| by non-compliant -fassociative-math) instantiates copy(s) of the |
| accumulator which it initializes with +0.0. Hence we expect that foo |
| returns +0.0. */ |
| if (__builtin_copysignf (1.0, foo (0.0 / -5.0, 10)) != 1.0) |
| abort (); |
| exit (0); |
| } |
| |
| /* { dg-final { scan-rtl-dump "Expanding Accumulator" "loop2_unroll" { xfail mmix-*-* } } } */ |