ipa-cp: Better representation of aggregate values in call contexts

This patch extends the previous one by using the same data structure
to represent aggregate values in classes ipa_auto_call_arg_values and
ipa_call_arg_values.

This usually simplifies handling and makes allocations of memory much
cheaper because only a single vectore is needed, as opposed to vectors
with each element pointing at other vecs.  The only functions which
unfortunately are a bit more complec are estimate_local_effects in
ipa-cp.cc and ipa_call_context::equal_to but I hope not too much - the
latter could probably be shorteneed at the expense of readability.

The patch removes types ipa_agg_value ipa_agg_value_set which is no
longer used with it.  This means that we could replace the "_argagg_"
part of the types introduced by the previous patches with more
reasonable "_agg_" - possibly as a follow-up patch.

gcc/ChangeLog:

2022-08-26  Martin Jambor  <mjambor@suse.cz>

	* ipa-prop.h (ipa_agg_value): Remove type.
	(ipa_agg_value_set): Likewise.
	(ipa_copy_agg_values): Remove function.
	(ipa_release_agg_values): Likewise.
	(ipa_auto_call_arg_values) Add a forward declaration.
	(ipa_call_arg_values): Likewise.
	(class ipa_argagg_value_list): New constructors, added member function
	value_for_index_p.
	(class ipa_auto_call_arg_values): Removed the destructor and member
	function safe_aggval_at.  Use ipa_argagg_values for m_known_aggs.
	(class ipa_call_arg_values): Removed member function safe_aggval_at.
	Use ipa_argagg_values for m_known_aggs.
	(ipa_get_indirect_edge_target): Removed declaration.
	(ipa_find_agg_cst_for_param): Likewise.
	(ipa_find_agg_cst_from_init): New declaration.
	(ipa_agg_value_from_jfunc): Likewise.
	(ipa_agg_value_set_from_jfunc): Removed declaration.
	(ipa_push_agg_values_from_jfunc): New declaration.
	* ipa-cp.cc (ipa_agg_value_from_node): Renamed to
	ipa_agg_value_from_jfunc, made public.
	(ipa_agg_value_set_from_jfunc): Removed.
	(ipa_push_agg_values_from_jfunc): New function.
	(ipa_get_indirect_edge_target_1): Removed known_aggs parameter, use
	avs for this purpose too.
	(ipa_get_indirect_edge_target): Removed the overload working on
	ipa_auto_call_arg_values, use ipa_argagg_value_list in the remaining
	one.
	(devirtualization_time_bonus): Use ipa_argagg_value_list and
	ipa_get_indirect_edge_target_1 instead of
	ipa_get_indirect_edge_target.
	(context_independent_aggregate_values): Removed function.
	(gather_context_independent_values): Work on ipa_argagg_value_list.
	(estimate_local_effects): Likewise, define some iterator variables
	only in the construct where necessary.
	(ipcp_discover_new_direct_edges): Adjust the call to
	ipa_get_indirect_edge_target_1.
	(push_agg_values_for_index_from_edge): Adjust the call
	ipa_agg_value_from_node which has been renamed to
	ipa_agg_value_from_jfunc.
	* ipa-fnsummary.cc (evaluate_conditions_for_known_args): Work on
	ipa_argagg_value_list.
	(evaluate_properties_for_edge): Replace manual filling in aggregate
	values with call to ipa_push_agg_values_from_jfunc.
	(estimate_calls_size_and_time): Work on ipa_argagg_value_list.
	(ipa_cached_call_context::duplicate_from): Likewise.
	(ipa_cached_call_context::release): Likewise.
	(ipa_call_context::equal_to): Likewise.
	* ipa-prop.cc (ipa_find_agg_cst_from_init): Make public.
	(ipa_find_agg_cst_for_param): Removed function.
	(ipa_find_agg_cst_from_jfunc_items): New function.
	(try_make_edge_direct_simple_call): Replace calls to
	ipa_agg_value_set_from_jfunc and ipa_find_agg_cst_for_param with
	ipa_find_agg_cst_from_init and ipa_find_agg_cst_from_jfunc_items.
	(try_make_edge_direct_virtual_call): Replace calls to
	ipa_agg_value_set_from_jfunc and ipa_find_agg_cst_for_param with
	simple query of constant jump function and a call to
	ipa_find_agg_cst_from_jfunc_items.
	(ipa_auto_call_arg_values::~ipa_auto_call_arg_values): Removed.
This commit is contained in:
Martin Jambor 2022-10-18 14:14:26 +02:00
parent e0403e9568
commit 656b2338c8
4 changed files with 218 additions and 403 deletions

View File

@ -1975,10 +1975,9 @@ ipa_value_range_from_jfunc (ipa_node_params *info, cgraph_edge *cs,
NODE and INFO describes the caller node or the one it is inlined to, and
its related info. */
static tree
ipa_agg_value_from_node (class ipa_node_params *info,
struct cgraph_node *node,
const ipa_agg_jf_item *item)
tree
ipa_agg_value_from_jfunc (ipa_node_params *info, cgraph_node *node,
const ipa_agg_jf_item *item)
{
tree value = NULL_TREE;
int src_idx;
@ -2061,37 +2060,38 @@ ipa_agg_value_from_node (class ipa_node_params *info,
item->type);
}
/* Determine whether AGG_JFUNC evaluates to a set of known constant value for
an aggregate and if so, return it. Otherwise return an empty set. NODE
and INFO describes the caller node or the one it is inlined to, and its
related info. */
/* Process all items in AGG_JFUNC relative to caller (or the node the original
caller is inlined to) NODE which described by INFO and push the results to
RES as describing values passed in parameter DST_INDEX. */
struct ipa_agg_value_set
ipa_agg_value_set_from_jfunc (class ipa_node_params *info, cgraph_node *node,
struct ipa_agg_jump_function *agg_jfunc)
void
ipa_push_agg_values_from_jfunc (ipa_node_params *info, cgraph_node *node,
ipa_agg_jump_function *agg_jfunc,
unsigned dst_index,
vec<ipa_argagg_value> *res)
{
struct ipa_agg_value_set agg;
struct ipa_agg_jf_item *item;
int i;
unsigned prev_unit_offset = 0;
bool first = true;
agg.items = vNULL;
agg.by_ref = agg_jfunc->by_ref;
FOR_EACH_VEC_SAFE_ELT (agg_jfunc->items, i, item)
for (const ipa_agg_jf_item &item : agg_jfunc->items)
{
tree value = ipa_agg_value_from_node (info, node, item);
tree value = ipa_agg_value_from_jfunc (info, node, &item);
if (!value)
continue;
if (value)
{
struct ipa_agg_value value_item;
ipa_argagg_value iav;
iav.value = value;
iav.unit_offset = item.offset / BITS_PER_UNIT;
iav.index = dst_index;
iav.by_ref = agg_jfunc->by_ref;
value_item.offset = item->offset;
value_item.value = value;
gcc_assert (first
|| iav.unit_offset > prev_unit_offset);
prev_unit_offset = iav.unit_offset;
first = false;
agg.items.safe_push (value_item);
}
res->safe_push (iav);
}
return agg;
}
/* If checking is enabled, verify that no lattice is in the TOP state, i.e. not
@ -3240,8 +3240,7 @@ static tree
ipa_get_indirect_edge_target_1 (struct cgraph_edge *ie,
const vec<tree> &known_csts,
const vec<ipa_polymorphic_call_context> &known_contexts,
const vec<ipa_agg_value_set> &known_aggs,
const ipa_argagg_value_list *avs,
const ipa_argagg_value_list &avs,
bool *speculative)
{
int param_index = ie->indirect_info->param_index;
@ -3261,31 +3260,16 @@ ipa_get_indirect_edge_target_1 (struct cgraph_edge *ie,
if (ie->indirect_info->agg_contents)
{
t = NULL;
if (avs && ie->indirect_info->guaranteed_unmodified)
t = avs->get_value (param_index,
ie->indirect_info->offset / BITS_PER_UNIT,
ie->indirect_info->by_ref);
if (!t)
{
const ipa_agg_value_set *agg;
if (known_aggs.length () > (unsigned int) param_index)
agg = &known_aggs[param_index];
else
agg = NULL;
bool from_global_constant;
t = ipa_find_agg_cst_for_param (agg,
(unsigned) param_index
< known_csts.length ()
? known_csts[param_index]
: NULL,
ie->indirect_info->offset,
ie->indirect_info->by_ref,
&from_global_constant);
if (t
&& !from_global_constant
&& !ie->indirect_info->guaranteed_unmodified)
t = NULL_TREE;
}
if ((unsigned) param_index < known_csts.length ()
&& known_csts[param_index])
t = ipa_find_agg_cst_from_init (known_csts[param_index],
ie->indirect_info->offset,
ie->indirect_info->by_ref);
if (!t && ie->indirect_info->guaranteed_unmodified)
t = avs.get_value (param_index,
ie->indirect_info->offset / BITS_PER_UNIT,
ie->indirect_info->by_ref);
}
else if ((unsigned) param_index < known_csts.length ())
t = known_csts[param_index];
@ -3302,28 +3286,22 @@ ipa_get_indirect_edge_target_1 (struct cgraph_edge *ie,
return NULL_TREE;
gcc_assert (!ie->indirect_info->agg_contents);
gcc_assert (!ie->indirect_info->by_ref);
anc_offset = ie->indirect_info->offset;
t = NULL;
/* Try to work out value of virtual table pointer value in replacements. */
if (!t && avs && !ie->indirect_info->by_ref)
t = avs->get_value (param_index,
ie->indirect_info->offset / BITS_PER_UNIT,
true);
if ((unsigned) param_index < known_csts.length ()
&& known_csts[param_index])
t = ipa_find_agg_cst_from_init (known_csts[param_index],
ie->indirect_info->offset, true);
/* Try to work out value of virtual table pointer value in known
aggregate values. */
if (!t && known_aggs.length () > (unsigned int) param_index
&& !ie->indirect_info->by_ref)
{
const ipa_agg_value_set *agg = &known_aggs[param_index];
t = ipa_find_agg_cst_for_param (agg,
(unsigned) param_index
< known_csts.length ()
? known_csts[param_index] : NULL,
ie->indirect_info->offset, true);
}
/* Try to work out value of virtual table pointer value in replacements. */
/* or known aggregate values. */
if (!t)
t = avs.get_value (param_index,
ie->indirect_info->offset / BITS_PER_UNIT,
true);
/* If we found the virtual table pointer, lookup the target. */
if (t)
@ -3442,23 +3420,10 @@ ipa_get_indirect_edge_target (struct cgraph_edge *ie,
ipa_call_arg_values *avals,
bool *speculative)
{
ipa_argagg_value_list avl (avals);
return ipa_get_indirect_edge_target_1 (ie, avals->m_known_vals,
avals->m_known_contexts,
avals->m_known_aggs,
NULL, speculative);
}
/* The same functionality as above overloaded for ipa_auto_call_arg_values. */
tree
ipa_get_indirect_edge_target (struct cgraph_edge *ie,
ipa_auto_call_arg_values *avals,
bool *speculative)
{
return ipa_get_indirect_edge_target_1 (ie, avals->m_known_vals,
avals->m_known_contexts,
avals->m_known_aggs,
NULL, speculative);
avl, speculative);
}
/* Calculate devirtualization time bonus for NODE, assuming we know information
@ -3479,7 +3444,10 @@ devirtualization_time_bonus (struct cgraph_node *node,
tree target;
bool speculative;
target = ipa_get_indirect_edge_target (ie, avals, &speculative);
ipa_argagg_value_list avl (avals);
target = ipa_get_indirect_edge_target_1 (ie, avals->m_known_vals,
avals->m_known_contexts,
avl, &speculative);
if (!target)
continue;
@ -3615,32 +3583,6 @@ good_cloning_opportunity_p (struct cgraph_node *node, sreal time_benefit,
}
}
/* Return all context independent values from aggregate lattices in PLATS in a
vector. Return NULL if there are none. */
static vec<ipa_agg_value>
context_independent_aggregate_values (class ipcp_param_lattices *plats)
{
vec<ipa_agg_value> res = vNULL;
if (plats->aggs_bottom
|| plats->aggs_contain_variable
|| plats->aggs_count == 0)
return vNULL;
for (struct ipcp_agg_lattice *aglat = plats->aggs;
aglat;
aglat = aglat->next)
if (aglat->is_single_const ())
{
struct ipa_agg_value item;
item.offset = aglat->offset;
item.value = aglat->values->value;
res.safe_push (item);
}
return res;
}
/* Grow vectors in AVALS and fill them with information about values of
parameters that are known to be independent of the context. Only calculate
m_known_aggs if CALCULATE_AGGS is true. INFO describes the function. If
@ -3660,8 +3602,6 @@ gather_context_independent_values (class ipa_node_params *info,
avals->m_known_vals.safe_grow_cleared (count, true);
avals->m_known_contexts.safe_grow_cleared (count, true);
if (calculate_aggs)
avals->m_known_aggs.safe_grow_cleared (count, true);
if (removable_params_cost)
*removable_params_cost = 0;
@ -3696,16 +3636,7 @@ gather_context_independent_values (class ipa_node_params *info,
avals->m_known_contexts[i] = ctxlat->values->value;
if (calculate_aggs)
{
vec<ipa_agg_value> agg_items;
struct ipa_agg_value_set *agg;
agg_items = context_independent_aggregate_values (plats);
agg = &avals->m_known_aggs[i];
agg->items = agg_items;
agg->by_ref = plats->aggs_by_ref;
ret |= !agg_items.is_empty ();
}
ret |= push_agg_values_from_plats (plats, i, 0, &avals->m_known_aggs);
}
return ret;
@ -3776,7 +3707,7 @@ static void
estimate_local_effects (struct cgraph_node *node)
{
ipa_node_params *info = ipa_node_params_sum->get (node);
int i, count = ipa_get_param_count (info);
int count = ipa_get_param_count (info);
bool always_const;
int removable_params_cost;
@ -3842,7 +3773,7 @@ estimate_local_effects (struct cgraph_node *node)
}
for (i = 0; i < count; i++)
for (int i = 0; i < count; i++)
{
class ipcp_param_lattices *plats = ipa_get_parm_lattices (info, i);
ipcp_lattice<tree> *lat = &plats->itself;
@ -3876,7 +3807,7 @@ estimate_local_effects (struct cgraph_node *node)
avals.m_known_vals[i] = NULL_TREE;
}
for (i = 0; i < count; i++)
for (int i = 0; i < count; i++)
{
class ipcp_param_lattices *plats = ipa_get_parm_lattices (info, i);
@ -3911,30 +3842,49 @@ estimate_local_effects (struct cgraph_node *node)
avals.m_known_contexts[i] = ipa_polymorphic_call_context ();
}
for (i = 0; i < count; i++)
unsigned all_ctx_len = avals.m_known_aggs.length ();
auto_vec<ipa_argagg_value, 32> all_ctx;
all_ctx.reserve_exact (all_ctx_len);
all_ctx.splice (avals.m_known_aggs);
avals.m_known_aggs.safe_grow_cleared (all_ctx_len + 1);
unsigned j = 0;
for (int index = 0; index < count; index++)
{
class ipcp_param_lattices *plats = ipa_get_parm_lattices (info, i);
class ipcp_param_lattices *plats = ipa_get_parm_lattices (info, index);
if (plats->aggs_bottom || !plats->aggs)
continue;
ipa_agg_value_set *agg = &avals.m_known_aggs[i];
for (ipcp_agg_lattice *aglat = plats->aggs; aglat; aglat = aglat->next)
{
ipcp_value<tree> *val;
if (aglat->bottom || !aglat->values
/* If the following is true, the one value is in known_aggs. */
/* If the following is true, the one value is already part of all
context estimations. */
|| (!plats->aggs_contain_variable
&& aglat->is_single_const ()))
continue;
unsigned unit_offset = aglat->offset / BITS_PER_UNIT;
while (j < all_ctx_len
&& (all_ctx[j].index < index
|| (all_ctx[j].index == index
&& all_ctx[j].unit_offset < unit_offset)))
{
avals.m_known_aggs[j] = all_ctx[j];
j++;
}
for (unsigned k = j; k < all_ctx_len; k++)
avals.m_known_aggs[k+1] = all_ctx[k];
for (val = aglat->values; val; val = val->next)
{
struct ipa_agg_value item;
item.offset = aglat->offset;
item.value = val->value;
agg->items.safe_push (item);
avals.m_known_aggs[j].value = val->value;
avals.m_known_aggs[j].unit_offset = unit_offset;
avals.m_known_aggs[j].index = index;
avals.m_known_aggs[j].by_ref = plats->aggs_by_ref;
perform_estimation_of_a_value (node, &avals,
removable_params_cost, 0, val);
@ -3944,7 +3894,7 @@ estimate_local_effects (struct cgraph_node *node)
fprintf (dump_file, " - estimates for value ");
print_ipcp_constant_value (dump_file, val->value);
fprintf (dump_file, " for ");
ipa_dump_param (dump_file, info, i);
ipa_dump_param (dump_file, info, index);
fprintf (dump_file, "[%soffset: " HOST_WIDE_INT_PRINT_DEC
"]: time_benefit: %g, size: %i\n",
plats->aggs_by_ref ? "ref " : "",
@ -3952,8 +3902,6 @@ estimate_local_effects (struct cgraph_node *node)
val->local_time_benefit.to_double (),
val->local_size_cost);
}
agg->items.pop ();
}
}
}
@ -4350,7 +4298,7 @@ ipcp_discover_new_direct_edges (struct cgraph_node *node,
next_ie = ie->next_callee;
ipa_argagg_value_list avs (aggvals);
target = ipa_get_indirect_edge_target_1 (ie, known_csts, known_contexts,
vNULL, &avs, &speculative);
avs, &speculative);
if (target)
{
bool agg_contents = ie->indirect_info->agg_contents;
@ -5779,8 +5727,8 @@ push_agg_values_for_index_from_edge (struct cgraph_edge *cs, int index,
agg_jf.value.pass_through.operand,
agg_jf.type);
else
value = ipa_agg_value_from_node (caller_info, cs->caller,
&agg_jf);
value = ipa_agg_value_from_jfunc (caller_info, cs->caller,
&agg_jf);
if (value)
{
struct ipa_argagg_value iav;

View File

@ -386,15 +386,6 @@ evaluate_conditions_for_known_args (struct cgraph_node *node,
int j;
struct expr_eval_op *op;
/* We allow call stmt to have fewer arguments than the callee function
(especially for K&R style programs). So bound check here (we assume
m_known_aggs vector is either empty or has the same length as
m_known_vals). */
gcc_checking_assert (!avals->m_known_aggs.length ()
|| !avals->m_known_vals.length ()
|| (avals->m_known_vals.length ()
== avals->m_known_aggs.length ()));
if (c->agg_contents)
{
if (c->code == ipa_predicate::changed
@ -402,14 +393,14 @@ evaluate_conditions_for_known_args (struct cgraph_node *node,
&& (avals->safe_sval_at(c->operand_num) == error_mark_node))
continue;
if (ipa_agg_value_set *agg = avals->safe_aggval_at (c->operand_num))
if (tree sval = avals->safe_sval_at (c->operand_num))
val = ipa_find_agg_cst_from_init (sval, c->offset, c->by_ref);
if (!val)
{
tree sval = avals->safe_sval_at (c->operand_num);
val = ipa_find_agg_cst_for_param (agg, sval, c->offset,
c->by_ref);
ipa_argagg_value_list avs (avals);
val = avs.get_value (c->operand_num, c->offset / BITS_PER_UNIT,
c->by_ref);
}
else
val = NULL_TREE;
}
else
{
@ -674,17 +665,9 @@ evaluate_properties_for_edge (struct cgraph_edge *e, bool inline_p,
/* Determine known aggregate values. */
if (fre_will_run_p (caller))
{
ipa_agg_value_set agg
= ipa_agg_value_set_from_jfunc (caller_parms_info,
caller, &jf->agg);
if (agg.items.length ())
{
if (!avals->m_known_aggs.length ())
avals->m_known_aggs.safe_grow_cleared (count, true);
avals->m_known_aggs[i] = agg;
}
}
ipa_push_agg_values_from_jfunc (caller_parms_info,
caller, &jf->agg, i,
&avals->m_known_aggs);
}
/* For calls used in polymorphic calls we further determine
@ -3446,8 +3429,7 @@ estimate_calls_size_and_time (struct cgraph_node *node, int *size,
{
if (ipa_is_param_used_by_indirect_call (params_summary, i)
&& (avals->safe_sval_at (i)
|| (avals->m_known_aggs.length () > i
&& avals->m_known_aggs[i].items.length ())))
|| (ipa_argagg_value_list (avals).value_for_index_p (i))))
use_table = false;
else if (ipa_is_param_used_by_polymorphic_call (params_summary, i)
&& (avals->m_known_contexts.length () > i
@ -3583,14 +3565,12 @@ ipa_cached_call_context::duplicate_from (const ipa_call_context &ctx)
m_avals.m_known_aggs = vNULL;
if (ctx.m_avals.m_known_aggs.exists ())
{
unsigned int n = MIN (ctx.m_avals.m_known_aggs.length (), nargs);
for (unsigned int i = 0; i < n; i++)
const ipa_argagg_value_list avl (&ctx.m_avals);
for (unsigned int i = 0; i < nargs; i++)
if (ipa_is_param_used_by_indirect_call (params_summary, i)
&& !ctx.m_avals.m_known_aggs[i].is_empty ())
&& avl.value_for_index_p (i))
{
m_avals.m_known_aggs
= ipa_copy_agg_values (ctx.m_avals.m_known_aggs);
m_avals.m_known_aggs = ctx.m_avals.m_known_aggs.copy ();
break;
}
}
@ -3607,7 +3587,7 @@ ipa_cached_call_context::release ()
/* See if context is initialized at first place. */
if (!m_node)
return;
ipa_release_agg_values (m_avals.m_known_aggs, true);
m_avals.m_known_aggs.release ();
m_avals.m_known_vals.release ();
m_avals.m_known_contexts.release ();
m_inline_param_summary.release ();
@ -3708,28 +3688,59 @@ ipa_call_context::equal_to (const ipa_call_context &ctx)
}
if (m_avals.m_known_aggs.exists () || ctx.m_avals.m_known_aggs.exists ())
{
for (unsigned int i = 0; i < nargs; i++)
unsigned i = 0, j = 0;
while (i < m_avals.m_known_aggs.length ()
|| j < ctx.m_avals.m_known_aggs.length ())
{
if (!ipa_is_param_used_by_indirect_call (params_summary, i))
continue;
if (i >= m_avals.m_known_aggs.length ()
|| m_avals.m_known_aggs[i].is_empty ())
if (i >= m_avals.m_known_aggs.length ())
{
if (i < ctx.m_avals.m_known_aggs.length ()
&& !ctx.m_avals.m_known_aggs[i].is_empty ())
int idx2 = ctx.m_avals.m_known_aggs[j].index;
if (ipa_is_param_used_by_indirect_call (params_summary, idx2))
return false;
j++;
continue;
}
if (i >= ctx.m_avals.m_known_aggs.length ()
|| ctx.m_avals.m_known_aggs[i].is_empty ())
if (j >= ctx.m_avals.m_known_aggs.length ())
{
if (i < m_avals.m_known_aggs.length ()
&& !m_avals.m_known_aggs[i].is_empty ())
int idx1 = m_avals.m_known_aggs[i].index;
if (ipa_is_param_used_by_indirect_call (params_summary, idx1))
return false;
i++;
continue;
}
if (!m_avals.m_known_aggs[i].equal_to (ctx.m_avals.m_known_aggs[i]))
int idx1 = m_avals.m_known_aggs[i].index;
int idx2 = ctx.m_avals.m_known_aggs[j].index;
if (idx1 < idx2)
{
if (ipa_is_param_used_by_indirect_call (params_summary, idx1))
return false;
i++;
continue;
}
if (idx1 > idx2)
{
if (ipa_is_param_used_by_indirect_call (params_summary, idx2))
return false;
j++;
continue;
}
if (!ipa_is_param_used_by_indirect_call (params_summary, idx1))
{
i++;
j++;
continue;
}
if ((m_avals.m_known_aggs[i].unit_offset
!= ctx.m_avals.m_known_aggs[j].unit_offset)
|| (m_avals.m_known_aggs[i].by_ref
!= ctx.m_avals.m_known_aggs[j].by_ref)
|| !operand_equal_p (m_avals.m_known_aggs[i].value,
ctx.m_avals.m_known_aggs[j].value))
return false;
i++;
j++;
}
}
return true;

View File

@ -3612,7 +3612,7 @@ find_constructor_constant_at_offset (tree constructor, HOST_WIDE_INT req_offset)
invariant from a static constructor and if so, return it. Otherwise return
NULL. */
static tree
tree
ipa_find_agg_cst_from_init (tree scalar, HOST_WIDE_INT offset, bool by_ref)
{
if (by_ref)
@ -3632,47 +3632,24 @@ ipa_find_agg_cst_from_init (tree scalar, HOST_WIDE_INT offset, bool by_ref)
return find_constructor_constant_at_offset (DECL_INITIAL (scalar), offset);
}
/* Retrieve value from AGG, a set of known offset/value for an aggregate or
static initializer of SCALAR (which can be NULL) for the given OFFSET or
return NULL if there is none. BY_REF specifies whether the value has to be
passed by reference or by value. If FROM_GLOBAL_CONSTANT is non-NULL, then
the boolean it points to is set to true if the value comes from an
initializer of a constant. */
/* Retrieve value from AGG_JFUNC for the given OFFSET or return NULL if there
is none. BY_REF specifies whether the value has to be passed by reference
or by value. */
tree
ipa_find_agg_cst_for_param (const ipa_agg_value_set *agg, tree scalar,
HOST_WIDE_INT offset, bool by_ref,
bool *from_global_constant)
static tree
ipa_find_agg_cst_from_jfunc_items (struct ipa_agg_jump_function *agg_jfunc,
ipa_node_params *src_info,
cgraph_node *src_node,
HOST_WIDE_INT offset, bool by_ref)
{
struct ipa_agg_value *item;
int i;
if (by_ref != agg_jfunc->by_ref)
return NULL_TREE;
if (scalar)
{
tree res = ipa_find_agg_cst_from_init (scalar, offset, by_ref);
if (res)
{
if (from_global_constant)
*from_global_constant = true;
return res;
}
}
for (const ipa_agg_jf_item &item : agg_jfunc->items)
if (item.offset == offset)
return ipa_agg_value_from_jfunc (src_info, src_node, &item);
if (!agg
|| by_ref != agg->by_ref)
return NULL;
FOR_EACH_VEC_ELT (agg->items, i, item)
if (item->offset == offset)
{
/* Currently we do not have clobber values, return NULL for them once
we do. */
gcc_checking_assert (is_gimple_ip_invariant (item->value));
if (from_global_constant)
*from_global_constant = false;
return item->value;
}
return NULL;
return NULL_TREE;
}
/* Remove a reference to SYMBOL from the list of references of a node given by
@ -3769,24 +3746,19 @@ try_make_edge_direct_simple_call (struct cgraph_edge *ie,
class ipa_node_params *new_root_info)
{
struct cgraph_edge *cs;
tree target;
tree target = NULL_TREE;
bool agg_contents = ie->indirect_info->agg_contents;
tree scalar = ipa_value_from_jfunc (new_root_info, jfunc, target_type);
if (agg_contents)
{
bool from_global_constant;
ipa_agg_value_set agg = ipa_agg_value_set_from_jfunc (new_root_info,
new_root,
&jfunc->agg);
target = ipa_find_agg_cst_for_param (&agg, scalar,
ie->indirect_info->offset,
ie->indirect_info->by_ref,
&from_global_constant);
agg.release ();
if (target
&& !from_global_constant
&& !ie->indirect_info->guaranteed_unmodified)
return NULL;
if (scalar)
target = ipa_find_agg_cst_from_init (scalar, ie->indirect_info->offset,
ie->indirect_info->by_ref);
if (!target && ie->indirect_info->guaranteed_unmodified)
target = ipa_find_agg_cst_from_jfunc_items (&jfunc->agg, new_root_info,
new_root,
ie->indirect_info->offset,
ie->indirect_info->by_ref);
}
else
target = scalar;
@ -3861,15 +3833,14 @@ try_make_edge_direct_virtual_call (struct cgraph_edge *ie,
{
tree vtable;
unsigned HOST_WIDE_INT offset;
tree scalar = (jfunc->type == IPA_JF_CONST) ? ipa_get_jf_constant (jfunc)
: NULL;
ipa_agg_value_set agg = ipa_agg_value_set_from_jfunc (new_root_info,
new_root,
&jfunc->agg);
tree t = ipa_find_agg_cst_for_param (&agg, scalar,
ie->indirect_info->offset,
true);
agg.release ();
tree t = NULL_TREE;
if (jfunc->type == IPA_JF_CONST)
t = ipa_find_agg_cst_from_init (ipa_get_jf_constant (jfunc),
ie->indirect_info->offset, true);
if (!t)
t = ipa_find_agg_cst_from_jfunc_items (&jfunc->agg, new_root_info,
new_root,
ie->indirect_info->offset, true);
if (t && vtable_pointer_value_to_vtable (t, &vtable, &offset))
{
bool can_refer;
@ -6064,21 +6035,4 @@ ipcp_transform_function (struct cgraph_node *node)
}
/* Return true if OTHER describes same agg value. */
bool
ipa_agg_value::equal_to (const ipa_agg_value &other)
{
return offset == other.offset
&& operand_equal_p (value, other.value, 0);
}
/* Destructor also removing individual aggregate values. */
ipa_auto_call_arg_values::~ipa_auto_call_arg_values ()
{
ipa_release_agg_values (m_known_aggs, false);
}
#include "gt-ipa-prop.h"

View File

@ -190,6 +190,8 @@ struct GTY(()) ipa_agg_jump_function
};
class ipcp_transformation;
class ipa_auto_call_arg_values;
class ipa_call_arg_values;
/* Element of a vector describing aggregate values for a number of arguments in
a particular context, be it a call or the aggregate constants that a node is
@ -224,6 +226,8 @@ public:
ipa_argagg_value_list (const vec<ipa_argagg_value> *values)
: m_elts (*values)
{}
ipa_argagg_value_list (const ipa_auto_call_arg_values *aavals);
ipa_argagg_value_list (const ipa_call_arg_values *gavals);
ipa_argagg_value_list (const ipcp_transformation *tinfo);
/* Return the aggregate constant stored for INDEX at UNIT_OFFSET, if it is
@ -243,12 +247,22 @@ public:
const ipa_argagg_value *get_elt (int index, unsigned unit_offset) const;
/* Return the first item describing a constant stored for parameter with
INDEX, regardless of offset or reference, or NULL if there is no such
constant. */
const ipa_argagg_value *get_elt_for_index (int index) const;
/* Return true if there is an aggregate constant referring to a value passed
in or by parameter with INDEX (at any offset, whether by reference or
not). */
bool value_for_index_p (int index) const
{
return !!get_elt_for_index (index);
}
/* Return true if all elements present in OTHER are also present in this
list. */
@ -275,105 +289,6 @@ public:
array_slice<const ipa_argagg_value> m_elts;
};
/* An element in an aggregate part describing a known value at a given offset.
All unlisted positions are assumed to be unknown and all listed values must
fulfill is_gimple_ip_invariant. */
struct ipa_agg_value
{
/* The offset at which the known value is located within the aggregate. */
HOST_WIDE_INT offset;
/* The known constant. */
tree value;
/* Return true if OTHER describes same agg value. */
bool equal_to (const ipa_agg_value &other);
};
/* Structure describing a set of known offset/value for aggregate. */
struct ipa_agg_value_set
{
/* Description of the individual item. */
vec<ipa_agg_value> items;
/* True if the data was passed by reference (as opposed to by value). */
bool by_ref;
/* Return true if OTHER describes same agg values. */
bool equal_to (const ipa_agg_value_set &other)
{
if (by_ref != other.by_ref)
return false;
if (items.length () != other.items.length ())
return false;
for (unsigned int i = 0; i < items.length (); i++)
if (!items[i].equal_to (other.items[i]))
return false;
return true;
}
/* Return true if there is any value for aggregate. */
bool is_empty () const
{
return items.is_empty ();
}
ipa_agg_value_set copy () const
{
ipa_agg_value_set new_copy;
new_copy.items = items.copy ();
new_copy.by_ref = by_ref;
return new_copy;
}
void release ()
{
items.release ();
}
};
/* Return copy of a vec<ipa_agg_value_set>. */
static inline vec<ipa_agg_value_set>
ipa_copy_agg_values (const vec<ipa_agg_value_set> &aggs)
{
vec<ipa_agg_value_set> aggs_copy = vNULL;
if (!aggs.is_empty ())
{
ipa_agg_value_set *agg;
int i;
aggs_copy.reserve_exact (aggs.length ());
FOR_EACH_VEC_ELT (aggs, i, agg)
aggs_copy.quick_push (agg->copy ());
}
return aggs_copy;
}
/* For vec<ipa_agg_value_set>, DO NOT call release(), use below function
instead. Because ipa_agg_value_set contains a field of vector type, we
should release this child vector in each element before reclaiming the
whole vector. */
static inline void
ipa_release_agg_values (vec<ipa_agg_value_set> &aggs,
bool release_vector = true)
{
ipa_agg_value_set *agg;
int i;
FOR_EACH_VEC_ELT (aggs, i, agg)
agg->release ();
if (release_vector)
aggs.release ();
}
/* Information about zero/non-zero bits. */
class GTY(()) ipa_bits
{
@ -551,28 +466,15 @@ ipa_get_jf_ancestor_keep_null (struct ipa_jump_func *jfunc)
class ipa_auto_call_arg_values
{
public:
~ipa_auto_call_arg_values ();
/* If m_known_vals (vector of known "scalar" values) is sufficiantly long,
return its element at INDEX, otherwise return NULL. */
tree safe_sval_at (int index)
{
/* TODO: Assert non-negative index here and test. */
if ((unsigned) index < m_known_vals.length ())
return m_known_vals[index];
return NULL;
}
/* If m_known_aggs is sufficiantly long, return the pointer rto its element
at INDEX, otherwise return NULL. */
ipa_agg_value_set *safe_aggval_at (int index)
{
/* TODO: Assert non-negative index here and test. */
if ((unsigned) index < m_known_aggs.length ())
return &m_known_aggs[index];
return NULL;
}
/* Vector describing known values of parameters. */
auto_vec<tree, 32> m_known_vals;
@ -580,15 +482,22 @@ public:
auto_vec<ipa_polymorphic_call_context, 32> m_known_contexts;
/* Vector describing known aggregate values. */
auto_vec<ipa_agg_value_set, 32> m_known_aggs;
auto_vec<ipa_argagg_value, 32> m_known_aggs;
/* Vector describing known value ranges of arguments. */
auto_vec<value_range, 32> m_known_value_ranges;
};
inline
ipa_argagg_value_list
::ipa_argagg_value_list (const ipa_auto_call_arg_values *aavals)
: m_elts (aavals->m_known_aggs)
{}
/* Class bundling the various potentially known properties about actual
arguments of a particular call. This variant does not deallocate the
bundled data in any way. */
bundled data in any way as the vectors can either be pointing to vectors in
ipa_auto_call_arg_values or be allocated independently. */
class ipa_call_arg_values
{
@ -613,22 +522,11 @@ public:
return its element at INDEX, otherwise return NULL. */
tree safe_sval_at (int index)
{
/* TODO: Assert non-negative index here and test. */
if ((unsigned) index < m_known_vals.length ())
return m_known_vals[index];
return NULL;
}
/* If m_known_aggs is sufficiantly long, return the pointer rto its element
at INDEX, otherwise return NULL. */
ipa_agg_value_set *safe_aggval_at (int index)
{
/* TODO: Assert non-negative index here and test. */
if ((unsigned) index < m_known_aggs.length ())
return &m_known_aggs[index];
return NULL;
}
/* Vector describing known values of parameters. */
vec<tree> m_known_vals = vNULL;
@ -636,12 +534,17 @@ public:
vec<ipa_polymorphic_call_context> m_known_contexts = vNULL;
/* Vector describing known aggregate values. */
vec<ipa_agg_value_set> m_known_aggs = vNULL;
vec<ipa_argagg_value> m_known_aggs = vNULL;
/* Vector describing known value ranges of arguments. */
vec<value_range> m_known_value_ranges = vNULL;
};
inline
ipa_argagg_value_list
::ipa_argagg_value_list (const ipa_call_arg_values *gavals)
: m_elts (gavals->m_known_aggs)
{}
/* Summary describing a single formal parameter. */
@ -1190,9 +1093,6 @@ bool ipa_propagate_indirect_call_infos (struct cgraph_edge *cs,
tree ipa_get_indirect_edge_target (struct cgraph_edge *ie,
ipa_call_arg_values *avals,
bool *speculative);
tree ipa_get_indirect_edge_target (struct cgraph_edge *ie,
ipa_auto_call_arg_values *avals,
bool *speculative);
struct cgraph_edge *ipa_make_edge_direct_to_target (struct cgraph_edge *, tree,
bool speculative = false);
tree ipa_impossible_devirt_target (struct cgraph_edge *, tree);
@ -1204,9 +1104,8 @@ ipa_bits *ipa_get_ipa_bits_for_value (const widest_int &value,
void ipa_analyze_node (struct cgraph_node *);
/* Aggregate jump function related functions. */
tree ipa_find_agg_cst_for_param (const ipa_agg_value_set *agg, tree scalar,
HOST_WIDE_INT offset, bool by_ref,
bool *from_global_constant = NULL);
tree ipa_find_agg_cst_from_init (tree scalar, HOST_WIDE_INT offset,
bool by_ref);
bool ipa_load_from_parm_agg (struct ipa_func_body_info *fbi,
vec<ipa_param_descriptor, va_gc> *descriptors,
gimple *stmt, tree op, int *index_p,
@ -1243,6 +1142,8 @@ void ipcp_read_transformation_summaries (void);
int ipa_get_param_decl_index (class ipa_node_params *, tree);
tree ipa_value_from_jfunc (class ipa_node_params *info,
struct ipa_jump_func *jfunc, tree type);
tree ipa_agg_value_from_jfunc (ipa_node_params *info, cgraph_node *node,
const ipa_agg_jf_item *item);
unsigned int ipcp_transform_function (struct cgraph_node *node);
ipa_polymorphic_call_context ipa_context_from_jfunc (ipa_node_params *,
cgraph_edge *,
@ -1250,9 +1151,10 @@ ipa_polymorphic_call_context ipa_context_from_jfunc (ipa_node_params *,
ipa_jump_func *);
value_range ipa_value_range_from_jfunc (ipa_node_params *, cgraph_edge *,
ipa_jump_func *, tree);
ipa_agg_value_set ipa_agg_value_set_from_jfunc (ipa_node_params *,
cgraph_node *,
ipa_agg_jump_function *);
void ipa_push_agg_values_from_jfunc (ipa_node_params *info, cgraph_node *node,
ipa_agg_jump_function *agg_jfunc,
unsigned dst_index,
vec<ipa_argagg_value> *res);
void ipa_dump_param (FILE *, class ipa_node_params *info, int i);
void ipa_release_body_info (struct ipa_func_body_info *);
tree ipa_get_callee_param_type (struct cgraph_edge *e, int i);