aboutsummaryrefslogtreecommitdiff
path: root/gcc/rtl-ssa/functions.h
diff options
context:
space:
mode:
Diffstat (limited to 'gcc/rtl-ssa/functions.h')
-rw-r--r--gcc/rtl-ssa/functions.h433
1 files changed, 433 insertions, 0 deletions
diff --git a/gcc/rtl-ssa/functions.h b/gcc/rtl-ssa/functions.h
new file mode 100644
index 0000000..b09d50e
--- /dev/null
+++ b/gcc/rtl-ssa/functions.h
@@ -0,0 +1,433 @@
+// Function-related RTL SSA classes -*- C++ -*-
+// Copyright (C) 2020 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 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/>.
+
+namespace rtl_ssa {
+
+// SSA-related information about a function. It contains three levels
+// of information, each in reverse postorder:
+//
+// - a list of extended basic blocks
+// - a list of basic blocks
+// - a list of instructions
+//
+// It also maintains a list of definitions of memory, and a list of
+// definitions of each register.
+//
+// See doc/rtl.texi for more details about the way this information
+// is organized and how changes to it are made.
+class function_info
+{
+ // The default obstack alignment takes long double into account.
+ // Since we have no use for that here, and since we allocate many
+ // relatively small objects, it's better to specify an alignment
+ // explicitly. The allocation routines assert that the alignment
+ // is enough for the objects being allocated.
+ //
+ // Because various structures use pointer_mux, we need at least 2 bytes
+ // of alignment.
+ static const size_t obstack_alignment = sizeof (void *);
+
+public:
+ // Construct SSA form for function FN.
+ function_info (function *fn);
+ ~function_info ();
+
+ // Return a list of all the extended basic blocks in the function, in reverse
+ // postorder. The list includes the entry and exit blocks.
+ iterator_range<ebb_iterator> ebbs () const;
+
+ // Like ebbs (), but in the reverse order.
+ iterator_range<reverse_ebb_iterator> reverse_ebbs () const;
+
+ // Return a list of all the basic blocks in the function, in reverse
+ // postorder. The list includes the entry and exit blocks.
+ iterator_range<bb_iterator> bbs () const;
+
+ // Like bbs (), but in the reverse order.
+ iterator_range<reverse_bb_iterator> reverse_bbs () const;
+
+ // Return the SSA information for the basic block with index INDEX.
+ bb_info *bb (unsigned int index) const { return m_bbs[index]; }
+
+ // Return the SSA information for CFG_BB.
+ bb_info *bb (basic_block cfg_bb) const { return m_bbs[cfg_bb->index]; }
+
+ // Return a list of all the instructions in the function, in reverse
+ // postorder. The list includes both real and artificial instructions.
+ //
+ // Iterations over the list will pick up any new instructions that are
+ // inserted after the iterator's current instruction.
+ iterator_range<any_insn_iterator> all_insns () const;
+
+ // Like all_insns (), but in the reverse order.
+ //
+ // Iterations over the list will pick up any new instructions that are
+ // inserted before the iterator's current instruction.
+ iterator_range<reverse_any_insn_iterator> reverse_all_insns () const;
+
+ // Like all_insns (), but without the debug instructions.
+ iterator_range<nondebug_insn_iterator> nondebug_insns () const;
+
+ // Like reverse_all_insns (), but without the debug instructions.
+ iterator_range<reverse_nondebug_insn_iterator>
+ reverse_nondebug_insns () const;
+
+ // Return the first and last instructions in insns ().
+ insn_info *first_insn () const { return m_first_insn; }
+ insn_info *last_insn () const { return m_last_insn; }
+
+ // Return a list of all definitions of memory, in reverse postorder.
+ // This includes both real stores by instructions and artificial
+ // definitions by things like phi nodes.
+ iterator_range<def_iterator> mem_defs () const;
+
+ // Return a list of all definitions of register REGNO, in reverse postorder.
+ // This includes both real stores by instructions and artificial
+ // definitions by things like phi nodes.
+ iterator_range<def_iterator> ref_defs (unsigned int regno) const;
+
+ // Check if all uses of register REGNO are either unconditionally undefined
+ // or use the same single dominating definition. Return the definition
+ // if so, otherwise return null.
+ set_info *single_dominating_def (unsigned int regno) const;
+
+ // Look for a definition of RESOURCE at INSN. Return the result of the
+ // search as a def_lookup; see the comments there for more details.
+ def_lookup find_def (resource_info resource, insn_info *insn);
+
+ // Return an RAII object that owns all temporary RTL SSA memory
+ // allocated during a change attempt. The object should remain in
+ // scope until the change has been aborted or successfully completed.
+ obstack_watermark new_change_attempt () { return &m_temp_obstack; }
+
+ // Make a best attempt to check whether the values used by USES are
+ // available on entry to BB, without solving a full dataflow problem.
+ // If all the values are already live on entry to BB or can be made
+ // available there, return a use_array that describes the uses as
+ // if they occured at the start of BB. These uses are purely temporary,
+ // and will not become permanent unless applied using change_insns.
+ //
+ // If the operation fails, return an invalid use_array.
+ //
+ // WATERMARK is a watermark returned by new_change_attempt ().
+ use_array make_uses_available (obstack_watermark &watermark,
+ use_array uses, bb_info *bb);
+
+ // If CHANGE doesn't already clobber REGNO, try to add such a clobber,
+ // limiting the movement range in order to make the clobber valid.
+ // When determining whether REGNO is live, ignore accesses made by an
+ // instruction I if IGNORE (I) is true. The caller then assumes the
+ // responsibility of ensuring that CHANGE and I are placed in a valid order.
+ //
+ // Return true on success. Leave CHANGE unmodified when returning false.
+ //
+ // WATERMARK is a watermark returned by new_change_attempt ().
+ template<typename IgnorePredicate>
+ bool add_regno_clobber (obstack_watermark &watermark, insn_change &change,
+ unsigned int regno, IgnorePredicate ignore);
+
+ // Return true if change_insns will be able to perform the changes
+ // described by CHANGES.
+ bool verify_insn_changes (array_slice<insn_change *const> changes);
+
+ // Perform all the changes in CHANGES, keeping the instructions in the
+ // order specified by the CHANGES array. On return, the SSA information
+ // remains up-to-date. The same is true for instruction-level DF
+ // information, although the block-level DF information might be
+ // marked dirty.
+ void change_insns (array_slice<insn_change *> changes);
+
+ // Like change_insns, but for a single change CHANGE.
+ void change_insn (insn_change &change);
+
+ // If the changes that have been made to instructions require updates
+ // to the CFG, perform those updates now. Return true if something changed.
+ // If it did:
+ //
+ // - The SSA information is now invalid and needs to be recomputed.
+ //
+ // - Dominance information is no longer available (in either direction).
+ //
+ // - The caller will need to call cleanup_cfg at some point.
+ //
+ // ??? We could probably update the SSA information for simple updates,
+ // but currently nothing would benefit. These late CFG changes are
+ // relatively rare anyway, since gimple optimisers should remove most
+ // unnecessary control flow.
+ bool perform_pending_updates ();
+
+ // Print the contents of the function to PP.
+ void print (pretty_printer *pp) const;
+
+private:
+ // Information about the values that are live on exit from a basic block.
+ // This class is only used when constructing the SSA form, it isn't
+ // designed for being kept up-to-date.
+ class bb_live_out_info
+ {
+ public:
+ // REG_VALUES contains all the registers that live out from the block,
+ // in order of increasing register number. There are NUM_REG_VALUES
+ // in total. Registers do not appear here if their values are known
+ // to be completely undefined; in that sense, the information is
+ // closer to DF_LIVE than to DF_LR.
+ unsigned int num_reg_values;
+ set_info **reg_values;
+
+ // The memory value that is live on exit from the block.
+ set_info *mem_value;
+ };
+
+ // Information used while constructing the SSA form and discarded
+ // afterwards.
+ class build_info
+ {
+ public:
+ set_info *current_reg_value (unsigned int) const;
+ set_info *current_mem_value () const;
+
+ void record_reg_def (unsigned int, def_info *);
+ void record_mem_def (def_info *);
+
+ // The block that we're currently processing.
+ bb_info *current_bb;
+
+ // The EBB that contains CURRENT_BB.
+ ebb_info *current_ebb;
+
+ // Except for the local exception noted below:
+ //
+ // - If register R has been defined in the current EBB, LAST_ACCESS[R + 1]
+ // is the last definition of R in the EBB.
+ //
+ // - If register R is currently live but has not yet been defined
+ // in the EBB, LAST_ACCESS[R + 1] is the current value of R,
+ // or null if the register's value is completely undefined.
+ //
+ // - The contents are not meaningful for other registers.
+ //
+ // Similarly:
+ //
+ // - If the current EBB has defined memory, LAST_ACCESS[0] is the last
+ // definition of memory in the EBB.
+ //
+ // - Otherwise LAST_ACCESS[0] is the value of memory that is live on
+ // - entry to the EBB.
+ //
+ // The exception is that while building instructions, LAST_ACCESS[I]
+ // can temporarily be the use of regno I - 1 by that instruction.
+ access_info **last_access;
+
+ // A bitmap of registers that are live on entry to this EBB, with a tree
+ // view for quick lookup. Only used if MAY_HAVE_DEBUG_INSNS.
+ bitmap ebb_live_in_for_debug;
+
+ // A conservative superset of the registers that are used by
+ // instructions in CURRENT_EBB. That is, all used registers
+ // are in the set, but some unused registers might be too.
+ bitmap ebb_use;
+
+ // A similarly conservative superset of the registers that are defined
+ // by instructions in CURRENT_EBB.
+ bitmap ebb_def;
+
+ // BB_LIVE_OUT[BI] gives the live-out values for the basic block
+ // with index BI.
+ bb_live_out_info *bb_live_out;
+ };
+
+ // Return an RAII object that owns all objects allocated by
+ // allocate_temp during its lifetime.
+ obstack_watermark temp_watermark () { return &m_temp_obstack; }
+
+ template<typename T, typename... Ts>
+ T *allocate (Ts... args);
+
+ template<typename T, typename... Ts>
+ T *allocate_temp (Ts... args);
+
+ access_array temp_access_array (access_array accesses);
+
+ clobber_group *need_clobber_group (clobber_info *);
+ def_node *need_def_node (def_info *);
+ def_splay_tree need_def_splay_tree (def_info *);
+
+ use_info *make_use_available (use_info *, bb_info *);
+ def_array insert_temp_clobber (obstack_watermark &, insn_info *,
+ unsigned int, def_array);
+
+ void insert_def_before (def_info *, def_info *);
+ void insert_def_after (def_info *, def_info *);
+ void remove_def_from_list (def_info *);
+
+ void add_clobber (clobber_info *, clobber_group *);
+ void remove_clobber (clobber_info *, clobber_group *);
+ void prepend_clobber_to_group (clobber_info *, clobber_group *);
+ void append_clobber_to_group (clobber_info *, clobber_group *);
+ void merge_clobber_groups (clobber_info *, clobber_info *,
+ def_info *);
+ clobber_info *split_clobber_group (clobber_group *, insn_info *);
+
+ void append_def (def_info *);
+ void add_def (def_info *);
+ void remove_def (def_info *);
+
+ void need_use_splay_tree (set_info *);
+
+ static void insert_use_before (use_info *, use_info *);
+ static void insert_use_after (use_info *, use_info *);
+
+ void add_use (use_info *);
+ void remove_use (use_info *);
+
+ insn_info::order_node *need_order_node (insn_info *);
+
+ void add_insn_after (insn_info *, insn_info *);
+ void append_insn (insn_info *);
+ void remove_insn (insn_info *);
+
+ insn_info *append_artificial_insn (bb_info *, rtx_insn * = nullptr);
+
+ void start_insn_accesses ();
+ void finish_insn_accesses (insn_info *);
+
+ void record_use (build_info &, insn_info *, rtx_obj_reference);
+ void record_call_clobbers (build_info &, insn_info *, rtx_call_insn *);
+ void record_def (build_info &, insn_info *, rtx_obj_reference);
+ void add_insn_to_block (build_info &, rtx_insn *);
+
+ void add_reg_unused_notes (insn_info *);
+
+ void add_live_out_use (bb_info *, set_info *);
+ set_info *live_out_value (bb_info *, set_info *);
+
+ void append_phi (ebb_info *, phi_info *);
+ void remove_phi (phi_info *);
+ void delete_phi (phi_info *);
+ void replace_phi (phi_info *, set_info *);
+ phi_info *create_phi (ebb_info *, resource_info, access_info **,
+ unsigned int);
+ phi_info *create_degenerate_phi (ebb_info *, set_info *);
+
+ bb_info *create_bb_info (basic_block);
+ void append_bb (bb_info *);
+ void calculate_potential_phi_regs ();
+
+ insn_info *add_placeholder_after (insn_info *);
+ void possibly_queue_changes (insn_change &);
+ void finalize_new_accesses (insn_change &);
+ void apply_changes_to_insn (insn_change &);
+
+ void init_function_data ();
+ void add_entry_block_defs (build_info &);
+ void add_phi_nodes (build_info &);
+ void add_artificial_accesses (build_info &, df_ref_flags);
+ void add_block_contents (build_info &);
+ void record_block_live_out (build_info &);
+ void populate_backedge_phis (build_info &);
+ void process_all_blocks ();
+
+ void simplify_phi_setup (phi_info *, set_info **, bitmap);
+ void simplify_phi_propagate (phi_info *, set_info **, bitmap, bitmap);
+ void simplify_phis ();
+
+ // The function that this object describes.
+ function *m_fn;
+
+ // The lowest (negative) in-use artificial insn uid minus one.
+ int m_next_artificial_uid;
+
+ // The highest in-use phi uid plus one.
+ unsigned int m_next_phi_uid;
+
+ // The highest in-use register number plus one.
+ unsigned int m_num_regs;
+
+ // M_DEFS[R] is the first definition of register R - 1 in a reverse
+ // postorder traversal of the function, or null if the function has
+ // no definition of R. Applying last () gives the last definition of R.
+ //
+ // M_DEFS[0] is for memory; MEM_REGNO + 1 == 0.
+ auto_vec<def_info *> m_defs;
+
+ // M_BBS[BI] gives the SSA information about the block with index BI.
+ auto_vec<bb_info *> m_bbs;
+
+ // An obstack used to allocate the main RTL SSA information.
+ obstack m_obstack;
+
+ // An obstack used for temporary work, such as while building up a list
+ // of possible instruction changes.
+ obstack m_temp_obstack;
+
+ // The start of each obstack, so that all memory in them can be freed.
+ char *m_obstack_start;
+ char *m_temp_obstack_start;
+
+ // The entry and exit blocks.
+ bb_info *m_first_bb;
+ bb_info *m_last_bb;
+
+ // The first and last instructions in a reverse postorder traversal
+ // of the function.
+ insn_info *m_first_insn;
+ insn_info *m_last_insn;
+
+ // The last nondebug instruction in the list of instructions.
+ // This is only different from m_last_insn when building the initial
+ // SSA information; after that, the last instruction is always a
+ // BB end instruction.
+ insn_info *m_last_nondebug_insn;
+
+ // Temporary working state when building up lists of definitions and uses.
+ // Keeping them around should reduce the number of unnecessary reallocations.
+ auto_vec<access_info *> m_temp_defs;
+ auto_vec<access_info *> m_temp_uses;
+
+ // The set of registers that might need to have phis associated with them.
+ // Registers outside this set are known to have a single definition that
+ // dominates all uses.
+ //
+ // Before RA, about 5% of registers are typically in the set.
+ auto_bitmap m_potential_phi_regs;
+
+ // A list of phis that are no longer in use. Their uids are still unique
+ // and so can be recycled.
+ phi_info *m_free_phis;
+
+ // A list of instructions that have been changed in ways that need
+ // further processing later, such as removing dead instructions or
+ // altering the CFG.
+ auto_vec<insn_info *> m_queued_insn_updates;
+
+ // The INSN_UIDs of all instructions in M_QUEUED_INSN_UPDATES.
+ auto_bitmap m_queued_insn_update_uids;
+
+ // A basic_block is in this bitmap if we need to call purge_dead_edges
+ // on it. As with M_QUEUED_INSN_UPDATES, these updates are queued until
+ // a convenient point.
+ auto_bitmap m_need_to_purge_dead_edges;
+};
+
+void pp_function (pretty_printer *, const function_info *);
+}
+
+void dump (FILE *, const rtl_ssa::function_info *);
+
+void DEBUG_FUNCTION debug (const rtl_ssa::function_info *);