726a989a8b
2008-07-28 Richard Guenther <rguenther@suse.de> Merge from gimple-tuples-branch. * ChangeLog.tuples: ChangeLog from gimple-tuples-branch. * gimple.def: New file. * gsstruct.def: Likewise. * gimple-iterator.c: Likewise. * gimple-pretty-print.c: Likewise. * tree-gimple.c: Removed. Merged into ... * gimple.c: ... here. New file. * tree-gimple.h: Removed. Merged into ... * gimple.h: ... here. New file. * Makefile.in: Add dependencies on GIMPLE_H and tree-iterator.h. * configure.ac: Added support for ENABLE_GIMPLE_CHECKING and the --enable-checking=gimple flag. * config.in: Likewise. * configure: Regenerated. * tree-ssa-operands.h: Tuplified. * tree-vrp.c: Likewise. * tree-loop-linear.c: Likewise. * tree-into-ssa.c: Likewise. * tree-ssa-loop-im.c: Likewise. * tree-dump.c: Likewise. * tree-complex.c: Likewise. * cgraphbuild.c: Likewise. * tree-ssa-threadupdate.c: Likewise. * tree-ssa-loop-niter.c: Likewise. * tree-pretty-print.c: Likewise. * tracer.c: Likewise. * gengtype.c: Likewise. * tree-loop-distribution.c: Likewise. * tree-ssa-loop-unswitch.c: Likewise. * cgraph.c: Likewise. * cgraph.h: Likewise. * tree-ssa-loop-manip.c: Likewise. * value-prof.c: Likewise. * tree-ssa-loop-ch.c: Likewise. * tree-tailcall.c: Likewise. * value-prof.h: Likewise. * tree.c: Likewise. * tree.h: Likewise. * tree-pass.h: Likewise. * ipa-cp.c: Likewise. * tree-scalar-evolution.c: Likewise. * tree-scalar-evolution.h: Likewise. * target.h: Likewise. * lambda-mat.c: Likewise. * tree-phinodes.c: Likewise. * diagnostic.h: Likewise. * builtins.c: Likewise. * tree-ssa-alias-warnings.c: Likewise. * cfghooks.c: Likewise. * fold-const.c: Likewise. * cfghooks.h: Likewise. * omp-low.c: Likewise. * tree-ssa-dse.c: Likewise. * ipa-reference.c: Likewise. * tree-ssa-uncprop.c: Likewise. * toplev.c: Likewise. * tree-gimple.c: Likewise. * tree-gimple.h: Likewise. * tree-chrec.c: Likewise. * tree-chrec.h: Likewise. * tree-ssa-sccvn.c: Likewise. * tree-ssa-sccvn.h: Likewise. * cgraphunit.c: Likewise. * tree-ssa-copyrename.c: Likewise. * tree-ssa-ccp.c: Likewise. * tree-ssa-loop-ivopts.c: Likewise. * tree-nomudflap.c: Likewise. * tree-call-cdce.c: Likewise. * ipa-pure-const.c: Likewise. * c-format.c: Likewise. * tree-stdarg.c: Likewise. * tree-ssa-math-opts.c: Likewise. * tree-ssa-dom.c: Likewise. * tree-nrv.c: Likewise. * tree-ssa-propagate.c: Likewise. * ipa-utils.c: Likewise. * tree-ssa-propagate.h: Likewise. * tree-ssa-alias.c: Likewise. * gimple-low.c: Likewise. * tree-ssa-sink.c: Likewise. * ipa-inline.c: Likewise. * c-semantics.c: Likewise. * dwarf2out.c: Likewise. * expr.c: Likewise. * tree-ssa-loop-ivcanon.c: Likewise. * predict.c: Likewise. * tree-ssa-loop.c: Likewise. * tree-parloops.c: Likewise. * tree-ssa-address.c: Likewise. * tree-ssa-ifcombine.c: Likewise. * matrix-reorg.c: Likewise. * c-decl.c: Likewise. * tree-eh.c: Likewise. * c-pretty-print.c: Likewise. * lambda-trans.c: Likewise. * function.c: Likewise. * langhooks.c: Likewise. * ebitmap.h: Likewise. * tree-vectorizer.c: Likewise. * function.h: Likewise. * langhooks.h: Likewise. * tree-vectorizer.h: Likewise. * ipa-type-escape.c: Likewise. * ipa-type-escape.h: Likewise. * domwalk.c: Likewise. * tree-if-conv.c: Likewise. * profile.c: Likewise. * domwalk.h: Likewise. * tree-data-ref.c: Likewise. * tree-data-ref.h: Likewise. * tree-flow-inline.h: Likewise. * tree-affine.c: Likewise. * tree-vect-analyze.c: Likewise. * c-typeck.c: Likewise. * gimplify.c: Likewise. * coretypes.h: Likewise. * tree-ssa-phiopt.c: Likewise. * calls.c: Likewise. * tree-ssa-coalesce.c: Likewise. * tree.def: Likewise. * tree-dfa.c: Likewise. * except.c: Likewise. * except.h: Likewise. * cfgexpand.c: Likewise. * tree-cfgcleanup.c: Likewise. * tree-ssa-pre.c: Likewise. * tree-ssa-live.c: Likewise. * tree-sra.c: Likewise. * tree-ssa-live.h: Likewise. * tree-predcom.c: Likewise. * lambda.h: Likewise. * tree-mudflap.c: Likewise. * ipa-prop.c: Likewise. * print-tree.c: Likewise. * tree-ssa-copy.c: Likewise. * ipa-prop.h: Likewise. * tree-ssa-forwprop.c: Likewise. * ggc-page.c: Likewise. * c-omp.c: Likewise. * tree-ssa-dce.c: Likewise. * tree-vect-patterns.c: Likewise. * tree-ssa-ter.c: Likewise. * tree-nested.c: Likewise. * tree-ssa.c: Likewise. * lambda-code.c: Likewise. * tree-ssa-loop-prefetch.c: Likewise. * tree-inline.c: Likewise. * tree-inline.h: Likewise. * tree-iterator.c: Likewise. * tree-optimize.c: Likewise. * tree-ssa-phiprop.c: Likewise. * tree-vect-transform.c: Likewise. * tree-object-size.c: Likewise. * tree-outof-ssa.c: Likewise. * cfgloop.c: Likewise. * system.h: Likewise. * tree-profile.c: Likewise. * cfgloop.h: Likewise. * c-gimplify.c: Likewise. * c-common.c: Likewise. * tree-vect-generic.c: Likewise. * tree-flow.h: Likewise. * c-common.h: Likewise. * basic-block.h: Likewise. * tree-ssa-structalias.c: Likewise. * tree-switch-conversion.c: Likewise. * tree-ssa-structalias.h: Likewise. * tree-cfg.c: Likewise. * passes.c: Likewise. * ipa-struct-reorg.c: Likewise. * ipa-struct-reorg.h: Likewise. * tree-ssa-reassoc.c: Likewise. * cfgrtl.c: Likewise. * varpool.c: Likewise. * stmt.c: Likewise. * tree-ssanames.c: Likewise. * tree-ssa-threadedge.c: Likewise. * langhooks-def.h: Likewise. * tree-ssa-operands.c: Likewise. * config/alpha/alpha.c: Likewise. * config/frv/frv.c: Likewise. * config/s390/s390.c: Likewise. * config/m32c/m32c.c: Likewise. * config/m32c/m32c-protos.h: Likewise. * config/spu/spu.c: Likewise. * config/sparc/sparc.c: Likewise. * config/i386/i386.c: Likewise. * config/sh/sh.c: Likewise. * config/xtensa/xtensa.c: Likewise. * config/stormy16/stormy16.c: Likewise. * config/ia64/ia64.c: Likewise. * config/rs6000/rs6000.c: Likewise. * config/pa/pa.c: Likewise. * config/mips/mips.c: Likewise. From-SVN: r138207
227 lines
6.1 KiB
C
227 lines
6.1 KiB
C
/* Utilities for ipa analysis.
|
||
Copyright (C) 2005, 2007 Free Software Foundation, Inc.
|
||
Contributed by Kenneth Zadeck <zadeck@naturalbridge.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/>. */
|
||
|
||
#include "config.h"
|
||
#include "system.h"
|
||
#include "coretypes.h"
|
||
#include "tm.h"
|
||
#include "tree.h"
|
||
#include "tree-flow.h"
|
||
#include "tree-inline.h"
|
||
#include "tree-pass.h"
|
||
#include "langhooks.h"
|
||
#include "pointer-set.h"
|
||
#include "ggc.h"
|
||
#include "ipa-utils.h"
|
||
#include "ipa-reference.h"
|
||
#include "c-common.h"
|
||
#include "gimple.h"
|
||
#include "cgraph.h"
|
||
#include "output.h"
|
||
#include "flags.h"
|
||
#include "timevar.h"
|
||
#include "diagnostic.h"
|
||
#include "langhooks.h"
|
||
|
||
/* Debugging function for postorder and inorder code. NOTE is a string
|
||
that is printed before the nodes are printed. ORDER is an array of
|
||
cgraph_nodes that has COUNT useful nodes in it. */
|
||
|
||
void
|
||
ipa_utils_print_order (FILE* out,
|
||
const char * note,
|
||
struct cgraph_node** order,
|
||
int count)
|
||
{
|
||
int i;
|
||
fprintf (out, "\n\n ordered call graph: %s\n", note);
|
||
|
||
for (i = count - 1; i >= 0; i--)
|
||
dump_cgraph_node(dump_file, order[i]);
|
||
fprintf (out, "\n");
|
||
fflush(out);
|
||
}
|
||
|
||
|
||
struct searchc_env {
|
||
struct cgraph_node **stack;
|
||
int stack_size;
|
||
struct cgraph_node **result;
|
||
int order_pos;
|
||
splay_tree nodes_marked_new;
|
||
bool reduce;
|
||
int count;
|
||
};
|
||
|
||
/* This is an implementation of Tarjan's strongly connected region
|
||
finder as reprinted in Aho Hopcraft and Ullman's The Design and
|
||
Analysis of Computer Programs (1975) pages 192-193. This version
|
||
has been customized for cgraph_nodes. The env parameter is because
|
||
it is recursive and there are no nested functions here. This
|
||
function should only be called from itself or
|
||
ipa_utils_reduced_inorder. ENV is a stack env and would be
|
||
unnecessary if C had nested functions. V is the node to start
|
||
searching from. */
|
||
|
||
static void
|
||
searchc (struct searchc_env* env, struct cgraph_node *v)
|
||
{
|
||
struct cgraph_edge *edge;
|
||
struct ipa_dfs_info *v_info = (struct ipa_dfs_info *) v->aux;
|
||
|
||
/* mark node as old */
|
||
v_info->new_node = false;
|
||
splay_tree_remove (env->nodes_marked_new, v->uid);
|
||
|
||
v_info->dfn_number = env->count;
|
||
v_info->low_link = env->count;
|
||
env->count++;
|
||
env->stack[(env->stack_size)++] = v;
|
||
v_info->on_stack = true;
|
||
|
||
for (edge = v->callees; edge; edge = edge->next_callee)
|
||
{
|
||
struct ipa_dfs_info * w_info;
|
||
struct cgraph_node *w = edge->callee;
|
||
/* Bypass the clones and only look at the master node. Skip
|
||
external and other bogus nodes. */
|
||
w = cgraph_master_clone (w);
|
||
if (w && w->aux)
|
||
{
|
||
w_info = (struct ipa_dfs_info *) w->aux;
|
||
if (w_info->new_node)
|
||
{
|
||
searchc (env, w);
|
||
v_info->low_link =
|
||
(v_info->low_link < w_info->low_link) ?
|
||
v_info->low_link : w_info->low_link;
|
||
}
|
||
else
|
||
if ((w_info->dfn_number < v_info->dfn_number)
|
||
&& (w_info->on_stack))
|
||
v_info->low_link =
|
||
(w_info->dfn_number < v_info->low_link) ?
|
||
w_info->dfn_number : v_info->low_link;
|
||
}
|
||
}
|
||
|
||
|
||
if (v_info->low_link == v_info->dfn_number)
|
||
{
|
||
struct cgraph_node *last = NULL;
|
||
struct cgraph_node *x;
|
||
struct ipa_dfs_info *x_info;
|
||
do {
|
||
x = env->stack[--(env->stack_size)];
|
||
x_info = (struct ipa_dfs_info *) x->aux;
|
||
x_info->on_stack = false;
|
||
|
||
if (env->reduce)
|
||
{
|
||
x_info->next_cycle = last;
|
||
last = x;
|
||
}
|
||
else
|
||
env->result[env->order_pos++] = x;
|
||
}
|
||
while (v != x);
|
||
if (env->reduce)
|
||
env->result[env->order_pos++] = v;
|
||
}
|
||
}
|
||
|
||
/* Topsort the call graph by caller relation. Put the result in ORDER.
|
||
|
||
The REDUCE flag is true if you want the cycles reduced to single
|
||
nodes. Only consider nodes that have the output bit set. */
|
||
|
||
int
|
||
ipa_utils_reduced_inorder (struct cgraph_node **order,
|
||
bool reduce, bool allow_overwritable)
|
||
{
|
||
struct cgraph_node *node;
|
||
struct searchc_env env;
|
||
splay_tree_node result;
|
||
env.stack = XCNEWVEC (struct cgraph_node *, cgraph_n_nodes);
|
||
env.stack_size = 0;
|
||
env.result = order;
|
||
env.order_pos = 0;
|
||
env.nodes_marked_new = splay_tree_new (splay_tree_compare_ints, 0, 0);
|
||
env.count = 1;
|
||
env.reduce = reduce;
|
||
|
||
for (node = cgraph_nodes; node; node = node->next)
|
||
if ((node->analyzed)
|
||
&& (cgraph_is_master_clone (node)
|
||
|| (allow_overwritable
|
||
&& (cgraph_function_body_availability (node) ==
|
||
AVAIL_OVERWRITABLE))))
|
||
{
|
||
/* Reuse the info if it is already there. */
|
||
struct ipa_dfs_info *info = (struct ipa_dfs_info *) node->aux;
|
||
if (!info)
|
||
info = XCNEW (struct ipa_dfs_info);
|
||
info->new_node = true;
|
||
info->on_stack = false;
|
||
info->next_cycle = NULL;
|
||
node->aux = info;
|
||
|
||
splay_tree_insert (env.nodes_marked_new,
|
||
(splay_tree_key)node->uid,
|
||
(splay_tree_value)node);
|
||
}
|
||
else
|
||
node->aux = NULL;
|
||
result = splay_tree_min (env.nodes_marked_new);
|
||
while (result)
|
||
{
|
||
node = (struct cgraph_node *)result->value;
|
||
searchc (&env, node);
|
||
result = splay_tree_min (env.nodes_marked_new);
|
||
}
|
||
splay_tree_delete (env.nodes_marked_new);
|
||
free (env.stack);
|
||
|
||
return env.order_pos;
|
||
}
|
||
|
||
|
||
/* Given a memory reference T, will return the variable at the bottom
|
||
of the access. Unlike get_base_address, this will recurse thru
|
||
INDIRECT_REFS. */
|
||
|
||
tree
|
||
get_base_var (tree t)
|
||
{
|
||
if ((TREE_CODE (t) == EXC_PTR_EXPR) || (TREE_CODE (t) == FILTER_EXPR))
|
||
return t;
|
||
|
||
while (!SSA_VAR_P (t)
|
||
&& (!CONSTANT_CLASS_P (t))
|
||
&& TREE_CODE (t) != LABEL_DECL
|
||
&& TREE_CODE (t) != FUNCTION_DECL
|
||
&& TREE_CODE (t) != CONST_DECL)
|
||
{
|
||
t = TREE_OPERAND (t, 0);
|
||
}
|
||
return t;
|
||
}
|
||
|