mirror of
git://gcc.gnu.org/git/gcc.git
synced 2025-04-06 05:20:26 +08:00
ipa-reference.c (ipa_obstack): Remove.
* ipa-reference.c (ipa_obstack): Remove. (local_info_obstack, global_info_obstack): New. (add_static_var): We now handle variables only. (mark_address_taken, mark_load, mark_store): New functions based on ... (check_operand): ... remove. (get_asm_stmt_operands): Rename to ... (check_asm_memory_clobber): ... this. Look only for memory clobber. (scan_stmt_for_static_refs): Rewrite. (scan_op_for_static_refs): Rename to ... (scan_initializer_for_static_refs): do not look for VAR_DECL initializers; stop recursion on types and decls. (ipa_init): Use proper obstacks. (analyze_variable): Use scan_initializer_for_static_refs. (init_function_info): Use local obstack. (analyze_function): Simplify. (add_new_function): We don't need visited_nodes obstack. (generate_summary): Use proper obstacks; cleanup after propagation. From-SVN: r140415
This commit is contained in:
parent
791f17147f
commit
ebcf9dc823
@ -1,3 +1,23 @@
|
||||
2008-09-17 Jan Hubicka <jh@suse.cz>
|
||||
|
||||
* ipa-reference.c (ipa_obstack): Remove.
|
||||
(local_info_obstack, global_info_obstack): New.
|
||||
(add_static_var): We now handle variables only.
|
||||
(mark_address_taken, mark_load, mark_store): New functions based on ...
|
||||
(check_operand): ... remove.
|
||||
(get_asm_stmt_operands): Rename to ...
|
||||
(check_asm_memory_clobber): ... this. Look only for memory clobber.
|
||||
(scan_stmt_for_static_refs): Rewrite.
|
||||
(scan_op_for_static_refs): Rename to ...
|
||||
(scan_initializer_for_static_refs): do not look for VAR_DECL
|
||||
initializers; stop recursion on types and decls.
|
||||
(ipa_init): Use proper obstacks.
|
||||
(analyze_variable): Use scan_initializer_for_static_refs.
|
||||
(init_function_info): Use local obstack.
|
||||
(analyze_function): Simplify.
|
||||
(add_new_function): We don't need visited_nodes obstack.
|
||||
(generate_summary): Use proper obstacks; cleanup after propagation.
|
||||
|
||||
2008-09-17 Richard Guenther <rguenther@suse.de>
|
||||
|
||||
PR middle-end/37385
|
||||
|
@ -93,7 +93,11 @@ static bitmap all_module_statics;
|
||||
|
||||
static struct pointer_set_t *visited_nodes;
|
||||
|
||||
static bitmap_obstack ipa_obstack;
|
||||
/* Obstack holding bitmaps of local analysis (live from analysis to
|
||||
propagation) */
|
||||
static bitmap_obstack local_info_obstack;
|
||||
/* Obstack holding global analysis live forever. */
|
||||
static bitmap_obstack global_info_obstack;
|
||||
|
||||
/* Holders of ipa cgraph hooks: */
|
||||
static struct cgraph_node_hook_list *function_insertion_hook_holder;
|
||||
@ -238,6 +242,7 @@ static inline void
|
||||
add_static_var (tree var)
|
||||
{
|
||||
int uid = DECL_UID (var);
|
||||
gcc_assert (TREE_CODE (var) == VAR_DECL);
|
||||
if (!bitmap_bit_p (all_module_statics, uid))
|
||||
{
|
||||
splay_tree_insert (reference_vars_to_consider,
|
||||
@ -281,147 +286,52 @@ has_proper_scope_for_analysis (tree t)
|
||||
return true;
|
||||
}
|
||||
|
||||
/* If T is a VAR_DECL for a static that we are interested in, add the
|
||||
uid to the bitmap. */
|
||||
/* Mark tree T as having address taken. */
|
||||
|
||||
static void
|
||||
check_operand (ipa_reference_local_vars_info_t local,
|
||||
tree t, bool checking_write)
|
||||
mark_address_taken (tree x)
|
||||
{
|
||||
if (!t) return;
|
||||
if (TREE_CODE (x) == VAR_DECL
|
||||
&& module_statics_escape && has_proper_scope_for_analysis (x))
|
||||
bitmap_set_bit (module_statics_escape, DECL_UID (x));
|
||||
}
|
||||
|
||||
if ((TREE_CODE (t) == VAR_DECL || TREE_CODE (t) == FUNCTION_DECL)
|
||||
&& (has_proper_scope_for_analysis (t)))
|
||||
/* Mark load of T. */
|
||||
|
||||
static void
|
||||
mark_load (ipa_reference_local_vars_info_t local,
|
||||
tree t)
|
||||
{
|
||||
if (TREE_CODE (t) == VAR_DECL
|
||||
&& has_proper_scope_for_analysis (t))
|
||||
bitmap_set_bit (local->statics_read, DECL_UID (t));
|
||||
}
|
||||
|
||||
/* Mark store of T. */
|
||||
|
||||
static void
|
||||
mark_store (ipa_reference_local_vars_info_t local,
|
||||
tree t)
|
||||
{
|
||||
if (TREE_CODE (t) == VAR_DECL
|
||||
&& has_proper_scope_for_analysis (t))
|
||||
{
|
||||
if (checking_write)
|
||||
{
|
||||
if (local)
|
||||
bitmap_set_bit (local->statics_written, DECL_UID (t));
|
||||
/* Mark the write so we can tell which statics are
|
||||
readonly. */
|
||||
if (module_statics_written)
|
||||
bitmap_set_bit (module_statics_written, DECL_UID (t));
|
||||
}
|
||||
else if (local)
|
||||
bitmap_set_bit (local->statics_read, DECL_UID (t));
|
||||
if (local)
|
||||
bitmap_set_bit (local->statics_written, DECL_UID (t));
|
||||
/* Mark the write so we can tell which statics are
|
||||
readonly. */
|
||||
if (module_statics_written)
|
||||
bitmap_set_bit (module_statics_written, DECL_UID (t));
|
||||
}
|
||||
}
|
||||
|
||||
/* Examine tree T for references to static variables. All internal
|
||||
references like array references or indirect references are added
|
||||
to the READ_BM. Direct references are added to either READ_BM or
|
||||
WRITE_BM depending on the value of CHECKING_WRITE. */
|
||||
/* Look for memory clobber and set read_all/write_all if present. */
|
||||
|
||||
static void
|
||||
check_tree (ipa_reference_local_vars_info_t local, tree t, bool checking_write)
|
||||
check_asm_memory_clobber (ipa_reference_local_vars_info_t local, gimple stmt)
|
||||
{
|
||||
if ((TREE_CODE (t) == EXC_PTR_EXPR) || (TREE_CODE (t) == FILTER_EXPR))
|
||||
return;
|
||||
|
||||
while (TREE_CODE (t) == REALPART_EXPR
|
||||
|| TREE_CODE (t) == IMAGPART_EXPR
|
||||
|| handled_component_p (t))
|
||||
{
|
||||
if (TREE_CODE (t) == ARRAY_REF)
|
||||
check_operand (local, TREE_OPERAND (t, 1), false);
|
||||
t = TREE_OPERAND (t, 0);
|
||||
}
|
||||
|
||||
/* The bottom of an indirect reference can only be read, not
|
||||
written. So just recurse and whatever we find, check it against
|
||||
the read bitmaps. */
|
||||
|
||||
/* if (INDIRECT_REF_P (t) || TREE_CODE (t) == MEM_REF) */
|
||||
/* FIXME when we have array_ref's of pointers. */
|
||||
if (INDIRECT_REF_P (t))
|
||||
check_tree (local, TREE_OPERAND (t, 0), false);
|
||||
|
||||
if (SSA_VAR_P (t))
|
||||
check_operand (local, t, checking_write);
|
||||
}
|
||||
|
||||
/* Scan tree T to see if there are any addresses taken in within T. */
|
||||
|
||||
static void
|
||||
look_for_address_of (tree t)
|
||||
{
|
||||
if (TREE_CODE (t) == ADDR_EXPR)
|
||||
{
|
||||
tree x = get_base_var (t);
|
||||
if (TREE_CODE (x) == VAR_DECL || TREE_CODE (x) == FUNCTION_DECL)
|
||||
if (has_proper_scope_for_analysis (x) && module_statics_escape)
|
||||
bitmap_set_bit (module_statics_escape, DECL_UID (x));
|
||||
}
|
||||
}
|
||||
|
||||
/* Check to see if T is a read or address of operation on a static var
|
||||
we are interested in analyzing. LOCAL is passed in to get access
|
||||
to its bit vectors. Local is NULL if this is called from a static
|
||||
initializer. */
|
||||
|
||||
static void
|
||||
check_rhs_var (ipa_reference_local_vars_info_t local, tree t)
|
||||
{
|
||||
look_for_address_of (t);
|
||||
|
||||
if (local == NULL)
|
||||
return;
|
||||
|
||||
check_tree(local, t, false);
|
||||
}
|
||||
|
||||
/* Check to see if T is an assignment to a static var we are
|
||||
interested in analyzing. LOCAL is passed in to get access to its bit
|
||||
vectors. */
|
||||
|
||||
static void
|
||||
check_lhs_var (ipa_reference_local_vars_info_t local, tree t)
|
||||
{
|
||||
if (local == NULL)
|
||||
return;
|
||||
|
||||
check_tree(local, t, true);
|
||||
}
|
||||
|
||||
/* This is a scaled down version of get_asm_expr_operands from
|
||||
tree_ssa_operands.c. The version there runs much later and assumes
|
||||
that aliasing information is already available. Here we are just
|
||||
trying to find if the set of inputs and outputs contain references
|
||||
or address of operations to local static variables. FN is the
|
||||
function being analyzed and STMT is the actual asm statement. */
|
||||
|
||||
static void
|
||||
get_asm_stmt_operands (ipa_reference_local_vars_info_t local, gimple stmt)
|
||||
{
|
||||
size_t noutputs = gimple_asm_noutputs (stmt);
|
||||
const char **oconstraints
|
||||
= (const char **) alloca ((noutputs) * sizeof (const char *));
|
||||
size_t i;
|
||||
tree op;
|
||||
const char *constraint;
|
||||
bool allows_mem, allows_reg, is_inout;
|
||||
|
||||
for (i = 0; i < noutputs; i++)
|
||||
{
|
||||
op = gimple_asm_output_op (stmt, i);
|
||||
oconstraints[i] = constraint
|
||||
= TREE_STRING_POINTER (TREE_VALUE (TREE_PURPOSE (op)));
|
||||
parse_output_constraint (&constraint, i, 0, 0,
|
||||
&allows_mem, &allows_reg, &is_inout);
|
||||
|
||||
check_lhs_var (local, TREE_VALUE (op));
|
||||
}
|
||||
|
||||
for (i = 0; i < gimple_asm_ninputs (stmt); i++)
|
||||
{
|
||||
op = gimple_asm_input_op (stmt, i);
|
||||
constraint
|
||||
= TREE_STRING_POINTER (TREE_VALUE (TREE_PURPOSE (op)));
|
||||
parse_input_constraint (&constraint, 0, 0, noutputs, 0,
|
||||
oconstraints, &allows_mem, &allows_reg);
|
||||
|
||||
check_rhs_var (local, TREE_VALUE (op));
|
||||
}
|
||||
|
||||
for (i = 0; i < gimple_asm_nclobbers (stmt); i++)
|
||||
{
|
||||
@ -435,27 +345,14 @@ get_asm_stmt_operands (ipa_reference_local_vars_info_t local, gimple stmt)
|
||||
}
|
||||
}
|
||||
|
||||
/* Check the parameters of a function call from CALLER to CALL_EXPR to
|
||||
see if any of them are static vars. Also check to see if this is
|
||||
either an indirect call, a call outside the compilation unit, or
|
||||
has special attributes that effect the clobbers. The caller
|
||||
parameter is the tree node for the caller and the second operand is
|
||||
the tree node for the entire call expression. */
|
||||
/* Look for external calls and set read_all/write_all correspondingly. */
|
||||
|
||||
static void
|
||||
check_call (ipa_reference_local_vars_info_t local, gimple stmt)
|
||||
{
|
||||
int flags = gimple_call_flags (stmt);
|
||||
tree operand;
|
||||
tree callee_t = gimple_call_fndecl (stmt);
|
||||
enum availability avail = AVAIL_NOT_AVAILABLE;
|
||||
size_t i;
|
||||
|
||||
if ((operand = gimple_call_lhs (stmt)) != NULL)
|
||||
check_lhs_var (local, operand);
|
||||
|
||||
for (i = 0; i < gimple_call_num_args (stmt); i++)
|
||||
check_rhs_var (local, gimple_call_arg (stmt, i));
|
||||
|
||||
if (callee_t)
|
||||
{
|
||||
@ -466,7 +363,9 @@ check_call (ipa_reference_local_vars_info_t local, gimple stmt)
|
||||
if (avail == AVAIL_NOT_AVAILABLE || avail == AVAIL_OVERWRITABLE)
|
||||
if (local)
|
||||
{
|
||||
if (flags & ECF_PURE)
|
||||
if (flags & ECF_CONST)
|
||||
;
|
||||
else if (flags & ECF_PURE)
|
||||
local->calls_read_all = true;
|
||||
else
|
||||
{
|
||||
@ -474,130 +373,78 @@ check_call (ipa_reference_local_vars_info_t local, gimple stmt)
|
||||
local->calls_write_all = true;
|
||||
}
|
||||
}
|
||||
/* TODO: To be able to produce sane results, we should also handle
|
||||
common builtins, in particular throw.
|
||||
Indirect calls hsould be only counted and as inliner is replacing them
|
||||
by direct calls, we can conclude if any indirect calls are left in body */
|
||||
}
|
||||
|
||||
/* TP is the part of the tree currently under the microscope.
|
||||
WALK_SUBTREES is part of the walk_tree api but is unused here.
|
||||
DATA is cgraph_node of the function being walked. */
|
||||
|
||||
/* FIXME: When this is converted to run over SSA form, this code
|
||||
should be converted to use the operand scanner. */
|
||||
|
||||
static tree
|
||||
scan_stmt_for_static_refs (gimple_stmt_iterator *gsip, bool *handled_ops_p,
|
||||
struct walk_stmt_info *data)
|
||||
scan_stmt_for_static_refs (gimple_stmt_iterator *gsip,
|
||||
struct cgraph_node *fn)
|
||||
{
|
||||
struct cgraph_node *fn = (struct cgraph_node *) data->info;
|
||||
gimple stmt = gsi_stmt (*gsip);
|
||||
ipa_reference_local_vars_info_t local = NULL;
|
||||
unsigned int i;
|
||||
bitmap_iterator bi;
|
||||
|
||||
if (fn)
|
||||
local = get_reference_vars_info_from_cgraph (fn)->local;
|
||||
|
||||
if (gimple_loaded_syms (stmt))
|
||||
EXECUTE_IF_SET_IN_BITMAP (gimple_loaded_syms (stmt), 0, i, bi)
|
||||
mark_load (local, referenced_var_lookup (i));
|
||||
if (gimple_stored_syms (stmt))
|
||||
EXECUTE_IF_SET_IN_BITMAP (gimple_stored_syms (stmt), 0, i, bi)
|
||||
mark_store (local, referenced_var_lookup (i));
|
||||
if (gimple_addresses_taken (stmt))
|
||||
EXECUTE_IF_SET_IN_BITMAP (gimple_addresses_taken (stmt), 0, i, bi)
|
||||
mark_address_taken (referenced_var_lookup (i));
|
||||
|
||||
switch (gimple_code (stmt))
|
||||
{
|
||||
case GIMPLE_ASSIGN:
|
||||
{
|
||||
/* First look on the lhs and see what variable is stored to */
|
||||
tree lhs = gimple_assign_lhs (stmt);
|
||||
tree rhs1 = gimple_assign_rhs1 (stmt);
|
||||
tree rhs2 = gimple_assign_rhs2 (stmt);
|
||||
enum tree_code code = gimple_assign_rhs_code (stmt);
|
||||
|
||||
check_lhs_var (local, lhs);
|
||||
|
||||
/* For the purposes of figuring out what the cast affects */
|
||||
|
||||
/* Next check the operands on the rhs to see if they are ok. */
|
||||
switch (TREE_CODE_CLASS (code))
|
||||
{
|
||||
case tcc_binary:
|
||||
case tcc_comparison:
|
||||
check_rhs_var (local, rhs1);
|
||||
check_rhs_var (local, rhs2);
|
||||
break;
|
||||
|
||||
case tcc_unary:
|
||||
case tcc_reference:
|
||||
case tcc_declaration:
|
||||
check_rhs_var (local, rhs1);
|
||||
break;
|
||||
|
||||
case tcc_expression:
|
||||
switch (code)
|
||||
{
|
||||
case ADDR_EXPR:
|
||||
check_rhs_var (local, rhs1);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
*handled_ops_p = true;
|
||||
}
|
||||
break;
|
||||
|
||||
case GIMPLE_LABEL:
|
||||
if (DECL_NONLOCAL (gimple_label_label (stmt)))
|
||||
{
|
||||
/* Target of long jump. */
|
||||
local->calls_read_all = true;
|
||||
local->calls_write_all = true;
|
||||
}
|
||||
break;
|
||||
|
||||
case GIMPLE_CALL:
|
||||
check_call (local, stmt);
|
||||
*handled_ops_p = true;
|
||||
break;
|
||||
|
||||
case GIMPLE_ASM:
|
||||
get_asm_stmt_operands (local, stmt);
|
||||
*handled_ops_p = true;
|
||||
check_asm_memory_clobber (local, stmt);
|
||||
break;
|
||||
|
||||
|
||||
/* We used to check nonlocal labels here and set them as potentially modifying
|
||||
everything. This is not needed, since we can get to nonlocal label only
|
||||
from callee and thus we will get info propagated. */
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/* Call-back to scan GIMPLE operands for static references. This is supposed
|
||||
to work with scan_stmt_for_static_refs so the real call-back data is stored
|
||||
inside a walk_stmt_info struct. Callers using the walk_tree interface must
|
||||
also wrap the call-back data in a walk_stmt_info struct. */
|
||||
/* Call-back to scan variable initializers for static references.
|
||||
Called using walk_tree. */
|
||||
|
||||
static tree
|
||||
scan_op_for_static_refs (tree *tp, int *walk_subtrees, void *data)
|
||||
scan_initializer_for_static_refs (tree *tp, int *walk_subtrees,
|
||||
void *data ATTRIBUTE_UNUSED)
|
||||
{
|
||||
struct walk_stmt_info *wi = (struct walk_stmt_info*) data;
|
||||
struct cgraph_node *fn = (struct cgraph_node *) wi->info;
|
||||
tree t = *tp;
|
||||
ipa_reference_local_vars_info_t local = NULL;
|
||||
if (fn)
|
||||
local = get_reference_vars_info_from_cgraph (fn)->local;
|
||||
|
||||
switch (TREE_CODE (t))
|
||||
if (TREE_CODE (t) == ADDR_EXPR)
|
||||
{
|
||||
case VAR_DECL:
|
||||
if (DECL_INITIAL (t))
|
||||
walk_tree (&DECL_INITIAL (t), scan_op_for_static_refs, data,
|
||||
wi->pset);
|
||||
mark_address_taken (get_base_var (t));
|
||||
*walk_subtrees = 0;
|
||||
break;
|
||||
|
||||
case ADDR_EXPR:
|
||||
/* This case is here to find addresses on rhs of constructors in
|
||||
decl_initial of static variables. */
|
||||
check_rhs_var (local, t);
|
||||
*walk_subtrees = 0;
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
/* Save some cycles by not walking types and declaration as we
|
||||
won't find anything useful there anyway. */
|
||||
else if (IS_TYPE_OR_DECL_P (*tp))
|
||||
*walk_subtrees = 0;
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
@ -687,9 +534,7 @@ propagate_bits (struct cgraph_node *x)
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
gcc_unreachable ();
|
||||
}
|
||||
gcc_unreachable ();
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -763,20 +608,16 @@ merge_callee_local_info (struct cgraph_node *target,
|
||||
static void
|
||||
ipa_init (void)
|
||||
{
|
||||
struct cgraph_node *node;
|
||||
memory_identifier_string = build_string(7, "memory");
|
||||
|
||||
reference_vars_to_consider =
|
||||
splay_tree_new_ggc (splay_tree_compare_ints);
|
||||
|
||||
bitmap_obstack_initialize (&ipa_obstack);
|
||||
module_statics_escape = BITMAP_ALLOC (&ipa_obstack);
|
||||
module_statics_written = BITMAP_ALLOC (&ipa_obstack);
|
||||
all_module_statics = BITMAP_ALLOC (&ipa_obstack);
|
||||
|
||||
/* This will add NODE->DECL to the splay trees. */
|
||||
for (node = cgraph_nodes; node; node = node->next)
|
||||
has_proper_scope_for_analysis (node->decl);
|
||||
bitmap_obstack_initialize (&local_info_obstack);
|
||||
bitmap_obstack_initialize (&global_info_obstack);
|
||||
module_statics_escape = BITMAP_ALLOC (&local_info_obstack);
|
||||
module_statics_written = BITMAP_ALLOC (&local_info_obstack);
|
||||
all_module_statics = BITMAP_ALLOC (&global_info_obstack);
|
||||
|
||||
/* There are some shared nodes, in particular the initializers on
|
||||
static declarations. We do not need to scan them more than once
|
||||
@ -799,7 +640,7 @@ analyze_variable (struct varpool_node *vnode)
|
||||
|
||||
memset (&wi, 0, sizeof (wi));
|
||||
wi.pset = visited_nodes;
|
||||
walk_tree (&DECL_INITIAL (global), scan_op_for_static_refs,
|
||||
walk_tree (&DECL_INITIAL (global), scan_initializer_for_static_refs,
|
||||
&wi, wi.pset);
|
||||
}
|
||||
|
||||
@ -818,8 +659,8 @@ init_function_info (struct cgraph_node *fn)
|
||||
get_function_ann (decl)->reference_vars_info = info;
|
||||
|
||||
info->local = l;
|
||||
l->statics_read = BITMAP_ALLOC (&ipa_obstack);
|
||||
l->statics_written = BITMAP_ALLOC (&ipa_obstack);
|
||||
l->statics_read = BITMAP_ALLOC (&local_info_obstack);
|
||||
l->statics_written = BITMAP_ALLOC (&local_info_obstack);
|
||||
|
||||
return l;
|
||||
}
|
||||
@ -830,15 +671,18 @@ init_function_info (struct cgraph_node *fn)
|
||||
static void
|
||||
analyze_function (struct cgraph_node *fn)
|
||||
{
|
||||
ipa_reference_local_vars_info_t l = init_function_info (fn);
|
||||
tree decl = fn->decl;
|
||||
struct walk_stmt_info wi;
|
||||
struct function *this_cfun = DECL_STRUCT_FUNCTION (decl);
|
||||
basic_block this_block;
|
||||
tree step;
|
||||
|
||||
if (dump_file)
|
||||
fprintf (dump_file, "\n local analysis of %s\n", cgraph_node_name (fn));
|
||||
|
||||
push_cfun (DECL_STRUCT_FUNCTION (decl));
|
||||
current_function_decl = decl;
|
||||
|
||||
init_function_info (fn);
|
||||
FOR_EACH_BB_FN (this_block, this_cfun)
|
||||
{
|
||||
gimple_stmt_iterator gsi;
|
||||
@ -857,39 +701,29 @@ analyze_function (struct cgraph_node *fn)
|
||||
{
|
||||
op = USE_FROM_PTR (use);
|
||||
if (TREE_CODE (op) == ADDR_EXPR)
|
||||
check_rhs_var (l, op);
|
||||
mark_address_taken (get_base_var (op));
|
||||
}
|
||||
}
|
||||
|
||||
memset (&wi, 0, sizeof (wi));
|
||||
wi.info = fn;
|
||||
wi.pset = visited_nodes;
|
||||
for (gsi = gsi_start_bb (this_block); !gsi_end_p (gsi); gsi_next (&gsi))
|
||||
walk_gimple_stmt (&gsi, scan_stmt_for_static_refs,
|
||||
scan_op_for_static_refs, &wi);
|
||||
scan_stmt_for_static_refs (&gsi, fn);
|
||||
}
|
||||
|
||||
/* There may be const decls with interesting right hand sides. */
|
||||
if (DECL_STRUCT_FUNCTION (decl))
|
||||
#ifdef ENABLE_CHECKING
|
||||
/* Verify that all local initializers was expanded by gimplifier. */
|
||||
for (step = DECL_STRUCT_FUNCTION (decl)->local_decls;
|
||||
step;
|
||||
step = TREE_CHAIN (step))
|
||||
{
|
||||
tree step;
|
||||
for (step = DECL_STRUCT_FUNCTION (decl)->local_decls;
|
||||
step;
|
||||
step = TREE_CHAIN (step))
|
||||
{
|
||||
tree var = TREE_VALUE (step);
|
||||
if (TREE_CODE (var) == VAR_DECL
|
||||
&& DECL_INITIAL (var)
|
||||
&& !TREE_STATIC (var))
|
||||
{
|
||||
memset (&wi, 0, sizeof (wi));
|
||||
wi.info = fn;
|
||||
wi.pset = visited_nodes;
|
||||
walk_tree (&DECL_INITIAL (var), scan_op_for_static_refs,
|
||||
&wi, wi.pset);
|
||||
}
|
||||
}
|
||||
tree var = TREE_VALUE (step);
|
||||
if (TREE_CODE (var) == VAR_DECL
|
||||
&& DECL_INITIAL (var)
|
||||
&& !TREE_STATIC (var))
|
||||
gcc_unreachable ();
|
||||
}
|
||||
#endif
|
||||
pop_cfun ();
|
||||
current_function_decl = NULL;
|
||||
}
|
||||
|
||||
/* If FN is avail == AVAIL_OVERWRITABLE, replace the effects bit
|
||||
@ -947,9 +781,7 @@ add_new_function (struct cgraph_node *node, void *data ATTRIBUTE_UNUSED)
|
||||
static declarations. We do not need to scan them more than once
|
||||
since all we would be interested in are the addressof
|
||||
operations. */
|
||||
visited_nodes = pointer_set_create ();
|
||||
analyze_function (node);
|
||||
pointer_set_destroy (visited_nodes);
|
||||
visited_nodes = NULL;
|
||||
}
|
||||
|
||||
@ -969,8 +801,8 @@ generate_summary (void)
|
||||
function_insertion_hook_holder =
|
||||
cgraph_add_function_insertion_hook (&add_new_function, NULL);
|
||||
ipa_init ();
|
||||
module_statics_readonly = BITMAP_ALLOC (&ipa_obstack);
|
||||
bm_temp = BITMAP_ALLOC (&ipa_obstack);
|
||||
module_statics_readonly = BITMAP_ALLOC (&local_info_obstack);
|
||||
bm_temp = BITMAP_ALLOC (&local_info_obstack);
|
||||
|
||||
/* Process all of the variables first. */
|
||||
FOR_EACH_STATIC_INITIALIZER (vnode)
|
||||
@ -1031,11 +863,6 @@ generate_summary (void)
|
||||
{
|
||||
tree var = get_static_decl (index);
|
||||
|
||||
/* Readonly on a function decl is very different from the
|
||||
variable. */
|
||||
if (TREE_CODE (var) == FUNCTION_DECL)
|
||||
continue;
|
||||
|
||||
/* Ignore variables in named sections - changing TREE_READONLY
|
||||
changes the section flags, potentially causing conflicts with
|
||||
other variables in the same named section. */
|
||||
@ -1186,7 +1013,7 @@ propagate (void)
|
||||
node_g->statics_read = all_module_statics;
|
||||
else
|
||||
{
|
||||
node_g->statics_read = BITMAP_ALLOC (&ipa_obstack);
|
||||
node_g->statics_read = BITMAP_ALLOC (&global_info_obstack);
|
||||
bitmap_copy (node_g->statics_read,
|
||||
node_l->statics_read);
|
||||
}
|
||||
@ -1195,7 +1022,7 @@ propagate (void)
|
||||
node_g->statics_written = all_module_statics;
|
||||
else
|
||||
{
|
||||
node_g->statics_written = BITMAP_ALLOC (&ipa_obstack);
|
||||
node_g->statics_written = BITMAP_ALLOC (&global_info_obstack);
|
||||
bitmap_copy (node_g->statics_written,
|
||||
node_l->statics_written);
|
||||
}
|
||||
@ -1346,8 +1173,8 @@ propagate (void)
|
||||
|
||||
/* Create the complimentary sets. These are more useful for
|
||||
certain apis. */
|
||||
node_g->statics_not_read = BITMAP_ALLOC (&ipa_obstack);
|
||||
node_g->statics_not_written = BITMAP_ALLOC (&ipa_obstack);
|
||||
node_g->statics_not_read = BITMAP_ALLOC (&global_info_obstack);
|
||||
node_g->statics_not_written = BITMAP_ALLOC (&global_info_obstack);
|
||||
|
||||
if (node_g->statics_read != all_module_statics)
|
||||
{
|
||||
@ -1367,6 +1194,8 @@ propagate (void)
|
||||
|
||||
for (node = cgraph_nodes; node; node = node->next)
|
||||
{
|
||||
ipa_reference_vars_info_t node_info;
|
||||
node_info = get_reference_vars_info_from_cgraph (node);
|
||||
/* Get rid of the aux information. */
|
||||
|
||||
if (node->aux)
|
||||
@ -1378,7 +1207,18 @@ propagate (void)
|
||||
if (node->analyzed
|
||||
&& (cgraph_function_body_availability (node) == AVAIL_OVERWRITABLE))
|
||||
clean_function (node);
|
||||
else if (node_info)
|
||||
{
|
||||
/* Remove local info we no longer need. */
|
||||
if (node_info->local->statics_read
|
||||
&& node_info->local->statics_read != all_module_statics)
|
||||
BITMAP_FREE (node_info->local->statics_read);
|
||||
if (node_info->local->statics_written
|
||||
&& node_info->local->statics_written != all_module_statics)
|
||||
BITMAP_FREE (node_info->local->statics_written);
|
||||
}
|
||||
}
|
||||
bitmap_obstack_release (&local_info_obstack);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user