a32b97a20d
* doc/tree-ssa.texi: Remove references to VDEF and add descriptions of V_MAY_DEF and V_MUST_DEF. * tree-dfa.c (dfa_stats_d): Add num_v_must_defs and rename num_vdefs to num_v_may_defs. (compute_immediate_uses_for_stmt): Rename occurences of vdef to v_may_def. (redirect_immediate_uses): Ditto. (dump_dfa_stats): Ditto. Also added code to dump num_v_must_defs. (collect_dfa_stats_r): Rename occurences of vdef to v_may_def. Also add code to sum up the number of v_must_defs. (vdefs_disappeared_p): Replace with... (v_may_defs_disappeared_p): This. (v_must_defs_disappeared_p): New function. (mark_new_vars_to_rename): Rename occurences of vdef to v_may_def. Also add code to mark new variables found in V_MUST_DEFs for renameing. * tree-flow.h (stmt_ann_d): Add v_must_def_ops and replace vdef_ops to v_may_def_ops. (get_vdef_ops): Replace with... (get_v_may_def_ops): This. * tree-flow-inline.h (get_vdef_ops): Replace with... (get_v_may_def_ops): This. (get_v_must_def_ops): New function. (get_vdef_result_ptr): Replace with... (get_v_may_def_result_ptr): This. (get_vdef_op_ptr): Ditto with... (get_v_may_def_op_ptr); This. (get_v_must_def_op_ptr): New function. * tree-into-ssa.c (mark_def_sites): Rename occurences of vdef to v_may_def. Also add code to mark statements with V_MUST_DEFs as definition sites. (rewrite_stmt): Rename occurences of vdef to v_may_def. Also add code to register new V_MUST_DEFs made by the statement. * tree-outof-ssa.c (VIRTUAL_PARTITION): Update comments. (check_replaceable): Rename occurences of vdef to v_may_def. Also add check for V_MUST_DEFs. (find_replaceable_in_bb): Ditto. * tree-pretty-print.c (dump_vops): Rename occurences of vdef to v_may_def. Also add code to dump V_MUST_DEFs. * tree-sra.c (mark_all_vdefs): Replace with... (mark_all_v_may_defs): This. (mark_all_v_must_defs): New function. (create_scalar_copies): Replace call to mark_all_vdefs with calls to mark_all_v_may_defs and mark_all_v_must_defs. (scalarize_structures): Rename occurences of vdef to v_may_def. Also add a check for V_MUST_DEFs. (scalarize_modify_expr): Rename occurences of vdef to v_may_def. * tree-ssa-alias.c (global_var): Update comment. (compute_may_aliases): Ditto. (compute_points_to_and_addr_escape): Rename occurences of vdef to v_may_def. Also add code to mark variables in V_MUST_DEF operands as being written to. (group_aliases): Update comment. (maybe_create_global_var): Ditto. * tree-ssa.c (verify_ssa): Rename occurences of vdef to v_may_def. Also add a check for V_MUST_DEFs on GIMPLE registers. (replace_immediate_uses): Rename occurences of vdef to v_may_def. * tree-ssa-ccp.c (visit_stmt): Rename occurences of vdef to v_may_def. Also add code to mark all V_MUST_DEF operands VARYING. (initialize): Ditto. (set_rhs): Rename occurences of vdef to v_may_def. Also add code to update SSA_NAMEs in V_MUST_DEFs. * tree-ssa-copy.c (cprop_into_stmt): Rename occurences of vdef to v_may_def. * tree-ssa-dce.c (mark_stmt_if_obviously_necessary): Rename occurences of vdef to v_may_def. Also add code to mark statements with V_MUST_DEFs as necessary. (propagate_necessity): Rename occurences of vdef to v_may_def. * tree-ssa-dom.c (redirect_edges_and_update_ssa_graph): Rename occurences of vdef to v_may_def. Also add code to mark operands in V_MUST_DEFs for renaming. (eliminate_redundant_computations): Rename occurences of vdef to v_may_def. (record_equivalences_from_stmt): Rename occurences of vdef to v_may_def. Also add code to record VUSEs for V_MUST_DEFs. (optimize_stmt): Remove unnesessary variable vdefs. Update comment. (register_definitions_for_stmt): Rename occurences of vdef to v_may_def. Also add code to register definitions made with V_MUST_DEFs. * tree-ssa-dse.c (fix_stmt_vdefs): Replace with... (fix_stmt_v_may_defs): This. (fix_phi_uses): Rename occurences of vdef to v_may_def. (dse_optimize_stmt): Ditto. * tree-ssa-live.c (create_ssa_var_map): Rename occurences of vdef to v_may_def. Also add code to mark V_MUST_DEF operands as being used in virtual operators. * tree-ssa-loop.c (mark_defs_for_rewrite): Rename occurences of vdef to v_may_def. Also add code to mark V_MUST_DEF operands for renaming. * tree-ssa-operands.c (opf_kill_def): New flag for killing definitions. (build_vdefs): Renamed to... (build_v_may_defs): This. (build_v_must_defs): New variable. (voperands_d): Add v_must_def_ops and replace vdef_ops with v_may_def_ops. (append_vdef): Replace with... (append_v_may_def): This. (append_v_must_def): New function. (NUM_FREE): Increment for V_MUST_DEF (optype_freelist): Increment its size for V_MUST_DEF (allocate_vdef_optype): Replace with... (allocate_v_may_def_optype): This. (allocate_v_must_def_optype): New function. (free_vdefs): Replace with... (free_v_may_defs): This. (free_v_must_defs): New function. (remove_vdefs): Replace with... (remove_v_may_defs): This. (remove_v_must_defs): New function. (init_ssa_operands): Rename occurences of vdef to v_may_def. Also add code to initialize build_v_must_defs. (finalize_ssa_vdefs): Replace with... (finalize_ssa_v_may_defs): This. (finalize_ssa_vuses): Rename occurences of vdef to v_may_def. (finalize_ssa_v_must_defs): New function. (finalize_ssa_stmt_operands): Replace call to finalize_ssa_vdefs with calls to finalize_ssa_v_may_defs and finalize_ssa_v_must_defs. (verify_start_operands): Rename occurences of vdef to v_may_def. Also add check for build_v_must_defs. (get_stmt_operands): Rename occurences of vdef to v_may_def. Also add code to handle V_MUST_DEFs and to use opf_kill_def for killing definitions. (get_expr_operands): Update comment and use opf_kill_def for killing definitions. (add_stmt_operand): Replace code that appends VDEFs with code that appends V_MUST_DEFs when opf_kill_def is set and V_MAY_DEFs otherwise. (add_call_clobber_ops): Update comments. * tree-ssa-operands.h (vdef_optype_d): Replace with... (v_may_def_optype_d): This. (v_must_def_optype_d): New structure. (VDEF_OPS): Replace with... (V_MAY_DEF_OPS): This. (STMT_VDEF_OPS): Same with... (STMT_V_MAY_DEF_OPS): This. (NUM_VDEFS): And... (NUM_V_MAY_DEFS): This. (VDEF_RESULT_PTR): As well as... (V_MAY_DEF_RESULT_PTR): This. (VDEF_RESULT): Same goes for... (V_MAY_DEF_RESULT): This. (VDEF_OP_PTR): And... (V_MAY_DEF_OP_PTR): This. (VDEF_OP): And... (V_MAY_DEF_OP): This. (V_MUST_DEF_OPS): New macro. (STMT_V_MUST_DEF_OPS): Ditto. (NUM_V_MUST_DEFS): Ditto. (V_MUST_DEF_OP_PTR): Ditto. (V_MUST_DEF_OP): Ditto. (remove_vdefs): Replace signature with... (remove_v_may_defs): This. (remove_v_must_defs): New function signature. * tree-ssa-pre.c (subst_phis): Replace call to remove_vdefs with calls to remove_v_may_defs and remove_v_must_defs. (process_left_occs_and_kills): Rename occurences of vdef to v_may_def. Also add code that marks left occurences of operands in V_MUST_DEFs. * tree-tailcall.c (find_tail_calls): Rename occurences of vdef to v_may_def. Also add check for V_MUST_DEFs. (eliminate_tail_call):Rename occurences of vdef to v_may_def. testsuite: * gcc.dg/tree-ssa/20031015-1.c: Scan for V_MAY_DEF instead of VDEF. * gcc.dg/tree-ssa/20040517-1.c: Ditto. From-SVN: r82947
447 lines
14 KiB
C
447 lines
14 KiB
C
/* Dead store elimination
|
|
Copyright (C) 2004 Free Software Foundation, Inc.
|
|
|
|
This file is part of GCC.
|
|
|
|
GCC is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2, or (at your option)
|
|
any later version.
|
|
|
|
GCC is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
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. */
|
|
|
|
#include "config.h"
|
|
#include "system.h"
|
|
#include "coretypes.h"
|
|
#include "tm.h"
|
|
#include "errors.h"
|
|
#include "ggc.h"
|
|
#include "tree.h"
|
|
#include "rtl.h"
|
|
#include "tm_p.h"
|
|
#include "basic-block.h"
|
|
#include "timevar.h"
|
|
#include "diagnostic.h"
|
|
#include "tree-flow.h"
|
|
#include "tree-pass.h"
|
|
#include "tree-dump.h"
|
|
#include "domwalk.h"
|
|
#include "flags.h"
|
|
|
|
/* This file implements dead store elimination.
|
|
|
|
A dead store is a store into a memory location which will later be
|
|
overwritten by another store without any intervening loads. In this
|
|
case the earlier store can be deleted.
|
|
|
|
In our SSA + virtual operand world we use immediate uses of virtual
|
|
operands to detect dead stores. If a store's virtual definition
|
|
is used precisely once by a later store to the same location which
|
|
post dominates the first store, then the first store is dead.
|
|
|
|
The single use of the store's virtual definition ensures that
|
|
there are no intervening aliased loads and the requirement that
|
|
the second load post dominate the first ensures that if the earlier
|
|
store executes, then the later stores will execute before the function
|
|
exits.
|
|
|
|
It may help to think of this as first moving the earlier store to
|
|
the point immediately before the later store. Again, the single
|
|
use of the virtual defintion and the post-dominance relationship
|
|
ensure that such movement would be safe. Clearly if there are
|
|
back to back stores, then the second is redundant.
|
|
|
|
Reviewing section 10.7.2 in Morgan's "Building an Optimizing Compiler"
|
|
may also help in understanding this code since it discusses the
|
|
relationship between dead store and redundant load elimination. In
|
|
fact, they are the same transformation applied to different views of
|
|
the CFG. */
|
|
|
|
|
|
struct dse_global_data
|
|
{
|
|
/* This is the global bitmap for store statements.
|
|
|
|
Each statement has a unique ID. When we encounter a store statement
|
|
that we want to record, set the bit corresponding to the statement's
|
|
unique ID in this bitmap. */
|
|
bitmap stores;
|
|
};
|
|
|
|
/* We allocate a bitmap-per-block for stores which are encountered
|
|
during the scan of that block. This allows us to restore the
|
|
global bitmap of stores when we finish processing a block. */
|
|
struct dse_block_local_data
|
|
{
|
|
bitmap stores;
|
|
};
|
|
|
|
static bool gate_dse (void);
|
|
static void tree_ssa_dse (void);
|
|
static void dse_initialize_block_local_data (struct dom_walk_data *,
|
|
basic_block,
|
|
bool);
|
|
static void dse_optimize_stmt (struct dom_walk_data *,
|
|
basic_block,
|
|
block_stmt_iterator);
|
|
static void dse_record_phis (struct dom_walk_data *, basic_block);
|
|
static void dse_finalize_block (struct dom_walk_data *, basic_block);
|
|
static void fix_phi_uses (tree, tree);
|
|
static void fix_stmt_v_may_defs (tree, tree);
|
|
static void record_voperand_set (bitmap, bitmap *, unsigned int);
|
|
|
|
/* Function indicating whether we ought to include information for 'var'
|
|
when calculating immediate uses. For this pass we only want use
|
|
information for virtual variables. */
|
|
|
|
static bool
|
|
need_imm_uses_for (tree var)
|
|
{
|
|
return !is_gimple_reg (var);
|
|
}
|
|
|
|
|
|
/* Replace uses in PHI which match V_MAY_DEF_RESULTs in STMT with the
|
|
corresponding V_MAY_DEF_OP in STMT. */
|
|
|
|
static void
|
|
fix_phi_uses (tree phi, tree stmt)
|
|
{
|
|
stmt_ann_t ann = stmt_ann (stmt);
|
|
v_may_def_optype v_may_defs;
|
|
unsigned int i;
|
|
int j;
|
|
|
|
get_stmt_operands (stmt);
|
|
v_may_defs = V_MAY_DEF_OPS (ann);
|
|
|
|
/* Walk each V_MAY_DEF in STMT. */
|
|
for (i = 0; i < NUM_V_MAY_DEFS (v_may_defs); i++)
|
|
{
|
|
tree v_may_def = V_MAY_DEF_RESULT (v_may_defs, i);
|
|
|
|
/* Find any uses in the PHI which match V_MAY_DEF and replace
|
|
them with the appropriate V_MAY_DEF_OP. */
|
|
for (j = 0; j < PHI_NUM_ARGS (phi); j++)
|
|
if (v_may_def == PHI_ARG_DEF (phi, j))
|
|
PHI_ARG_DEF (phi, j) = V_MAY_DEF_OP (v_may_defs, i);
|
|
}
|
|
}
|
|
|
|
/* Replace the V_MAY_DEF_OPs in STMT1 which match V_MAY_DEF_RESULTs
|
|
in STMT2 with the appropriate V_MAY_DEF_OPs from STMT2. */
|
|
|
|
static void
|
|
fix_stmt_v_may_defs (tree stmt1, tree stmt2)
|
|
{
|
|
stmt_ann_t ann1 = stmt_ann (stmt1);
|
|
stmt_ann_t ann2 = stmt_ann (stmt2);
|
|
v_may_def_optype v_may_defs1;
|
|
v_may_def_optype v_may_defs2;
|
|
unsigned int i, j;
|
|
|
|
get_stmt_operands (stmt1);
|
|
get_stmt_operands (stmt2);
|
|
v_may_defs1 = V_MAY_DEF_OPS (ann1);
|
|
v_may_defs2 = V_MAY_DEF_OPS (ann2);
|
|
|
|
/* Walk each V_MAY_DEF_OP in stmt1. */
|
|
for (i = 0; i < NUM_V_MAY_DEFS (v_may_defs1); i++)
|
|
{
|
|
tree v_may_def1 = V_MAY_DEF_OP (v_may_defs1, i);
|
|
|
|
/* Find the appropriate V_MAY_DEF_RESULT in STMT2. */
|
|
for (j = 0; j < NUM_V_MAY_DEFS (v_may_defs2); j++)
|
|
{
|
|
if (v_may_def1 == V_MAY_DEF_RESULT (v_may_defs2, j))
|
|
{
|
|
/* Update. */
|
|
*V_MAY_DEF_OP_PTR (v_may_defs1, i) =
|
|
V_MAY_DEF_OP (v_may_defs2, j);
|
|
break;
|
|
}
|
|
}
|
|
|
|
#ifdef ENABLE_CHECKING
|
|
/* If we did not find a corresponding V_MAY_DEF_RESULT, then something
|
|
has gone terribly wrong. */
|
|
if (j == NUM_V_MAY_DEFS (v_may_defs2))
|
|
abort ();
|
|
#endif
|
|
|
|
}
|
|
}
|
|
|
|
|
|
/* Set bit UID in bitmaps GLOBAL and *LOCAL, creating *LOCAL as needed. */
|
|
static void
|
|
record_voperand_set (bitmap global, bitmap *local, unsigned int uid)
|
|
{
|
|
/* Lazily allocate the bitmap. Note that we do not get a notification
|
|
when the block local data structures die, so we allocate the local
|
|
bitmap backed by the GC system. */
|
|
if (*local == NULL)
|
|
*local = BITMAP_GGC_ALLOC ();
|
|
|
|
/* Set the bit in the local and global bitmaps. */
|
|
bitmap_set_bit (*local, uid);
|
|
bitmap_set_bit (global, uid);
|
|
}
|
|
/* Initialize block local data structures. */
|
|
|
|
static void
|
|
dse_initialize_block_local_data (struct dom_walk_data *walk_data,
|
|
basic_block bb ATTRIBUTE_UNUSED,
|
|
bool recycled)
|
|
{
|
|
struct dse_block_local_data *bd
|
|
= VARRAY_TOP_GENERIC_PTR (walk_data->block_data_stack);
|
|
|
|
/* If we are given a recycled block local data structure, ensure any
|
|
bitmap associated with the block is cleared. */
|
|
if (recycled)
|
|
{
|
|
if (bd->stores)
|
|
bitmap_clear (bd->stores);
|
|
}
|
|
}
|
|
|
|
/* Attempt to eliminate dead stores in the statement referenced by BSI.
|
|
|
|
A dead store is a store into a memory location which will later be
|
|
overwritten by another store without any intervening loads. In this
|
|
case the earlier store can be deleted.
|
|
|
|
In our SSA + virtual operand world we use immediate uses of virtual
|
|
operands to detect dead stores. If a store's virtual definition
|
|
is used precisely once by a later store to the same location which
|
|
post dominates the first store, then the first store is dead. */
|
|
|
|
static void
|
|
dse_optimize_stmt (struct dom_walk_data *walk_data,
|
|
basic_block bb ATTRIBUTE_UNUSED,
|
|
block_stmt_iterator bsi)
|
|
{
|
|
struct dse_block_local_data *bd
|
|
= VARRAY_TOP_GENERIC_PTR (walk_data->block_data_stack);
|
|
struct dse_global_data *dse_gd = walk_data->global_data;
|
|
tree stmt = bsi_stmt (bsi);
|
|
stmt_ann_t ann = stmt_ann (stmt);
|
|
v_may_def_optype v_may_defs;
|
|
|
|
get_stmt_operands (stmt);
|
|
v_may_defs = V_MAY_DEF_OPS (ann);
|
|
|
|
/* If this statement has no virtual uses, then there is nothing
|
|
to do. */
|
|
if (NUM_V_MAY_DEFS (v_may_defs) == 0)
|
|
return;
|
|
|
|
/* We know we have virtual definitions. If this is a MODIFY_EXPR, then
|
|
record it into our table. */
|
|
if (TREE_CODE (stmt) == MODIFY_EXPR
|
|
&& TREE_CODE (TREE_OPERAND (stmt, 1)) != CALL_EXPR)
|
|
{
|
|
dataflow_t df = get_immediate_uses (stmt);
|
|
unsigned int num_uses = num_immediate_uses (df);
|
|
tree use;
|
|
tree skipped_phi;
|
|
|
|
|
|
/* If there are no uses then there is nothing left to do. */
|
|
if (num_uses == 0)
|
|
{
|
|
record_voperand_set (dse_gd->stores, &bd->stores, ann->uid);
|
|
return;
|
|
}
|
|
|
|
use = immediate_use (df, 0);
|
|
skipped_phi = NULL;
|
|
|
|
/* Skip through any PHI nodes we have already seen if the PHI
|
|
represents the only use of this store.
|
|
|
|
Note this does not handle the case where the store has
|
|
multiple V_MAY_DEFs which all reach a set of PHI nodes in the
|
|
same block. */
|
|
while (num_uses == 1
|
|
&& TREE_CODE (use) == PHI_NODE
|
|
&& bitmap_bit_p (dse_gd->stores, stmt_ann (use)->uid))
|
|
{
|
|
/* Record the first PHI we skip so that we can fix its
|
|
uses if we find that STMT is a dead store. */
|
|
if (!skipped_phi)
|
|
skipped_phi = use;
|
|
|
|
/* Skip past this PHI and loop again in case we had a PHI
|
|
chain. */
|
|
df = get_immediate_uses (use);
|
|
num_uses = num_immediate_uses (df);
|
|
use = immediate_use (df, 0);
|
|
}
|
|
|
|
/* If we have precisely one immediate use at this point, then we may
|
|
have found redundant store. */
|
|
if (num_uses == 1
|
|
&& bitmap_bit_p (dse_gd->stores, stmt_ann (use)->uid)
|
|
&& operand_equal_p (TREE_OPERAND (stmt, 0),
|
|
TREE_OPERAND (use, 0), 0))
|
|
{
|
|
/* We need to fix the operands if either the first PHI we
|
|
skipped, or the store which we are not deleting if we did
|
|
not skip any PHIs. */
|
|
if (skipped_phi)
|
|
fix_phi_uses (skipped_phi, stmt);
|
|
else
|
|
fix_stmt_v_may_defs (use, stmt);
|
|
|
|
if (dump_file && (dump_flags & TDF_DETAILS))
|
|
{
|
|
fprintf (dump_file, " Deleted dead store '");
|
|
print_generic_expr (dump_file, bsi_stmt (bsi), dump_flags);
|
|
fprintf (dump_file, "'\n");
|
|
}
|
|
|
|
/* Any immediate uses which reference STMT need to instead
|
|
reference the new consumer, either SKIPPED_PHI or USE.
|
|
This allows us to cascade dead stores. */
|
|
redirect_immediate_uses (stmt, skipped_phi ? skipped_phi : use);
|
|
|
|
/* Finally remove the dead store. */
|
|
bsi_remove (&bsi);
|
|
}
|
|
|
|
record_voperand_set (dse_gd->stores, &bd->stores, ann->uid);
|
|
}
|
|
}
|
|
|
|
/* Record that we have seen the PHIs at the start of BB which correspond
|
|
to virtual operands. */
|
|
static void
|
|
dse_record_phis (struct dom_walk_data *walk_data, basic_block bb)
|
|
{
|
|
struct dse_block_local_data *bd
|
|
= VARRAY_TOP_GENERIC_PTR (walk_data->block_data_stack);
|
|
struct dse_global_data *dse_gd = walk_data->global_data;
|
|
tree phi;
|
|
|
|
for (phi = phi_nodes (bb); phi; phi = TREE_CHAIN (phi))
|
|
if (need_imm_uses_for (PHI_RESULT (phi)))
|
|
record_voperand_set (dse_gd->stores,
|
|
&bd->stores,
|
|
get_stmt_ann (phi)->uid);
|
|
}
|
|
|
|
static void
|
|
dse_finalize_block (struct dom_walk_data *walk_data,
|
|
basic_block bb ATTRIBUTE_UNUSED)
|
|
{
|
|
struct dse_block_local_data *bd
|
|
= VARRAY_TOP_GENERIC_PTR (walk_data->block_data_stack);
|
|
struct dse_global_data *dse_gd = walk_data->global_data;
|
|
bitmap stores = dse_gd->stores;
|
|
unsigned int i;
|
|
|
|
/* Unwind the stores noted in this basic block. */
|
|
if (bd->stores)
|
|
EXECUTE_IF_SET_IN_BITMAP (bd->stores, 0, i, bitmap_clear_bit (stores, i););
|
|
}
|
|
|
|
static void
|
|
tree_ssa_dse (void)
|
|
{
|
|
struct dom_walk_data walk_data;
|
|
struct dse_global_data dse_gd;
|
|
unsigned int uid = 0;
|
|
basic_block bb;
|
|
|
|
/* Create a UID for each statement in the function. Ordering of the
|
|
UIDs is not important for this pass. */
|
|
FOR_EACH_BB (bb)
|
|
{
|
|
block_stmt_iterator bsi;
|
|
tree phi;
|
|
|
|
for (bsi = bsi_start (bb); !bsi_end_p (bsi); bsi_next (&bsi))
|
|
stmt_ann (bsi_stmt (bsi))->uid = uid++;
|
|
|
|
for (phi = phi_nodes (bb); phi; phi = TREE_CHAIN (phi))
|
|
stmt_ann (phi)->uid = uid++;
|
|
}
|
|
|
|
/* We might consider making this a property of each pass so that it
|
|
can be [re]computed on an as-needed basis. Particularly since
|
|
this pass could be seen as an extension of DCE which needs post
|
|
dominators. */
|
|
calculate_dominance_info (CDI_POST_DOMINATORS);
|
|
|
|
/* We also need immediate use information for virtual operands. */
|
|
compute_immediate_uses (TDFA_USE_VOPS, need_imm_uses_for);
|
|
|
|
/* Dead store elimination is fundamentally a walk of the post-dominator
|
|
tree and a backwards walk of statements within each block. */
|
|
walk_data.walk_stmts_backward = true;
|
|
walk_data.dom_direction = CDI_POST_DOMINATORS;
|
|
walk_data.initialize_block_local_data = dse_initialize_block_local_data;
|
|
walk_data.before_dom_children_before_stmts = NULL;
|
|
walk_data.before_dom_children_walk_stmts = dse_optimize_stmt;
|
|
walk_data.before_dom_children_after_stmts = dse_record_phis;
|
|
walk_data.after_dom_children_before_stmts = NULL;
|
|
walk_data.after_dom_children_walk_stmts = NULL;
|
|
walk_data.after_dom_children_after_stmts = dse_finalize_block;
|
|
|
|
walk_data.block_local_data_size = sizeof (struct dse_block_local_data);
|
|
|
|
/* This is the main hash table for the dead store elimination pass. */
|
|
dse_gd.stores = BITMAP_XMALLOC ();
|
|
walk_data.global_data = &dse_gd;
|
|
|
|
/* Initialize the dominator walker. */
|
|
init_walk_dominator_tree (&walk_data);
|
|
|
|
/* Recursively walk the dominator tree. */
|
|
walk_dominator_tree (&walk_data, EXIT_BLOCK_PTR);
|
|
|
|
/* Finalize the dominator walker. */
|
|
fini_walk_dominator_tree (&walk_data);
|
|
|
|
/* Release the main bitmap. */
|
|
BITMAP_XFREE (dse_gd.stores);
|
|
|
|
/* Free dataflow information. It's probably out of date now anyway. */
|
|
free_df ();
|
|
|
|
/* For now, just wipe the post-dominator information. */
|
|
free_dominance_info (CDI_POST_DOMINATORS);
|
|
}
|
|
|
|
static bool
|
|
gate_dse (void)
|
|
{
|
|
return flag_tree_dse != 0;
|
|
}
|
|
|
|
struct tree_opt_pass pass_dse = {
|
|
"dse", /* name */
|
|
gate_dse, /* gate */
|
|
tree_ssa_dse, /* execute */
|
|
NULL, /* sub */
|
|
NULL, /* next */
|
|
0, /* static_pass_number */
|
|
TV_TREE_DSE, /* tv_id */
|
|
PROP_cfg | PROP_ssa, /* properties_required */
|
|
0, /* properties_provided */
|
|
0, /* properties_destroyed */
|
|
0, /* todo_flags_start */
|
|
TODO_dump_func | TODO_ggc_collect /* todo_flags_finish */
|
|
| TODO_verify_ssa
|
|
};
|