116 lines
4.3 KiB
C
116 lines
4.3 KiB
C
|
/* Header file for jump threading path solver.
|
||
|
Copyright (C) 2021-2023 Free Software Foundation, Inc.
|
||
|
Contributed by Aldy Hernandez <aldyh@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_TREE_SSA_THREADSOLVER_H
|
||
|
#define GCC_TREE_SSA_THREADSOLVER_H
|
||
|
|
||
|
// This class is a basic block path solver. Given a set of BBs
|
||
|
// indicating a path through the CFG, range_of_expr and range_of_stmt
|
||
|
// will calculate the range of an SSA or STMT as if the BBs in the
|
||
|
// path would have been executed in order.
|
||
|
//
|
||
|
// Note that the blocks are in reverse order, thus the exit block is
|
||
|
// path[0].
|
||
|
|
||
|
class path_range_query : public range_query
|
||
|
{
|
||
|
public:
|
||
|
path_range_query (class gimple_ranger &ranger,
|
||
|
const vec<basic_block> &path,
|
||
|
const bitmap_head *dependencies = NULL,
|
||
|
bool resolve = true);
|
||
|
path_range_query (gimple_ranger &ranger, bool resolve = true);
|
||
|
virtual ~path_range_query ();
|
||
|
void reset_path (const vec<basic_block> &, const bitmap_head *dependencies);
|
||
|
bool range_of_expr (vrange &r, tree name, gimple * = NULL) override;
|
||
|
bool range_of_stmt (vrange &r, gimple *, tree name = NULL) override;
|
||
|
bool unreachable_path_p ();
|
||
|
void dump (FILE *) override;
|
||
|
void debug ();
|
||
|
|
||
|
private:
|
||
|
bool internal_range_of_expr (vrange &r, tree name, gimple *);
|
||
|
void compute_ranges (const bitmap_head *dependencies);
|
||
|
void compute_exit_dependencies (bitmap_head *dependencies);
|
||
|
bool defined_outside_path (tree name);
|
||
|
void range_on_path_entry (vrange &r, tree name);
|
||
|
path_oracle *get_path_oracle () { return (path_oracle *)m_oracle; }
|
||
|
|
||
|
// Cache manipulation.
|
||
|
void set_cache (const vrange &r, tree name);
|
||
|
bool get_cache (vrange &r, tree name);
|
||
|
void clear_cache (tree name);
|
||
|
|
||
|
// Methods to compute ranges for the given path.
|
||
|
bool range_defined_in_block (vrange &, tree name, basic_block bb);
|
||
|
void compute_ranges_in_block (basic_block bb);
|
||
|
void compute_ranges_in_phis (basic_block bb);
|
||
|
void adjust_for_non_null_uses (basic_block bb);
|
||
|
void ssa_range_in_phi (vrange &r, gphi *phi);
|
||
|
void compute_outgoing_relations (basic_block bb, basic_block next);
|
||
|
void compute_phi_relations (basic_block bb, basic_block prev);
|
||
|
void maybe_register_phi_relation (gphi *, edge e);
|
||
|
bool add_to_exit_dependencies (tree name, bitmap dependencies);
|
||
|
bool exit_dependency_p (tree name);
|
||
|
bool ssa_defined_in_bb (tree name, basic_block bb);
|
||
|
bool relations_may_be_invalidated (edge);
|
||
|
|
||
|
// Path navigation.
|
||
|
basic_block entry_bb () { return m_path[m_path.length () - 1]; }
|
||
|
basic_block exit_bb () { return m_path[0]; }
|
||
|
basic_block curr_bb () { return m_path[m_pos]; }
|
||
|
basic_block prev_bb () { return m_path[m_pos + 1]; }
|
||
|
basic_block next_bb () { return m_path[m_pos - 1]; }
|
||
|
bool at_entry () { return m_pos == m_path.length () - 1; }
|
||
|
bool at_exit () { return m_pos == 0; }
|
||
|
void move_next () { --m_pos; }
|
||
|
|
||
|
// Range cache for SSA names.
|
||
|
ssa_global_cache *m_cache;
|
||
|
|
||
|
// Set for each SSA that has an active entry in the cache.
|
||
|
bitmap m_has_cache_entry;
|
||
|
|
||
|
// Path being analyzed.
|
||
|
auto_vec<basic_block> m_path;
|
||
|
|
||
|
// This is a list of SSA names that may have relevant context
|
||
|
// information for solving the final conditional along the path.
|
||
|
// Ranges for these SSA names are pre-calculated and cached during a
|
||
|
// top-down traversal of the path, and are then used to answer
|
||
|
// questions at the path exit.
|
||
|
auto_bitmap m_exit_dependencies;
|
||
|
|
||
|
// A ranger used to resolve ranges for SSA names whose values come
|
||
|
// from outside the path.
|
||
|
gimple_ranger &m_ranger;
|
||
|
|
||
|
// Current path position.
|
||
|
unsigned m_pos;
|
||
|
|
||
|
// Use ranger to resolve anything not known on entry.
|
||
|
bool m_resolve;
|
||
|
|
||
|
// Set if there were any undefined expressions while pre-calculating path.
|
||
|
bool m_undefined_path;
|
||
|
};
|
||
|
|
||
|
#endif // GCC_TREE_SSA_THREADSOLVER_H
|