When there's an asm goto in the latch of a loop we may not use
IP_END IVs since instantiating those would (need to) split the
latch edge which in turn invalidates IP_NORMAL position handling.
This is a revision of the PR107997 fix.
PR tree-optimization/107997
PR tree-optimization/121844
* tree-ssa-loop-ivopts.cc (allow_ip_end_pos_p): Do not allow
IP_END for latches ending with a control stmt.
(create_new_iv): Do not split the latch edge, instead assert
that's not necessary.
* gcc.dg/torture/pr121844.c: New testcase.
--- /dev/null
+/* { dg-do compile } */
+
+typedef unsigned long a;
+int b[] = {};
+int **c;
+short d(a *e, int f)
+{
+ *c = &f;
+ for (;;)
+ asm goto("" : : : : g);
+ for (; f; f--) {
+ asm goto("" : : : : g);
+ g:
+ *e ^= b[f + 1];
+ }
+}
static bool
allow_ip_end_pos_p (class loop *loop)
{
+ /* Do not allow IP_END when creating the IV would need to split the
+ latch edge as that makes all IP_NORMAL invalid. */
+ auto pos = gsi_last_bb (ip_end_pos (loop));
+ if (!gsi_end_p (pos) && stmt_ends_bb_p (*pos))
+ return false;
+
if (!ip_normal_pos (loop))
return true;
case IP_END:
incr_pos = gsi_last_bb (ip_end_pos (data->current_loop));
after = true;
- if (!gsi_end_p (incr_pos) && stmt_ends_bb_p (gsi_stmt (incr_pos)))
- {
- edge e = find_edge (gsi_bb (incr_pos), data->current_loop->header);
- incr_pos = gsi_after_labels (split_edge (e));
- after = false;
- }
+ gcc_assert (gsi_end_p (incr_pos) || !stmt_ends_bb_p (*incr_pos));
break;
case IP_AFTER_USE: