diff options
Diffstat (limited to 'contrib/gcc/df.c')
-rw-r--r-- | contrib/gcc/df.c | 542 |
1 files changed, 286 insertions, 256 deletions
diff --git a/contrib/gcc/df.c b/contrib/gcc/df.c index ed0fd77..fb434ff 100644 --- a/contrib/gcc/df.c +++ b/contrib/gcc/df.c @@ -153,8 +153,6 @@ 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 - #include "config.h" #include "system.h" #include "rtl.h" @@ -171,32 +169,14 @@ Perhaps there should be a bitmap argument to df_analyse to specify #include "df.h" #include "fibheap.h" -#define FOR_ALL_BBS(BB, CODE) \ -do { \ - int node_; \ - for (node_ = 0; node_ < n_basic_blocks; node_++) \ - {(BB) = BASIC_BLOCK (node_); CODE;};} while (0) - -#define FOR_EACH_BB_IN_BITMAP(BITMAP, MIN, BB, CODE) \ -do { \ - unsigned int node_; \ - EXECUTE_IF_SET_IN_BITMAP (BITMAP, MIN, node_, \ - {(BB) = BASIC_BLOCK (node_); CODE;});} while (0) - -#define FOR_EACH_BB_IN_BITMAP_REV(BITMAP, MIN, BB, CODE) \ -do { \ - unsigned int node_; \ - EXECUTE_IF_SET_IN_BITMAP_REV (BITMAP, node_, \ - {(BB) = BASIC_BLOCK (node_); CODE;});} while (0) - -#define FOR_EACH_BB_IN_SBITMAP(BITMAP, MIN, BB, CODE) \ -do { \ - unsigned int node_; \ - EXECUTE_IF_SET_IN_SBITMAP (BITMAP, MIN, node_, \ - {(BB) = BASIC_BLOCK (node_); CODE;});} while (0) - -#define obstack_chunk_alloc xmalloc -#define obstack_chunk_free free +#define FOR_EACH_BB_IN_BITMAP(BITMAP, MIN, BB, CODE) \ + do \ + { \ + unsigned int node_; \ + EXECUTE_IF_SET_IN_BITMAP (BITMAP, MIN, node_, \ + {(BB) = BASIC_BLOCK (node_); CODE;}); \ + } \ + while (0) static struct obstack df_ref_obstack; static struct df *ddf; @@ -205,7 +185,7 @@ static void df_reg_table_realloc PARAMS((struct df *, int)); #if 0 static void df_def_table_realloc PARAMS((struct df *, int)); #endif -static void df_insn_table_realloc PARAMS((struct df *, int)); +static void df_insn_table_realloc PARAMS((struct df *, unsigned int)); static void df_bitmaps_alloc PARAMS((struct df *, int)); static void df_bitmaps_free PARAMS((struct df *, int)); static void df_free PARAMS((struct df *)); @@ -295,16 +275,16 @@ 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, +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, +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, +static void df_lr_transfer_function PARAMS ((int, int *, bitmap, bitmap, bitmap, bitmap, void *)); -static void hybrid_search_bitmap PARAMS ((basic_block, bitmap *, bitmap *, - bitmap *, bitmap *, enum df_flow_dir, - enum df_confluence_op, - transfer_function_bitmap, +static void hybrid_search_bitmap PARAMS ((basic_block, bitmap *, bitmap *, + bitmap *, bitmap *, enum df_flow_dir, + enum df_confluence_op, + transfer_function_bitmap, sbitmap, sbitmap, void *)); static void hybrid_search_sbitmap PARAMS ((basic_block, sbitmap *, sbitmap *, sbitmap *, sbitmap *, enum df_flow_dir, @@ -317,17 +297,20 @@ static inline bool read_modify_subreg_p PARAMS ((rtx)); /* Local memory allocation/deallocation routines. */ -/* Increase the insn info table by SIZE more elements. */ +/* Increase the insn info table to have space for at least SIZE + 1 + elements. */ static void df_insn_table_realloc (df, size) struct df *df; - int size; + unsigned int size; { - /* Make table 25 percent larger by default. */ - if (! size) - size = df->insn_size / 4; + size++; + if (size <= df->insn_size) + return; - size += df->insn_size; + /* Make the table a little larger than requested, so we don't need + to enlarge it so often. */ + size += df->insn_size / 4; df->insns = (struct insn_info *) xrealloc (df->insns, size * sizeof (struct insn_info)); @@ -356,6 +339,8 @@ df_reg_table_realloc (df, size) size = df->reg_size / 4; size += df->reg_size; + if (size < max_reg_num ()) + size = max_reg_num (); df->regs = (struct reg_info *) xrealloc (df->regs, size * sizeof (struct reg_info)); @@ -393,7 +378,7 @@ df_def_table_realloc (df, size) /* Link all the new refs together, overloading the chain field. */ for (i = 0; i < size - 1; i++) - refs[i].chain = (struct df_link *)(refs + i + 1); + refs[i].chain = (struct df_link *) (refs + i + 1); refs[size - 1].chain = 0; } #endif @@ -406,11 +391,11 @@ df_bitmaps_alloc (df, flags) struct df *df; int flags; { - unsigned int i; int dflags = 0; + basic_block bb; /* Free the bitmaps if they need resizing. */ - if ((flags & DF_LR) && df->n_regs < (unsigned int)max_reg_num ()) + if ((flags & DF_LR) && df->n_regs < (unsigned int) max_reg_num ()) dflags |= DF_LR | DF_RU; if ((flags & DF_RU) && df->n_uses < df->use_id) dflags |= DF_RU; @@ -423,9 +408,8 @@ df_bitmaps_alloc (df, flags) df->n_defs = df->def_id; df->n_uses = df->use_id; - for (i = 0; i < df->n_bbs; i++) + FOR_EACH_BB (bb) { - basic_block bb = BASIC_BLOCK (i); struct bb_info *bb_info = DF_BB_INFO (df, bb); if (flags & DF_RD && ! bb_info->rd_in) @@ -474,11 +458,10 @@ df_bitmaps_free (df, flags) struct df *df ATTRIBUTE_UNUSED; int flags; { - unsigned int i; + basic_block bb; - for (i = 0; i < df->n_bbs; i++) + FOR_EACH_BB (bb) { - basic_block bb = BASIC_BLOCK (i); struct bb_info *bb_info = DF_BB_INFO (df, bb); if (!bb_info) @@ -527,14 +510,14 @@ df_bitmaps_free (df, flags) } -/* Allocate and initialise dataflow memory. */ +/* Allocate and initialize dataflow memory. */ static void df_alloc (df, n_regs) struct df *df; int n_regs; { int n_insns; - int i; + basic_block bb; gcc_obstack_init (&df_ref_obstack); @@ -555,7 +538,7 @@ df_alloc (df, n_regs) df->uses = xmalloc (df->use_size * sizeof (*df->uses)); df->n_regs = n_regs; - df->n_bbs = n_basic_blocks; + df->n_bbs = last_basic_block; /* Allocate temporary working array used during local dataflow analysis. */ df->reg_def_last = xmalloc (df->n_regs * sizeof (struct ref *)); @@ -569,11 +552,11 @@ df_alloc (df, n_regs) df->flags = 0; - df->bbs = xcalloc (df->n_bbs, sizeof (struct bb_info)); + df->bbs = xcalloc (last_basic_block, sizeof (struct bb_info)); df->all_blocks = BITMAP_XMALLOC (); - for (i = 0; i < n_basic_blocks; i++) - bitmap_set_bit (df->all_blocks, i); + FOR_EACH_BB (bb) + bitmap_set_bit (df->all_blocks, bb->index); } @@ -633,8 +616,7 @@ static rtx df_reg_use_gen (regno) rtx reg; rtx use; - reg = regno >= FIRST_PSEUDO_REGISTER - ? regno_reg_rtx[regno] : gen_rtx_REG (reg_raw_mode[regno], regno); + reg = regno_reg_rtx[regno]; use = gen_rtx_USE (GET_MODE (reg), reg); return use; @@ -648,8 +630,7 @@ static rtx df_reg_clobber_gen (regno) rtx reg; rtx use; - reg = regno >= FIRST_PSEUDO_REGISTER - ? regno_reg_rtx[regno] : gen_rtx_REG (reg_raw_mode[regno], regno); + reg = regno_reg_rtx[regno]; use = gen_rtx_CLOBBER (GET_MODE (reg), reg); return use; @@ -881,7 +862,7 @@ df_ref_record (df, reg, loc, insn, ref_type, ref_flags) XXX Is that true? We could also use the global word_mode variable. */ if (GET_CODE (reg) == SUBREG && (GET_MODE_SIZE (GET_MODE (reg)) < GET_MODE_SIZE (word_mode) - || GET_MODE_SIZE (GET_MODE (reg)) + || GET_MODE_SIZE (GET_MODE (reg)) >= GET_MODE_SIZE (GET_MODE (SUBREG_REG (reg))))) { loc = &SUBREG_REG (reg); @@ -902,10 +883,14 @@ df_ref_record (df, reg, loc, insn, ref_type, ref_flags) are really referenced. E.g. a (subreg:SI (reg:DI 0) 0) does _not_ reference the whole reg 0 in DI mode (which would also include reg 1, at least, if 0 and 1 are SImode registers). */ - endregno = regno + HARD_REGNO_NREGS (regno, GET_MODE (reg)); + endregno = HARD_REGNO_NREGS (regno, GET_MODE (reg)); + if (GET_CODE (reg) == SUBREG) + regno += subreg_regno_offset (regno, GET_MODE (SUBREG_REG (reg)), + SUBREG_BYTE (reg), GET_MODE (reg)); + endregno += regno; for (i = regno; i < endregno; i++) - df_ref_record_1 (df, gen_rtx_REG (reg_raw_mode[i], i), + df_ref_record_1 (df, regno_reg_rtx[i], loc, insn, ref_type, ref_flags); } else @@ -914,18 +899,23 @@ df_ref_record (df, reg, loc, insn, ref_type, ref_flags) } } -/* Writes to SUBREG of inndermode wider than word and outermode shorter than - word are read-modify-write. */ +/* Writes to paradoxical subregs, or subregs which are too narrow + are read-modify-write. */ static inline bool read_modify_subreg_p (x) rtx x; { + unsigned int isize, osize; if (GET_CODE (x) != SUBREG) return false; - if (GET_MODE_SIZE (GET_MODE (SUBREG_REG (x))) <= UNITS_PER_WORD) + isize = GET_MODE_SIZE (GET_MODE (SUBREG_REG (x))); + osize = GET_MODE_SIZE (GET_MODE (x)); + if (isize <= osize) + return true; + if (isize <= UNITS_PER_WORD) return false; - if (GET_MODE_SIZE (GET_MODE (x)) > UNITS_PER_WORD) + if (osize >= UNITS_PER_WORD) return false; return true; } @@ -949,14 +939,21 @@ df_def_record_1 (df, x, bb, insn) int i; for (i = XVECLEN (dst, 0) - 1; i >= 0; i--) - df_def_record_1 (df, XVECEXP (dst, 0, i), bb, insn); + df_def_record_1 (df, XVECEXP (dst, 0, i), bb, insn); return; } +#ifdef CLASS_CANNOT_CHANGE_MODE + if (GET_CODE (dst) == SUBREG + && CLASS_CANNOT_CHANGE_MODE_P (GET_MODE (SUBREG_REG (dst)), + GET_MODE (dst))) + flags |= DF_REF_MODE_CHANGE; +#endif + /* May be, we should flag the use of strict_low_part somehow. Might be handy for the reg allocator. */ while (GET_CODE (dst) == STRICT_LOW_PART - || GET_CODE (dst) == ZERO_EXTRACT + || GET_CODE (dst) == ZERO_EXTRACT || GET_CODE (dst) == SIGN_EXTRACT || read_modify_subreg_p (dst)) { @@ -967,14 +964,20 @@ df_def_record_1 (df, x, bb, insn) loc = &XEXP (dst, 0); dst = *loc; } +#ifdef CLASS_CANNOT_CHANGE_MODE + if (GET_CODE (dst) == SUBREG + && CLASS_CANNOT_CHANGE_MODE_P (GET_MODE (SUBREG_REG (dst)), + GET_MODE (dst))) + flags |= DF_REF_MODE_CHANGE; +#endif loc = &XEXP (dst, 0); dst = *loc; flags |= DF_REF_READ_WRITE; } - - if (GET_CODE (dst) == REG - || (GET_CODE (dst) == SUBREG && GET_CODE (SUBREG_REG (dst)) == REG)) - df_ref_record (df, dst, loc, insn, DF_REF_REG_DEF, flags); + + if (GET_CODE (dst) == REG + || (GET_CODE (dst) == SUBREG && GET_CODE (SUBREG_REG (dst)) == REG)) + df_ref_record (df, dst, loc, insn, DF_REF_REG_DEF, flags); } @@ -1034,6 +1037,7 @@ df_uses_record (df, loc, ref_type, bb, insn, flags) case CONST_DOUBLE: case CONST_VECTOR: case PC: + case CC0: case ADDR_VEC: case ADDR_DIFF_VEC: return; @@ -1062,6 +1066,11 @@ df_uses_record (df, loc, ref_type, bb, insn, flags) df_uses_record (df, loc, ref_type, bb, insn, flags); return; } +#ifdef CLASS_CANNOT_CHANGE_MODE + if (CLASS_CANNOT_CHANGE_MODE_P (GET_MODE (x), + GET_MODE (SUBREG_REG (x)))) + flags |= DF_REF_MODE_CHANGE; +#endif /* ... Fall through ... */ @@ -1078,19 +1087,28 @@ df_uses_record (df, loc, ref_type, bb, insn, flags) switch (GET_CODE (dst)) { + enum df_ref_flags use_flags; case SUBREG: if (read_modify_subreg_p (dst)) { + use_flags = DF_REF_READ_WRITE; +#ifdef CLASS_CANNOT_CHANGE_MODE + if (CLASS_CANNOT_CHANGE_MODE_P (GET_MODE (dst), + GET_MODE (SUBREG_REG (dst)))) + use_flags |= DF_REF_MODE_CHANGE; +#endif df_uses_record (df, &SUBREG_REG (dst), DF_REF_REG_USE, bb, - insn, DF_REF_READ_WRITE); + insn, use_flags); break; } /* ... FALLTHRU ... */ case REG: + case PARALLEL: case PC: - break; + case CC0: + break; case MEM: - df_uses_record (df, &XEXP (dst, 0), + df_uses_record (df, &XEXP (dst, 0), DF_REF_REG_MEM_STORE, bb, insn, 0); break; @@ -1099,8 +1117,14 @@ df_uses_record (df, loc, ref_type, bb, insn, flags) dst = XEXP (dst, 0); if (GET_CODE (dst) != SUBREG) abort (); + use_flags = DF_REF_READ_WRITE; +#ifdef CLASS_CANNOT_CHANGE_MODE + if (CLASS_CANNOT_CHANGE_MODE_P (GET_MODE (dst), + GET_MODE (SUBREG_REG (dst)))) + use_flags |= DF_REF_MODE_CHANGE; +#endif df_uses_record (df, &SUBREG_REG (dst), DF_REF_REG_USE, bb, - insn, DF_REF_READ_WRITE); + insn, use_flags); break; case ZERO_EXTRACT: case SIGN_EXTRACT: @@ -1135,7 +1159,7 @@ df_uses_record (df, loc, ref_type, bb, insn, flags) For now, just mark any regs we can find in ASM_OPERANDS as used. */ - /* For all ASM_OPERANDS, we must traverse the vector of input operands. + /* For all ASM_OPERANDS, we must traverse the vector of input operands. We can not just fall through here since then we would be confused by the ASM_INPUT rtx inside ASM_OPERANDS, which do not indicate traditional asms unlike their normal usage. */ @@ -1217,12 +1241,12 @@ df_insn_refs_record (df, bb, insn) { switch (REG_NOTE_KIND (note)) { - case REG_EQUIV: - case REG_EQUAL: - df_uses_record (df, &XEXP (note, 0), DF_REF_REG_USE, - bb, insn, 0); - default: - break; + case REG_EQUIV: + case REG_EQUAL: + df_uses_record (df, &XEXP (note, 0), DF_REF_REG_USE, + bb, insn, 0); + default: + break; } } @@ -1253,7 +1277,7 @@ df_insn_refs_record (df, bb, insn) { x = df_reg_use_gen (i); df_uses_record (df, &SET_DEST (x), - DF_REF_REG_USE, bb, insn, 0); + DF_REF_REG_USE, bb, insn, 0); } } } @@ -1355,6 +1379,11 @@ df_bb_reg_def_chain_create (df, bb) { struct ref *def = link->ref; unsigned int dregno = DF_REF_REGNO (def); + /* Don't add ref's to the chain two times. I.e. only add + new refs. XXX the same could be done by testing if the current + insn is a modified (or a new) one. This would be faster. */ + if (DF_REF_ID (def) < df->def_id_save) + continue; df->regs[dregno].defs = df_link_create (def, df->regs[dregno].defs); @@ -1404,6 +1433,11 @@ df_bb_reg_use_chain_create (df, bb) { struct ref *use = link->ref; unsigned int uregno = DF_REF_REGNO (use); + /* Don't add ref's to the chain two times. I.e. only add + new refs. XXX the same could be done by testing if the current + insn is a modified (or a new) one. This would be faster. */ + if (DF_REF_ID (use) < df->use_id_save) + continue; df->regs[uregno].uses = df_link_create (use, df->regs[uregno].uses); @@ -1673,7 +1707,7 @@ df_bb_rd_local_compute (df, bb) bitmap_set_bit (bb_info->rd_gen, DF_REF_ID (def)); } } - + bb_info->rd_valid = 1; } @@ -1703,7 +1737,7 @@ 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; @@ -1946,6 +1980,8 @@ df_analyse_1 (df, blocks, flags, update) int aflags; int dflags; int i; + basic_block bb; + dflags = 0; aflags = flags; if (flags & DF_UD_CHAIN) @@ -1961,7 +1997,7 @@ df_analyse_1 (df, blocks, flags, update) aflags |= DF_LR; if (! blocks) - blocks = df->all_blocks; + blocks = df->all_blocks; df->flags = flags; if (update) @@ -2009,39 +2045,38 @@ df_analyse_1 (df, blocks, flags, update) df_reg_use_chain_create (df, blocks); } - 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); - + 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) * last_basic_block); + df->inverse_rc_map = xmalloc (sizeof (int) * last_basic_block); + df->inverse_rts_map = xmalloc (sizeof (int) * last_basic_block); + 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; - } + 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); { - 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 ++) + bitmap *in = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *out = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *gen = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *kill = xmalloc (sizeof (bitmap) * last_basic_block); + FOR_EACH_BB (bb) { - 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; + in[bb->index] = DF_BB_INFO (df, bb)->rd_in; + out[bb->index] = DF_BB_INFO (df, bb)->rd_out; + gen[bb->index] = DF_BB_INFO (df, bb)->rd_gen; + kill[bb->index] = DF_BB_INFO (df, bb)->rd_kill; } - iterative_dataflow_bitmap (in, out, gen, kill, df->all_blocks, + iterative_dataflow_bitmap (in, out, gen, kill, df->all_blocks, FORWARD, UNION, df_rd_transfer_function, df->inverse_rc_map, NULL); free (in); @@ -2066,19 +2101,18 @@ df_analyse_1 (df, blocks, flags, update) uses in each bb. */ df_ru_local_compute (df, df->flags & DF_RU ? blocks : 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 ++) + bitmap *in = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *out = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *gen = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *kill = xmalloc (sizeof (bitmap) * last_basic_block); + FOR_EACH_BB (bb) { - 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; + in[bb->index] = DF_BB_INFO (df, bb)->ru_in; + out[bb->index] = DF_BB_INFO (df, bb)->ru_out; + gen[bb->index] = DF_BB_INFO (df, bb)->ru_gen; + kill[bb->index] = DF_BB_INFO (df, bb)->ru_kill; } - iterative_dataflow_bitmap (in, out, gen, kill, df->all_blocks, + iterative_dataflow_bitmap (in, out, gen, kill, df->all_blocks, BACKWARD, UNION, df_ru_transfer_function, df->inverse_rts_map, NULL); free (in); @@ -2099,26 +2133,25 @@ df_analyse_1 (df, blocks, flags, update) /* Free up bitmaps that are no longer required. */ if (dflags) - df_bitmaps_free (df, dflags); + df_bitmaps_free (df, dflags); 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); + 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 ++) + bitmap *in = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *out = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *use = xmalloc (sizeof (bitmap) * last_basic_block); + bitmap *def = xmalloc (sizeof (bitmap) * last_basic_block); + FOR_EACH_BB (bb) { - 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; + in[bb->index] = DF_BB_INFO (df, bb)->lr_in; + out[bb->index] = DF_BB_INFO (df, bb)->lr_out; + use[bb->index] = DF_BB_INFO (df, bb)->lr_use; + def[bb->index] = DF_BB_INFO (df, bb)->lr_def; } - iterative_dataflow_bitmap (in, out, use, def, df->all_blocks, + iterative_dataflow_bitmap (in, out, use, def, df->all_blocks, BACKWARD, UNION, df_lr_transfer_function, df->inverse_rts_map, NULL); free (in); @@ -2141,7 +2174,7 @@ df_analyse_1 (df, blocks, flags, update) } -/* Initialise dataflow analysis. */ +/* Initialize dataflow analysis. */ struct df * df_init () { @@ -2229,8 +2262,6 @@ df_bb_refs_update (df, bb) /* Scan the insn for refs. */ df_insn_refs_record (df, bb, insn); - - bitmap_clear_bit (df->insns_modified, uid); count++; } if (insn == bb->end) @@ -2248,7 +2279,7 @@ df_refs_update (df) basic_block bb; int count = 0; - if ((unsigned int)max_reg_num () >= df->reg_size) + if ((unsigned int) max_reg_num () >= df->reg_size) df_reg_table_realloc (df, 0); df_refs_queue (df); @@ -2263,19 +2294,22 @@ df_refs_update (df) } -/* Return non-zero if any of the requested blocks in the bitmap +/* Return nonzero if any of the requested blocks in the bitmap BLOCKS have been modified. */ static int df_modified_p (df, blocks) struct df *df; bitmap blocks; { - unsigned int j; int update = 0; + basic_block bb; + + if (!df->n_bbs) + return 0; - for (j = 0; j < df->n_bbs; j++) - if (bitmap_bit_p (df->bbs_modified, j) - && (! blocks || (blocks == (bitmap) -1) || bitmap_bit_p (blocks, j))) + FOR_EACH_BB (bb) + if (bitmap_bit_p (df->bbs_modified, bb->index) + && (! blocks || (blocks == (bitmap) -1) || bitmap_bit_p (blocks, bb->index))) { update = 1; break; @@ -2298,7 +2332,7 @@ df_analyse (df, blocks, flags) /* We could deal with additional basic blocks being created by rescanning everything again. */ - if (df->n_bbs && df->n_bbs != (unsigned int)n_basic_blocks) + if (df->n_bbs && df->n_bbs != (unsigned int) last_basic_block) abort (); update = df_modified_p (df, blocks); @@ -2311,7 +2345,7 @@ df_analyse (df, blocks, flags) /* Recompute everything from scratch. */ df_free (df); } - /* Allocate and initialise data structures. */ + /* Allocate and initialize data structures. */ df_alloc (df, max_reg_num ()); df_analyse_1 (df, 0, flags, 0); update = 1; @@ -2326,6 +2360,7 @@ df_analyse (df, blocks, flags) df_analyse_1 (df, blocks, flags, 1); bitmap_zero (df->bbs_modified); + bitmap_zero (df->insns_modified); } } return update; @@ -2408,10 +2443,8 @@ df_refs_unlink (df, blocks) } else { - FOR_ALL_BBS (bb, - { + FOR_EACH_BB (bb) df_bb_refs_unlink (df, bb); - }); } } #endif @@ -2455,9 +2488,8 @@ df_insn_modify (df, bb, insn) unsigned int uid; uid = INSN_UID (insn); - if (uid >= df->insn_size) - df_insn_table_realloc (df, 0); + df_insn_table_realloc (df, uid); bitmap_set_bit (df->bbs_modified, bb->index); bitmap_set_bit (df->insns_modified, uid); @@ -2470,8 +2502,7 @@ df_insn_modify (df, bb, insn) } -typedef struct replace_args -{ +typedef struct replace_args { rtx match; rtx replacement; rtx insn; @@ -2744,7 +2775,7 @@ df_insns_modify (df, bb, first_insn, last_insn) uid = INSN_UID (insn); if (uid >= df->insn_size) - df_insn_table_realloc (df, 0); + df_insn_table_realloc (df, uid); df_insn_modify (df, bb, insn); @@ -2924,7 +2955,7 @@ df_insn_dominates_all_uses_p (df, bb, insn) } -/* Return non-zero if all DF dominates all the uses within the bitmap +/* Return nonzero if all DF dominates all the uses within the bitmap BLOCKS. */ static int df_def_dominates_uses_p (df, def, blocks) @@ -2955,7 +2986,7 @@ df_def_dominates_uses_p (df, def, blocks) } -/* Return non-zero if all the defs of INSN within BB dominates +/* Return nonzero if all the defs of INSN within BB dominates all the corresponding uses. */ int df_insn_dominates_uses_p (df, bb, insn, blocks) @@ -3002,7 +3033,7 @@ df_regno_bb (df, regno) } -/* Return non-zero if REG used in multiple basic blocks. */ +/* Return nonzero if REG used in multiple basic blocks. */ int df_reg_global_p (df, reg) struct df *df; @@ -3022,7 +3053,7 @@ df_reg_lifetime (df, reg) } -/* Return non-zero if REG live at start of BB. */ +/* Return nonzero if REG live at start of BB. */ int df_bb_reg_live_start_p (df, bb, reg) struct df *df ATTRIBUTE_UNUSED; @@ -3040,7 +3071,7 @@ df_bb_reg_live_start_p (df, bb, reg) } -/* Return non-zero if REG live at end of BB. */ +/* Return nonzero if REG live at end of BB. */ int df_bb_reg_live_end_p (df, bb, reg) struct df *df ATTRIBUTE_UNUSED; @@ -3260,9 +3291,9 @@ df_chain_dump_regno (link, file) for (; link; link = link->next) { fprintf (file, "%c%d(%d) ", - DF_REF_REG_DEF_P (link->ref) ? 'd' : 'u', - DF_REF_ID (link->ref), - DF_REF_REGNO (link->ref)); + DF_REF_REG_DEF_P (link->ref) ? 'd' : 'u', + DF_REF_ID (link->ref), + DF_REF_REGNO (link->ref)); } fprintf (file, "}"); } @@ -3274,8 +3305,8 @@ df_dump (df, flags, file) int flags; FILE *file; { - unsigned int i; unsigned int j; + basic_block bb; if (! df || ! file) return; @@ -3286,22 +3317,23 @@ df_dump (df, flags, file) if (flags & DF_RD) { + basic_block bb; + fprintf (file, "Reaching defs:\n"); - for (i = 0; i < df->n_bbs; i++) + FOR_EACH_BB (bb) { - basic_block bb = BASIC_BLOCK (i); struct bb_info *bb_info = DF_BB_INFO (df, bb); if (! bb_info->rd_in) continue; - fprintf (file, "bb %d in \t", i); + fprintf (file, "bb %d in \t", bb->index); dump_bitmap (file, bb_info->rd_in); - fprintf (file, "bb %d gen \t", i); + fprintf (file, "bb %d gen \t", bb->index); dump_bitmap (file, bb_info->rd_gen); - fprintf (file, "bb %d kill\t", i); + fprintf (file, "bb %d kill\t", bb->index); dump_bitmap (file, bb_info->rd_kill); - fprintf (file, "bb %d out \t", i); + fprintf (file, "bb %d out \t", bb->index); dump_bitmap (file, bb_info->rd_out); } } @@ -3329,21 +3361,20 @@ df_dump (df, flags, file) if (flags & DF_RU) { fprintf (file, "Reaching uses:\n"); - for (i = 0; i < df->n_bbs; i++) + FOR_EACH_BB (bb) { - basic_block bb = BASIC_BLOCK (i); struct bb_info *bb_info = DF_BB_INFO (df, bb); if (! bb_info->ru_in) continue; - fprintf (file, "bb %d in \t", i); + fprintf (file, "bb %d in \t", bb->index); dump_bitmap (file, bb_info->ru_in); - fprintf (file, "bb %d gen \t", i); + fprintf (file, "bb %d gen \t", bb->index); dump_bitmap (file, bb_info->ru_gen); - fprintf (file, "bb %d kill\t", i); + fprintf (file, "bb %d kill\t", bb->index); dump_bitmap (file, bb_info->ru_kill); - fprintf (file, "bb %d out \t", i); + fprintf (file, "bb %d out \t", bb->index); dump_bitmap (file, bb_info->ru_out); } } @@ -3371,21 +3402,20 @@ df_dump (df, flags, file) if (flags & DF_LR) { fprintf (file, "Live regs:\n"); - for (i = 0; i < df->n_bbs; i++) + FOR_EACH_BB (bb) { - basic_block bb = BASIC_BLOCK (i); struct bb_info *bb_info = DF_BB_INFO (df, bb); if (! bb_info->lr_in) continue; - fprintf (file, "bb %d in \t", i); + fprintf (file, "bb %d in \t", bb->index); dump_bitmap (file, bb_info->lr_in); - fprintf (file, "bb %d use \t", i); + fprintf (file, "bb %d use \t", bb->index); dump_bitmap (file, bb_info->lr_use); - fprintf (file, "bb %d def \t", i); + fprintf (file, "bb %d def \t", bb->index); dump_bitmap (file, bb_info->lr_def); - fprintf (file, "bb %d out \t", i); + fprintf (file, "bb %d out \t", bb->index); dump_bitmap (file, bb_info->lr_out); } } @@ -3401,42 +3431,42 @@ df_dump (df, flags, file) && (reg_info[j].n_uses || reg_info[j].n_defs)) || ((flags & DF_RD_CHAIN) && reg_info[j].defs) || ((flags & DF_RU_CHAIN) && reg_info[j].uses)) - { - fprintf (file, "reg %d", j); - if ((flags & DF_RD_CHAIN) && (flags & DF_RU_CHAIN)) - { - basic_block bb = df_regno_bb (df, j); + { + fprintf (file, "reg %d", j); + if ((flags & DF_RD_CHAIN) && (flags & DF_RU_CHAIN)) + { + basic_block bb = df_regno_bb (df, j); - if (bb) - fprintf (file, " bb %d", bb->index); - else - fprintf (file, " bb ?"); - } - if (flags & DF_REG_INFO) - { - fprintf (file, " life %d", reg_info[j].lifetime); - } + if (bb) + fprintf (file, " bb %d", bb->index); + else + fprintf (file, " bb ?"); + } + if (flags & DF_REG_INFO) + { + fprintf (file, " life %d", reg_info[j].lifetime); + } - if ((flags & DF_REG_INFO) || (flags & DF_RD_CHAIN)) - { - fprintf (file, " defs "); - if (flags & DF_REG_INFO) - fprintf (file, "%d ", reg_info[j].n_defs); - if (flags & DF_RD_CHAIN) - df_chain_dump (reg_info[j].defs, file); - } + if ((flags & DF_REG_INFO) || (flags & DF_RD_CHAIN)) + { + fprintf (file, " defs "); + if (flags & DF_REG_INFO) + fprintf (file, "%d ", reg_info[j].n_defs); + if (flags & DF_RD_CHAIN) + df_chain_dump (reg_info[j].defs, file); + } - if ((flags & DF_REG_INFO) || (flags & DF_RU_CHAIN)) - { - fprintf (file, " uses "); - if (flags & DF_REG_INFO) - fprintf (file, "%d ", reg_info[j].n_uses); - if (flags & DF_RU_CHAIN) - df_chain_dump (reg_info[j].uses, file); - } + if ((flags & DF_REG_INFO) || (flags & DF_RU_CHAIN)) + { + fprintf (file, " uses "); + if (flags & DF_REG_INFO) + fprintf (file, "%d ", reg_info[j].n_uses); + if (flags & DF_RU_CHAIN) + df_chain_dump (reg_info[j].uses, file); + } - fprintf (file, "\n"); - } + fprintf (file, "\n"); + } } } fprintf (file, "\n"); @@ -3458,7 +3488,7 @@ df_insn_debug (df, insn, file) if (df->insns[uid].defs) bbi = DF_REF_BBNO (df->insns[uid].defs->ref); - else if (df->insns[uid].uses) + else if (df->insns[uid].uses) bbi = DF_REF_BBNO (df->insns[uid].uses->ref); else bbi = -1; @@ -3486,13 +3516,13 @@ df_insn_debug_regno (df, insn, file) if (df->insns[uid].defs) bbi = DF_REF_BBNO (df->insns[uid].defs->ref); - else if (df->insns[uid].uses) + else if (df->insns[uid].uses) bbi = DF_REF_BBNO (df->insns[uid].uses->ref); else bbi = -1; fprintf (file, "insn %d bb %d luid %d defs ", - uid, bbi, DF_INSN_LUID (df, insn)); + uid, bbi, DF_INSN_LUID (df, insn)); df_chain_dump_regno (df->insns[uid].defs, file); fprintf (file, " uses "); df_chain_dump_regno (df->insns[uid].uses, file); @@ -3595,9 +3625,9 @@ debug_df_chain (link) /* Hybrid search algorithm from "Implementation Techniques for Efficient Data-Flow Analysis of Large Programs". */ -static void -hybrid_search_bitmap (block, in, out, gen, kill, dir, - conf_op, transfun, visited, pending, +static void +hybrid_search_bitmap (block, in, out, gen, kill, dir, + conf_op, transfun, visited, pending, data) basic_block block; bitmap *in, *out, *gen, *kill; @@ -3611,7 +3641,7 @@ hybrid_search_bitmap (block, in, out, gen, kill, dir, int changed; int i = block->index; edge e; - basic_block bb= block; + basic_block bb = block; SET_BIT (visited, block->index); if (TEST_BIT (pending, block->index)) { @@ -3634,16 +3664,16 @@ hybrid_search_bitmap (block, in, out, gen, kill, dir, } } } - else + else { /* Calculate <conf_op> of successor ins */ - bitmap_zero(out[i]); + 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; @@ -3652,7 +3682,7 @@ hybrid_search_bitmap (block, in, out, gen, kill, dir, break; } } - } + } /* Common part */ (*transfun)(i, &changed, in[i], out[i], gen[i], kill[i], data); RESET_BIT (pending, i); @@ -3685,8 +3715,8 @@ hybrid_search_bitmap (block, in, out, gen, kill, dir, if (e->dest == EXIT_BLOCK_PTR || e->dest->index == i) continue; if (!TEST_BIT (visited, e->dest->index)) - hybrid_search_bitmap (e->dest, in, out, gen, kill, dir, - conf_op, transfun, visited, pending, + hybrid_search_bitmap (e->dest, in, out, gen, kill, dir, + conf_op, transfun, visited, pending, data); } } @@ -3697,8 +3727,8 @@ hybrid_search_bitmap (block, in, out, gen, kill, dir, if (e->src == ENTRY_BLOCK_PTR || e->src->index == i) continue; if (!TEST_BIT (visited, e->src->index)) - hybrid_search_bitmap (e->src, in, out, gen, kill, dir, - conf_op, transfun, visited, pending, + hybrid_search_bitmap (e->src, in, out, gen, kill, dir, + conf_op, transfun, visited, pending, data); } } @@ -3706,8 +3736,8 @@ hybrid_search_bitmap (block, in, out, gen, kill, dir, /* Hybrid search for sbitmaps, rather than bitmaps. */ -static void -hybrid_search_sbitmap (block, in, out, gen, kill, dir, +static void +hybrid_search_sbitmap (block, in, out, gen, kill, dir, conf_op, transfun, visited, pending, data) basic_block block; @@ -3722,7 +3752,7 @@ hybrid_search_sbitmap (block, in, out, gen, kill, dir, int changed; int i = block->index; edge e; - basic_block bb= block; + basic_block bb = block; SET_BIT (visited, block->index); if (TEST_BIT (pending, block->index)) { @@ -3745,16 +3775,16 @@ hybrid_search_sbitmap (block, in, out, gen, kill, dir, } } } - else + else { /* Calculate <conf_op> of successor ins */ - sbitmap_zero(out[i]); + 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; @@ -3763,7 +3793,7 @@ hybrid_search_sbitmap (block, in, out, gen, kill, dir, break; } } - } + } /* Common part */ (*transfun)(i, &changed, in[i], out[i], gen[i], kill[i], data); RESET_BIT (pending, i); @@ -3796,8 +3826,8 @@ hybrid_search_sbitmap (block, in, out, gen, kill, dir, if (e->dest == EXIT_BLOCK_PTR || e->dest->index == i) continue; if (!TEST_BIT (visited, e->dest->index)) - hybrid_search_sbitmap (e->dest, in, out, gen, kill, dir, - conf_op, transfun, visited, pending, + hybrid_search_sbitmap (e->dest, in, out, gen, kill, dir, + conf_op, transfun, visited, pending, data); } } @@ -3808,8 +3838,8 @@ hybrid_search_sbitmap (block, in, out, gen, kill, dir, if (e->src == ENTRY_BLOCK_PTR || e->src->index == i) continue; if (!TEST_BIT (visited, e->src->index)) - hybrid_search_sbitmap (e->src, in, out, gen, kill, dir, - conf_op, transfun, visited, pending, + hybrid_search_sbitmap (e->src, in, out, gen, kill, dir, + conf_op, transfun, visited, pending, data); } } @@ -3827,20 +3857,20 @@ hybrid_search_sbitmap (block, in, out, gen, kill, dir, 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) +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; @@ -3853,14 +3883,14 @@ iterative_dataflow_sbitmap (in, out, gen, kill, blocks, fibheap_t worklist; basic_block bb; sbitmap visited, pending; - pending = sbitmap_alloc (n_basic_blocks); - visited = sbitmap_alloc (n_basic_blocks); + pending = sbitmap_alloc (last_basic_block); + visited = sbitmap_alloc (last_basic_block); sbitmap_zero (pending); sbitmap_zero (visited); worklist = fibheap_new (); EXECUTE_IF_SET_IN_BITMAP (blocks, 0, i, { - fibheap_insert (worklist, order[i], (void *) (size_t) i); + fibheap_insert (worklist, order[i], (void *) (size_t) i); SET_BIT (pending, i); if (dir == FORWARD) sbitmap_copy (out[i], gen[i]); @@ -3874,7 +3904,7 @@ iterative_dataflow_sbitmap (in, out, gen, kill, blocks, i = (size_t) fibheap_extract_min (worklist); bb = BASIC_BLOCK (i); if (!TEST_BIT (visited, bb->index)) - hybrid_search_sbitmap (bb, in, out, gen, kill, dir, + hybrid_search_sbitmap (bb, in, out, gen, kill, dir, conf_op, transfun, visited, pending, data); } if (sbitmap_first_set_bit (pending) != -1) @@ -3888,7 +3918,7 @@ iterative_dataflow_sbitmap (in, out, gen, kill, blocks, else { break; - } + } } sbitmap_free (pending); sbitmap_free (visited); @@ -3898,8 +3928,8 @@ iterative_dataflow_sbitmap (in, out, gen, kill, blocks, /* 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) +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; @@ -3912,8 +3942,8 @@ iterative_dataflow_bitmap (in, out, gen, kill, blocks, fibheap_t worklist; basic_block bb; sbitmap visited, pending; - pending = sbitmap_alloc (n_basic_blocks); - visited = sbitmap_alloc (n_basic_blocks); + pending = sbitmap_alloc (last_basic_block); + visited = sbitmap_alloc (last_basic_block); sbitmap_zero (pending); sbitmap_zero (visited); worklist = fibheap_new (); @@ -3933,7 +3963,7 @@ iterative_dataflow_bitmap (in, out, gen, kill, blocks, i = (size_t) fibheap_extract_min (worklist); bb = BASIC_BLOCK (i); if (!TEST_BIT (visited, bb->index)) - hybrid_search_bitmap (bb, in, out, gen, kill, dir, + hybrid_search_bitmap (bb, in, out, gen, kill, dir, conf_op, transfun, visited, pending, data); } if (sbitmap_first_set_bit (pending) != -1) @@ -3947,9 +3977,9 @@ iterative_dataflow_bitmap (in, out, gen, kill, blocks, else { break; - } + } } sbitmap_free (pending); sbitmap_free (visited); - fibheap_delete (worklist); + fibheap_delete (worklist); } |