diff options
author | Richard Biener <rguenther@suse.de> | 2022-04-12 09:40:15 +0200 |
---|---|---|
committer | Richard Biener <rguenther@suse.de> | 2022-04-12 10:53:46 +0200 |
commit | 62d5bb0f35fb6ec373eaac942755585a633528a0 (patch) | |
tree | a24dfa5a8db17ddaef4fabb55f3a7884fa942b58 | |
parent | 152f243fdacf8ad97df3d4ae5b51b792796e6740 (diff) | |
download | gcc-62d5bb0f35fb6ec373eaac942755585a633528a0.zip gcc-62d5bb0f35fb6ec373eaac942755585a633528a0.tar.gz gcc-62d5bb0f35fb6ec373eaac942755585a633528a0.tar.bz2 |
tree-optimization/105226 - avoid splitting abnormal edges
Vectorizer loop versioning tries to version outer loops if possible
but fails to check whether it can actually split the single exit
edge as it will do.
2022-04-12 Richard Biener <rguenther@suse.de>
PR tree-optimization/105226
* tree-vect-loop-manip.cc (vect_loop_versioning): Verify
we can split the exit of an outer loop we choose to version.
* gcc.dg/pr105226.c: New testcase.
-rw-r--r-- | gcc/testsuite/gcc.dg/pr105226.c | 23 | ||||
-rw-r--r-- | gcc/tree-vect-loop-manip.cc | 3 |
2 files changed, 25 insertions, 1 deletions
diff --git a/gcc/testsuite/gcc.dg/pr105226.c b/gcc/testsuite/gcc.dg/pr105226.c new file mode 100644 index 0000000..9c4941d --- /dev/null +++ b/gcc/testsuite/gcc.dg/pr105226.c @@ -0,0 +1,23 @@ +/* { dg-do compile } */ +/* { dg-options "-Ofast" } */ +/* { dg-require-effective-target indirect_jumps } */ + +#include <setjmp.h> +struct longjmp_buf { + jmp_buf buf; +}; +void g (); +void f () +{ + int i, n; + long *a; + long *args; + struct longjmp_buf b; + setjmp (b.buf); + for (;;) + { + for (i = 0; i < n; i++) + a[i] = args[i]; + g (); + } +} diff --git a/gcc/tree-vect-loop-manip.cc b/gcc/tree-vect-loop-manip.cc index a7bbc91..63fb6f6 100644 --- a/gcc/tree-vect-loop-manip.cc +++ b/gcc/tree-vect-loop-manip.cc @@ -3524,7 +3524,8 @@ vect_loop_versioning (loop_vec_info loop_vinfo, outermost = superloop_at_depth (loop, 1); /* And avoid applying versioning on non-perfect nests. */ while (loop_to_version != outermost - && single_exit (loop_outer (loop_to_version)) + && (e = single_exit (loop_outer (loop_to_version))) + && !(e->flags & EDGE_COMPLEX) && (!loop_outer (loop_to_version)->inner->next || vect_loop_vectorized_call (loop_to_version)) && (!loop_outer (loop_to_version)->inner->next |