PR libstdc++/56278
[official-gcc.git] / gcc / ipa-reference.c
blob98c2056839ca70d87fef86d0a32228316525d538
1 /* Callgraph based analysis of static variables.
2 Copyright (C) 2004-2013 Free Software Foundation, Inc.
3 Contributed by Kenneth Zadeck <zadeck@naturalbridge.com>
5 This file is part of GCC.
7 GCC is free software; you can redistribute it and/or modify it under
8 the terms of the GNU General Public License as published by the Free
9 Software Foundation; either version 3, or (at your option) any later
10 version.
12 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
15 for more details.
17 You should have received a copy of the GNU General Public License
18 along with GCC; see the file COPYING3. If not see
19 <http://www.gnu.org/licenses/>. */
21 /* This file gathers information about how variables whose scope is
22 confined to the compilation unit are used.
24 The transitive call site specific clobber effects are computed
25 for the variables whose scope is contained within this compilation
26 unit.
28 First each function and static variable initialization is analyzed
29 to determine which local static variables are either read, written,
30 or have their address taken. Any local static that has its address
31 taken is removed from consideration. Once the local read and
32 writes are determined, a transitive closure of this information is
33 performed over the call graph to determine the worst case set of
34 side effects of each call. In later parts of the compiler, these
35 local and global sets are examined to make the call clobbering less
36 traumatic, promote some statics to registers, and improve aliasing
37 information. */
39 #include "config.h"
40 #include "system.h"
41 #include "coretypes.h"
42 #include "tm.h"
43 #include "tree.h"
44 #include "tree-flow.h"
45 #include "tree-inline.h"
46 #include "tree-pass.h"
47 #include "pointer-set.h"
48 #include "splay-tree.h"
49 #include "ggc.h"
50 #include "ipa-utils.h"
51 #include "ipa-reference.h"
52 #include "gimple.h"
53 #include "cgraph.h"
54 #include "flags.h"
55 #include "diagnostic.h"
56 #include "data-streamer.h"
57 #include "lto-streamer.h"
59 static void remove_node_data (struct cgraph_node *node,
60 void *data ATTRIBUTE_UNUSED);
61 static void duplicate_node_data (struct cgraph_node *src,
62 struct cgraph_node *dst,
63 void *data ATTRIBUTE_UNUSED);
65 /* The static variables defined within the compilation unit that are
66 loaded or stored directly by function that owns this structure. */
68 struct ipa_reference_local_vars_info_d
70 bitmap statics_read;
71 bitmap statics_written;
74 /* Statics that are read and written by some set of functions. The
75 local ones are based on the loads and stores local to the function.
76 The global ones are based on the local info as well as the
77 transitive closure of the functions that are called. */
79 struct ipa_reference_global_vars_info_d
81 bitmap statics_read;
82 bitmap statics_written;
85 /* Information we save about every function after ipa-reference is completed. */
87 struct ipa_reference_optimization_summary_d
89 bitmap statics_not_read;
90 bitmap statics_not_written;
93 typedef struct ipa_reference_local_vars_info_d *ipa_reference_local_vars_info_t;
94 typedef struct ipa_reference_global_vars_info_d *ipa_reference_global_vars_info_t;
95 typedef struct ipa_reference_optimization_summary_d *ipa_reference_optimization_summary_t;
97 struct ipa_reference_vars_info_d
99 struct ipa_reference_local_vars_info_d local;
100 struct ipa_reference_global_vars_info_d global;
103 typedef struct ipa_reference_vars_info_d *ipa_reference_vars_info_t;
105 /* This splay tree contains all of the static variables that are
106 being considered by the compilation level alias analysis. */
107 static splay_tree reference_vars_to_consider;
109 /* Set of all interesting module statics. A bit is set for every module
110 static we are considering. This is added to the local info when asm
111 code is found that clobbers all memory. */
112 static bitmap all_module_statics;
114 /* Obstack holding bitmaps of local analysis (live from analysis to
115 propagation) */
116 static bitmap_obstack local_info_obstack;
117 /* Obstack holding global analysis live forever. */
118 static bitmap_obstack optimization_summary_obstack;
120 /* Holders of ipa cgraph hooks: */
121 static struct cgraph_2node_hook_list *node_duplication_hook_holder;
122 static struct cgraph_node_hook_list *node_removal_hook_holder;
124 /* Vector where the reference var infos are actually stored.
125 Indexed by UID of call graph nodes. */
126 static vec<ipa_reference_vars_info_t> ipa_reference_vars_vector;
128 static vec<ipa_reference_optimization_summary_t> ipa_reference_opt_sum_vector;
130 /* Return the ipa_reference_vars structure starting from the cgraph NODE. */
131 static inline ipa_reference_vars_info_t
132 get_reference_vars_info (struct cgraph_node *node)
134 if (!ipa_reference_vars_vector.exists ()
135 || ipa_reference_vars_vector.length () <= (unsigned int) node->uid)
136 return NULL;
137 return ipa_reference_vars_vector[node->uid];
140 /* Return the ipa_reference_vars structure starting from the cgraph NODE. */
141 static inline ipa_reference_optimization_summary_t
142 get_reference_optimization_summary (struct cgraph_node *node)
144 if (!ipa_reference_opt_sum_vector.exists ()
145 || (ipa_reference_opt_sum_vector.length () <= (unsigned int) node->uid))
146 return NULL;
147 return ipa_reference_opt_sum_vector[node->uid];
150 /* Return the ipa_reference_vars structure starting from the cgraph NODE. */
151 static inline void
152 set_reference_vars_info (struct cgraph_node *node,
153 ipa_reference_vars_info_t info)
155 if (!ipa_reference_vars_vector.exists ()
156 || ipa_reference_vars_vector.length () <= (unsigned int) node->uid)
157 ipa_reference_vars_vector.safe_grow_cleared (node->uid + 1);
158 ipa_reference_vars_vector[node->uid] = info;
161 /* Return the ipa_reference_vars structure starting from the cgraph NODE. */
162 static inline void
163 set_reference_optimization_summary (struct cgraph_node *node,
164 ipa_reference_optimization_summary_t info)
166 if (!ipa_reference_opt_sum_vector.exists ()
167 || (ipa_reference_opt_sum_vector.length () <= (unsigned int) node->uid))
168 ipa_reference_opt_sum_vector.safe_grow_cleared (node->uid + 1);
169 ipa_reference_opt_sum_vector[node->uid] = info;
172 /* Return a bitmap indexed by DECL_UID for the static variables that
173 are *not* read during the execution of the function FN. Returns
174 NULL if no data is available. */
176 bitmap
177 ipa_reference_get_not_read_global (struct cgraph_node *fn)
179 ipa_reference_optimization_summary_t info =
180 get_reference_optimization_summary (cgraph_function_node (fn, NULL));
181 if (info)
182 return info->statics_not_read;
183 else if (flags_from_decl_or_type (fn->symbol.decl) & ECF_LEAF)
184 return all_module_statics;
185 else
186 return NULL;
189 /* Return a bitmap indexed by DECL_UID for the static variables that
190 are *not* written during the execution of the function FN. Note
191 that variables written may or may not be read during the function
192 call. Returns NULL if no data is available. */
194 bitmap
195 ipa_reference_get_not_written_global (struct cgraph_node *fn)
197 ipa_reference_optimization_summary_t info =
198 get_reference_optimization_summary (fn);
199 if (info)
200 return info->statics_not_written;
201 else if (flags_from_decl_or_type (fn->symbol.decl) & ECF_LEAF)
202 return all_module_statics;
203 else
204 return NULL;
209 /* Add VAR to all_module_statics and the two
210 reference_vars_to_consider* sets. */
212 static inline void
213 add_static_var (tree var)
215 int uid = DECL_UID (var);
216 gcc_assert (TREE_CODE (var) == VAR_DECL);
217 if (dump_file)
218 splay_tree_insert (reference_vars_to_consider,
219 uid, (splay_tree_value)var);
220 bitmap_set_bit (all_module_statics, uid);
223 /* Return true if the variable T is the right kind of static variable to
224 perform compilation unit scope escape analysis. */
226 static inline bool
227 is_proper_for_analysis (tree t)
229 /* If the variable has the "used" attribute, treat it as if it had a
230 been touched by the devil. */
231 if (DECL_PRESERVE_P (t))
232 return false;
234 /* Do not want to do anything with volatile except mark any
235 function that uses one to be not const or pure. */
236 if (TREE_THIS_VOLATILE (t))
237 return false;
239 /* We do not need to analyze readonly vars, we already know they do not
240 alias. */
241 if (TREE_READONLY (t))
242 return false;
244 /* This is a variable we care about. Check if we have seen it
245 before, and if not add it the set of variables we care about. */
246 if (all_module_statics
247 && !bitmap_bit_p (all_module_statics, DECL_UID (t)))
248 add_static_var (t);
250 return true;
253 /* Lookup the tree node for the static variable that has UID and
254 convert the name to a string for debugging. */
256 static const char *
257 get_static_name (int index)
259 splay_tree_node stn =
260 splay_tree_lookup (reference_vars_to_consider, index);
261 return fndecl_name ((tree)(stn->value));
264 /* Dump a set of static vars to FILE. */
265 static void
266 dump_static_vars_set_to_file (FILE *f, bitmap set)
268 unsigned int index;
269 bitmap_iterator bi;
270 if (set == NULL)
271 return;
272 else if (set == all_module_statics)
273 fprintf (f, "ALL");
274 else
275 EXECUTE_IF_SET_IN_BITMAP (set, 0, index, bi)
277 fprintf (f, "%s ", get_static_name (index));
281 /* Compute X |= Y, taking into account the possibility that
282 either X or Y is already the maximum set.
283 Return true if X is the maximum set after taking the union with Y. */
285 static bool
286 union_static_var_sets (bitmap &x, bitmap y)
288 if (x != all_module_statics)
290 if (y == all_module_statics)
292 BITMAP_FREE (x);
293 x = all_module_statics;
295 else if (bitmap_ior_into (x, y))
297 /* The union may have reduced X to the maximum set.
298 In that case, we want to make that visible explicitly.
299 Even though bitmap_equal_p can be very expensive, it
300 turns out to be an overall win to check this here for
301 an LTO bootstrap of GCC itself. Liberally extrapoliate
302 that result to be applicable to all cases. */
303 if (bitmap_equal_p (x, all_module_statics))
305 BITMAP_FREE (x);
306 x = all_module_statics;
310 return x == all_module_statics;
313 /* Compute X &= Y, taking into account the possibility that
314 X may become the maximum set. */
316 static bool
317 intersect_static_var_sets (bitmap &x, bitmap y)
319 if (x != all_module_statics)
321 bitmap_and_into (x, y);
322 /* As with union_static_var_sets, reducing to the maximum
323 set as early as possible is an overall win. */
324 if (bitmap_equal_p (x, all_module_statics))
326 BITMAP_FREE (x);
327 x = all_module_statics;
330 return x == all_module_statics;
333 /* Return a copy of SET on the bitmap obstack containing SET.
334 But if SET is NULL or the maximum set, return that instead. */
336 static bitmap
337 copy_static_var_set (bitmap set)
339 if (set == NULL || set == all_module_statics)
340 return set;
341 bitmap_obstack *o = set->obstack;
342 gcc_checking_assert (o);
343 bitmap copy = BITMAP_ALLOC (o);
344 bitmap_copy (copy, set);
345 return copy;
348 /* Compute the union all of the statics read and written by every callee of X
349 into X_GLOBAL->statics_read and X_GLOBAL->statics_written. X_GLOBAL is
350 actually the set representing the cycle containing X. If the read and
351 written sets of X_GLOBAL has been reduced to the maximum set, we don't
352 have to look at the remaining callees. */
354 static void
355 propagate_bits (ipa_reference_global_vars_info_t x_global, struct cgraph_node *x)
357 struct cgraph_edge *e;
358 bool read_all = x_global->statics_read == all_module_statics;
359 bool write_all = x_global->statics_written == all_module_statics;
360 for (e = x->callees;
361 e && !(read_all && write_all);
362 e = e->next_callee)
364 enum availability avail;
365 struct cgraph_node *y = cgraph_function_node (e->callee, &avail);
366 if (!y)
367 continue;
369 /* Only look into nodes we can propagate something. */
370 int flags = flags_from_decl_or_type (y->symbol.decl);
371 if (avail > AVAIL_OVERWRITABLE
372 || (avail == AVAIL_OVERWRITABLE && (flags & ECF_LEAF)))
374 if (get_reference_vars_info (y))
376 ipa_reference_vars_info_t y_info = get_reference_vars_info (y);
377 ipa_reference_global_vars_info_t y_global = &y_info->global;
379 /* Calls in the current cycle do not have their global set
380 computed yet (but everything else does because we're
381 visiting nodes in topological order). */
382 if (!y_global->statics_read)
383 continue;
385 /* If the function is const, it reads no memory even if it
386 seems so to local analysis. */
387 if (flags & ECF_CONST)
388 continue;
390 union_static_var_sets (x_global->statics_read,
391 y_global->statics_read);
393 /* If the function is pure, it has no stores even if it
394 seems so to local analysis. If we cannot return from
395 the function, we can safely ignore the call. */
396 if ((flags & ECF_PURE)
397 || cgraph_edge_cannot_lead_to_return (e))
398 continue;
400 union_static_var_sets (x_global->statics_written,
401 y_global->statics_written);
403 else
404 gcc_unreachable ();
409 /* The init routine for analyzing global static variable usage. See
410 comments at top for description. */
411 static void
412 ipa_init (void)
414 static bool init_p = false;
416 if (init_p)
417 return;
419 init_p = true;
421 if (dump_file)
422 reference_vars_to_consider = splay_tree_new (splay_tree_compare_ints, 0, 0);
424 bitmap_obstack_initialize (&local_info_obstack);
425 bitmap_obstack_initialize (&optimization_summary_obstack);
426 all_module_statics = BITMAP_ALLOC (&optimization_summary_obstack);
428 node_removal_hook_holder =
429 cgraph_add_node_removal_hook (&remove_node_data, NULL);
430 node_duplication_hook_holder =
431 cgraph_add_node_duplication_hook (&duplicate_node_data, NULL);
435 /* Set up the persistent info for FN. */
437 static ipa_reference_local_vars_info_t
438 init_function_info (struct cgraph_node *fn)
440 ipa_reference_vars_info_t info
441 = XCNEW (struct ipa_reference_vars_info_d);
443 /* Add the info to the tree's annotation. */
444 set_reference_vars_info (fn, info);
446 info->local.statics_read = BITMAP_ALLOC (&local_info_obstack);
447 info->local.statics_written = BITMAP_ALLOC (&local_info_obstack);
449 return &info->local;
453 /* This is the main routine for finding the reference patterns for
454 global variables within a function FN. */
456 static void
457 analyze_function (struct cgraph_node *fn)
459 ipa_reference_local_vars_info_t local;
460 struct ipa_ref *ref;
461 int i;
462 tree var;
464 local = init_function_info (fn);
465 for (i = 0; ipa_ref_list_reference_iterate (&fn->symbol.ref_list, i, ref); i++)
467 if (!is_a <varpool_node> (ref->referred))
468 continue;
469 var = ipa_ref_varpool_node (ref)->symbol.decl;
470 if (!is_proper_for_analysis (var))
471 continue;
472 switch (ref->use)
474 case IPA_REF_LOAD:
475 bitmap_set_bit (local->statics_read, DECL_UID (var));
476 break;
477 case IPA_REF_STORE:
478 if (ipa_ref_cannot_lead_to_return (ref))
479 break;
480 bitmap_set_bit (local->statics_written, DECL_UID (var));
481 break;
482 case IPA_REF_ADDR:
483 break;
487 if (cgraph_node_cannot_return (fn))
488 bitmap_clear (local->statics_written);
492 /* Called when new clone is inserted to callgraph late. */
494 static void
495 duplicate_node_data (struct cgraph_node *src, struct cgraph_node *dst,
496 void *data ATTRIBUTE_UNUSED)
498 ipa_reference_optimization_summary_t ginfo;
499 ipa_reference_optimization_summary_t dst_ginfo;
501 ginfo = get_reference_optimization_summary (src);
502 if (!ginfo)
503 return;
504 dst_ginfo = XCNEW (struct ipa_reference_optimization_summary_d);
505 set_reference_optimization_summary (dst, dst_ginfo);
506 dst_ginfo->statics_not_read =
507 copy_static_var_set (ginfo->statics_not_read);
508 dst_ginfo->statics_not_written =
509 copy_static_var_set (ginfo->statics_not_written);
512 /* Called when node is removed. */
514 static void
515 remove_node_data (struct cgraph_node *node, void *data ATTRIBUTE_UNUSED)
517 ipa_reference_optimization_summary_t ginfo;
518 ginfo = get_reference_optimization_summary (node);
519 if (ginfo)
521 if (ginfo->statics_not_read
522 && ginfo->statics_not_read != all_module_statics)
523 BITMAP_FREE (ginfo->statics_not_read);
525 if (ginfo->statics_not_written
526 && ginfo->statics_not_written != all_module_statics)
527 BITMAP_FREE (ginfo->statics_not_written);
528 free (ginfo);
529 set_reference_optimization_summary (node, NULL);
533 /* Analyze each function in the cgraph to see which global or statics
534 are read or written. */
536 static void
537 generate_summary (void)
539 struct cgraph_node *node;
540 unsigned int index;
541 bitmap_iterator bi;
543 ipa_init ();
545 /* Process all of the functions next. */
546 FOR_EACH_DEFINED_FUNCTION (node)
547 analyze_function (node);
549 if (dump_file)
550 EXECUTE_IF_SET_IN_BITMAP (all_module_statics, 0, index, bi)
552 fprintf (dump_file, "\nPromotable global:%s (uid=%u)\n",
553 get_static_name (index), index);
556 if (dump_file)
557 FOR_EACH_DEFINED_FUNCTION (node)
558 if (cgraph_function_body_availability (node) >= AVAIL_OVERWRITABLE)
560 ipa_reference_local_vars_info_t l;
561 unsigned int index;
562 bitmap_iterator bi;
564 l = &get_reference_vars_info (node)->local;
565 fprintf (dump_file,
566 "\nFunction name:%s/%i:",
567 cgraph_node_asm_name (node), node->symbol.order);
568 fprintf (dump_file, "\n locals read: ");
569 if (l->statics_read)
570 EXECUTE_IF_SET_IN_BITMAP (l->statics_read,
571 0, index, bi)
573 fprintf (dump_file, "%s ",
574 get_static_name (index));
576 fprintf (dump_file, "\n locals written: ");
577 if (l->statics_written)
578 EXECUTE_IF_SET_IN_BITMAP (l->statics_written,
579 0, index, bi)
581 fprintf(dump_file, "%s ",
582 get_static_name (index));
587 /* Set READ_ALL/WRITE_ALL based on decl flags of NODE. */
589 static void
590 read_write_all_from_decl (struct cgraph_node *node,
591 bool &read_all, bool &write_all)
593 tree decl = node->symbol.decl;
594 int flags = flags_from_decl_or_type (decl);
595 if ((flags & ECF_LEAF)
596 && cgraph_function_body_availability (node) <= AVAIL_OVERWRITABLE)
598 else if (flags & ECF_CONST)
600 else if ((flags & ECF_PURE)
601 || cgraph_node_cannot_return (node))
603 read_all = true;
604 if (dump_file && (dump_flags & TDF_DETAILS))
605 fprintf (dump_file, " %s/%i -> read all\n",
606 cgraph_node_asm_name (node), node->symbol.order);
608 else
610 /* TODO: To be able to produce sane results, we should also handle
611 common builtins, in particular throw. */
612 read_all = true;
613 write_all = true;
614 if (dump_file && (dump_flags & TDF_DETAILS))
615 fprintf (dump_file, " %s/%i -> read all, write all\n",
616 cgraph_node_asm_name (node), node->symbol.order);
620 /* Set READ_ALL/WRITE_ALL based on decl flags of NODE or any member
621 in the cycle of NODE. */
623 static void
624 get_read_write_all_from_node (struct cgraph_node *node,
625 bool &read_all, bool &write_all)
627 struct cgraph_edge *e, *ie;
629 /* When function is overwritable, we can not assume anything. */
630 if (cgraph_function_body_availability (node) <= AVAIL_OVERWRITABLE)
631 read_write_all_from_decl (node, read_all, write_all);
633 for (e = node->callees;
634 e && !(read_all && write_all);
635 e = e->next_callee)
637 enum availability avail;
638 struct cgraph_node *callee = cgraph_function_node (e->callee, &avail);
639 gcc_checking_assert (callee);
640 if (avail <= AVAIL_OVERWRITABLE)
641 read_write_all_from_decl (callee, read_all, write_all);
644 for (ie = node->indirect_calls;
645 ie && !(read_all && write_all);
646 ie = ie->next_callee)
647 if (!(ie->indirect_info->ecf_flags & ECF_CONST))
649 read_all = true;
650 if (dump_file && (dump_flags & TDF_DETAILS))
651 fprintf (dump_file, " indirect call -> read all\n");
652 if (!cgraph_edge_cannot_lead_to_return (ie)
653 && !(ie->indirect_info->ecf_flags & ECF_PURE))
655 if (dump_file && (dump_flags & TDF_DETAILS))
656 fprintf (dump_file, " indirect call -> write all\n");
657 write_all = true;
662 /* Produce the global information by preforming a transitive closure
663 on the local information that was produced by ipa_analyze_function. */
665 static unsigned int
666 propagate (void)
668 struct cgraph_node *node;
669 struct varpool_node *vnode;
670 struct cgraph_node **order =
671 XCNEWVEC (struct cgraph_node *, cgraph_n_nodes);
672 int order_pos;
673 int i;
675 if (dump_file)
676 dump_cgraph (dump_file);
678 ipa_discover_readonly_nonaddressable_vars ();
679 generate_summary ();
681 /* Now we know what vars are really statics; prune out those that aren't. */
682 FOR_EACH_VARIABLE (vnode)
683 if (vnode->symbol.externally_visible
684 || TREE_ADDRESSABLE (vnode->symbol.decl)
685 || TREE_READONLY (vnode->symbol.decl)
686 || !is_proper_for_analysis (vnode->symbol.decl)
687 || !vnode->analyzed)
688 bitmap_clear_bit (all_module_statics, DECL_UID (vnode->symbol.decl));
690 /* Forget info we collected "just for fun" on variables that turned out to be
691 non-local. */
692 FOR_EACH_DEFINED_FUNCTION (node)
694 ipa_reference_local_vars_info_t node_l;
695 node_l = &get_reference_vars_info (node)->local;
696 intersect_static_var_sets (node_l->statics_read, all_module_statics);
697 intersect_static_var_sets (node_l->statics_written, all_module_statics);
700 /* Propagate the local information through the call graph to produce
701 the global information. All the nodes within a cycle will have
702 the same info so we collapse cycles first. Then we can do the
703 propagation in one pass from the leaves to the roots. */
704 order_pos = ipa_reduced_postorder (order, true, true, NULL);
705 if (dump_file)
706 ipa_print_order (dump_file, "reduced", order, order_pos);
708 for (i = 0; i < order_pos; i++ )
710 unsigned x;
711 struct cgraph_node *w;
712 ipa_reference_vars_info_t node_info;
713 ipa_reference_global_vars_info_t node_g;
714 ipa_reference_local_vars_info_t node_l;
715 bool read_all = false;
716 bool write_all = false;
718 node = order[i];
719 if (node->alias)
720 continue;
722 node_info = get_reference_vars_info (node);
723 gcc_assert (node_info);
724 node_l = &node_info->local;
725 node_g = &node_info->global;
727 if (dump_file && (dump_flags & TDF_DETAILS))
728 fprintf (dump_file, "Starting cycle with %s/%i\n",
729 cgraph_node_asm_name (node), node->symbol.order);
731 vec<cgraph_node_ptr> cycle_nodes = ipa_get_nodes_in_cycle (node);
733 /* If any node in a cycle is read_all or write_all, they all are. */
734 FOR_EACH_VEC_ELT (cycle_nodes, x, w)
736 if (dump_file && (dump_flags & TDF_DETAILS))
737 fprintf (dump_file, " Visiting %s/%i\n",
738 cgraph_node_asm_name (w), w->symbol.order);
739 get_read_write_all_from_node (w, read_all, write_all);
740 if (read_all && write_all)
741 break;
744 /* Initialized the bitmaps global sets for the reduced node. */
745 if (read_all)
746 node_g->statics_read = all_module_statics;
747 else
748 node_g->statics_read = copy_static_var_set (node_l->statics_read);
749 if (write_all)
750 node_g->statics_written = all_module_statics;
751 else
752 node_g->statics_written = copy_static_var_set (node_l->statics_written);
754 /* Merge the sets of this cycle with all sets of callees reached
755 from this cycle. */
756 FOR_EACH_VEC_ELT (cycle_nodes, x, w)
758 if (read_all && write_all)
759 break;
761 if (w != node)
763 ipa_reference_vars_info_t w_ri = get_reference_vars_info (w);
764 ipa_reference_local_vars_info_t w_l = &w_ri->local;
765 int flags = flags_from_decl_or_type (w->symbol.decl);
767 if (!(flags & ECF_CONST))
768 read_all = union_static_var_sets (node_g->statics_read,
769 w_l->statics_read);
770 if (!(flags & ECF_PURE)
771 && !cgraph_node_cannot_return (w))
772 write_all = union_static_var_sets (node_g->statics_written,
773 w_l->statics_written);
776 propagate_bits (node_g, w);
779 /* All nodes within a cycle have the same global info bitmaps. */
780 FOR_EACH_VEC_ELT (cycle_nodes, x, w)
782 ipa_reference_vars_info_t w_ri = get_reference_vars_info (w);
783 w_ri->global = *node_g;
786 cycle_nodes.release ();
789 if (dump_file)
791 for (i = 0; i < order_pos; i++)
793 unsigned x;
794 struct cgraph_node *w;
796 node = order[i];
797 if (node->alias)
798 continue;
800 fprintf (dump_file,
801 "\nFunction name:%s/%i:",
802 cgraph_node_asm_name (node), node->symbol.order);
804 ipa_reference_vars_info_t node_info = get_reference_vars_info (node);
805 ipa_reference_global_vars_info_t node_g = &node_info->global;
807 vec<cgraph_node_ptr> cycle_nodes = ipa_get_nodes_in_cycle (node);
808 FOR_EACH_VEC_ELT (cycle_nodes, x, w)
810 ipa_reference_vars_info_t w_ri = get_reference_vars_info (w);
811 ipa_reference_local_vars_info_t w_l = &w_ri->local;
812 if (w != node)
813 fprintf (dump_file, "\n next cycle: %s/%i ",
814 cgraph_node_asm_name (w), w->symbol.order);
815 fprintf (dump_file, "\n locals read: ");
816 dump_static_vars_set_to_file (dump_file, w_l->statics_read);
817 fprintf (dump_file, "\n locals written: ");
818 dump_static_vars_set_to_file (dump_file, w_l->statics_written);
820 cycle_nodes.release ();
822 fprintf (dump_file, "\n globals read: ");
823 dump_static_vars_set_to_file (dump_file, node_g->statics_read);
824 fprintf (dump_file, "\n globals written: ");
825 dump_static_vars_set_to_file (dump_file, node_g->statics_written);
826 fprintf (dump_file, "\n");
830 /* Cleanup. */
831 FOR_EACH_DEFINED_FUNCTION (node)
833 ipa_reference_vars_info_t node_info;
834 ipa_reference_global_vars_info_t node_g;
835 ipa_reference_optimization_summary_t opt;
837 if (node->alias)
838 continue;
840 node_info = get_reference_vars_info (node);
841 if (cgraph_function_body_availability (node) > AVAIL_OVERWRITABLE
842 || (flags_from_decl_or_type (node->symbol.decl) & ECF_LEAF))
844 node_g = &node_info->global;
846 opt = XCNEW (struct ipa_reference_optimization_summary_d);
847 set_reference_optimization_summary (node, opt);
849 /* Create the complimentary sets. */
851 if (bitmap_empty_p (node_g->statics_read))
852 opt->statics_not_read = all_module_statics;
853 else
855 opt->statics_not_read
856 = BITMAP_ALLOC (&optimization_summary_obstack);
857 if (node_g->statics_read != all_module_statics)
858 bitmap_and_compl (opt->statics_not_read,
859 all_module_statics,
860 node_g->statics_read);
863 if (bitmap_empty_p (node_g->statics_written))
864 opt->statics_not_written = all_module_statics;
865 else
867 opt->statics_not_written
868 = BITMAP_ALLOC (&optimization_summary_obstack);
869 if (node_g->statics_written != all_module_statics)
870 bitmap_and_compl (opt->statics_not_written,
871 all_module_statics,
872 node_g->statics_written);
875 free (node_info);
878 ipa_free_postorder_info ();
879 free (order);
881 bitmap_obstack_release (&local_info_obstack);
882 ipa_reference_vars_vector.release ();
883 if (dump_file)
884 splay_tree_delete (reference_vars_to_consider);
885 reference_vars_to_consider = NULL;
886 return 0;
889 /* Return true if we need to write summary of NODE. */
891 static bool
892 write_node_summary_p (struct cgraph_node *node,
893 lto_symtab_encoder_t encoder,
894 bitmap ltrans_statics)
896 ipa_reference_optimization_summary_t info;
898 /* See if we have (non-empty) info. */
899 if (!node->analyzed || node->global.inlined_to)
900 return false;
901 info = get_reference_optimization_summary (node);
902 if (!info || (bitmap_empty_p (info->statics_not_read)
903 && bitmap_empty_p (info->statics_not_written)))
904 return false;
906 /* See if we want to encode it.
907 Encode also referenced functions since constant folding might turn it into
908 a direct call.
910 In future we might also want to include summaries of functions references
911 by initializers of constant variables references in current unit. */
912 if (!reachable_from_this_partition_p (node, encoder)
913 && !referenced_from_this_partition_p (&node->symbol.ref_list, encoder))
914 return false;
916 /* See if the info has non-empty intersections with vars we want to encode. */
917 if (!bitmap_intersect_p (info->statics_not_read, ltrans_statics)
918 && !bitmap_intersect_p (info->statics_not_written, ltrans_statics))
919 return false;
920 return true;
923 /* Stream out BITS&LTRANS_STATICS as list of decls to OB.
924 LTRANS_STATICS_BITCOUNT specify number of bits in LTRANS_STATICS
925 or -1. When it is positive, just output -1 when
926 BITS&LTRANS_STATICS == BITS&LTRANS_STATICS. */
928 static void
929 stream_out_bitmap (struct lto_simple_output_block *ob,
930 bitmap bits, bitmap ltrans_statics,
931 int ltrans_statics_bitcount)
933 int count = 0;
934 unsigned int index;
935 bitmap_iterator bi;
936 if (bits == all_module_statics)
938 streamer_write_hwi_stream (ob->main_stream, -1);
939 return;
941 EXECUTE_IF_AND_IN_BITMAP (bits, ltrans_statics, 0, index, bi)
942 count ++;
943 if (count == ltrans_statics_bitcount)
945 streamer_write_hwi_stream (ob->main_stream, -1);
946 return;
948 streamer_write_hwi_stream (ob->main_stream, count);
949 if (!count)
950 return;
951 EXECUTE_IF_AND_IN_BITMAP (bits, ltrans_statics, 0, index, bi)
953 tree decl = (tree)splay_tree_lookup (reference_vars_to_consider, index)->value;
954 lto_output_var_decl_index(ob->decl_state, ob->main_stream, decl);
958 /* Serialize the ipa info for lto. */
960 static void
961 ipa_reference_write_optimization_summary (void)
963 struct lto_simple_output_block *ob
964 = lto_create_simple_output_block (LTO_section_ipa_reference);
965 unsigned int count = 0;
966 int ltrans_statics_bitcount = 0;
967 lto_symtab_encoder_t encoder = ob->decl_state->symtab_node_encoder;
968 bitmap ltrans_statics = BITMAP_ALLOC (NULL);
969 int i;
971 reference_vars_to_consider = splay_tree_new (splay_tree_compare_ints, 0, 0);
973 /* See what variables we are interested in. */
974 for (i = 0; i < lto_symtab_encoder_size (encoder); i++)
976 symtab_node snode = lto_symtab_encoder_deref (encoder, i);
977 varpool_node *vnode = dyn_cast <varpool_node> (snode);
978 if (vnode
979 && bitmap_bit_p (all_module_statics, DECL_UID (vnode->symbol.decl))
980 && referenced_from_this_partition_p (&vnode->symbol.ref_list, encoder))
982 tree decl = vnode->symbol.decl;
983 bitmap_set_bit (ltrans_statics, DECL_UID (decl));
984 splay_tree_insert (reference_vars_to_consider,
985 DECL_UID (decl), (splay_tree_value)decl);
986 ltrans_statics_bitcount ++;
991 if (ltrans_statics_bitcount)
992 for (i = 0; i < lto_symtab_encoder_size (encoder); i++)
994 symtab_node snode = lto_symtab_encoder_deref (encoder, i);
995 cgraph_node *cnode = dyn_cast <cgraph_node> (snode);
996 if (cnode && write_node_summary_p (cnode, encoder, ltrans_statics))
997 count++;
1000 streamer_write_uhwi_stream (ob->main_stream, count);
1001 if (count)
1002 stream_out_bitmap (ob, ltrans_statics, ltrans_statics,
1003 -1);
1005 /* Process all of the functions. */
1006 if (ltrans_statics_bitcount)
1007 for (i = 0; i < lto_symtab_encoder_size (encoder); i++)
1009 symtab_node snode = lto_symtab_encoder_deref (encoder, i);
1010 cgraph_node *cnode = dyn_cast <cgraph_node> (snode);
1011 if (cnode && write_node_summary_p (cnode, encoder, ltrans_statics))
1013 ipa_reference_optimization_summary_t info;
1014 int node_ref;
1016 info = get_reference_optimization_summary (cnode);
1017 node_ref = lto_symtab_encoder_encode (encoder, snode);
1018 streamer_write_uhwi_stream (ob->main_stream, node_ref);
1020 stream_out_bitmap (ob, info->statics_not_read, ltrans_statics,
1021 ltrans_statics_bitcount);
1022 stream_out_bitmap (ob, info->statics_not_written, ltrans_statics,
1023 ltrans_statics_bitcount);
1026 BITMAP_FREE (ltrans_statics);
1027 lto_destroy_simple_output_block (ob);
1028 splay_tree_delete (reference_vars_to_consider);
1031 /* Deserialize the ipa info for lto. */
1033 static void
1034 ipa_reference_read_optimization_summary (void)
1036 struct lto_file_decl_data ** file_data_vec
1037 = lto_get_file_decl_data ();
1038 struct lto_file_decl_data * file_data;
1039 unsigned int j = 0;
1040 bitmap_obstack_initialize (&optimization_summary_obstack);
1042 node_removal_hook_holder =
1043 cgraph_add_node_removal_hook (&remove_node_data, NULL);
1044 node_duplication_hook_holder =
1045 cgraph_add_node_duplication_hook (&duplicate_node_data, NULL);
1046 all_module_statics = BITMAP_ALLOC (&optimization_summary_obstack);
1048 while ((file_data = file_data_vec[j++]))
1050 const char *data;
1051 size_t len;
1052 struct lto_input_block *ib
1053 = lto_create_simple_input_block (file_data,
1054 LTO_section_ipa_reference,
1055 &data, &len);
1056 if (ib)
1058 unsigned int i;
1059 unsigned int f_count = streamer_read_uhwi (ib);
1060 int b_count;
1061 if (!f_count)
1062 continue;
1063 b_count = streamer_read_hwi (ib);
1064 if (dump_file)
1065 fprintf (dump_file, "all module statics:");
1066 for (i = 0; i < (unsigned int)b_count; i++)
1068 unsigned int var_index = streamer_read_uhwi (ib);
1069 tree v_decl = lto_file_decl_data_get_var_decl (file_data,
1070 var_index);
1071 bitmap_set_bit (all_module_statics, DECL_UID (v_decl));
1072 if (dump_file)
1073 fprintf (dump_file, " %s", fndecl_name (v_decl));
1076 for (i = 0; i < f_count; i++)
1078 unsigned int j, index;
1079 struct cgraph_node *node;
1080 ipa_reference_optimization_summary_t info;
1081 int v_count;
1082 lto_symtab_encoder_t encoder;
1084 index = streamer_read_uhwi (ib);
1085 encoder = file_data->symtab_node_encoder;
1086 node = cgraph (lto_symtab_encoder_deref (encoder, index));
1087 info = XCNEW (struct ipa_reference_optimization_summary_d);
1088 set_reference_optimization_summary (node, info);
1089 info->statics_not_read = BITMAP_ALLOC (&optimization_summary_obstack);
1090 info->statics_not_written = BITMAP_ALLOC (&optimization_summary_obstack);
1091 if (dump_file)
1092 fprintf (dump_file,
1093 "\nFunction name:%s/%i:\n static not read:",
1094 cgraph_node_asm_name (node), node->symbol.order);
1096 /* Set the statics not read. */
1097 v_count = streamer_read_hwi (ib);
1098 if (v_count == -1)
1100 info->statics_not_read = all_module_statics;
1101 if (dump_file)
1102 fprintf (dump_file, " all module statics");
1104 else
1105 for (j = 0; j < (unsigned int)v_count; j++)
1107 unsigned int var_index = streamer_read_uhwi (ib);
1108 tree v_decl = lto_file_decl_data_get_var_decl (file_data,
1109 var_index);
1110 bitmap_set_bit (info->statics_not_read, DECL_UID (v_decl));
1111 if (dump_file)
1112 fprintf (dump_file, " %s", fndecl_name (v_decl));
1115 if (dump_file)
1116 fprintf (dump_file,
1117 "\n static not written:");
1118 /* Set the statics not written. */
1119 v_count = streamer_read_hwi (ib);
1120 if (v_count == -1)
1122 info->statics_not_written = all_module_statics;
1123 if (dump_file)
1124 fprintf (dump_file, " all module statics");
1126 else
1127 for (j = 0; j < (unsigned int)v_count; j++)
1129 unsigned int var_index = streamer_read_uhwi (ib);
1130 tree v_decl = lto_file_decl_data_get_var_decl (file_data,
1131 var_index);
1132 bitmap_set_bit (info->statics_not_written, DECL_UID (v_decl));
1133 if (dump_file)
1134 fprintf (dump_file, " %s", fndecl_name (v_decl));
1136 if (dump_file)
1137 fprintf (dump_file, "\n");
1140 lto_destroy_simple_input_block (file_data,
1141 LTO_section_ipa_reference,
1142 ib, data, len);
1144 else
1145 /* Fatal error here. We do not want to support compiling ltrans units with
1146 different version of compiler or different flags than the WPA unit, so
1147 this should never happen. */
1148 fatal_error ("ipa reference summary is missing in ltrans unit");
1152 static bool
1153 gate_reference (void)
1155 return (flag_ipa_reference
1156 /* Don't bother doing anything if the program has errors. */
1157 && !seen_error ());
1160 struct ipa_opt_pass_d pass_ipa_reference =
1163 IPA_PASS,
1164 "static-var", /* name */
1165 OPTGROUP_NONE, /* optinfo_flags */
1166 gate_reference, /* gate */
1167 propagate, /* execute */
1168 NULL, /* sub */
1169 NULL, /* next */
1170 0, /* static_pass_number */
1171 TV_IPA_REFERENCE, /* tv_id */
1172 0, /* properties_required */
1173 0, /* properties_provided */
1174 0, /* properties_destroyed */
1175 0, /* todo_flags_start */
1176 0 /* todo_flags_finish */
1178 NULL, /* generate_summary */
1179 NULL, /* write_summary */
1180 NULL, /* read_summary */
1181 ipa_reference_write_optimization_summary,/* write_optimization_summary */
1182 ipa_reference_read_optimization_summary,/* read_optimization_summary */
1183 NULL, /* stmt_fixup */
1184 0, /* TODOs */
1185 NULL, /* function_transform */
1186 NULL /* variable_transform */