xref: /dflybsd-src/contrib/gcc-4.7/gcc/ipa-ref.c (revision 04febcfb30580676d3e95f58a16c5137ee478b32)
1*e4b17023SJohn Marino /* Interprocedural reference lists.
2*e4b17023SJohn Marino    Copyright (C) 2010
3*e4b17023SJohn Marino    Free Software Foundation, Inc.
4*e4b17023SJohn Marino    Contributed by Jan Hubicka
5*e4b17023SJohn Marino 
6*e4b17023SJohn Marino This file is part of GCC.
7*e4b17023SJohn Marino 
8*e4b17023SJohn Marino GCC is free software; you can redistribute it and/or modify it under
9*e4b17023SJohn Marino the terms of the GNU General Public License as published by the Free
10*e4b17023SJohn Marino Software Foundation; either version 3, or (at your option) any later
11*e4b17023SJohn Marino version.
12*e4b17023SJohn Marino 
13*e4b17023SJohn Marino GCC is distributed in the hope that it will be useful, but WITHOUT ANY
14*e4b17023SJohn Marino WARRANTY; without even the implied warranty of MERCHANTABILITY or
15*e4b17023SJohn Marino FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
16*e4b17023SJohn Marino for more details.
17*e4b17023SJohn Marino 
18*e4b17023SJohn Marino You should have received a copy of the GNU General Public License
19*e4b17023SJohn Marino along with GCC; see the file COPYING3.  If not see
20*e4b17023SJohn Marino <http://www.gnu.org/licenses/>.  */
21*e4b17023SJohn Marino 
22*e4b17023SJohn Marino #include "config.h"
23*e4b17023SJohn Marino #include "system.h"
24*e4b17023SJohn Marino #include "coretypes.h"
25*e4b17023SJohn Marino #include "tree.h"
26*e4b17023SJohn Marino #include "ggc.h"
27*e4b17023SJohn Marino #include "target.h"
28*e4b17023SJohn Marino #include "cgraph.h"
29*e4b17023SJohn Marino 
30*e4b17023SJohn Marino static const char *ipa_ref_use_name[] = {"read","write","addr","alias"};
31*e4b17023SJohn Marino 
32*e4b17023SJohn Marino /* Return ipa reference from REFERING_NODE or REFERING_VARPOOL_NODE
33*e4b17023SJohn Marino    to REFERED_NODE or REFERED_VARPOOL_NODE. USE_TYPE specify type
34*e4b17023SJohn Marino    of the use and STMT the statement (if it exists).  */
35*e4b17023SJohn Marino 
36*e4b17023SJohn Marino struct ipa_ref *
ipa_record_reference(struct cgraph_node * refering_node,struct varpool_node * refering_varpool_node,struct cgraph_node * refered_node,struct varpool_node * refered_varpool_node,enum ipa_ref_use use_type,gimple stmt)37*e4b17023SJohn Marino ipa_record_reference (struct cgraph_node *refering_node,
38*e4b17023SJohn Marino 		      struct varpool_node *refering_varpool_node,
39*e4b17023SJohn Marino 		      struct cgraph_node *refered_node,
40*e4b17023SJohn Marino 		      struct varpool_node *refered_varpool_node,
41*e4b17023SJohn Marino 		      enum ipa_ref_use use_type, gimple stmt)
42*e4b17023SJohn Marino {
43*e4b17023SJohn Marino   struct ipa_ref *ref;
44*e4b17023SJohn Marino   struct ipa_ref_list *list, *list2;
45*e4b17023SJohn Marino   VEC(ipa_ref_t,gc) *old_references;
46*e4b17023SJohn Marino   gcc_assert ((!refering_node) ^ (!refering_varpool_node));
47*e4b17023SJohn Marino   gcc_assert ((!refered_node) ^ (!refered_varpool_node));
48*e4b17023SJohn Marino   gcc_assert (!stmt || refering_node);
49*e4b17023SJohn Marino   gcc_assert (use_type != IPA_REF_ALIAS || !stmt);
50*e4b17023SJohn Marino 
51*e4b17023SJohn Marino   list = (refering_node ? &refering_node->ref_list
52*e4b17023SJohn Marino 	  : &refering_varpool_node->ref_list);
53*e4b17023SJohn Marino   old_references = list->references;
54*e4b17023SJohn Marino   VEC_safe_grow (ipa_ref_t, gc, list->references,
55*e4b17023SJohn Marino 		 VEC_length (ipa_ref_t, list->references) + 1);
56*e4b17023SJohn Marino   ref = VEC_last (ipa_ref_t, list->references);
57*e4b17023SJohn Marino 
58*e4b17023SJohn Marino   list2 = (refered_node ? &refered_node->ref_list
59*e4b17023SJohn Marino 	   : &refered_varpool_node->ref_list);
60*e4b17023SJohn Marino   VEC_safe_push (ipa_ref_ptr, heap, list2->refering, ref);
61*e4b17023SJohn Marino   ref->refered_index = VEC_length (ipa_ref_ptr, list2->refering) - 1;
62*e4b17023SJohn Marino   if (refering_node)
63*e4b17023SJohn Marino     {
64*e4b17023SJohn Marino       ref->refering.cgraph_node = refering_node;
65*e4b17023SJohn Marino       ref->refering_type = IPA_REF_CGRAPH;
66*e4b17023SJohn Marino     }
67*e4b17023SJohn Marino   else
68*e4b17023SJohn Marino     {
69*e4b17023SJohn Marino       ref->refering.varpool_node = refering_varpool_node;
70*e4b17023SJohn Marino       ref->refering_type = IPA_REF_VARPOOL;
71*e4b17023SJohn Marino       gcc_assert (use_type == IPA_REF_ADDR || use_type == IPA_REF_ALIAS);
72*e4b17023SJohn Marino     }
73*e4b17023SJohn Marino   if (refered_node)
74*e4b17023SJohn Marino     {
75*e4b17023SJohn Marino       ref->refered.cgraph_node = refered_node;
76*e4b17023SJohn Marino       ref->refered_type = IPA_REF_CGRAPH;
77*e4b17023SJohn Marino       gcc_assert (use_type == IPA_REF_ADDR || use_type == IPA_REF_ALIAS);
78*e4b17023SJohn Marino     }
79*e4b17023SJohn Marino   else
80*e4b17023SJohn Marino     {
81*e4b17023SJohn Marino       varpool_mark_needed_node (refered_varpool_node);
82*e4b17023SJohn Marino       ref->refered.varpool_node = refered_varpool_node;
83*e4b17023SJohn Marino       ref->refered_type = IPA_REF_VARPOOL;
84*e4b17023SJohn Marino     }
85*e4b17023SJohn Marino   ref->stmt = stmt;
86*e4b17023SJohn Marino   ref->use = use_type;
87*e4b17023SJohn Marino 
88*e4b17023SJohn Marino   /* If vector was moved in memory, update pointers.  */
89*e4b17023SJohn Marino   if (old_references != list->references)
90*e4b17023SJohn Marino     {
91*e4b17023SJohn Marino       int i;
92*e4b17023SJohn Marino       for (i = 0; ipa_ref_list_reference_iterate (list, i, ref); i++)
93*e4b17023SJohn Marino 	VEC_replace (ipa_ref_ptr,
94*e4b17023SJohn Marino 		     ipa_ref_refered_ref_list (ref)->refering,
95*e4b17023SJohn Marino 		     ref->refered_index, ref);
96*e4b17023SJohn Marino     }
97*e4b17023SJohn Marino   return ref;
98*e4b17023SJohn Marino }
99*e4b17023SJohn Marino 
100*e4b17023SJohn Marino /* Remove reference REF.  */
101*e4b17023SJohn Marino 
102*e4b17023SJohn Marino void
ipa_remove_reference(struct ipa_ref * ref)103*e4b17023SJohn Marino ipa_remove_reference (struct ipa_ref *ref)
104*e4b17023SJohn Marino {
105*e4b17023SJohn Marino   struct ipa_ref_list *list = ipa_ref_refered_ref_list (ref);
106*e4b17023SJohn Marino   struct ipa_ref_list *list2 = ipa_ref_refering_ref_list (ref);
107*e4b17023SJohn Marino   VEC(ipa_ref_t,gc) *old_references = list2->references;
108*e4b17023SJohn Marino   struct ipa_ref *last;
109*e4b17023SJohn Marino 
110*e4b17023SJohn Marino   gcc_assert (VEC_index (ipa_ref_ptr, list->refering, ref->refered_index) == ref);
111*e4b17023SJohn Marino   last = VEC_last (ipa_ref_ptr, list->refering);
112*e4b17023SJohn Marino   if (ref != last)
113*e4b17023SJohn Marino     {
114*e4b17023SJohn Marino       VEC_replace (ipa_ref_ptr, list->refering,
115*e4b17023SJohn Marino 		   ref->refered_index,
116*e4b17023SJohn Marino 		   VEC_last (ipa_ref_ptr, list->refering));
117*e4b17023SJohn Marino       VEC_index (ipa_ref_ptr, list->refering,
118*e4b17023SJohn Marino 		 ref->refered_index)->refered_index = ref->refered_index;
119*e4b17023SJohn Marino     }
120*e4b17023SJohn Marino   VEC_pop (ipa_ref_ptr, list->refering);
121*e4b17023SJohn Marino 
122*e4b17023SJohn Marino   last = VEC_last (ipa_ref_t, list2->references);
123*e4b17023SJohn Marino   if (ref != last)
124*e4b17023SJohn Marino     {
125*e4b17023SJohn Marino       *ref = *last;
126*e4b17023SJohn Marino       VEC_replace (ipa_ref_ptr,
127*e4b17023SJohn Marino 		   ipa_ref_refered_ref_list (ref)->refering,
128*e4b17023SJohn Marino 		   ref->refered_index, ref);
129*e4b17023SJohn Marino     }
130*e4b17023SJohn Marino   VEC_pop (ipa_ref_t, list2->references);
131*e4b17023SJohn Marino   gcc_assert (list2->references == old_references);
132*e4b17023SJohn Marino }
133*e4b17023SJohn Marino 
134*e4b17023SJohn Marino /* Remove all references in ref list LIST.  */
135*e4b17023SJohn Marino 
136*e4b17023SJohn Marino void
ipa_remove_all_references(struct ipa_ref_list * list)137*e4b17023SJohn Marino ipa_remove_all_references (struct ipa_ref_list *list)
138*e4b17023SJohn Marino {
139*e4b17023SJohn Marino   while (VEC_length (ipa_ref_t, list->references))
140*e4b17023SJohn Marino     ipa_remove_reference (VEC_last (ipa_ref_t, list->references));
141*e4b17023SJohn Marino   VEC_free (ipa_ref_t, gc, list->references);
142*e4b17023SJohn Marino   list->references = NULL;
143*e4b17023SJohn Marino }
144*e4b17023SJohn Marino 
145*e4b17023SJohn Marino /* Remove all references in ref list LIST.  */
146*e4b17023SJohn Marino 
147*e4b17023SJohn Marino void
ipa_remove_all_refering(struct ipa_ref_list * list)148*e4b17023SJohn Marino ipa_remove_all_refering (struct ipa_ref_list *list)
149*e4b17023SJohn Marino {
150*e4b17023SJohn Marino   while (VEC_length (ipa_ref_ptr, list->refering))
151*e4b17023SJohn Marino     ipa_remove_reference (VEC_last (ipa_ref_ptr, list->refering));
152*e4b17023SJohn Marino   VEC_free (ipa_ref_ptr, heap, list->refering);
153*e4b17023SJohn Marino   list->refering = NULL;
154*e4b17023SJohn Marino }
155*e4b17023SJohn Marino 
156*e4b17023SJohn Marino /* Dump references in LIST to FILE.  */
157*e4b17023SJohn Marino 
158*e4b17023SJohn Marino void
ipa_dump_references(FILE * file,struct ipa_ref_list * list)159*e4b17023SJohn Marino ipa_dump_references (FILE * file, struct ipa_ref_list *list)
160*e4b17023SJohn Marino {
161*e4b17023SJohn Marino   struct ipa_ref *ref;
162*e4b17023SJohn Marino   int i;
163*e4b17023SJohn Marino   for (i = 0; ipa_ref_list_reference_iterate (list, i, ref); i++)
164*e4b17023SJohn Marino     {
165*e4b17023SJohn Marino       if (ref->refered_type == IPA_REF_CGRAPH)
166*e4b17023SJohn Marino 	{
167*e4b17023SJohn Marino 	  fprintf (file, " fn:%s/%i (%s)", cgraph_node_name (ipa_ref_node (ref)),
168*e4b17023SJohn Marino 		   ipa_ref_node (ref)->uid,
169*e4b17023SJohn Marino 		   ipa_ref_use_name [ref->use]);
170*e4b17023SJohn Marino 	}
171*e4b17023SJohn Marino       else
172*e4b17023SJohn Marino 	fprintf (file, " var:%s (%s)",
173*e4b17023SJohn Marino 		 varpool_node_name (ipa_ref_varpool_node (ref)),
174*e4b17023SJohn Marino 		 ipa_ref_use_name [ref->use]);
175*e4b17023SJohn Marino     }
176*e4b17023SJohn Marino   fprintf (file, "\n");
177*e4b17023SJohn Marino }
178*e4b17023SJohn Marino 
179*e4b17023SJohn Marino /* Dump refering in LIST to FILE.  */
180*e4b17023SJohn Marino 
181*e4b17023SJohn Marino void
ipa_dump_refering(FILE * file,struct ipa_ref_list * list)182*e4b17023SJohn Marino ipa_dump_refering (FILE * file, struct ipa_ref_list *list)
183*e4b17023SJohn Marino {
184*e4b17023SJohn Marino   struct ipa_ref *ref;
185*e4b17023SJohn Marino   int i;
186*e4b17023SJohn Marino   for (i = 0; ipa_ref_list_refering_iterate (list, i, ref); i++)
187*e4b17023SJohn Marino     {
188*e4b17023SJohn Marino       if (ref->refering_type == IPA_REF_CGRAPH)
189*e4b17023SJohn Marino 	fprintf (file, " fn:%s/%i (%s)",
190*e4b17023SJohn Marino 		 cgraph_node_name (ipa_ref_refering_node (ref)),
191*e4b17023SJohn Marino 		 ipa_ref_refering_node (ref)->uid,
192*e4b17023SJohn Marino 		 ipa_ref_use_name [ref->use]);
193*e4b17023SJohn Marino       else
194*e4b17023SJohn Marino 	fprintf (file, " var:%s (%s)",
195*e4b17023SJohn Marino 		 varpool_node_name (ipa_ref_refering_varpool_node (ref)),
196*e4b17023SJohn Marino 		 ipa_ref_use_name [ref->use]);
197*e4b17023SJohn Marino     }
198*e4b17023SJohn Marino   fprintf (file, "\n");
199*e4b17023SJohn Marino }
200*e4b17023SJohn Marino 
201*e4b17023SJohn Marino /* Clone all references from SRC to DEST_NODE or DEST_VARPOOL_NODE.  */
202*e4b17023SJohn Marino 
203*e4b17023SJohn Marino void
ipa_clone_references(struct cgraph_node * dest_node,struct varpool_node * dest_varpool_node,struct ipa_ref_list * src)204*e4b17023SJohn Marino ipa_clone_references (struct cgraph_node *dest_node,
205*e4b17023SJohn Marino 		      struct varpool_node *dest_varpool_node,
206*e4b17023SJohn Marino 		      struct ipa_ref_list *src)
207*e4b17023SJohn Marino {
208*e4b17023SJohn Marino   struct ipa_ref *ref;
209*e4b17023SJohn Marino   int i;
210*e4b17023SJohn Marino   for (i = 0; ipa_ref_list_reference_iterate (src, i, ref); i++)
211*e4b17023SJohn Marino     ipa_record_reference (dest_node, dest_varpool_node,
212*e4b17023SJohn Marino 			  ref->refered_type == IPA_REF_CGRAPH
213*e4b17023SJohn Marino 			  ? ipa_ref_node (ref) : NULL,
214*e4b17023SJohn Marino 			  ref->refered_type == IPA_REF_VARPOOL
215*e4b17023SJohn Marino 			  ? ipa_ref_varpool_node (ref) : NULL,
216*e4b17023SJohn Marino 			  ref->use, ref->stmt);
217*e4b17023SJohn Marino }
218*e4b17023SJohn Marino 
219*e4b17023SJohn Marino /* Clone all refering from SRC to DEST_NODE or DEST_VARPOOL_NODE.  */
220*e4b17023SJohn Marino 
221*e4b17023SJohn Marino void
ipa_clone_refering(struct cgraph_node * dest_node,struct varpool_node * dest_varpool_node,struct ipa_ref_list * src)222*e4b17023SJohn Marino ipa_clone_refering (struct cgraph_node *dest_node,
223*e4b17023SJohn Marino 		    struct varpool_node *dest_varpool_node,
224*e4b17023SJohn Marino 		    struct ipa_ref_list *src)
225*e4b17023SJohn Marino {
226*e4b17023SJohn Marino   struct ipa_ref *ref;
227*e4b17023SJohn Marino   int i;
228*e4b17023SJohn Marino   for (i = 0; ipa_ref_list_refering_iterate (src, i, ref); i++)
229*e4b17023SJohn Marino     ipa_record_reference (
230*e4b17023SJohn Marino 			  ref->refering_type == IPA_REF_CGRAPH
231*e4b17023SJohn Marino 			  ? ipa_ref_refering_node (ref) : NULL,
232*e4b17023SJohn Marino 			  ref->refering_type == IPA_REF_VARPOOL
233*e4b17023SJohn Marino 			  ? ipa_ref_refering_varpool_node (ref) : NULL,
234*e4b17023SJohn Marino 			  dest_node, dest_varpool_node,
235*e4b17023SJohn Marino 			  ref->use, ref->stmt);
236*e4b17023SJohn Marino }
237*e4b17023SJohn Marino 
238*e4b17023SJohn Marino /* Return true when execution of REF can load to return from
239*e4b17023SJohn Marino    function. */
240*e4b17023SJohn Marino bool
ipa_ref_cannot_lead_to_return(struct ipa_ref * ref)241*e4b17023SJohn Marino ipa_ref_cannot_lead_to_return (struct ipa_ref *ref)
242*e4b17023SJohn Marino {
243*e4b17023SJohn Marino   return cgraph_node_cannot_return (ipa_ref_refering_node (ref));
244*e4b17023SJohn Marino }
245*e4b17023SJohn Marino 
246*e4b17023SJohn Marino /* Return true if list contains an alias.  */
247*e4b17023SJohn Marino bool
ipa_ref_has_aliases_p(struct ipa_ref_list * ref_list)248*e4b17023SJohn Marino ipa_ref_has_aliases_p (struct ipa_ref_list *ref_list)
249*e4b17023SJohn Marino {
250*e4b17023SJohn Marino   struct ipa_ref *ref;
251*e4b17023SJohn Marino   int i;
252*e4b17023SJohn Marino   for (i = 0; ipa_ref_list_refering_iterate (ref_list, i, ref); i++)
253*e4b17023SJohn Marino     if (ref->use == IPA_REF_ALIAS)
254*e4b17023SJohn Marino       return true;
255*e4b17023SJohn Marino   return false;
256*e4b17023SJohn Marino }
257