| /* Header file for SSA range interface. |
| Copyright (C) 2017-2018 Free Software Foundation, Inc. |
| Contributed by Andrew MacLeod <amacleod@redhat.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/>. */ |
| |
| #ifndef GCC_SSA_RANGE_H |
| #define GCC_SSA_RANGE_H |
| |
| #include "grange.h" |
| #include "ssa-range-gori.h" |
| #include "ssa-range-cache.h" |
| |
| // This is the basic range generator interface. |
| // |
| // This base class provides all the API entry points, but only provides |
| // functionality at the statement level. Ie, it can calculate ranges on |
| // statements, but does no additonal lookup. |
| // |
| // All the range_of_* methods will return a range if the types is supported |
| // by the range engine. It may be the full range for the type, AKA varying_p |
| // or it may be a refined range. |
| // If the range type is not supported, then false is returned. |
| // The basic ssa_ranger class will also return false for range_on_entry and |
| // range_on_exit as those make no sense at the statement level. |
| // |
| // outgoing_edge_range_p will only return a range if the edge specified |
| // defines a range for the specified name. Otherwise false is returned. |
| // |
| // |
| class stmt_ranger |
| { |
| public: |
| |
| bool range_of_expr (irange &r, tree expr, gimple *s = NULL); |
| virtual irange range_of_ssa_name (tree name, gimple *s = NULL); |
| virtual bool range_of_stmt (irange &r, gimple *s, tree name = NULL_TREE); |
| virtual bool range_of_stmt_with_range (irange &r, gimple *s, tree name, |
| const irange &name_range); |
| protected: |
| bool range_of_grange (irange &r, grange *s, tree name = NULL_TREE, |
| const irange *name_range = NULL); |
| |
| virtual bool range_of_phi (irange &r, gphi *phi, tree name = NULL_TREE, |
| const irange *name_range = NULL); |
| |
| bool range_of_call (irange &r, gcall *call, tree name = NULL_TREE, |
| const irange *name_range = NULL); |
| |
| bool range_of_cond_expr (irange &r, gassign* call, tree name = NULL_TREE, |
| const irange *name_range = NULL); |
| }; |
| |
| class ssa_ranger : public stmt_ranger |
| { |
| public: |
| virtual void range_on_edge (irange &r, edge e, tree name); |
| virtual void range_on_entry (irange &r, basic_block bb, tree name); |
| virtual void range_on_exit (irange &r, basic_block bb, tree name); |
| |
| virtual bool outgoing_edge_range_p (irange &r, edge e, tree name, |
| irange *name_range = NULL); |
| |
| protected: |
| virtual bool range_of_phi (irange &r, gphi *phi, tree name = NULL_TREE, |
| const irange *name_range = NULL); |
| }; |
| |
| // This class utilizes the gori summary to query the range |
| // of SSA_NAMEs across multiple basic blocks and edges. It builds a cache |
| // of range on entry to blocks. All work is done on-demand so it is relatively |
| // lightweight until used. |
| // |
| // GORI stands for Generates Outgoing Range Info |
| // Itutilizes the engine from ssa-range-gori.h to build |
| // defintion chains on demand which enables the calculation of ranges for |
| // various ssa names that are related to those that actually generate ranges. |
| // |
| // It is lightweight to declare in this case, but for each basic block that is |
| // queried, it will scan some or all of the statements in the block to |
| // determine what ssa-names can have range information generated for them and |
| // cache this information. |
| // |
| // terminal_name () provides the "input" name to the chain of statements for |
| // name that can change the calculation of its range. This is usually outside |
| // the basic block the name is defind in.. ie, |
| // bb4: |
| // a_2 = b_6 + 5 |
| // c_3 = (a_2 < 25) |
| // in this small example, b_6 is the |
| // "terminal_name" returned for both a_2 and c_3 since a change in the |
| // range of b_6 to calculate their results may produce a different range. |
| |
| |
| class global_ranger : public ssa_ranger |
| { |
| public: |
| global_ranger (); |
| ~global_ranger (); |
| |
| virtual irange range_of_ssa_name (tree name, gimple *s = NULL); |
| virtual bool range_of_stmt (irange &r, gimple *s, tree name = NULL_TREE); |
| virtual void range_on_entry (irange &r, basic_block bb, tree name); |
| |
| virtual bool outgoing_edge_range_p (irange &r, edge e, tree name, |
| irange *name_range = NULL); |
| tree terminal_name (tree name); |
| |
| void export_global_ranges (); |
| |
| void dump (FILE *f); |
| void calculate_and_dump (FILE *f); /* Calculate all stmts and dump */ |
| |
| protected: |
| gori_cache m_gori; /* Generates Outgoing Range Info. */ |
| }; |
| |
| // FIXME: Forward declaration for loop_ranger. |
| class vr_values; |
| |
| // A global ranger that uses SCEV/loop (if available) to refine PHI results. |
| |
| class loop_ranger : public global_ranger |
| { |
| public: |
| loop_ranger (); |
| ~loop_ranger (); |
| private: |
| void adjust_phi_with_loop_info (irange &r, gphi *phi); |
| virtual bool range_of_phi (irange &r, gphi *phi, tree name, |
| const irange *name_range); |
| |
| vr_values *m_vr_values; |
| }; |
| |
| class trace_ranger : public loop_ranger |
| { |
| public: |
| trace_ranger(); |
| |
| virtual irange range_of_ssa_name (tree name, gimple *s = NULL); |
| virtual bool range_of_stmt (irange &r, gimple *s, tree name = NULL_TREE); |
| virtual void range_on_edge (irange &r, edge e, tree name); |
| virtual void range_on_entry (irange &r, basic_block bb, tree name); |
| virtual void range_on_exit (irange &r, basic_block bb, tree name); |
| |
| // Calculate a range on edge E only if it is defined by E. |
| virtual bool outgoing_edge_range_p (irange &r, edge e, tree name, |
| irange *name_range = NULL); |
| private: |
| typedef global_ranger super; // Inherited from class for easy changing. |
| static const unsigned bump = 2; |
| unsigned indent; |
| unsigned trace_count; // Current trace index count. |
| |
| bool dumping (unsigned counter, bool trailing = false); |
| bool trailer (unsigned counter, const char *caller, bool result, tree name, |
| const irange &r); |
| }; |
| |
| |
| |
| // Like global_ranger::range_of_expr (), but make an on-the-fly ranger. |
| // Return TRUE if SSA as seen from within STMT has a known range the is not |
| // varying. Set this range in R. |
| // |
| // NOTE: There is overhead involved with this function, so it |
| // should only be used for very lightweight or unrelated range |
| // queries. This function is mostly meant for range queries that |
| // don't need caching in subsequent calls. */ |
| |
| static inline bool |
| on_demand_get_range_on_stmt (irange &r, tree ssa, gimple *stmt) |
| { |
| if (!cfun->cfg) |
| return false; |
| loop_ranger ranger; |
| bool ret; |
| ret = ranger.range_of_expr (r, ssa, stmt); |
| if (ret && r.varying_p ()) |
| return false; |
| return ret; |
| } |
| #endif // GCC_SSA_RANGE_H |