diff options
Diffstat (limited to 'gcc')
-rw-r--r-- | gcc/ChangeLog | 22 | ||||
-rw-r--r-- | gcc/Makefile.in | 4 | ||||
-rw-r--r-- | gcc/df.c | 664 | ||||
-rw-r--r-- | gcc/df.h | 44 |
4 files changed, 453 insertions, 281 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog index 03dbec951f4..df947461d34 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,25 @@ +2001-11-07 Daniel Berlin <dan@cgsoftware.com> + + * Makefile.in (df.o): Add fibheap.h to dependencies. + + * df.h: Add prototypes for transfer functions, iterative_dataflow + functions. + (enum df_flow_dir): New enum. + (enum df_confluence_op): New enum. + (struct df): Add inverse_rts_map. + + * df.c: Add sbitmap.h to the list of includes. + (df_rd_global_compute): Removed. + (df_ru_global_compute): Removed. + (df_lr_global_compute): Removed. + (df_rd_transfer_function): New function. + (df_ru_transfer_function): New function. + (df_lr_transfer_function): New function. + (df_analyse_1): allocate/compute/free df->inverse_rts_map. + Use iterative_dataflow_bitmap instead of df_*_global_compute. + (iterative_dataflow_sbitmap): New function. + (iterative_dataflow_bitmap): New function. + 2001-11-07 Joseph S. Myers <jsm28@cam.ac.uk> * doc/gcc.texi: Move terminology and spelling conventions to diff --git a/gcc/Makefile.in b/gcc/Makefile.in index 73596948bb9..8a9bf0e58e5 100644 --- a/gcc/Makefile.in +++ b/gcc/Makefile.in @@ -231,6 +231,7 @@ SYSTEM_HEADER_DIR = /usr/include HASHTAB_H = $(srcdir)/../include/hashtab.h OBSTACK_H = $(srcdir)/../include/obstack.h SPLAY_TREE_H= $(srcdir)/../include/splay-tree.h +FIBHEAP_H = $(srcdir)/../include/fibheap.h # Default cross SYSTEM_HEADER_DIR, to be overridden by targets. CROSS_SYSTEM_HEADER_DIR = $(tooldir)/sys-include @@ -1480,7 +1481,8 @@ ssa-ccp.o : ssa-ccp.c $(CONFIG_H) system.h $(RTL_H) hard-reg-set.h \ $(BASIC_BLOCK_H) ssa.h insn-config.h $(RECOG_H) output.h \ errors.h $(GGC_H) df.h function.h df.o : df.c $(CONFIG_H) system.h $(RTL_H) insn-config.h $(RECOG_H) \ - function.h $(REGS_H) $(OBSTACK_H) hard-reg-set.h $(BASIC_BLOCK_H) df.h + function.h $(REGS_H) $(OBSTACK_H) hard-reg-set.h $(BASIC_BLOCK_H) df.h \ + $(FIBHEAP_H) conflict.o : conflict.c $(CONFIG_H) $(SYSTEM_H) $(OBSTACK_H) $(HASHTAB_H) \ $(RTL_H) hard-reg-set.h $(BASIC_BLOCK_H) profile.o : profile.c $(CONFIG_H) $(SYSTEM_H) $(RTL_H) $(TREE_H) flags.h \ @@ -153,7 +153,7 @@ when optimising a loop, only certain registers are of interest. Perhaps there should be a bitmap argument to df_analyse to specify which registers should be analysed? */ -#define HANDLE_SUBREG +#define HANDLE_SUBREG #include "config.h" #include "system.h" @@ -166,9 +166,10 @@ Perhaps there should be a bitmap argument to df_analyse to specify #include "obstack.h" #include "hard-reg-set.h" #include "basic-block.h" +#include "sbitmap.h" #include "bitmap.h" #include "df.h" - +#include "fibheap.h" #define FOR_ALL_BBS(BB, CODE) \ do { \ @@ -239,8 +240,6 @@ static void df_insn_refs_record PARAMS((struct df *, basic_block, rtx)); static void df_bb_refs_record PARAMS((struct df *, basic_block)); static void df_refs_record PARAMS((struct df *, bitmap)); -static int df_visit_next_rc PARAMS ((struct df *, sbitmap)); -static int df_visit_next_rts PARAMS ((struct df *, sbitmap)); static void df_bb_reg_def_chain_create PARAMS((struct df *, basic_block)); static void df_reg_def_chain_create PARAMS((struct df *, bitmap)); static void df_bb_reg_use_chain_create PARAMS((struct df *, basic_block)); @@ -249,9 +248,6 @@ static void df_bb_du_chain_create PARAMS((struct df *, basic_block, bitmap)); static void df_du_chain_create PARAMS((struct df *, bitmap)); static void df_bb_ud_chain_create PARAMS((struct df *, basic_block)); static void df_ud_chain_create PARAMS((struct df *, bitmap)); -static void df_rd_global_compute PARAMS((struct df *, bitmap)); -static void df_ru_global_compute PARAMS((struct df *, bitmap)); -static void df_lr_global_compute PARAMS((struct df *, bitmap)); static void df_bb_rd_local_compute PARAMS((struct df *, basic_block)); static void df_rd_local_compute PARAMS((struct df *, bitmap)); static void df_bb_ru_local_compute PARAMS((struct df *, basic_block)); @@ -296,6 +292,12 @@ static void df_chain_dump PARAMS((struct df_link *, FILE *file)); static void df_chain_dump_regno PARAMS((struct df_link *, FILE *file)); static void df_regno_debug PARAMS ((struct df *, unsigned int, FILE *)); static void df_ref_debug PARAMS ((struct df *, struct ref *, FILE *)); +static void df_rd_transfer_function PARAMS ((int, int *, bitmap, bitmap, + bitmap, bitmap, void *)); +static void df_ru_transfer_function PARAMS ((int, int *, bitmap, bitmap, + bitmap, bitmap, void *)); +static void df_lr_transfer_function PARAMS ((int, int *, bitmap, bitmap, + bitmap, bitmap, void *)); /* Local memory allocation/deallocation routines. */ @@ -898,7 +900,6 @@ df_ref_record (df, reg, loc, bb, insn, ref_type) } } - /* Process all the registers defined in the rtx, X. */ static void df_def_record_1 (df, x, bb, insn) @@ -951,9 +952,9 @@ df_def_record_1 (df, x, bb, insn) dst = *loc; } #endif - - if (GET_CODE (dst) == REG - || (GET_CODE (dst) == SUBREG && GET_CODE (SUBREG_REG (dst)) == REG)) + + if (GET_CODE (dst) == REG + || (GET_CODE (dst) == SUBREG && GET_CODE (SUBREG_REG (dst)) == REG)) df_ref_record (df, dst, loc, bb, insn, DF_REF_REG_DEF); } @@ -1040,6 +1041,7 @@ df_uses_record (df, loc, ref_type, bb, insn) df_uses_record (df, loc, ref_type, bb, insn); return; } + #else loc = &SUBREG_REG (x); x = *loc; @@ -1049,7 +1051,6 @@ df_uses_record (df, loc, ref_type, bb, insn) return; } #endif - /* ... Fall through ... */ case REG: @@ -1619,260 +1620,34 @@ df_ud_chain_create (df, blocks) } -/* Use reverse completion order, and the worklist, to figure out what block - to look at next. */ - -static int -df_visit_next_rc (df, blocks) - struct df *df ATTRIBUTE_UNUSED; - sbitmap blocks; -{ - int i=0; - for (i = 0; i < n_basic_blocks; i++) - if (TEST_BIT (blocks, df->rc_order[i])) - return df->rc_order[i]; - return sbitmap_first_set_bit (blocks); -} - -/* Use reverse topsort order, and the worklist, to figure out what block - to look at next. */ - -static int -df_visit_next_rts (df, blocks) - struct df *df ATTRIBUTE_UNUSED; - sbitmap blocks; -{ - int i=0; - for (i = 0; i < n_basic_blocks; i++) - if (TEST_BIT (blocks, df->rts_order[i])) - return df->rts_order[i]; - return sbitmap_first_set_bit (blocks); -} - -/* Calculate reaching defs for each basic block in BLOCKS, i.e., the - defs that are live at the start of a basic block. */ static void -df_rd_global_compute (df, blocks) - struct df *df ATTRIBUTE_UNUSED; - bitmap blocks; +df_rd_transfer_function (bb, changed, in, out, gen, kill, data) + int bb ATTRIBUTE_UNUSED; + int *changed; + bitmap in, out, gen, kill; + void *data ATTRIBUTE_UNUSED; { - int i; - basic_block bb; - sbitmap worklist; - - worklist = sbitmap_alloc (n_basic_blocks); - sbitmap_zero (worklist); - - /* Copy the blocklist to the worklist */ - EXECUTE_IF_SET_IN_BITMAP (blocks, 0, i, - { - SET_BIT (worklist, i); - }); - - /* We assume that only the basic blocks in WORKLIST have been - modified. */ - FOR_EACH_BB_IN_SBITMAP (worklist, 0, bb, - { - struct bb_info *bb_info = DF_BB_INFO (df, bb); - - bitmap_copy (bb_info->rd_out, bb_info->rd_gen); - }); - - while ((i = df_visit_next_rc (df, worklist)) >= 0) - { - struct bb_info *bb_info; - edge e; - int changed; - - /* Remove this block from the worklist. */ - RESET_BIT (worklist, i); - - - bb = BASIC_BLOCK (i); - bb_info = DF_BB_INFO (df, bb); - - /* Calculate union of predecessor outs. */ - bitmap_zero (bb_info->rd_in); - for (e = bb->pred; e != 0; e = e->pred_next) - { - struct bb_info *pred_refs = DF_BB_INFO (df, e->src); - - if (e->src == ENTRY_BLOCK_PTR) - continue; - - bitmap_a_or_b (bb_info->rd_in, bb_info->rd_in, - pred_refs->rd_out); - } - - /* RD_OUT is the set of defs that are live at the end of the - BB. These are the defs that are either generated by defs - (RD_GEN) within the BB or are live at the start (RD_IN) - and are not killed by other defs (RD_KILL). */ - changed = bitmap_union_of_diff (bb_info->rd_out, bb_info->rd_gen, - bb_info->rd_in, bb_info->rd_kill); - - if (changed) - { - /* Add each of this block's successors to the worklist. */ - for (e = bb->succ; e != 0; e = e->succ_next) - { - if (e->dest == EXIT_BLOCK_PTR) - continue; - - SET_BIT (worklist, e->dest->index); - } - } - } - sbitmap_free (worklist); + *changed = bitmap_union_of_diff (out, gen, in, kill); } - - -/* Calculate reaching uses for each basic block within BLOCKS, i.e., - the uses that are live at the start of a basic block. */ static void -df_ru_global_compute (df, blocks) - struct df *df ATTRIBUTE_UNUSED; - bitmap blocks; +df_ru_transfer_function (bb, changed, in, out, gen, kill, data) + int bb ATTRIBUTE_UNUSED; + int *changed; + bitmap in, out, gen, kill; + void *data ATTRIBUTE_UNUSED; { - int i; - basic_block bb; - sbitmap worklist; - - worklist = sbitmap_alloc (n_basic_blocks); - sbitmap_zero (worklist); - - EXECUTE_IF_SET_IN_BITMAP (blocks, 0, i, - { - SET_BIT (worklist, i); - }); - - /* We assume that only the basic blocks in WORKLIST have been - modified. */ - FOR_EACH_BB_IN_SBITMAP (worklist, 0, bb, - { - struct bb_info *bb_info = DF_BB_INFO (df, bb); - - bitmap_copy (bb_info->ru_in, bb_info->ru_gen); - }); - - - while ((i = df_visit_next_rts (df, worklist)) >= 0) - { - struct bb_info *bb_info; - edge e; - int changed; - - /* Remove this block from the worklist. */ - RESET_BIT (worklist, i); - - bb = BASIC_BLOCK (i); - bb_info = DF_BB_INFO (df, bb); - - /* Calculate union of successor ins. */ - bitmap_zero (bb_info->ru_out); - for (e = bb->succ; e != 0; e = e->succ_next) - { - struct bb_info *succ_refs = DF_BB_INFO (df, e->dest); - - if (e->dest == EXIT_BLOCK_PTR) - continue; - - bitmap_a_or_b (bb_info->ru_out, bb_info->ru_out, - succ_refs->ru_in); - } - - /* RU_IN is the set of uses that are live at the start of the - BB. These are the uses that are either generated within the - BB (RU_GEN) or are live at the end (RU_OUT) and are not uses - killed by defs within the BB (RU_KILL). */ - changed = bitmap_union_of_diff (bb_info->ru_in, bb_info->ru_gen, - bb_info->ru_out, bb_info->ru_kill); - - if (changed) - { - /* Add each of this block's predecessors to the worklist. */ - for (e = bb->pred; e != 0; e = e->pred_next) - { - if (e->src == ENTRY_BLOCK_PTR) - continue; - - SET_BIT (worklist, e->src->index); - } - } - } - - sbitmap_free (worklist); + *changed = bitmap_union_of_diff (in, gen, out, kill); } - -/* Calculate live registers for each basic block within BLOCKS. */ static void -df_lr_global_compute (df, blocks) - struct df *df ATTRIBUTE_UNUSED; - bitmap blocks; +df_lr_transfer_function (bb, changed, in, out, use, def, data) + int bb ATTRIBUTE_UNUSED; + int *changed; + bitmap in, out, use, def; + void *data ATTRIBUTE_UNUSED; { - int i; - basic_block bb; - bitmap worklist; - - worklist = BITMAP_XMALLOC (); - bitmap_copy (worklist, blocks); - - /* We assume that only the basic blocks in WORKLIST have been - modified. */ - FOR_EACH_BB_IN_BITMAP (worklist, 0, bb, - { - struct bb_info *bb_info = DF_BB_INFO (df, bb); - - bitmap_copy (bb_info->lr_in, bb_info->lr_use); - }); - - while ((i = bitmap_last_set_bit (worklist)) >= 0) - { - struct bb_info *bb_info = DF_BB_INFO (df, bb); - edge e; - int changed; - - /* Remove this block from the worklist. */ - bitmap_clear_bit (worklist, i); - - bb = BASIC_BLOCK (i); - bb_info = DF_BB_INFO (df, bb); - - /* Calculate union of successor ins. */ - bitmap_zero (bb_info->lr_out); - for (e = bb->succ; e != 0; e = e->succ_next) - { - struct bb_info *succ_refs = DF_BB_INFO (df, e->dest); - - if (e->dest == EXIT_BLOCK_PTR) - continue; - - bitmap_a_or_b (bb_info->lr_out, bb_info->lr_out, - succ_refs->lr_in); - } - - /* LR_IN is the set of uses that are live at the start of the - BB. These are the uses that are either generated by uses - (LR_USE) within the BB or are live at the end (LR_OUT) - and are not killed by other uses (LR_DEF). */ - changed = bitmap_union_of_diff (bb_info->lr_in, bb_info->lr_use, - bb_info->lr_out, bb_info->lr_def); - - if (changed) - { - /* Add each of this block's predecessors to the worklist. */ - for (e = bb->pred; e != 0; e = e->pred_next) - { - if (e->src == ENTRY_BLOCK_PTR) - continue; - - bitmap_set_bit (worklist, e->src->index); - } - } - } - BITMAP_XFREE (worklist); + *changed = bitmap_union_of_diff (in, use, out, def); } @@ -1918,7 +1693,7 @@ df_bb_rd_local_compute (df, bb) bitmap_set_bit (bb_info->rd_gen, DF_REF_ID (def)); } } - + bb_info->rd_valid = 1; } @@ -1948,10 +1723,11 @@ df_bb_ru_local_compute (df, bb) /* This is much more tricky than computing reaching defs. With reaching defs, defs get killed by other defs. With upwards exposed uses, these get killed by defs with the same regno. */ - + struct bb_info *bb_info = DF_BB_INFO (df, bb); rtx insn; + for (insn = bb->end; insn && insn != PREV_INSN (bb->head); insn = PREV_INSN (insn)) { @@ -2071,7 +1847,7 @@ static void df_bb_reg_info_compute (df, bb, live) struct df *df; basic_block bb; - bitmap live; + bitmap live; { struct reg_info *reg_info = df->regs; struct bb_info *bb_info = DF_BB_INFO (df, bb); @@ -2190,7 +1966,7 @@ df_analyse_1 (df, blocks, flags, update) { int aflags; int dflags; - + int i; dflags = 0; aflags = flags; if (flags & DF_UD_CHAIN) @@ -2257,16 +2033,43 @@ df_analyse_1 (df, blocks, flags, update) df->dfs_order = xmalloc (sizeof(int) * n_basic_blocks); df->rc_order = xmalloc (sizeof(int) * n_basic_blocks); df->rts_order = xmalloc (sizeof(int) * n_basic_blocks); - + df->inverse_dfs_map = xmalloc (sizeof(int) * n_basic_blocks); + df->inverse_rc_map = xmalloc (sizeof(int) * n_basic_blocks); + df->inverse_rts_map = xmalloc (sizeof(int) * n_basic_blocks); + flow_depth_first_order_compute (df->dfs_order, df->rc_order); flow_reverse_top_sort_order_compute (df->rts_order); + for (i = 0; i < n_basic_blocks; i ++) + { + df->inverse_dfs_map[df->dfs_order[i]] = i; + df->inverse_rc_map[df->rc_order[i]] = i; + df->inverse_rts_map[df->rts_order[i]] = i; + } if (aflags & DF_RD) { /* Compute the sets of gens and kills for the defs of each bb. */ df_rd_local_compute (df, df->flags & DF_RD ? blocks : df->all_blocks); - - /* Compute the global reaching definitions. */ - df_rd_global_compute (df, df->all_blocks); + { + int i; + bitmap *in = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *out = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *gen = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *kill = xmalloc (sizeof (bitmap) * n_basic_blocks); + for (i = 0; i < n_basic_blocks; i ++) + { + in[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->rd_in; + out[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->rd_out; + gen[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->rd_gen; + kill[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->rd_kill; + } + iterative_dataflow_bitmap (in, out, gen, kill, df->all_blocks, + FORWARD, UNION, df_rd_transfer_function, + df->inverse_rc_map, NULL); + free (in); + free (out); + free (gen); + free (kill); + } } if (aflags & DF_UD_CHAIN) @@ -2283,9 +2086,27 @@ df_analyse_1 (df, blocks, flags, update) /* Compute the sets of gens and kills for the upwards exposed uses in each bb. */ df_ru_local_compute (df, df->flags & DF_RU ? blocks : df->all_blocks); - - /* Compute the global reaching uses. */ - df_ru_global_compute (df, df->all_blocks); + { + int i; + bitmap *in = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *out = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *gen = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *kill = xmalloc (sizeof (bitmap) * n_basic_blocks); + for (i = 0; i < n_basic_blocks; i ++) + { + in[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->ru_in; + out[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->ru_out; + gen[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->ru_gen; + kill[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->ru_kill; + } + iterative_dataflow_bitmap (in, out, gen, kill, df->all_blocks, + BACKWARD, UNION, df_ru_transfer_function, + df->inverse_rts_map, NULL); + free (in); + free (out); + free (gen); + free (kill); + } } if (aflags & DF_DU_CHAIN) @@ -2304,10 +2125,28 @@ df_analyse_1 (df, blocks, flags, update) if (aflags & DF_LR) { /* Compute the sets of defs and uses of live variables. */ - df_lr_local_compute (df, df->flags & DF_LR ? blocks : df->all_blocks); - - /* Compute the global live variables. */ - df_lr_global_compute (df, df->all_blocks); + df_lr_local_compute (df, df->flags & DF_LR ? blocks : df->all_blocks); + { + int i; + bitmap *in = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *out = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *use = xmalloc (sizeof (bitmap) * n_basic_blocks); + bitmap *def = xmalloc (sizeof (bitmap) * n_basic_blocks); + for (i = 0; i < n_basic_blocks; i ++) + { + in[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->lr_in; + out[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->lr_out; + use[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->lr_use; + def[i] = DF_BB_INFO (df, BASIC_BLOCK (i))->lr_def; + } + iterative_dataflow_bitmap (in, out, use, def, df->all_blocks, + BACKWARD, UNION, df_lr_transfer_function, + df->inverse_rts_map, NULL); + free (in); + free (out); + free (use); + free (def); + } } if (aflags & DF_REG_INFO) @@ -2317,6 +2156,9 @@ df_analyse_1 (df, blocks, flags, update) free (df->dfs_order); free (df->rc_order); free (df->rts_order); + free (df->inverse_rc_map); + free (df->inverse_dfs_map); + free (df->inverse_rts_map); } @@ -2641,14 +2483,11 @@ df_insn_modify (df, bb, insn) bitmap_set_bit (df->bbs_modified, bb->index); bitmap_set_bit (df->insns_modified, uid); -#if 0 /* For incremental updating on the fly, perhaps we could make a copy of all the refs of the original insn and turn them into anti-refs. When df_refs_update finds these anti-refs, it annihilates the original refs. If validate_change fails then these anti-refs will just get ignored. */ - */ -#endif } @@ -3770,3 +3609,280 @@ debug_df_chain (link) df_chain_dump (link, stderr); fputc ('\n', stderr); } + +/* gen = GEN set. + kill = KILL set. + in, out = Filled in by function. + blocks = Blocks to analyze. + dir = Dataflow direction. + conf_op = Confluence operation. + transfun = Transfer function. + order = Order to iterate in. (Should map block numbers -> order) + data = Whatever you want. It's passed to the transfer function. + + This function will perform iterative bitvector dataflow, producing + the in and out sets. Even if you only want to perform it for a + small number of blocks, the vectors for in and out must be large + enough for *all* blocks, because changing one block might affect + others. However, it'll only put what you say to analyze on the + initial worklist. + + For forward problems, you probably want to pass in a mapping of + block number to rc_order (like df->inverse_rc_map). +*/ + +void +iterative_dataflow_sbitmap (in, out, gen, kill, blocks, + dir, conf_op, transfun, order, data) + sbitmap *in, *out, *gen, *kill; + bitmap blocks; + enum df_flow_dir dir; + enum df_confluence_op conf_op; + transfer_function_sbitmap transfun; + int *order; + void *data; +{ + int changed; + int i; + fibheap_t worklist; + sbitmap onqueue; + basic_block bb; + onqueue = sbitmap_alloc (n_basic_blocks); + sbitmap_zero (onqueue); + worklist = fibheap_new (); + EXECUTE_IF_SET_IN_BITMAP (blocks, 0, i, + { + fibheap_insert (worklist, order[i], (void *) i); + SET_BIT (onqueue, i); + if (dir == FORWARD) + { + sbitmap_copy (out[i], gen[i]); + } + else + { + sbitmap_copy (in[i], gen[i]); + } + + }); + while (!fibheap_empty (worklist)) + { + edge e; + i = (int) fibheap_extract_min (worklist); + changed = 0; + bb = BASIC_BLOCK (i); + RESET_BIT (onqueue, i); + if (dir == FORWARD) + { + /* Calculate <conf_op> of predecessor_outs */ + for (e = bb->pred; e != 0; e = e->pred_next) + { + if (e->src == ENTRY_BLOCK_PTR) + { + sbitmap_zero (in[i]); + continue; + } + sbitmap_copy (in[i], out[e->src->index]); + break; + } + if (e == 0) + sbitmap_zero (in[i]); + for (e = bb->pred; e != 0; e = e->pred_next) + { + if (e->src == ENTRY_BLOCK_PTR) + continue; + switch (conf_op) + { + case UNION: + sbitmap_a_or_b (in[i], in[i], out[e->src->index]); + break; + case INTERSECTION: + sbitmap_a_and_b (in[i], in[i], out[e->src->index]); + break; + } + } + } + else + { + /* Calculate <conf_op> of successor ins */ + sbitmap_zero (out[i]); + for (e = bb->succ; e != 0; e = e->succ_next) + { + if (e->dest == EXIT_BLOCK_PTR) + continue; + switch (conf_op) + { + case UNION: + sbitmap_a_or_b (out[i], out[i], in[e->dest->index]); + break; + case INTERSECTION: + sbitmap_a_and_b (out[i], out[i], in[e->dest->index]); + break; + } + } + } + /* Common part */ + (*transfun)(i, &changed, in[i], out[i], gen[i], kill[i], data); + + if (changed) + { + if (dir == FORWARD) + { + for (e = bb->succ; e != 0; e = e->succ_next) + { + if (e->dest == EXIT_BLOCK_PTR) + continue; + if (!TEST_BIT (onqueue, e->dest->index)) + { + SET_BIT (onqueue, e->dest->index); + fibheap_insert (worklist, + order[e->dest->index], + (void *)e->dest->index); + } + } + } + else + { + for (e = bb->pred; e != 0; e = e->pred_next) + { + if (e->src == ENTRY_BLOCK_PTR) + continue; + if (!TEST_BIT (onqueue, e->src->index)) + { + SET_BIT (onqueue, e->src->index); + fibheap_insert (worklist, + order[e->src->index], + (void *)e->src->index); + } + + } + } + } + } + sbitmap_free (onqueue); + fibheap_delete (worklist); + +} +/* Exactly the same as iterative_dataflow_sbitmap, except it works on + bitmaps instead */ +void +iterative_dataflow_bitmap (in, out, gen, kill, blocks, + dir, conf_op, transfun, order, data) + bitmap *in, *out, *gen, *kill; + bitmap blocks; + enum df_flow_dir dir; + enum df_confluence_op conf_op; + transfer_function_bitmap transfun; + int *order; + void *data; +{ + int changed; + int i; + fibheap_t worklist; + sbitmap onqueue; + basic_block bb; + + onqueue = sbitmap_alloc (n_basic_blocks); + sbitmap_zero (onqueue); + worklist = fibheap_new (); + EXECUTE_IF_SET_IN_BITMAP (blocks, 0, i, + { + fibheap_insert (worklist, order[i], (void *) i); + SET_BIT (onqueue, i); + if (dir == FORWARD) + { + bitmap_copy (out[i], gen[i]); + } + else + { + bitmap_copy (in[i], gen[i]); + } + + }); + while (!fibheap_empty (worklist)) + { + edge e; + i = (int) fibheap_extract_min (worklist); + changed = 0; + bb = BASIC_BLOCK (i); + RESET_BIT (onqueue, i); + + if (dir == FORWARD) + { + /* Calculate <conf_op> of predecessor_outs */ + bitmap_zero (in[i]); + for (e = bb->pred; e != 0; e = e->pred_next) + { + if (e->src == ENTRY_BLOCK_PTR) + continue; + switch (conf_op) + { + case UNION: + bitmap_a_or_b (in[i], in[i], out[e->src->index]); + break; + case INTERSECTION: + bitmap_a_and_b (in[i], in[i], out[e->src->index]); + break; + } + } + } + else + { + /* Calculate <conf_op> of successor ins */ + bitmap_zero(out[i]); + for (e = bb->succ; e != 0; e = e->succ_next) + { + if (e->dest == EXIT_BLOCK_PTR) + continue; + switch (conf_op) + { + case UNION: + bitmap_a_or_b (out[i], out[i], in[e->dest->index]); + break; + case INTERSECTION: + bitmap_a_and_b (out[i], out[i], in[e->dest->index]); + break; + } + } + } + /* Common part */ + (*transfun)(i, &changed, in[i], out[i], gen[i], kill[i], data); + + if (changed) + { + if (dir == FORWARD) + { + for (e = bb->succ; e != 0; e = e->succ_next) + { + if (e->dest == EXIT_BLOCK_PTR) + continue; + if (!TEST_BIT (onqueue, e->dest->index)) + { + SET_BIT (onqueue, e->dest->index); + fibheap_insert (worklist, + order[e->dest->index], + (void *)e->dest->index); + } + } + } + else + { + for (e = bb->pred; e != 0; e = e->pred_next) + { + if (e->src == ENTRY_BLOCK_PTR) + continue; + if (!TEST_BIT (onqueue, e->src->index)) + { + SET_BIT (onqueue, e->src->index); + fibheap_insert (worklist, + order[e->src->index], + (void *)e->src->index); + } + + } + } + } + } + sbitmap_free (onqueue); + fibheap_delete (worklist); + +} @@ -20,7 +20,6 @@ along with GCC; see the file COPYING. If not, write to the Free Software Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ - #define DF_RD 1 /* Reaching definitions. */ #define DF_RU 2 /* Reaching uses. */ #define DF_LR 4 /* Live registers. */ @@ -136,12 +135,15 @@ struct df bitmap insns_modified; /* Insns that (may) have changed. */ bitmap bbs_modified; /* Blocks that (may) have changed. */ bitmap all_blocks; /* All blocks in CFG. */ - /* The bitmap vector of dominators or NULL if not computed. + /* The sbitmap vector of dominators or NULL if not computed. Ideally, this should be a pointer to a CFG object. */ - bitmap *dom; - int * dfs_order; - int * rc_order; - int * rts_order; + sbitmap *dom; + int * dfs_order; /* DFS order -> block number */ + int * rc_order; /* reverse completion order -> block number */ + int * rts_order; /* reverse top sort order -> block number */ + int * inverse_rc_map; /* block number -> reverse completion order */ + int * inverse_dfs_map; /* block number -> DFS order */ + int * inverse_rts_map; /* block number -> reverse top-sort order */ }; @@ -297,3 +299,33 @@ extern void debug_df_ref PARAMS ((struct ref *)); extern void debug_df_chain PARAMS ((struct df_link *)); extern void df_insn_debug PARAMS ((struct df *, rtx, FILE *)); extern void df_insn_debug_regno PARAMS ((struct df *, rtx, FILE *)); +/* Meet over any path (UNION) or meet over all paths (INTERSECTION) */ +enum df_confluence_op + { + UNION, + INTERSECTION + }; +/* Dataflow direction */ +enum df_flow_dir + { + FORWARD, + BACKWARD + }; + +typedef void (*transfer_function_sbitmap) (int, int *, sbitmap, sbitmap, + sbitmap, sbitmap, void *); +typedef void (*transfer_function_bitmap) (int, int *, bitmap, bitmap, + bitmap, bitmap, void *); + +extern void iterative_dataflow_sbitmap PARAMS ((sbitmap *, sbitmap *, + sbitmap *, sbitmap *, + bitmap, enum df_flow_dir, + enum df_confluence_op, + transfer_function_sbitmap, + int *, void *)); +extern void iterative_dataflow_bitmap PARAMS ((bitmap *, bitmap *, bitmap *, + bitmap *, bitmap, + enum df_flow_dir, + enum df_confluence_op, + transfer_function_bitmap, + int *, void *)); |