This is the mail archive of the
gcc-patches@gcc.gnu.org
mailing list for the GCC project.
New function split_block
- To: gcc-patches at gcc dot gnu dot org
- Subject: New function split_block
- From: Michael Hayes <mhayes at cygnus dot com>
- Date: Tue, 12 Sep 2000 19:31:33 +1200 (NZST)
This patch provides a new function to split a basic block at a given
insn. It creates a new fallthru edge between the blocks and returns
the new edge. This function is used in a forthcoming loop patch that
adds fake edges to the CFG after each call insn that may exit.
Michael.
2000-09-12 Michael Hayes <mhayes@cygnus.com>
* basic-block.h (split_block, update_bb_for_insn): New prototypes.
* flow.c (split_block, update_bb_for_insn): New functions.
*** fsf-patch/gcc/basic-block.h Tue Sep 12 19:24:20 2000
--- fsf-local/gcc/basic-block.h Tue Sep 12 19:22:03 2000
*************** extern varray_type basic_block_for_insn;
*** 235,245 ****
--- 235,247 ----
#define BLOCK_NUM(INSN) (BLOCK_FOR_INSN (INSN)->index + 0)
extern void compute_bb_for_insn PARAMS ((int));
+ extern void update_bb_for_insn PARAMS ((basic_block));
extern void set_block_for_insn PARAMS ((rtx, basic_block));
extern void set_block_num PARAMS ((rtx, int));
extern void free_basic_block_vars PARAMS ((int));
+ extern edge split_block PARAMS ((basic_block, rtx));
extern basic_block split_edge PARAMS ((edge));
extern void insert_insn_on_edge PARAMS ((rtx, edge));
extern void commit_edge_insertions PARAMS ((void));
*** fsf-patch/gcc/flow.c Tue Sep 12 19:24:21 2000
--- fsf-local/gcc/flow.c Tue Sep 12 19:22:04 2000
*************** mark_critical_edges ()
*** 1422,1427 ****
--- 1422,1523 ----
}
}
+ /* Split a block BB after insn INSN creating a new fallthru edge.
+ Return the new edge. Note that to keep other parts of the compiler happy,
+ this function renumbers all the basic blocks so that the new
+ one has a number one greater than the block split. */
+
+ edge
+ split_block (bb, insn)
+ basic_block bb;
+ rtx insn;
+ {
+ basic_block new_bb;
+ edge new_edge;
+ edge e;
+ rtx bb_note;
+ int i, j;
+
+ if (BLOCK_FOR_INSN (insn) != bb)
+ abort ();
+
+ /* There is no point splitting the block after its end. */
+ if (bb->end == insn)
+ return 0;
+
+ /* Create the new structures. */
+ new_bb = (basic_block) obstack_alloc (function_obstack, sizeof (*new_bb));
+ new_edge = (edge) xcalloc (1, sizeof (*new_edge));
+ n_edges++;
+
+ memset (new_bb, 0, sizeof (*new_bb));
+
+ new_bb->head = NEXT_INSN (insn);
+ new_bb->end = bb->end;
+ bb->end = insn;
+
+ new_bb->succ = bb->succ;
+ bb->succ = new_edge;
+ new_bb->pred = new_edge;
+ new_bb->count = bb->count;
+ new_bb->loop_depth = bb->loop_depth;
+
+ new_edge->src = bb;
+ new_edge->dest = new_bb;
+ new_edge->flags = EDGE_FALLTHRU;
+ new_edge->probability = REG_BR_PROB_BASE;
+ new_edge->count = bb->count;
+
+ /* Redirect the src of the successor edges of bb to point to new_bb. */
+ for (e = new_bb->succ; e; e = e->succ_next)
+ e->src = new_bb;
+
+ /* Place the new block just after the block being split. */
+ VARRAY_GROW (basic_block_info, ++n_basic_blocks);
+
+ /* Some parts of the compiler expect blocks to be number in
+ sequential order so insert the new block immediately after the
+ block being split.. */
+ j = bb->index;
+ for (i = n_basic_blocks - 1; i > j + 1; --i)
+ {
+ basic_block tmp = BASIC_BLOCK (i - 1);
+ BASIC_BLOCK (i) = tmp;
+ tmp->index = i;
+ }
+
+ BASIC_BLOCK (i) = new_bb;
+ new_bb->index = i;
+
+ /* Create the basic block note. */
+ bb_note = emit_note_before (NOTE_INSN_BASIC_BLOCK,
+ new_bb->head);
+ NOTE_BASIC_BLOCK (bb_note) = new_bb;
+ new_bb->head = bb_note;
+
+ update_bb_for_insn (new_bb);
+
+ if (bb->global_live_at_start)
+ {
+ new_bb->global_live_at_start = OBSTACK_ALLOC_REG_SET (function_obstack);
+ new_bb->global_live_at_end = OBSTACK_ALLOC_REG_SET (function_obstack);
+ COPY_REG_SET (new_bb->global_live_at_end, bb->global_live_at_end);
+
+ /* We now have to calculate which registers are live at the end
+ of the split basic block and at the start of the new basic
+ block. Start with those registers that are known to be live
+ at the end of the original basic block and get
+ propagate_block to determine which registers are live. */
+ COPY_REG_SET (new_bb->global_live_at_start, bb->global_live_at_end);
+ propagate_block (new_bb, new_bb->global_live_at_start, NULL, 0);
+ COPY_REG_SET (new_bb->global_live_at_end,
+ new_bb->global_live_at_start);
+ }
+
+ return new_edge;
+ }
+
+
/* Split a (typically critical) edge. Return the new block.
Abort on abnormal edges.
*************** count_or_remove_death_notes (blocks, kil
*** 6427,6432 ****
--- 6523,6550 ----
return count;
}
+
+ /* Update insns block within BB. */
+
+ void
+ update_bb_for_insn (bb)
+ basic_block bb;
+ {
+ rtx insn;
+
+ if (! basic_block_for_insn)
+ return;
+
+ for (insn = bb->head; ; insn = NEXT_INSN (insn))
+ {
+ set_block_for_insn (insn, bb);
+
+ if (insn == bb->end)
+ break;
+ }
+ }
+
+
/* Record INSN's block as BB. */
void