summaryrefslogtreecommitdiff
path: root/gcc/tree-ssa-threadedge.c
diff options
context:
space:
mode:
authoraoliva <aoliva@138bc75d-0d04-0410-961f-82ee72b054a4>2012-11-04 18:44:13 +0000
committeraoliva <aoliva@138bc75d-0d04-0410-961f-82ee72b054a4>2012-11-04 18:44:13 +0000
commit1ea5fe8f3fd83e8fb02c48d80b7ed8d8920eb845 (patch)
tree9b2764e159a783f231719c21b79e4b32df016978 /gcc/tree-ssa-threadedge.c
parent5b01786639cf92fa8fe2c4715238ead778e8da96 (diff)
downloadgcc-1ea5fe8f3fd83e8fb02c48d80b7ed8d8920eb845.tar.gz
PR debug/54693
* tree-ssa-threadedge.c (propagate_threaded_block_debug_into): New, rewritten from debug stmt copying code... (thread_around_empty_block): ... removed from here. (thread_across_edge): Call propagate_threaded_block_debug_into. git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@193138 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc/tree-ssa-threadedge.c')
-rw-r--r--gcc/tree-ssa-threadedge.c106
1 files changed, 87 insertions, 19 deletions
diff --git a/gcc/tree-ssa-threadedge.c b/gcc/tree-ssa-threadedge.c
index f43a564d786..a9c671e4ecc 100644
--- a/gcc/tree-ssa-threadedge.c
+++ b/gcc/tree-ssa-threadedge.c
@@ -610,6 +610,85 @@ cond_arg_set_in_bb (edge e, basic_block bb)
return false;
}
+/* Copy debug stmts from DEST's chain of single predecessors up to
+ SRC, so that we don't lose the bindings as PHI nodes are introduced
+ when DEST gains new predecessors. */
+static void
+propagate_threaded_block_debug_into (basic_block dest, basic_block src)
+{
+ if (!MAY_HAVE_DEBUG_STMTS)
+ return;
+
+ if (!single_pred_p (dest))
+ return;
+
+ gcc_checking_assert (dest != src);
+
+ gimple_stmt_iterator gsi = gsi_after_labels (dest);
+ pointer_set_t *vars = pointer_set_create ();
+
+ for (gimple_stmt_iterator si = gsi;
+ !gsi_end_p (si); gsi_next (&si))
+ {
+ gimple stmt = gsi_stmt (si);
+ if (!is_gimple_debug (stmt))
+ break;
+
+ tree var;
+
+ if (gimple_debug_bind_p (stmt))
+ var = gimple_debug_bind_get_var (stmt);
+ else if (gimple_debug_source_bind_p (stmt))
+ var = gimple_debug_source_bind_get_var (stmt);
+ else
+ gcc_unreachable ();
+
+ pointer_set_insert (vars, var);
+ }
+
+ basic_block bb = dest;
+
+ do
+ {
+ bb = single_pred (bb);
+ for (gimple_stmt_iterator si = gsi_last_bb (bb);
+ !gsi_end_p (si); gsi_prev (&si))
+ {
+ gimple stmt = gsi_stmt (si);
+ if (!is_gimple_debug (stmt))
+ continue;
+
+ tree var;
+
+ if (gimple_debug_bind_p (stmt))
+ var = gimple_debug_bind_get_var (stmt);
+ else if (gimple_debug_source_bind_p (stmt))
+ var = gimple_debug_source_bind_get_var (stmt);
+ else
+ gcc_unreachable ();
+
+ /* Discard debug bind overlaps. ??? Unlike stmts from src,
+ copied into a new block that will precede BB, debug bind
+ stmts in bypassed BBs may actually be discarded if
+ they're overwritten by subsequent debug bind stmts, which
+ might be a problem once we introduce stmt frontier notes
+ or somesuch. Adding `&& bb == src' to the condition
+ below will preserve all potentially relevant debug
+ notes. */
+ if (pointer_set_insert (vars, var))
+ continue;
+
+ stmt = gimple_copy (stmt);
+ /* ??? Should we drop the location of the copy to denote
+ they're artificial bindings? */
+ gsi_insert_before (&gsi, stmt, GSI_NEW_STMT);
+ }
+ }
+ while (bb != src && single_pred_p (bb));
+
+ pointer_set_destroy (vars);
+}
+
/* TAKEN_EDGE represents the an edge taken as a result of jump threading.
See if we can thread around TAKEN_EDGE->dest as well. If so, return
the edge out of TAKEN_EDGE->dest that we can statically compute will be
@@ -637,24 +716,6 @@ thread_around_empty_block (edge taken_edge,
if (!single_pred_p (bb))
return NULL;
- /* Before threading, copy DEBUG stmts from the predecessor, so that
- we don't lose the bindings as we redirect the edges. */
- if (MAY_HAVE_DEBUG_STMTS)
- {
- gsi = gsi_after_labels (bb);
- for (gimple_stmt_iterator si = gsi_last_bb (taken_edge->src);
- !gsi_end_p (si); gsi_prev (&si))
- {
- stmt = gsi_stmt (si);
- if (!is_gimple_debug (stmt))
- continue;
-
- stmt = gimple_copy (stmt);
- /* ??? Should we drop the location of the copy? */
- gsi_insert_before (&gsi, stmt, GSI_NEW_STMT);
- }
- }
-
/* This block must have more than one successor. */
if (single_succ_p (bb))
return NULL;
@@ -827,6 +888,9 @@ thread_across_edge (gimple dummy_cond,
}
remove_temporary_equivalences (stack);
+ if (!taken_edge)
+ return;
+ propagate_threaded_block_debug_into (taken_edge->dest, e->dest);
register_jump_thread (e, taken_edge, NULL);
return;
}
@@ -892,7 +956,11 @@ thread_across_edge (gimple dummy_cond,
same. */
tmp = find_edge (taken_edge->src, e3->dest);
if (!tmp || phi_args_equal_on_edges (tmp, e3))
- register_jump_thread (e, taken_edge, e3);
+ {
+ propagate_threaded_block_debug_into (e3->dest,
+ taken_edge->dest);
+ register_jump_thread (e, taken_edge, e3);
+ }
}
}