GCC Middle and Back End API Reference
|
#include "config.h"
#include "system.h"
#include "coretypes.h"
#include "backend.h"
#include "target.h"
#include "rtl.h"
#include "tree.h"
#include "gimple.h"
#include "predict.h"
#include "df.h"
#include "memmodel.h"
#include "tm_p.h"
#include "gimple-ssa.h"
#include "expmed.h"
#include "optabs.h"
#include "emit-rtl.h"
#include "recog.h"
#include "alias.h"
#include "stor-layout.h"
#include "cfgrtl.h"
#include "cselib.h"
#include "tree-pass.h"
#include "explow.h"
#include "expr.h"
#include "dbgcnt.h"
#include "rtl-iter.h"
#include "cfgcleanup.h"
#include "calls.h"
Data Structures | |
class | store_info |
class | read_info_type |
struct | insn_info_type |
struct | dse_bb_info_type |
struct | group_info |
struct | deferred_change |
struct | invariant_group_base_hasher |
struct | note_add_store_info |
Macros | |
#define | MAX_OFFSET (64 * 1024) |
Typedefs | |
typedef class read_info_type * | read_info_t |
typedef struct insn_info_type * | insn_info_t |
typedef struct dse_bb_info_type * | bb_info_t |
#define MAX_OFFSET (64 * 1024) |
RTL dead store elimination. Copyright (C) 2005-2024 Free Software Foundation, Inc. Contributed by Richard Sandiford <rsandifor@codesourcery.com> and Kenneth Zadeck <zadeck@naturalbridge.com> 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 3, 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 COPYING3. If not see <http://www.gnu.org/licenses/>.
This file contains three techniques for performing Dead Store Elimination (dse). * The first technique performs dse locally on any base address. It is based on the cselib which is a local value numbering technique. This technique is local to a basic block but deals with a fairly general addresses. * The second technique performs dse globally but is restricted to base addresses that are either constant or are relative to the frame_pointer. * The third technique, (which is only done after register allocation) processes the spill slots. This differs from the second technique because it takes advantage of the fact that spilling is completely free from the effects of aliasing. Logically, dse is a backwards dataflow problem. A store can be deleted if it if cannot be reached in the backward direction by any use of the value being stored. However, the local technique uses a forwards scan of the basic block because cselib requires that the block be processed in that order. The pass is logically broken into 7 steps: 0) Initialization. 1) The local algorithm, as well as scanning the insns for the two global algorithms. 2) Analysis to see if the global algs are necessary. In the case of stores base on a constant address, there must be at least two stores to that address, to make it possible to delete some of the stores. In the case of stores off of the frame or spill related stores, only one store to an address is necessary because those stores die at the end of the function. 3) Set up the global dataflow equations based on processing the info parsed in the first step. 4) Solve the dataflow equations. 5) Delete the insns that the global analysis has indicated are unnecessary. 6) Delete insns that store the same value as preceding store where the earlier store couldn't be eliminated. 7) Cleanup. This step uses cselib and canon_rtx to build the largest expression possible for each address. This pass is a forwards pass through each basic block. From the point of view of the global technique, the first pass could examine a block in either direction. The forwards ordering is to accommodate cselib. We make a simplifying assumption: addresses fall into four broad categories: 1) base has rtx_varies_p == false, offset is constant. 2) base has rtx_varies_p == false, offset variable. 3) base has rtx_varies_p == true, offset constant. 4) base has rtx_varies_p == true, offset variable. The local passes are able to process all 4 kinds of addresses. The global pass only handles 1). The global problem is formulated as follows: A store, S1, to address A, where A is not relative to the stack frame, can be eliminated if all paths from S1 to the end of the function contain another store to A before a read to A. If the address A is relative to the stack frame, a store S2 to A can be eliminated if there are no paths from S2 that reach the end of the function that read A before another store to A. In this case S2 can be deleted if there are paths from S2 to the end of the function that have no reads or writes to A. This second case allows stores to the stack frame to be deleted that would otherwise die when the function returns. This cannot be done if stores_off_frame_dead_at_return is not true. See the doc for that variable for when this variable is false. The global problem is formulated as a backwards set union dataflow problem where the stores are the gens and reads are the kills. Set union problems are rare and require some special handling given our representation of bitmaps. A straightforward implementation requires a lot of bitmaps filled with 1s. These are expensive and cumbersome in our bitmap formulation so care has been taken to avoid large vectors filled with 1s. See the comments in bb_info and in the dataflow confluence functions for details. There are two places for further enhancements to this algorithm: 1) The original dse which was embedded in a pass called flow also did local address forwarding. For example in A <- r100 ... <- A flow would replace the right hand side of the second insn with a reference to r100. Most of the information is available to add this to this pass. It has not done it because it is a lot of work in the case that either r100 is assigned to between the first and second insn and/or the second insn is a load of part of the value stored by the first insn. insn 5 in gcc.c-torture/compile/990203-1.c simple case. insn 15 in gcc.c-torture/execute/20001017-2.c simple case. insn 25 in gcc.c-torture/execute/20001026-1.c simple case. insn 44 in gcc.c-torture/execute/20010910-1.c simple case. 2) The cleaning up of spill code is quite profitable. It currently depends on reading tea leaves and chicken entrails left by reload. This pass depends on reload creating a singleton alias set for each spill slot and telling the next dse pass which of these alias sets are the singletons. Rather than analyze the addresses of the spills, dse's spill processing just does analysis of the loads and stores that use those alias sets. There are three cases where this falls short: a) Reload sometimes creates the slot for one mode of access, and then inserts loads and/or stores for a smaller mode. In this case, the current code just punts on the slot. The proper thing to do is to back out and use one bit vector position for each byte of the entity associated with the slot. This depends on KNOWING that reload always generates the accesses for each of the bytes in some canonical (read that easy to understand several passes after reload happens) way. b) Reload sometimes decides that spill slot it allocated was not large enough for the mode and goes back and allocates more slots with the same mode and alias set. The backout in this case is a little more graceful than (a). In this case the slot is unmarked as being a spill slot and if final address comes out to be based off the frame pointer, the global algorithm handles this slot. c) For any pass that may prespill, there is currently no mechanism to tell the dse pass that the slot being used has the special properties that reload uses. It may be that all that is required is to have those passes make the same calls that reload does, assuming that the alias sets can be manipulated in the same way.
There are limits to the size of constant offsets we model for the global problem. There are certainly test cases, that exceed this limit, however, it is unlikely that there are important programs that really have constant offsets this size.
Referenced by record_store(), and set_usage_bits().
typedef struct dse_bb_info_type* bb_info_t |
typedef struct insn_info_type* insn_info_t |
typedef class read_info_type* read_info_t |
|
static |
Set the BB_INFO so that the last insn is marked as a wild read of non-frame locations.
References free_read_records(), dse_bb_info_type::last_insn, insn_info_type::non_frame_wild_read, and reset_active_stores().
Referenced by scan_insn().
|
static |
Set the BB_INFO so that the last insn is marked as a wild read.
References free_read_records(), dse_bb_info_type::last_insn, reset_active_stores(), and insn_info_type::wild_read.
Referenced by check_mem_read_rtx(), record_store(), and scan_insn().
|
inlinestatic |
Return TRUE if all bytes START through START+WIDTH-1 from S_INFO store are known to be needed.
References bitmap_bit_p, store_info::bmap, store_info::count, gcc_assert, i, poly_int< N, C >::is_constant(), store_info::is_large, store_info::large, lowpart_bitmask(), store_info::positions_needed, store_info::rhs, store_info::small_bitmask, UNLIKELY, and store_info::width.
Referenced by check_mem_read_rtx(), and record_store().
|
inlinestatic |
Return TRUE if any bytes from S_INFO store are needed.
References store_info::bmap, store_info::count, gcc_checking_assert, HOST_WIDE_INT_0U, poly_int< N, C >::is_constant(), store_info::is_large, store_info::large, store_info::positions_needed, store_info::small_bitmask, UNLIKELY, and store_info::width.
Referenced by record_store().
Return whether EXPR can possibly escape the current function scope.
References DECL_EXTERNAL, DECL_P, get_base_address(), local_variable_can_escape(), may_be_aliased(), TREE_STATIC, and VAR_P.
Referenced by set_usage_bits().
|
static |
Take all reasonable action to put the address of MEM into the form that we can do analysis on. The gold standard is to get the address into the form: address + OFFSET where address is something that rtx_varies_p considers a constant. When we can get the address in this form, we can do global analysis on it. Note that for constant bases, address is not actually returned, only the group_id. The address can be obtained from that. If that fails, we try cselib to get a value we can at least use locally. If that fails we return false. The GROUP_ID is set to -1 for cselib bases and the index of the group for non_varying bases. FOR_READ is true if this is a mem read and false if not.
References ADDR_SPACE_GENERIC_P, canon_rtx(), const_or_frame_p(), cselib_expand_value_rtx(), cselib_lookup(), dump_file, dump_flags, get_address_mode(), GET_CODE, get_group_info(), GET_MODE, group_info::id, MEM_ADDR_SPACE, NULL, NULL_RTX, offset, print_dec(), print_inline_rtx(), scratch, strip_offset_and_add(), TDF_DETAILS, and XEXP.
Referenced by check_mem_read_rtx(), and record_store().
Entry point for postreload. If you work on reload_cse, or you need this anywhere else, consider if you can provide register liveness information and add a parameter to this function so that it can be passed down in insn_info.fixed_regs_live.
References emit_inc_dec_insn_before(), find_reg_note(), insn_info_type::fixed_regs_live, for_each_inc_dec(), FOR_EACH_SUBRTX, GET_CODE, GET_RTX_CLASS, insn_info_type::insn, NULL, NULL_RTX, PATTERN(), and RTX_AUTOINC.
Referenced by reload_cse_simplify().
|
static |
Before we delete INSN_INFO->INSN, make sure that the auto inc/dec, if it is there, is split into a separate insn. Return true on success (or if there was nothing to do), false on failure.
References emit_inc_dec_insn_before(), find_reg_note(), for_each_inc_dec(), FOR_EACH_SUBRTX, GET_CODE, GET_RTX_CLASS, insn_info_type::insn, NULL_RTX, PATTERN(), and RTX_AUTOINC.
Referenced by delete_dead_store_insn(), and dse_step5().
Check the address of MEM *LOC and kill any appropriate stores that may be active.
References active_local_stores, active_local_stores_len, add_wild_read(), ALIAS_SET_MEMORY_BARRIER, all_positions_needed_p(), insn_info_type::cannot_delete, canon_address(), group_info::canon_base_addr, canon_true_dependence(), crtl, store_info::cse_base, dump_file, dump_flags, dump_insn_info(), get_addr(), get_address_mode(), GET_MODE, GET_MODE_SIZE(), store_info::group_id, insn_info_type::insn, INSN_UID(), store_info::is_set, known_eq, last, dse_bb_info_type::last_insn, store_info::mem, store_info::mem_addr, MEM_ALIAS_SET, MEM_EXPR, MEM_READONLY_P, MEM_VOLATILE_P, store_info::next, insn_info_type::next_local_store, NULL, offset, store_info::offset, plus_constant(), print_inline_rtx(), print_range(), read_info_type_pool, insn_info_type::read_rec, replace_read(), store_info::rhs, rtx_group_vec, insn_info_type::store_rec, TDF_DETAILS, cselib_val::val_rtx, and store_info::width.
Referenced by check_mem_read_use(), and scan_insn().
|
static |
A note_uses callback in which DATA points the INSN_INFO for as check_mem_read_rtx. Nullify the pointer if i_m_r_m_r returns true for any part of *LOC.
References check_mem_read_rtx(), FOR_EACH_SUBRTX_PTR, and MEM_P.
Referenced by scan_insn().
|
static |
Clear the rhs field from the active_local_stores array.
References active_local_stores, store_info::const_rhs, store_info::is_set, store_info::next, insn_info_type::next_local_store, NULL, store_info::rhs, and insn_info_type::store_rec.
Referenced by record_store(), and scan_insn().
Return true if X is a constant or one of the registers that behave as a constant over the life of a function. This is equivalent to !rtx_varies_p for memory addresses.
References arg_pointer_rtx, CONSTANT_P, fixed_regs, frame_pointer_rtx, GET_CODE, hard_frame_pointer_rtx, and pic_offset_table_rtx.
Referenced by canon_address().
|
static |
Return a bitmap of the fixed registers contained in IN.
References ALLOC_REG_SET, bitmap_and(), fixed_reg_set, and NULL.
Referenced by scan_insn().
|
static |
Delete the insn and free all of the fields inside INSN_INFO.
References check_for_inc_dec_1(), dbg_cnt(), delete_insn(), dump_file, dump_flags, free_store_info(), insn_info_type::insn, INSN_UID(), locally_deleted, NULL, read_info_type_pool, insn_info_type::read_rec, TDF_DETAILS, and insn_info_type::wild_read.
Referenced by dse_step1(), dse_step6(), and record_store().
|
static |
Confluence function for blocks with no successors. Create an out set from the gen set of the exit block. This block logically has the exit block as a successor.
References bb_table, BITMAP_ALLOC, bitmap_copy(), dse_bitmap_obstack, EXIT_BLOCK, dse_bb_info_type::gen, basic_block_def::index, and dse_bb_info_type::out.
Referenced by dse_step4().
Propagate the information from the in set of the dest of E to the out set of the src of E. If the various in or out sets are not there, that means they are all ones.
References bb_table, BITMAP_ALLOC, bitmap_and_into(), bitmap_copy(), dse_bitmap_obstack, dse_bb_info_type::in, and dse_bb_info_type::out.
Referenced by dse_step4().
|
static |
Initialization of data structures.
References bb_table, BITMAP_ALLOC, bitmap_obstack_initialize(), cfun, dse_bitmap_obstack, dse_obstack, gcc_obstack_init, globally_deleted, init_alias_analysis(), kill_on_calls, last_basic_block_for_fn, locally_deleted, reg_obstack, rtx_group_next_id, rtx_group_table, scratch, and stores_off_frame_dead_at_return.
Referenced by rest_of_handle_dse().
|
static |
Do all of step 1.
References active_local_stores, active_local_stores_len, all_blocks, bb_table, BITMAP_ALLOC, bitmap_copy(), BITMAP_FREE, bitmap_set_bit, store_info::bmap, insn_info_type::cannot_delete, cfun, insn_info_type::contains_cselib_groups, crtl, cse_store_info_pool, cselib_clear_table(), cselib_discard_hook, cselib_finish(), cselib_init(), cselib_process_insn(), deferred_change_list, deferred_change_pool, delete_dead_store_insn(), DF_LR_IN, df_simulate_initialize_forwards(), df_simulate_one_insn_forwards(), dse_bb_info_type_pool, dump_file, dump_flags, EDGE_COUNT, ENTRY_BLOCK, EXIT_BLOCK, EXIT_BLOCK_PTR_FOR_FN, FOR_ALL_BB_FN, FOR_BB_INSNS, group_info::frame_related, free_store_info(), store_info::group_id, basic_block_def::index, insn_info_type::insn, INSN_P, INSN_UID(), store_info::is_large, store_info::is_set, store_info::large, dse_bb_info_type::last_insn, deferred_change::loc, deferred_change::next, store_info::next, insn_info_type::next_local_store, NULL, NUM_FIXED_BLOCKS, store_info::positions_needed, insn_info_type::prev_insn, store_info::redundant_reason, deferred_change::reg, reg_obstack, dse_bb_info_type::regs_live, remove_useless_values(), rtx_group_table, rtx_group_vec, scan_insn(), single_succ(), single_succ_p(), insn_info_type::store_rec, stores_off_frame_dead_at_return, basic_block_def::succs, and TDF_DETAILS.
Referenced by rest_of_handle_dse().
|
static |
Init the offset tables.
References bitmap_bit_p, bitmap_clear(), bitmap_set_bit, current_position, group_info::escaped_n, group_info::escaped_p, EXECUTE_IF_SET_IN_BITMAP, FOR_EACH_VEC_ELT, group_info::group_kill, i, kill_on_calls, group_info::offset_map_n, group_info::offset_map_p, group_info::offset_map_size_n, group_info::offset_map_size_p, group_info::process_globally, rtx_group_vec, group_info::store2_n, and group_info::store2_p.
Referenced by rest_of_handle_dse().
|
static |
References bitmap_count_bits(), bitmap_ior_into(), bitmap_print(), dse_obstack, dump_file, dump_flags, FOR_EACH_VEC_ELT, group_info::frame_related, i, group_info::offset_map_n, group_info::offset_map_p, group_info::offset_map_size_n, group_info::offset_map_size_p, group_info::process_globally, rtx_group_vec, group_info::store1_n, group_info::store1_p, group_info::store2_n, group_info::store2_p, stores_off_frame_dead_at_return, and TDF_DETAILS.
Referenced by rest_of_handle_dse().
|
static |
Build the transfer functions for the function.
References all_blocks, bb_table, BITMAP_ALLOC, bitmap_bit_p, bitmap_clear(), bitmap_copy(), BITMAP_FREE, bitmap_ior_into(), bitmap_ones(), cfun, dse_bitmap_obstack, dse_step3_exit_block_scan(), dse_step3_scan(), EDGE_COUNT, ENTRY_BLOCK, EXECUTE_IF_SET_IN_BITMAP, EXIT_BLOCK, FOR_ALL_BB_FN, FOR_EACH_VEC_ELT, dse_bb_info_type::gen, group_info::group_kill, i, dse_bb_info_type::in, basic_block_def::index, last_basic_block_for_fn, mark_reachable_blocks(), NULL, dse_bb_info_type::out, rtx_group_vec, and basic_block_def::succs.
Referenced by rest_of_handle_dse().
|
static |
Set the gen set of the exit block, and also any block with no successors that does not have a wild read.
References bitmap_ior_into(), FOR_EACH_VEC_ELT, group_info::frame_related, dse_bb_info_type::gen, group_info::group_kill, i, group_info::process_globally, rtx_group_vec, and stores_off_frame_dead_at_return.
Referenced by dse_step3().
|
static |
Scan the insns in BB_INFO starting at PTR and going to the top of the block in order to build the gen and kill sets for the block. We start at ptr which may be the last insn in the block or may be the first insn with a wild read. In the latter case we are able to skip the rest of the block because it just does not matter: anything that happens is hidden by the wild read.
References bb_table, BITMAP_ALLOC, bitmap_clear(), BITMAP_FREE, dse_bitmap_obstack, find_insn_before_first_wild_read(), dse_bb_info_type::gen, basic_block_def::index, insn_info_type::insn, INSN_P, dse_bb_info_type::kill, dse_bb_info_type::last_insn, insn_info_type::prev_insn, scan_reads(), scan_stores(), and insn_info_type::store_rec.
Referenced by dse_step3().
|
static |
Solve the dataflow equations.
References all_blocks, bb_table, bitmap_print(), cfun, DF_BACKWARD, df_get_n_blocks(), df_get_postorder(), df_print_bb_index(), df_simple_dataflow(), dse_confluence_0(), dse_confluence_n(), dse_transfer_function(), dump_file, dump_flags, FOR_ALL_BB_FN, dse_bb_info_type::gen, dse_bb_info_type::in, basic_block_def::index, dse_bb_info_type::kill, NULL, dse_bb_info_type::out, and TDF_DETAILS.
Referenced by rest_of_handle_dse().
|
static |
References bb_table, bitmap_bit_p, bitmap_clear(), bitmap_empty_p(), bitmap_print(), insn_info_type::cannot_delete, cfun, check_for_inc_dec_1(), dbg_cnt(), delete_insn(), deleted, dump_file, dump_flags, end(), FOR_EACH_BB_FN, insn_info_type::frame_read, get_bitmap_index(), globally_deleted, store_info::group_id, i, basic_block_def::index, insn_info_type::insn, INSN_P, INSN_UID(), poly_int< N, C >::is_constant(), store_info::is_set, dse_bb_info_type::last_insn, store_info::next, insn_info_type::non_frame_wild_read, NULL, offset, store_info::offset, dse_bb_info_type::out, insn_info_type::prev_insn, insn_info_type::read_rec, rtx_group_vec, scan_reads(), scan_stores(), insn_info_type::store_rec, TDF_DETAILS, store_info::width, and insn_info_type::wild_read.
Referenced by rest_of_handle_dse().
|
static |
References bb_table, insn_info_type::cannot_delete, cfun, delete_dead_store_insn(), dump_file, dump_flags, FOR_ALL_BB_FN, basic_block_def::index, insn_info_type::insn, INSN_P, INSN_UID(), store_info::is_set, dse_bb_info_type::last_insn, store_info::next, insn_info_type::prev_insn, store_info::redundant_reason, insn_info_type::store_rec, and TDF_DETAILS.
Referenced by rest_of_handle_dse().
|
static |
References all_blocks, bb_table, BITMAP_FREE, bitmap_obstack_release(), deferred_change_pool, dse_bb_info_type_pool, dse_bitmap_obstack, dse_obstack, end_alias_analysis(), free(), group_info_pool, insn_info_type_pool, NULL, read_info_type_pool, rtx_group_table, rtx_group_vec, rtx_store_info_pool, and scratch.
Referenced by rest_of_handle_dse().
|
static |
Propagate the info from the out to the in set of BB_INDEX's basic block. There are three cases: 1) The block has no kill set. In this case the kill set is all ones. It does not matter what the out set of the block is, none of the info can reach the top. The only thing that reaches the top is the gen set and we just copy the set. 2) There is a kill set but no out set and bb has successors. In this case we just return. Eventually an out set will be created and it is better to wait than to create a set of ones. 3) There is both a kill and out set. We apply the obvious transfer function.
References bb_table, BITMAP_ALLOC, bitmap_copy(), bitmap_ior_and_compl(), dse_bitmap_obstack, dse_bb_info_type::gen, dse_bb_info_type::in, dse_bb_info_type::kill, and dse_bb_info_type::out.
Referenced by dse_step4().
|
static |
References dump_file, insn_info_type::insn, INSN_UID(), and insn_info_type::store_rec.
Referenced by check_mem_read_rtx(), and scan_insn().
|
static |
Callback for for_each_inc_dec that emits an INSN that sets DEST to SRC + SRCOFF before insn ARG.
References note_add_store_info::current, emit_insn(), emit_insn_before(), end_sequence(), note_add_store_info::failure, note_add_store_info::first, insn_info_type::fixed_regs_live, note_add_store_info::fixed_regs_live, gen_add3_insn(), gen_move_insn(), get_insns(), insn_info_type::insn, NEXT_INSN(), note_add_store(), note_stores(), and start_sequence().
Referenced by check_for_inc_dec(), and check_for_inc_dec_1().
|
static |
Return the insn in BB_INFO before the first wild read or if there are no wild reads in the block, return the last insn.
References dse_bb_info_type::last_insn, NULL, insn_info_type::prev_insn, and insn_info_type::wild_read.
Referenced by dse_step3_scan().
|
static |
If the modes are different and the value's source and target do not line up, we need to extract the value from lower part of the rhs of the store, shift it, and then put it into a form that can be shoved into the read_insn. This function generates a right SHIFT of a value that is at least ACCESS_BYTES bytes wide of READ_MODE. The shift sequence is returned or NULL if we failed to find a shift.
References BITS_PER_WORD, store_info::const_rhs, CONSTANT_P, copy_rtx(), COSTS_N_INSNS, emit_insn(), emit_move_insn(), end_sequence(), expand_binop(), extract_low_bits(), FOR_EACH_MODE_IN_CLASS, gen_int_shift_amount(), gen_reg_rtx(), get_insns(), GET_MODE, GET_MODE_BITSIZE(), GET_MODE_SIZE(), insn_cost(), INSN_P, known_le, store_info::mem, new_mode(), NEXT_INSN(), NULL, NULL_RTX, offset, OPTAB_DIRECT, opt_mode< T >::require(), store_info::rhs, set_src_cost(), shift, simplify_const_binary_operation(), simplify_gen_subreg(), simplify_subreg(), smallest_int_mode_for_size(), start_sequence(), subreg_lowpart_offset(), subreg_offset_from_lsb(), targetm, and TRULY_NOOP_TRUNCATION_MODES_P.
Referenced by get_stored_val().
|
static |
Free all READ_REC of the LAST_INSN of BB_INFO.
References dse_bb_info_type::last_insn, read_info_type_pool, and insn_info_type::read_rec.
Referenced by add_non_frame_wild_read(), and add_wild_read().
|
static |
Delete all of the store_info recs from INSN_INFO.
References BITMAP_FREE, store_info::bmap, insn_info_type::cannot_delete, insn_info_type::contains_cselib_groups, store_info::cse_base, cse_store_info_pool, store_info::is_large, store_info::large, store_info::next, NULL, store_info::positions_needed, rtx_store_info_pool, and insn_info_type::store_rec.
Referenced by delete_dead_store_insn(), dse_step1(), and remove_useless_values().
|
static |
Look up the bitmap index for OFFSET in GROUP_INFO. If it is not there, return 0.
References offset, group_info::offset_map_n, group_info::offset_map_p, group_info::offset_map_size_n, and group_info::offset_map_size_p.
Referenced by dse_step5(), scan_reads(), and scan_stores().
Get arguments passed to CALL_INSN. Return TRUE if successful. So far it only handles arguments passed in registers.
References CALL_INSN_FUNCTION_USAGE, CONST_INT_P, cselib_expand_value_rtx(), gen_int_mode(), GET_CODE, GET_MODE, GET_MODE_SIZE(), INTVAL, is_int_mode(), NULL_RTX, REG_P, REGNO, scratch, targetm, TREE_CHAIN, TREE_TYPE, TREE_VALUE, TYPE_ARG_TYPES, TYPE_MODE, void_list_node, and XEXP.
Referenced by scan_insn().
|
static |
Get the GROUP for BASE. Add a new group if it is not there.
References arg_pointer_rtx, BITMAP_ALLOC, canon_rtx(), dse_bitmap_obstack, fixed_regs, frame_pointer_rtx, gcc_assert, gen_rtx_MEM(), group_info_pool, hard_frame_pointer_rtx, NULL, NULL_RTX, group_info::rtx_base, rtx_group_next_id, rtx_group_table, and rtx_group_vec.
Referenced by canon_address().
|
static |
Helper function for replace_read and record_store. Attempt to return a value of mode READ_MODE stored in STORE_INFO, consisting of READ_WIDTH bytes starting from READ_OFFSET. Return NULL if not successful. If REQUIRE_CST is true, return always constant.
References const0_rtx, CONST_INT_P, store_info::const_rhs, CONSTANT_P, copy_rtx(), extract_low_bits(), find_shift_sequence(), gcc_assert, gen_int_mode(), gen_lowpart, GET_MODE, GET_MODE_BITSIZE(), GET_MODE_CLASS, GET_MODE_SIZE(), HOST_BITS_PER_WIDE_INT, HOST_WIDE_INT_1, int_mode_for_mode(), INTVAL, poly_int< N, C >::is_constant(), known_le, store_info::mem, NULL_RTX, store_info::offset, optimize_bb_for_speed_p(), store_info::rhs, shift, subreg_lowpart_offset(), targetm, validate_subreg(), VECTOR_MODE_P, and store_info::width.
Referenced by record_store(), and replace_read().
Return whether DECL, a local variable, can possibly escape the current function scope.
References cfun, NULL, and TREE_ADDRESSABLE.
Referenced by can_escape().
Call back for note_stores to find the hard regs set or clobbered by insn. Data is a bitmap of the hardregs set so far.
References bitmap_set_range(), HARD_REGISTER_P, REG_NREGS, REG_P, REGNO, and regs_set.
Referenced by replace_read().
|
static |
Return a bitmask with the first N low bits set.
References HOST_BITS_PER_WIDE_INT, and HOST_WIDE_INT_M1U.
Referenced by all_positions_needed_p(), and record_store().
rtl_opt_pass * make_pass_rtl_dse1 | ( | gcc::context * | ctxt | ) |
rtl_opt_pass * make_pass_rtl_dse2 | ( | gcc::context * | ctxt | ) |
|
static |
Find all of the blocks that are not backwards reachable from the exit block or any block with no successors (BB). These are the infinite loops or infinite self loops. These blocks will still have their bits set in UNREACHABLE_BLOCKS.
References bitmap_bit_p, bitmap_clear_bit(), FOR_EACH_EDGE, basic_block_def::index, mark_reachable_blocks(), and basic_block_def::preds.
Referenced by dse_step3(), and mark_reachable_blocks().
Callback for emit_inc_dec_insn_before via note_stores. Check if a register is clobbered which is live afterwards.
References note_add_store_info::current, END_REGNO(), note_add_store_info::failure, note_add_store_info::first, note_add_store_info::fixed_regs_live, NEXT_INSN(), PATTERN(), REG_P, reg_referenced_p(), REGNO, and REGNO_REG_SET_P.
Referenced by emit_inc_dec_insn_before().
|
static |
Print offset range [OFFSET, OFFSET + WIDTH) to FILE.
References offset, print_dec(), and SIGNED.
Referenced by check_mem_read_rtx(), and record_store().
BODY is an instruction pattern that belongs to INSN. Return 1 if there is a candidate store, after adding it to the appropriate local store group if so.
References active_local_stores, active_local_stores_len, add_wild_read(), store_info::addrspace, all_positions_needed_p(), any_positions_needed_p(), BITMAP_ALLOC, BLOCK_FOR_INSN(), store_info::bmap, insn_info_type::cannot_delete, canon_address(), group_info::canon_base_addr, canon_output_dependence(), canon_rtx(), clear_rhs_from_active_local_stores(), const0_rtx, CONST_INT_P, store_info::const_rhs, CONSTANT_P, insn_info_type::contains_cselib_groups, store_info::count, store_info::cse_base, cse_store_info_pool, cselib_expand_value_rtx(), delete_dead_store_insn(), df_find_single_def_src(), dse_bitmap_obstack, dump_file, dump_flags, end_sequence(), find_reg_note(), FLOAT_MODE_P, get_address_mode(), GET_CODE, get_insns(), GET_MODE, GET_MODE_SIZE(), get_stored_val(), store_info::group_id, HOST_BITS_PER_WIDE_INT, i, insn_info_type::insn, INSN_UID(), poly_int< N, C >::is_constant(), store_info::is_large, store_info::is_set, known_eq, store_info::large, last, dse_bb_info_type::last_insn, lowpart_bitmask(), MAX, MAX_OFFSET, may_be_sp_based_p(), store_info::mem, store_info::mem_addr, MEM_ADDR_SPACE, MEM_EXPR, MEM_P, MEM_SIZE, MEM_SIZE_KNOWN_P, MEM_VOLATILE_P, mems_same_for_tbaa_p(), MIN, store_info::next, insn_info_type::next_local_store, NULL, NULL_RTX, offset, store_info::offset, PATTERN(), plus_constant(), store_info::positions_needed, print_range(), store_info::redundant_reason, REG_P, reload_completed, store_info::rhs, rtx_equal_p(), rtx_group_vec, rtx_store_info_pool, scratch, SET, set_all_positions_unneeded(), SET_DEST, set_position_unneeded(), SET_SRC, set_usage_bits(), store_info::small_bitmask, insn_info_type::stack_pointer_based, start_sequence(), insn_info_type::store_rec, TDF_DETAILS, cselib_val::val_rtx, store_info::width, and XEXP.
Referenced by scan_insn().
|
static |
Remove BASE from the set of active_local_stores. This is a callback from cselib that is used to get rid of the stores in active_local_stores.
References active_local_stores, active_local_stores_len, store_info::cse_base, free_store_info(), store_info::group_id, last, store_info::next, insn_info_type::next_local_store, NULL, and insn_info_type::store_rec.
|
static |
Take a sequence of: A <- r1 ... ... <- A and change it into r2 <- r1 A <- r1 ... ... <- r2 or r3 <- extract (r1) r3 <- r3 >> shift r2 <- extract (r3) ... <- r2 or r2 <- extract (r1) ... <- r2 Depending on the alignment and the mode of the store and subsequent load. The STORE_INFO and STORE_INSN are for the store and READ_INFO and READ_INSN are for the read. Return true if the replacement went ok.
References BITMAP_ALLOC, bitmap_and_into(), bitmap_empty_p(), BITMAP_FREE, BLOCK_FOR_INSN(), copy_to_mode_reg(), dbg_cnt(), deferred_change_list, deferred_change_pool, df_print_regset(), dump_file, dump_flags, emit_insn_before(), end_sequence(), insn_info_type::fixed_regs_live, FOR_EACH_SUBRTX, GET_CODE, get_insns(), GET_MODE, GET_MODE_NAME, GET_RTX_CLASS, get_stored_val(), insn_info_type::insn, insn_invalid_p(), INSN_UID(), insns, deferred_change::loc, look_for_hardregs(), store_info::mem, deferred_change::next, NEXT_INSN(), note_stores(), NULL_RTX, store_info::offset, print_simple_rtl(), r, read_info_type_pool, insn_info_type::read_rec, deferred_change::reg, reg_obstack, REG_P, regs_set, RTX_AUTOINC, shallow_copy_rtx(), start_sequence(), SUBREG_REG, TDF_DETAILS, this_insn, validate_change(), word_mode, and WORD_REGISTER_OPERATIONS.
Referenced by check_mem_read_rtx().
|
static |
References active_local_stores, active_local_stores_len, and NULL.
Referenced by add_non_frame_wild_read(), and add_wild_read().
|
static |
------------------------------------------------------------------------- DSE -------------------------------------------------------------------------
Callback for running pass_rtl_dse.
References CDI_DOMINATORS, cfun, cleanup_cfg(), delete_unreachable_blocks(), df_analyze(), DF_DEFER_INSN_RESCAN, DF_LR_RUN_DCE, df_note_add_problem(), df_set_flags(), dse_step0(), dse_step1(), dse_step2(), dse_step2_init(), dse_step3(), dse_step4(), dse_step5(), dse_step6(), dse_step7(), dump_file, dump_flags, free_dominance_info(), globally_deleted, locally_deleted, purge_all_dead_edges(), and TDF_DETAILS.
Apply record_store to all candidate stores in INSN. Mark INSN if some part of it is not a candidate store and assigns to a non-register target.
References active_local_stores, active_local_stores_len, add_non_frame_wild_read(), add_wild_read(), CALL_INSN_FUNCTION_USAGE, CALL_P, insn_info_type::cannot_delete, cfun, check_mem_read_rtx(), check_mem_read_use(), clear_rhs_from_active_local_stores(), CONST_INT_P, copy_fixed_regs(), DEBUG_INSN_P, dump_file, dump_flags, dump_insn_info(), find_reg_note(), insn_info_type::fixed_regs_live, fndecl_built_in_p(), insn_info_type::frame_read, gen_rtx_MEM(), get_call_args(), get_call_rtx_from(), GET_CODE, store_info::group_id, HARD_FRAME_POINTER_IS_ARG_POINTER, i, insn_info_type::insn, insn_info_type_pool, insn_nothrow_p(), INSN_UID(), INTVAL, store_info::is_set, last, dse_bb_info_type::last_insn, store_info::mem, MEM_P, store_info::next, insn_info_type::next_local_store, note_uses(), NULL, NULL_RTX, NULL_TREE, PATTERN(), insn_info_type::prev_insn, record_store(), dse_bb_info_type::regs_live, reload_completed, RTL_CONST_CALL_P, RTX_FRAME_RELATED_P, rtx_group_vec, set_mem_size(), SIBLING_CALL_P, insn_info_type::stack_pointer_based, insn_info_type::store_rec, SYMBOL_REF_DECL, TDF_DETAILS, TREE_CODE, volatile_refs_p(), XEXP, XVECEXP, and XVECLEN.
Referenced by dse_step1().
|
static |
Process the READ_INFOs into the bitmaps into GEN and KILL. KILL may be NULL.
References group_info::base_mem, bitmap_and_compl_into(), bitmap_clear_bit(), bitmap_ior_into(), bitmap_set_bit, group_info::canon_base_addr, canon_true_dependence(), end(), FOR_EACH_VEC_ELT, insn_info_type::frame_read, group_info::frame_related, get_bitmap_index(), GET_MODE, group_info::group_kill, i, kill, kill_on_calls, insn_info_type::non_frame_wild_read, NULL_RTX, offset, group_info::process_globally, insn_info_type::read_rec, rtx_group_vec, and store_info::width.
Referenced by dse_step3_scan(), and dse_step5().
|
static |
Process the STORE_INFOs into the bitmaps into GEN and KILL. KILL may be NULL.
References bitmap_clear_bit(), bitmap_set_bit, end(), get_bitmap_index(), store_info::group_id, i, poly_int< N, C >::is_constant(), kill, store_info::next, offset, store_info::offset, group_info::process_globally, rtx_group_vec, and store_info::width.
Referenced by dse_step3_scan(), and dse_step5().
|
inlinestatic |
Mark the whole store S_INFO as unneeded.
References bitmap_set_range(), store_info::bmap, store_info::count, gcc_checking_assert, HOST_WIDE_INT_0U, poly_int< N, C >::is_constant(), store_info::is_large, store_info::large, store_info::positions_needed, store_info::small_bitmask, UNLIKELY, and store_info::width.
Referenced by record_store().
|
inlinestatic |
Mark byte POS bytes from the beginning of store S_INFO as unneeded.
References bitmap_set_bit, store_info::bmap, store_info::count, HOST_WIDE_INT_1U, store_info::is_large, store_info::large, store_info::positions_needed, store_info::small_bitmask, and UNLIKELY.
Referenced by record_store().
|
static |
Set the store* bitmaps offset_map_size* fields in GROUP based on OFFSET and WIDTH.
References bitmap_set_bit, can_escape(), group_info::escaped_n, group_info::escaped_p, i, poly_int< N, C >::is_constant(), MAX_OFFSET, offset, group_info::offset_map_size_n, group_info::offset_map_size_p, group_info::store1_n, group_info::store1_p, group_info::store2_n, and group_info::store2_p.
Referenced by record_store().
|
static |
The linked list of stores that are under consideration in this basic block.
Referenced by check_mem_read_rtx(), clear_rhs_from_active_local_stores(), dse_step1(), record_store(), remove_useless_values(), reset_active_stores(), and scan_insn().
|
static |
Referenced by check_mem_read_rtx(), dse_step1(), record_store(), remove_useless_values(), reset_active_stores(), and scan_insn().
|
static |
Referenced by all_blocks(), df_chain_finalize(), df_live_finalize(), df_live_init(), df_live_reset(), df_live_verify_transfer_functions(), df_lr_dce_finalize(), df_lr_init(), df_lr_reset(), df_lr_verify_transfer_functions(), df_md_alloc(), df_md_init(), df_md_local_compute(), df_md_reset(), df_mir_alloc(), df_mir_init(), df_mir_local_compute(), df_mir_reset(), df_note_compute(), df_rd_alloc(), df_rd_init_solution(), df_rd_local_compute(), df_word_lr_init(), df_word_lr_reset(), dse_step1(), dse_step3(), dse_step4(), dse_step7(), fast_dce(), and get_block_vector().
|
static |
Table to hold all bb_infos.
Referenced by dse_confluence_0(), dse_confluence_n(), dse_step0(), dse_step1(), dse_step3(), dse_step3_scan(), dse_step4(), dse_step5(), dse_step6(), dse_step7(), and dse_transfer_function().
|
static |
Referenced by dse_step1(), free_store_info(), and record_store().
|
static |
The number of bits used in the global bitmaps.
Referenced by dse_step2(), and gcov_write_length().
|
static |
Referenced by dse_step1(), and replace_read().
|
static |
Referenced by dse_step1(), dse_step7(), and replace_read().
|
static |
Referenced by dse_step1(), and dse_step7().
|
static |
Obstack for the DSE dataflow bitmaps. We don't want to put these on the default obstack because these bitmaps can grow quite large (~2GB for the small (!) test case of PR54146) and we'll hold on to all that memory until the end of the compiler run. As a bonus, delete_tree_live_info can destroy all the bitmaps by just releasing the whole obstack.
Referenced by dse_confluence_0(), dse_confluence_n(), dse_step0(), dse_step3(), dse_step3_scan(), dse_step7(), dse_transfer_function(), get_group_info(), and record_store().
|
static |
Obstack for other data. As for above: Kinda nice to be able to throw it all away at the end in one big sweep.
Referenced by dse_step0(), dse_step2_init(), and dse_step7().
|
static |
Counter for stats.
Referenced by dse_step0(), dse_step5(), and rest_of_handle_dse().
|
static |
Referenced by dse_step7(), and get_group_info().
|
static |
Referenced by dse_step7(), and scan_insn().
|
static |
Locations that are killed by calls in the global phase.
Referenced by dse_step0(), dse_step2(), and scan_reads().
|
static |
Referenced by delete_dead_store_insn(), dse_step0(), and rest_of_handle_dse().
|
static |
Referenced by check_mem_read_rtx(), delete_dead_store_insn(), dse_step7(), free_read_records(), and replace_read().
|
static |
Index into the rtx_group_vec.
Referenced by dse_step0(), and get_group_info().
|
static |
Tables of group_info structures, hashed by base value.
Referenced by dse_step0(), dse_step1(), dse_step7(), and get_group_info().
|
static |
|
static |
Referenced by dse_step7(), free_store_info(), and record_store().
Scratch bitmap for cselib's cselib_expand_value_rtx.
Referenced by canon_address(), wi::int_traits< double_int >::decompose(), wi::int_traits< wi::hwi_with_prec >::decompose(), wi::primitive_int_traits< T, signed_p >::decompose(), dse_step0(), dse_step7(), gcc_qsort(), gcc_sort_r(), get_call_args(), and record_store().
|
static |
This is true except if cfun->stdarg -- i.e. we cannot do this for vararg functions because they play games with the frame.
Referenced by dse_step0(), dse_step1(), dse_step2_init(), and dse_step3_exit_block_scan().