/* Support routines for value queries. Copyright (C) 2020-2021 Free Software Foundation, Inc. Contributed by Aldy Hernandez and Andrew MacLeod . 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 . */ #include "config.h" #include "system.h" #include "coretypes.h" #include "backend.h" #include "tree.h" #include "gimple.h" #include "ssa.h" #include "tree-pretty-print.h" #include "fold-const.h" #include "value-range-equiv.h" #include "value-query.h" #include "alloc-pool.h" // value_query default methods. tree value_query::value_on_edge (edge, tree name) { return value_of_expr (name); } tree value_query::value_of_stmt (gimple *stmt, tree name) { if (!name) name = gimple_get_lhs (stmt); gcc_checking_assert (!name || name == gimple_get_lhs (stmt)); if (name) return value_of_expr (name); return NULL_TREE; } // range_query default methods. bool range_query::range_on_edge (irange &r, edge, tree name) { return range_of_expr (r, name); } bool range_query::range_of_stmt (irange &r, gimple *stmt, tree name) { if (!name) name = gimple_get_lhs (stmt); gcc_checking_assert (!name || name == gimple_get_lhs (stmt)); if (name) return range_of_expr (r, name); return false; } tree range_query::value_of_expr (tree name, gimple *stmt) { tree t; int_range_max r; if (!irange::supports_type_p (TREE_TYPE (name))) return NULL_TREE; if (range_of_expr (r, name, stmt)) { // A constant used in an unreachable block oftens returns as UNDEFINED. // If the result is undefined, check the global value for a constant. if (r.undefined_p ()) range_of_expr (r, name); if (r.singleton_p (&t)) return t; } return NULL_TREE; } tree range_query::value_on_edge (edge e, tree name) { tree t; int_range_max r; if (!irange::supports_type_p (TREE_TYPE (name))) return NULL_TREE; if (range_on_edge (r, e, name)) { // A constant used in an unreachable block oftens returns as UNDEFINED. // If the result is undefined, check the global value for a constant. if (r.undefined_p ()) range_of_expr (r, name); if (r.singleton_p (&t)) return t; } return NULL_TREE; } tree range_query::value_of_stmt (gimple *stmt, tree name) { tree t; int_range_max r; if (!name) name = gimple_get_lhs (stmt); gcc_checking_assert (!name || name == gimple_get_lhs (stmt)); if (!name || !irange::supports_type_p (TREE_TYPE (name))) return NULL_TREE; if (range_of_stmt (r, stmt, name) && r.singleton_p (&t)) return t; return NULL_TREE; } // valuation_query support routines for value_range_equiv's. class equiv_allocator : public object_allocator { public: equiv_allocator () : object_allocator ("equiv_allocator pool") { } }; value_range_equiv * range_query::allocate_value_range_equiv () { return new (equiv_alloc->allocate ()) value_range_equiv; } void range_query::free_value_range_equiv (value_range_equiv *v) { equiv_alloc->remove (v); } const class value_range_equiv * range_query::get_value_range (const_tree expr, gimple *stmt) { int_range_max r; if (range_of_expr (r, const_cast (expr), stmt)) return new (equiv_alloc->allocate ()) value_range_equiv (r); return new (equiv_alloc->allocate ()) value_range_equiv (TREE_TYPE (expr)); } range_query::range_query () { equiv_alloc = new equiv_allocator; } range_query::~range_query () { equiv_alloc->release (); delete equiv_alloc; }