annotate gcc/ipa-fnsummary.c @ 128:fe568345ddd5

fix CbC-example
author mir3636
date Wed, 11 Apr 2018 19:32:28 +0900
parents 04ced10e8804
children 84e7813d76e9
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 /* Function summary pass.
kono
parents:
diff changeset
2 Copyright (C) 2003-2017 Free Software Foundation, Inc.
kono
parents:
diff changeset
3 Contributed by Jan Hubicka
kono
parents:
diff changeset
4
kono
parents:
diff changeset
5 This file is part of GCC.
kono
parents:
diff changeset
6
kono
parents:
diff changeset
7 GCC is free software; you can redistribute it and/or modify it under
kono
parents:
diff changeset
8 the terms of the GNU General Public License as published by the Free
kono
parents:
diff changeset
9 Software Foundation; either version 3, or (at your option) any later
kono
parents:
diff changeset
10 version.
kono
parents:
diff changeset
11
kono
parents:
diff changeset
12 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
kono
parents:
diff changeset
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
kono
parents:
diff changeset
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
kono
parents:
diff changeset
15 for more details.
kono
parents:
diff changeset
16
kono
parents:
diff changeset
17 You should have received a copy of the GNU General Public License
kono
parents:
diff changeset
18 along with GCC; see the file COPYING3. If not see
kono
parents:
diff changeset
19 <http://www.gnu.org/licenses/>. */
kono
parents:
diff changeset
20
kono
parents:
diff changeset
21 /* Analysis of function bodies used by inter-procedural passes
kono
parents:
diff changeset
22
kono
parents:
diff changeset
23 We estimate for each function
kono
parents:
diff changeset
24 - function body size and size after specializing into given context
kono
parents:
diff changeset
25 - average function execution time in a given context
kono
parents:
diff changeset
26 - function frame size
kono
parents:
diff changeset
27 For each call
kono
parents:
diff changeset
28 - call statement size, time and how often the parameters change
kono
parents:
diff changeset
29
kono
parents:
diff changeset
30 ipa_fn_summary data structures store above information locally (i.e.
kono
parents:
diff changeset
31 parameters of the function itself) and globally (i.e. parameters of
kono
parents:
diff changeset
32 the function created by applying all the inline decisions already
kono
parents:
diff changeset
33 present in the callgraph).
kono
parents:
diff changeset
34
kono
parents:
diff changeset
35 We provide access to the ipa_fn_summary data structure and
kono
parents:
diff changeset
36 basic logic updating the parameters when inlining is performed.
kono
parents:
diff changeset
37
kono
parents:
diff changeset
38 The summaries are context sensitive. Context means
kono
parents:
diff changeset
39 1) partial assignment of known constant values of operands
kono
parents:
diff changeset
40 2) whether function is inlined into the call or not.
kono
parents:
diff changeset
41 It is easy to add more variants. To represent function size and time
kono
parents:
diff changeset
42 that depends on context (i.e. it is known to be optimized away when
kono
parents:
diff changeset
43 context is known either by inlining or from IP-CP and cloning),
kono
parents:
diff changeset
44 we use predicates.
kono
parents:
diff changeset
45
kono
parents:
diff changeset
46 estimate_edge_size_and_time can be used to query
kono
parents:
diff changeset
47 function size/time in the given context. ipa_merge_fn_summary_after_inlining merges
kono
parents:
diff changeset
48 properties of caller and callee after inlining.
kono
parents:
diff changeset
49
kono
parents:
diff changeset
50 Finally pass_inline_parameters is exported. This is used to drive
kono
parents:
diff changeset
51 computation of function parameters used by the early inliner. IPA
kono
parents:
diff changeset
52 inlined performs analysis via its analyze_function method. */
kono
parents:
diff changeset
53
kono
parents:
diff changeset
54 #include "config.h"
kono
parents:
diff changeset
55 #include "system.h"
kono
parents:
diff changeset
56 #include "coretypes.h"
kono
parents:
diff changeset
57 #include "backend.h"
kono
parents:
diff changeset
58 #include "tree.h"
kono
parents:
diff changeset
59 #include "gimple.h"
kono
parents:
diff changeset
60 #include "alloc-pool.h"
kono
parents:
diff changeset
61 #include "tree-pass.h"
kono
parents:
diff changeset
62 #include "ssa.h"
kono
parents:
diff changeset
63 #include "tree-streamer.h"
kono
parents:
diff changeset
64 #include "cgraph.h"
kono
parents:
diff changeset
65 #include "diagnostic.h"
kono
parents:
diff changeset
66 #include "fold-const.h"
kono
parents:
diff changeset
67 #include "print-tree.h"
kono
parents:
diff changeset
68 #include "tree-inline.h"
kono
parents:
diff changeset
69 #include "gimple-pretty-print.h"
kono
parents:
diff changeset
70 #include "params.h"
kono
parents:
diff changeset
71 #include "cfganal.h"
kono
parents:
diff changeset
72 #include "gimple-iterator.h"
kono
parents:
diff changeset
73 #include "tree-cfg.h"
kono
parents:
diff changeset
74 #include "tree-ssa-loop-niter.h"
kono
parents:
diff changeset
75 #include "tree-ssa-loop.h"
kono
parents:
diff changeset
76 #include "symbol-summary.h"
kono
parents:
diff changeset
77 #include "ipa-prop.h"
kono
parents:
diff changeset
78 #include "ipa-fnsummary.h"
kono
parents:
diff changeset
79 #include "cfgloop.h"
kono
parents:
diff changeset
80 #include "tree-scalar-evolution.h"
kono
parents:
diff changeset
81 #include "ipa-utils.h"
kono
parents:
diff changeset
82 #include "cilk.h"
kono
parents:
diff changeset
83 #include "cfgexpand.h"
kono
parents:
diff changeset
84 #include "gimplify.h"
kono
parents:
diff changeset
85 #include "stringpool.h"
kono
parents:
diff changeset
86 #include "attribs.h"
kono
parents:
diff changeset
87
kono
parents:
diff changeset
88 /* Summaries. */
kono
parents:
diff changeset
89 function_summary <ipa_fn_summary *> *ipa_fn_summaries;
kono
parents:
diff changeset
90 call_summary <ipa_call_summary *> *ipa_call_summaries;
kono
parents:
diff changeset
91
kono
parents:
diff changeset
92 /* Edge predicates goes here. */
kono
parents:
diff changeset
93 static object_allocator<predicate> edge_predicate_pool ("edge predicates");
kono
parents:
diff changeset
94
kono
parents:
diff changeset
95
kono
parents:
diff changeset
96 /* Dump IPA hints. */
kono
parents:
diff changeset
97 void
kono
parents:
diff changeset
98 ipa_dump_hints (FILE *f, ipa_hints hints)
kono
parents:
diff changeset
99 {
kono
parents:
diff changeset
100 if (!hints)
kono
parents:
diff changeset
101 return;
kono
parents:
diff changeset
102 fprintf (f, "IPA hints:");
kono
parents:
diff changeset
103 if (hints & INLINE_HINT_indirect_call)
kono
parents:
diff changeset
104 {
kono
parents:
diff changeset
105 hints &= ~INLINE_HINT_indirect_call;
kono
parents:
diff changeset
106 fprintf (f, " indirect_call");
kono
parents:
diff changeset
107 }
kono
parents:
diff changeset
108 if (hints & INLINE_HINT_loop_iterations)
kono
parents:
diff changeset
109 {
kono
parents:
diff changeset
110 hints &= ~INLINE_HINT_loop_iterations;
kono
parents:
diff changeset
111 fprintf (f, " loop_iterations");
kono
parents:
diff changeset
112 }
kono
parents:
diff changeset
113 if (hints & INLINE_HINT_loop_stride)
kono
parents:
diff changeset
114 {
kono
parents:
diff changeset
115 hints &= ~INLINE_HINT_loop_stride;
kono
parents:
diff changeset
116 fprintf (f, " loop_stride");
kono
parents:
diff changeset
117 }
kono
parents:
diff changeset
118 if (hints & INLINE_HINT_same_scc)
kono
parents:
diff changeset
119 {
kono
parents:
diff changeset
120 hints &= ~INLINE_HINT_same_scc;
kono
parents:
diff changeset
121 fprintf (f, " same_scc");
kono
parents:
diff changeset
122 }
kono
parents:
diff changeset
123 if (hints & INLINE_HINT_in_scc)
kono
parents:
diff changeset
124 {
kono
parents:
diff changeset
125 hints &= ~INLINE_HINT_in_scc;
kono
parents:
diff changeset
126 fprintf (f, " in_scc");
kono
parents:
diff changeset
127 }
kono
parents:
diff changeset
128 if (hints & INLINE_HINT_cross_module)
kono
parents:
diff changeset
129 {
kono
parents:
diff changeset
130 hints &= ~INLINE_HINT_cross_module;
kono
parents:
diff changeset
131 fprintf (f, " cross_module");
kono
parents:
diff changeset
132 }
kono
parents:
diff changeset
133 if (hints & INLINE_HINT_declared_inline)
kono
parents:
diff changeset
134 {
kono
parents:
diff changeset
135 hints &= ~INLINE_HINT_declared_inline;
kono
parents:
diff changeset
136 fprintf (f, " declared_inline");
kono
parents:
diff changeset
137 }
kono
parents:
diff changeset
138 if (hints & INLINE_HINT_array_index)
kono
parents:
diff changeset
139 {
kono
parents:
diff changeset
140 hints &= ~INLINE_HINT_array_index;
kono
parents:
diff changeset
141 fprintf (f, " array_index");
kono
parents:
diff changeset
142 }
kono
parents:
diff changeset
143 if (hints & INLINE_HINT_known_hot)
kono
parents:
diff changeset
144 {
kono
parents:
diff changeset
145 hints &= ~INLINE_HINT_known_hot;
kono
parents:
diff changeset
146 fprintf (f, " known_hot");
kono
parents:
diff changeset
147 }
kono
parents:
diff changeset
148 gcc_assert (!hints);
kono
parents:
diff changeset
149 }
kono
parents:
diff changeset
150
kono
parents:
diff changeset
151
kono
parents:
diff changeset
152 /* Record SIZE and TIME to SUMMARY.
kono
parents:
diff changeset
153 The accounted code will be executed when EXEC_PRED is true.
kono
parents:
diff changeset
154 When NONCONST_PRED is false the code will evaulate to constant and
kono
parents:
diff changeset
155 will get optimized out in specialized clones of the function. */
kono
parents:
diff changeset
156
kono
parents:
diff changeset
157 void
kono
parents:
diff changeset
158 ipa_fn_summary::account_size_time (int size, sreal time,
kono
parents:
diff changeset
159 const predicate &exec_pred,
kono
parents:
diff changeset
160 const predicate &nonconst_pred_in)
kono
parents:
diff changeset
161 {
kono
parents:
diff changeset
162 size_time_entry *e;
kono
parents:
diff changeset
163 bool found = false;
kono
parents:
diff changeset
164 int i;
kono
parents:
diff changeset
165 predicate nonconst_pred;
kono
parents:
diff changeset
166
kono
parents:
diff changeset
167 if (exec_pred == false)
kono
parents:
diff changeset
168 return;
kono
parents:
diff changeset
169
kono
parents:
diff changeset
170 nonconst_pred = nonconst_pred_in & exec_pred;
kono
parents:
diff changeset
171
kono
parents:
diff changeset
172 if (nonconst_pred == false)
kono
parents:
diff changeset
173 return;
kono
parents:
diff changeset
174
kono
parents:
diff changeset
175 /* We need to create initial empty unconitional clause, but otherwie
kono
parents:
diff changeset
176 we don't need to account empty times and sizes. */
kono
parents:
diff changeset
177 if (!size && time == 0 && size_time_table)
kono
parents:
diff changeset
178 return;
kono
parents:
diff changeset
179
kono
parents:
diff changeset
180 gcc_assert (time >= 0);
kono
parents:
diff changeset
181
kono
parents:
diff changeset
182 for (i = 0; vec_safe_iterate (size_time_table, i, &e); i++)
kono
parents:
diff changeset
183 if (e->exec_predicate == exec_pred
kono
parents:
diff changeset
184 && e->nonconst_predicate == nonconst_pred)
kono
parents:
diff changeset
185 {
kono
parents:
diff changeset
186 found = true;
kono
parents:
diff changeset
187 break;
kono
parents:
diff changeset
188 }
kono
parents:
diff changeset
189 if (i == 256)
kono
parents:
diff changeset
190 {
kono
parents:
diff changeset
191 i = 0;
kono
parents:
diff changeset
192 found = true;
kono
parents:
diff changeset
193 e = &(*size_time_table)[0];
kono
parents:
diff changeset
194 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
195 fprintf (dump_file,
kono
parents:
diff changeset
196 "\t\tReached limit on number of entries, "
kono
parents:
diff changeset
197 "ignoring the predicate.");
kono
parents:
diff changeset
198 }
kono
parents:
diff changeset
199 if (dump_file && (dump_flags & TDF_DETAILS) && (time != 0 || size))
kono
parents:
diff changeset
200 {
kono
parents:
diff changeset
201 fprintf (dump_file,
kono
parents:
diff changeset
202 "\t\tAccounting size:%3.2f, time:%3.2f on %spredicate exec:",
kono
parents:
diff changeset
203 ((double) size) / ipa_fn_summary::size_scale,
kono
parents:
diff changeset
204 (time.to_double ()), found ? "" : "new ");
kono
parents:
diff changeset
205 exec_pred.dump (dump_file, conds, 0);
kono
parents:
diff changeset
206 if (exec_pred != nonconst_pred)
kono
parents:
diff changeset
207 {
kono
parents:
diff changeset
208 fprintf (dump_file, " nonconst:");
kono
parents:
diff changeset
209 nonconst_pred.dump (dump_file, conds);
kono
parents:
diff changeset
210 }
kono
parents:
diff changeset
211 else
kono
parents:
diff changeset
212 fprintf (dump_file, "\n");
kono
parents:
diff changeset
213 }
kono
parents:
diff changeset
214 if (!found)
kono
parents:
diff changeset
215 {
kono
parents:
diff changeset
216 struct size_time_entry new_entry;
kono
parents:
diff changeset
217 new_entry.size = size;
kono
parents:
diff changeset
218 new_entry.time = time;
kono
parents:
diff changeset
219 new_entry.exec_predicate = exec_pred;
kono
parents:
diff changeset
220 new_entry.nonconst_predicate = nonconst_pred;
kono
parents:
diff changeset
221 vec_safe_push (size_time_table, new_entry);
kono
parents:
diff changeset
222 }
kono
parents:
diff changeset
223 else
kono
parents:
diff changeset
224 {
kono
parents:
diff changeset
225 e->size += size;
kono
parents:
diff changeset
226 e->time += time;
kono
parents:
diff changeset
227 }
kono
parents:
diff changeset
228 }
kono
parents:
diff changeset
229
kono
parents:
diff changeset
230 /* We proved E to be unreachable, redirect it to __bultin_unreachable. */
kono
parents:
diff changeset
231
kono
parents:
diff changeset
232 static struct cgraph_edge *
kono
parents:
diff changeset
233 redirect_to_unreachable (struct cgraph_edge *e)
kono
parents:
diff changeset
234 {
kono
parents:
diff changeset
235 struct cgraph_node *callee = !e->inline_failed ? e->callee : NULL;
kono
parents:
diff changeset
236 struct cgraph_node *target = cgraph_node::get_create
kono
parents:
diff changeset
237 (builtin_decl_implicit (BUILT_IN_UNREACHABLE));
kono
parents:
diff changeset
238
kono
parents:
diff changeset
239 if (e->speculative)
kono
parents:
diff changeset
240 e = e->resolve_speculation (target->decl);
kono
parents:
diff changeset
241 else if (!e->callee)
kono
parents:
diff changeset
242 e->make_direct (target);
kono
parents:
diff changeset
243 else
kono
parents:
diff changeset
244 e->redirect_callee (target);
kono
parents:
diff changeset
245 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
246 e->inline_failed = CIF_UNREACHABLE;
kono
parents:
diff changeset
247 e->frequency = 0;
kono
parents:
diff changeset
248 e->count = profile_count::zero ();
kono
parents:
diff changeset
249 es->call_stmt_size = 0;
kono
parents:
diff changeset
250 es->call_stmt_time = 0;
kono
parents:
diff changeset
251 if (callee)
kono
parents:
diff changeset
252 callee->remove_symbol_and_inline_clones ();
kono
parents:
diff changeset
253 return e;
kono
parents:
diff changeset
254 }
kono
parents:
diff changeset
255
kono
parents:
diff changeset
256 /* Set predicate for edge E. */
kono
parents:
diff changeset
257
kono
parents:
diff changeset
258 static void
kono
parents:
diff changeset
259 edge_set_predicate (struct cgraph_edge *e, predicate *predicate)
kono
parents:
diff changeset
260 {
kono
parents:
diff changeset
261 /* If the edge is determined to be never executed, redirect it
kono
parents:
diff changeset
262 to BUILTIN_UNREACHABLE to make it clear to IPA passes the call will
kono
parents:
diff changeset
263 be optimized out. */
kono
parents:
diff changeset
264 if (predicate && *predicate == false
kono
parents:
diff changeset
265 /* When handling speculative edges, we need to do the redirection
kono
parents:
diff changeset
266 just once. Do it always on the direct edge, so we do not
kono
parents:
diff changeset
267 attempt to resolve speculation while duplicating the edge. */
kono
parents:
diff changeset
268 && (!e->speculative || e->callee))
kono
parents:
diff changeset
269 e = redirect_to_unreachable (e);
kono
parents:
diff changeset
270
kono
parents:
diff changeset
271 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
272 if (predicate && *predicate != true)
kono
parents:
diff changeset
273 {
kono
parents:
diff changeset
274 if (!es->predicate)
kono
parents:
diff changeset
275 es->predicate = edge_predicate_pool.allocate ();
kono
parents:
diff changeset
276 *es->predicate = *predicate;
kono
parents:
diff changeset
277 }
kono
parents:
diff changeset
278 else
kono
parents:
diff changeset
279 {
kono
parents:
diff changeset
280 if (es->predicate)
kono
parents:
diff changeset
281 edge_predicate_pool.remove (es->predicate);
kono
parents:
diff changeset
282 es->predicate = NULL;
kono
parents:
diff changeset
283 }
kono
parents:
diff changeset
284 }
kono
parents:
diff changeset
285
kono
parents:
diff changeset
286 /* Set predicate for hint *P. */
kono
parents:
diff changeset
287
kono
parents:
diff changeset
288 static void
kono
parents:
diff changeset
289 set_hint_predicate (predicate **p, predicate new_predicate)
kono
parents:
diff changeset
290 {
kono
parents:
diff changeset
291 if (new_predicate == false || new_predicate == true)
kono
parents:
diff changeset
292 {
kono
parents:
diff changeset
293 if (*p)
kono
parents:
diff changeset
294 edge_predicate_pool.remove (*p);
kono
parents:
diff changeset
295 *p = NULL;
kono
parents:
diff changeset
296 }
kono
parents:
diff changeset
297 else
kono
parents:
diff changeset
298 {
kono
parents:
diff changeset
299 if (!*p)
kono
parents:
diff changeset
300 *p = edge_predicate_pool.allocate ();
kono
parents:
diff changeset
301 **p = new_predicate;
kono
parents:
diff changeset
302 }
kono
parents:
diff changeset
303 }
kono
parents:
diff changeset
304
kono
parents:
diff changeset
305
kono
parents:
diff changeset
306 /* Compute what conditions may or may not hold given invormation about
kono
parents:
diff changeset
307 parameters. RET_CLAUSE returns truths that may hold in a specialized copy,
kono
parents:
diff changeset
308 whie RET_NONSPEC_CLAUSE returns truths that may hold in an nonspecialized
kono
parents:
diff changeset
309 copy when called in a given context. It is a bitmask of conditions. Bit
kono
parents:
diff changeset
310 0 means that condition is known to be false, while bit 1 means that condition
kono
parents:
diff changeset
311 may or may not be true. These differs - for example NOT_INLINED condition
kono
parents:
diff changeset
312 is always false in the second and also builtin_constant_p tests can not use
kono
parents:
diff changeset
313 the fact that parameter is indeed a constant.
kono
parents:
diff changeset
314
kono
parents:
diff changeset
315 KNOWN_VALS is partial mapping of parameters of NODE to constant values.
kono
parents:
diff changeset
316 KNOWN_AGGS is a vector of aggreggate jump functions for each parameter.
kono
parents:
diff changeset
317 Return clause of possible truths. When INLINE_P is true, assume that we are
kono
parents:
diff changeset
318 inlining.
kono
parents:
diff changeset
319
kono
parents:
diff changeset
320 ERROR_MARK means compile time invariant. */
kono
parents:
diff changeset
321
kono
parents:
diff changeset
322 static void
kono
parents:
diff changeset
323 evaluate_conditions_for_known_args (struct cgraph_node *node,
kono
parents:
diff changeset
324 bool inline_p,
kono
parents:
diff changeset
325 vec<tree> known_vals,
kono
parents:
diff changeset
326 vec<ipa_agg_jump_function_p>
kono
parents:
diff changeset
327 known_aggs,
kono
parents:
diff changeset
328 clause_t *ret_clause,
kono
parents:
diff changeset
329 clause_t *ret_nonspec_clause)
kono
parents:
diff changeset
330 {
kono
parents:
diff changeset
331 clause_t clause = inline_p ? 0 : 1 << predicate::not_inlined_condition;
kono
parents:
diff changeset
332 clause_t nonspec_clause = 1 << predicate::not_inlined_condition;
kono
parents:
diff changeset
333 struct ipa_fn_summary *info = ipa_fn_summaries->get (node);
kono
parents:
diff changeset
334 int i;
kono
parents:
diff changeset
335 struct condition *c;
kono
parents:
diff changeset
336
kono
parents:
diff changeset
337 for (i = 0; vec_safe_iterate (info->conds, i, &c); i++)
kono
parents:
diff changeset
338 {
kono
parents:
diff changeset
339 tree val;
kono
parents:
diff changeset
340 tree res;
kono
parents:
diff changeset
341
kono
parents:
diff changeset
342 /* We allow call stmt to have fewer arguments than the callee function
kono
parents:
diff changeset
343 (especially for K&R style programs). So bound check here (we assume
kono
parents:
diff changeset
344 known_aggs vector, if non-NULL, has the same length as
kono
parents:
diff changeset
345 known_vals). */
kono
parents:
diff changeset
346 gcc_checking_assert (!known_aggs.exists ()
kono
parents:
diff changeset
347 || (known_vals.length () == known_aggs.length ()));
kono
parents:
diff changeset
348 if (c->operand_num >= (int) known_vals.length ())
kono
parents:
diff changeset
349 {
kono
parents:
diff changeset
350 clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
351 nonspec_clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
352 continue;
kono
parents:
diff changeset
353 }
kono
parents:
diff changeset
354
kono
parents:
diff changeset
355 if (c->agg_contents)
kono
parents:
diff changeset
356 {
kono
parents:
diff changeset
357 struct ipa_agg_jump_function *agg;
kono
parents:
diff changeset
358
kono
parents:
diff changeset
359 if (c->code == predicate::changed
kono
parents:
diff changeset
360 && !c->by_ref
kono
parents:
diff changeset
361 && (known_vals[c->operand_num] == error_mark_node))
kono
parents:
diff changeset
362 continue;
kono
parents:
diff changeset
363
kono
parents:
diff changeset
364 if (known_aggs.exists ())
kono
parents:
diff changeset
365 {
kono
parents:
diff changeset
366 agg = known_aggs[c->operand_num];
kono
parents:
diff changeset
367 val = ipa_find_agg_cst_for_param (agg, known_vals[c->operand_num],
kono
parents:
diff changeset
368 c->offset, c->by_ref);
kono
parents:
diff changeset
369 }
kono
parents:
diff changeset
370 else
kono
parents:
diff changeset
371 val = NULL_TREE;
kono
parents:
diff changeset
372 }
kono
parents:
diff changeset
373 else
kono
parents:
diff changeset
374 {
kono
parents:
diff changeset
375 val = known_vals[c->operand_num];
kono
parents:
diff changeset
376 if (val == error_mark_node && c->code != predicate::changed)
kono
parents:
diff changeset
377 val = NULL_TREE;
kono
parents:
diff changeset
378 }
kono
parents:
diff changeset
379
kono
parents:
diff changeset
380 if (!val)
kono
parents:
diff changeset
381 {
kono
parents:
diff changeset
382 clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
383 nonspec_clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
384 continue;
kono
parents:
diff changeset
385 }
kono
parents:
diff changeset
386 if (c->code == predicate::changed)
kono
parents:
diff changeset
387 {
kono
parents:
diff changeset
388 nonspec_clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
389 continue;
kono
parents:
diff changeset
390 }
kono
parents:
diff changeset
391
kono
parents:
diff changeset
392 if (tree_to_shwi (TYPE_SIZE (TREE_TYPE (val))) != c->size)
kono
parents:
diff changeset
393 {
kono
parents:
diff changeset
394 clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
395 nonspec_clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
396 continue;
kono
parents:
diff changeset
397 }
kono
parents:
diff changeset
398 if (c->code == predicate::is_not_constant)
kono
parents:
diff changeset
399 {
kono
parents:
diff changeset
400 nonspec_clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
401 continue;
kono
parents:
diff changeset
402 }
kono
parents:
diff changeset
403
kono
parents:
diff changeset
404 val = fold_unary (VIEW_CONVERT_EXPR, TREE_TYPE (c->val), val);
kono
parents:
diff changeset
405 res = val
kono
parents:
diff changeset
406 ? fold_binary_to_constant (c->code, boolean_type_node, val, c->val)
kono
parents:
diff changeset
407 : NULL;
kono
parents:
diff changeset
408
kono
parents:
diff changeset
409 if (res && integer_zerop (res))
kono
parents:
diff changeset
410 continue;
kono
parents:
diff changeset
411
kono
parents:
diff changeset
412 clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
413 nonspec_clause |= 1 << (i + predicate::first_dynamic_condition);
kono
parents:
diff changeset
414 }
kono
parents:
diff changeset
415 *ret_clause = clause;
kono
parents:
diff changeset
416 if (ret_nonspec_clause)
kono
parents:
diff changeset
417 *ret_nonspec_clause = nonspec_clause;
kono
parents:
diff changeset
418 }
kono
parents:
diff changeset
419
kono
parents:
diff changeset
420
kono
parents:
diff changeset
421 /* Work out what conditions might be true at invocation of E. */
kono
parents:
diff changeset
422
kono
parents:
diff changeset
423 void
kono
parents:
diff changeset
424 evaluate_properties_for_edge (struct cgraph_edge *e, bool inline_p,
kono
parents:
diff changeset
425 clause_t *clause_ptr,
kono
parents:
diff changeset
426 clause_t *nonspec_clause_ptr,
kono
parents:
diff changeset
427 vec<tree> *known_vals_ptr,
kono
parents:
diff changeset
428 vec<ipa_polymorphic_call_context>
kono
parents:
diff changeset
429 *known_contexts_ptr,
kono
parents:
diff changeset
430 vec<ipa_agg_jump_function_p> *known_aggs_ptr)
kono
parents:
diff changeset
431 {
kono
parents:
diff changeset
432 struct cgraph_node *callee = e->callee->ultimate_alias_target ();
kono
parents:
diff changeset
433 struct ipa_fn_summary *info = ipa_fn_summaries->get (callee);
kono
parents:
diff changeset
434 vec<tree> known_vals = vNULL;
kono
parents:
diff changeset
435 vec<ipa_agg_jump_function_p> known_aggs = vNULL;
kono
parents:
diff changeset
436
kono
parents:
diff changeset
437 if (clause_ptr)
kono
parents:
diff changeset
438 *clause_ptr = inline_p ? 0 : 1 << predicate::not_inlined_condition;
kono
parents:
diff changeset
439 if (known_vals_ptr)
kono
parents:
diff changeset
440 known_vals_ptr->create (0);
kono
parents:
diff changeset
441 if (known_contexts_ptr)
kono
parents:
diff changeset
442 known_contexts_ptr->create (0);
kono
parents:
diff changeset
443
kono
parents:
diff changeset
444 if (ipa_node_params_sum
kono
parents:
diff changeset
445 && !e->call_stmt_cannot_inline_p
kono
parents:
diff changeset
446 && ((clause_ptr && info->conds) || known_vals_ptr || known_contexts_ptr))
kono
parents:
diff changeset
447 {
kono
parents:
diff changeset
448 struct ipa_node_params *parms_info;
kono
parents:
diff changeset
449 struct ipa_edge_args *args = IPA_EDGE_REF (e);
kono
parents:
diff changeset
450 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
451 int i, count = ipa_get_cs_argument_count (args);
kono
parents:
diff changeset
452
kono
parents:
diff changeset
453 if (e->caller->global.inlined_to)
kono
parents:
diff changeset
454 parms_info = IPA_NODE_REF (e->caller->global.inlined_to);
kono
parents:
diff changeset
455 else
kono
parents:
diff changeset
456 parms_info = IPA_NODE_REF (e->caller);
kono
parents:
diff changeset
457
kono
parents:
diff changeset
458 if (count && (info->conds || known_vals_ptr))
kono
parents:
diff changeset
459 known_vals.safe_grow_cleared (count);
kono
parents:
diff changeset
460 if (count && (info->conds || known_aggs_ptr))
kono
parents:
diff changeset
461 known_aggs.safe_grow_cleared (count);
kono
parents:
diff changeset
462 if (count && known_contexts_ptr)
kono
parents:
diff changeset
463 known_contexts_ptr->safe_grow_cleared (count);
kono
parents:
diff changeset
464
kono
parents:
diff changeset
465 for (i = 0; i < count; i++)
kono
parents:
diff changeset
466 {
kono
parents:
diff changeset
467 struct ipa_jump_func *jf = ipa_get_ith_jump_func (args, i);
kono
parents:
diff changeset
468 tree cst = ipa_value_from_jfunc (parms_info, jf);
kono
parents:
diff changeset
469
kono
parents:
diff changeset
470 if (!cst && e->call_stmt
kono
parents:
diff changeset
471 && i < (int)gimple_call_num_args (e->call_stmt))
kono
parents:
diff changeset
472 {
kono
parents:
diff changeset
473 cst = gimple_call_arg (e->call_stmt, i);
kono
parents:
diff changeset
474 if (!is_gimple_min_invariant (cst))
kono
parents:
diff changeset
475 cst = NULL;
kono
parents:
diff changeset
476 }
kono
parents:
diff changeset
477 if (cst)
kono
parents:
diff changeset
478 {
kono
parents:
diff changeset
479 gcc_checking_assert (TREE_CODE (cst) != TREE_BINFO);
kono
parents:
diff changeset
480 if (known_vals.exists ())
kono
parents:
diff changeset
481 known_vals[i] = cst;
kono
parents:
diff changeset
482 }
kono
parents:
diff changeset
483 else if (inline_p && !es->param[i].change_prob)
kono
parents:
diff changeset
484 known_vals[i] = error_mark_node;
kono
parents:
diff changeset
485
kono
parents:
diff changeset
486 if (known_contexts_ptr)
kono
parents:
diff changeset
487 (*known_contexts_ptr)[i] = ipa_context_from_jfunc (parms_info, e,
kono
parents:
diff changeset
488 i, jf);
kono
parents:
diff changeset
489 /* TODO: When IPA-CP starts propagating and merging aggregate jump
kono
parents:
diff changeset
490 functions, use its knowledge of the caller too, just like the
kono
parents:
diff changeset
491 scalar case above. */
kono
parents:
diff changeset
492 known_aggs[i] = &jf->agg;
kono
parents:
diff changeset
493 }
kono
parents:
diff changeset
494 }
kono
parents:
diff changeset
495 else if (e->call_stmt && !e->call_stmt_cannot_inline_p
kono
parents:
diff changeset
496 && ((clause_ptr && info->conds) || known_vals_ptr))
kono
parents:
diff changeset
497 {
kono
parents:
diff changeset
498 int i, count = (int)gimple_call_num_args (e->call_stmt);
kono
parents:
diff changeset
499
kono
parents:
diff changeset
500 if (count && (info->conds || known_vals_ptr))
kono
parents:
diff changeset
501 known_vals.safe_grow_cleared (count);
kono
parents:
diff changeset
502 for (i = 0; i < count; i++)
kono
parents:
diff changeset
503 {
kono
parents:
diff changeset
504 tree cst = gimple_call_arg (e->call_stmt, i);
kono
parents:
diff changeset
505 if (!is_gimple_min_invariant (cst))
kono
parents:
diff changeset
506 cst = NULL;
kono
parents:
diff changeset
507 if (cst)
kono
parents:
diff changeset
508 known_vals[i] = cst;
kono
parents:
diff changeset
509 }
kono
parents:
diff changeset
510 }
kono
parents:
diff changeset
511
kono
parents:
diff changeset
512 evaluate_conditions_for_known_args (callee, inline_p,
kono
parents:
diff changeset
513 known_vals, known_aggs, clause_ptr,
kono
parents:
diff changeset
514 nonspec_clause_ptr);
kono
parents:
diff changeset
515
kono
parents:
diff changeset
516 if (known_vals_ptr)
kono
parents:
diff changeset
517 *known_vals_ptr = known_vals;
kono
parents:
diff changeset
518 else
kono
parents:
diff changeset
519 known_vals.release ();
kono
parents:
diff changeset
520
kono
parents:
diff changeset
521 if (known_aggs_ptr)
kono
parents:
diff changeset
522 *known_aggs_ptr = known_aggs;
kono
parents:
diff changeset
523 else
kono
parents:
diff changeset
524 known_aggs.release ();
kono
parents:
diff changeset
525 }
kono
parents:
diff changeset
526
kono
parents:
diff changeset
527
kono
parents:
diff changeset
528 /* Allocate the function summary. */
kono
parents:
diff changeset
529
kono
parents:
diff changeset
530 static void
kono
parents:
diff changeset
531 ipa_fn_summary_alloc (void)
kono
parents:
diff changeset
532 {
kono
parents:
diff changeset
533 gcc_checking_assert (!ipa_fn_summaries);
kono
parents:
diff changeset
534 ipa_fn_summaries = ipa_fn_summary_t::create_ggc (symtab);
kono
parents:
diff changeset
535 ipa_call_summaries = new ipa_call_summary_t (symtab, false);
kono
parents:
diff changeset
536 }
kono
parents:
diff changeset
537
kono
parents:
diff changeset
538 /* We are called multiple time for given function; clear
kono
parents:
diff changeset
539 data from previous run so they are not cumulated. */
kono
parents:
diff changeset
540
kono
parents:
diff changeset
541 void
kono
parents:
diff changeset
542 ipa_call_summary::reset ()
kono
parents:
diff changeset
543 {
kono
parents:
diff changeset
544 call_stmt_size = call_stmt_time = 0;
kono
parents:
diff changeset
545 if (predicate)
kono
parents:
diff changeset
546 edge_predicate_pool.remove (predicate);
kono
parents:
diff changeset
547 predicate = NULL;
kono
parents:
diff changeset
548 param.release ();
kono
parents:
diff changeset
549 }
kono
parents:
diff changeset
550
kono
parents:
diff changeset
551 /* We are called multiple time for given function; clear
kono
parents:
diff changeset
552 data from previous run so they are not cumulated. */
kono
parents:
diff changeset
553
kono
parents:
diff changeset
554 void
kono
parents:
diff changeset
555 ipa_fn_summary::reset (struct cgraph_node *node)
kono
parents:
diff changeset
556 {
kono
parents:
diff changeset
557 struct cgraph_edge *e;
kono
parents:
diff changeset
558
kono
parents:
diff changeset
559 self_size = 0;
kono
parents:
diff changeset
560 estimated_stack_size = 0;
kono
parents:
diff changeset
561 estimated_self_stack_size = 0;
kono
parents:
diff changeset
562 stack_frame_offset = 0;
kono
parents:
diff changeset
563 size = 0;
kono
parents:
diff changeset
564 time = 0;
kono
parents:
diff changeset
565 growth = 0;
kono
parents:
diff changeset
566 scc_no = 0;
kono
parents:
diff changeset
567 if (loop_iterations)
kono
parents:
diff changeset
568 {
kono
parents:
diff changeset
569 edge_predicate_pool.remove (loop_iterations);
kono
parents:
diff changeset
570 loop_iterations = NULL;
kono
parents:
diff changeset
571 }
kono
parents:
diff changeset
572 if (loop_stride)
kono
parents:
diff changeset
573 {
kono
parents:
diff changeset
574 edge_predicate_pool.remove (loop_stride);
kono
parents:
diff changeset
575 loop_stride = NULL;
kono
parents:
diff changeset
576 }
kono
parents:
diff changeset
577 if (array_index)
kono
parents:
diff changeset
578 {
kono
parents:
diff changeset
579 edge_predicate_pool.remove (array_index);
kono
parents:
diff changeset
580 array_index = NULL;
kono
parents:
diff changeset
581 }
kono
parents:
diff changeset
582 vec_free (conds);
kono
parents:
diff changeset
583 vec_free (size_time_table);
kono
parents:
diff changeset
584 for (e = node->callees; e; e = e->next_callee)
kono
parents:
diff changeset
585 ipa_call_summaries->get (e)->reset ();
kono
parents:
diff changeset
586 for (e = node->indirect_calls; e; e = e->next_callee)
kono
parents:
diff changeset
587 ipa_call_summaries->get (e)->reset ();
kono
parents:
diff changeset
588 fp_expressions = false;
kono
parents:
diff changeset
589 }
kono
parents:
diff changeset
590
kono
parents:
diff changeset
591 /* Hook that is called by cgraph.c when a node is removed. */
kono
parents:
diff changeset
592
kono
parents:
diff changeset
593 void
kono
parents:
diff changeset
594 ipa_fn_summary_t::remove (cgraph_node *node, ipa_fn_summary *info)
kono
parents:
diff changeset
595 {
kono
parents:
diff changeset
596 info->reset (node);
kono
parents:
diff changeset
597 }
kono
parents:
diff changeset
598
kono
parents:
diff changeset
599 /* Same as remap_predicate_after_duplication but handle hint predicate *P.
kono
parents:
diff changeset
600 Additionally care about allocating new memory slot for updated predicate
kono
parents:
diff changeset
601 and set it to NULL when it becomes true or false (and thus uninteresting).
kono
parents:
diff changeset
602 */
kono
parents:
diff changeset
603
kono
parents:
diff changeset
604 static void
kono
parents:
diff changeset
605 remap_hint_predicate_after_duplication (predicate **p,
kono
parents:
diff changeset
606 clause_t possible_truths)
kono
parents:
diff changeset
607 {
kono
parents:
diff changeset
608 predicate new_predicate;
kono
parents:
diff changeset
609
kono
parents:
diff changeset
610 if (!*p)
kono
parents:
diff changeset
611 return;
kono
parents:
diff changeset
612
kono
parents:
diff changeset
613 new_predicate = (*p)->remap_after_duplication (possible_truths);
kono
parents:
diff changeset
614 /* We do not want to free previous predicate; it is used by node origin. */
kono
parents:
diff changeset
615 *p = NULL;
kono
parents:
diff changeset
616 set_hint_predicate (p, new_predicate);
kono
parents:
diff changeset
617 }
kono
parents:
diff changeset
618
kono
parents:
diff changeset
619
kono
parents:
diff changeset
620 /* Hook that is called by cgraph.c when a node is duplicated. */
kono
parents:
diff changeset
621 void
kono
parents:
diff changeset
622 ipa_fn_summary_t::duplicate (cgraph_node *src,
kono
parents:
diff changeset
623 cgraph_node *dst,
kono
parents:
diff changeset
624 ipa_fn_summary *,
kono
parents:
diff changeset
625 ipa_fn_summary *info)
kono
parents:
diff changeset
626 {
kono
parents:
diff changeset
627 memcpy (info, ipa_fn_summaries->get (src), sizeof (ipa_fn_summary));
kono
parents:
diff changeset
628 /* TODO: as an optimization, we may avoid copying conditions
kono
parents:
diff changeset
629 that are known to be false or true. */
kono
parents:
diff changeset
630 info->conds = vec_safe_copy (info->conds);
kono
parents:
diff changeset
631
kono
parents:
diff changeset
632 /* When there are any replacements in the function body, see if we can figure
kono
parents:
diff changeset
633 out that something was optimized out. */
kono
parents:
diff changeset
634 if (ipa_node_params_sum && dst->clone.tree_map)
kono
parents:
diff changeset
635 {
kono
parents:
diff changeset
636 vec<size_time_entry, va_gc> *entry = info->size_time_table;
kono
parents:
diff changeset
637 /* Use SRC parm info since it may not be copied yet. */
kono
parents:
diff changeset
638 struct ipa_node_params *parms_info = IPA_NODE_REF (src);
kono
parents:
diff changeset
639 vec<tree> known_vals = vNULL;
kono
parents:
diff changeset
640 int count = ipa_get_param_count (parms_info);
kono
parents:
diff changeset
641 int i, j;
kono
parents:
diff changeset
642 clause_t possible_truths;
kono
parents:
diff changeset
643 predicate true_pred = true;
kono
parents:
diff changeset
644 size_time_entry *e;
kono
parents:
diff changeset
645 int optimized_out_size = 0;
kono
parents:
diff changeset
646 bool inlined_to_p = false;
kono
parents:
diff changeset
647 struct cgraph_edge *edge, *next;
kono
parents:
diff changeset
648
kono
parents:
diff changeset
649 info->size_time_table = 0;
kono
parents:
diff changeset
650 known_vals.safe_grow_cleared (count);
kono
parents:
diff changeset
651 for (i = 0; i < count; i++)
kono
parents:
diff changeset
652 {
kono
parents:
diff changeset
653 struct ipa_replace_map *r;
kono
parents:
diff changeset
654
kono
parents:
diff changeset
655 for (j = 0; vec_safe_iterate (dst->clone.tree_map, j, &r); j++)
kono
parents:
diff changeset
656 {
kono
parents:
diff changeset
657 if (((!r->old_tree && r->parm_num == i)
kono
parents:
diff changeset
658 || (r->old_tree && r->old_tree == ipa_get_param (parms_info, i)))
kono
parents:
diff changeset
659 && r->replace_p && !r->ref_p)
kono
parents:
diff changeset
660 {
kono
parents:
diff changeset
661 known_vals[i] = r->new_tree;
kono
parents:
diff changeset
662 break;
kono
parents:
diff changeset
663 }
kono
parents:
diff changeset
664 }
kono
parents:
diff changeset
665 }
kono
parents:
diff changeset
666 evaluate_conditions_for_known_args (dst, false,
kono
parents:
diff changeset
667 known_vals,
kono
parents:
diff changeset
668 vNULL,
kono
parents:
diff changeset
669 &possible_truths,
kono
parents:
diff changeset
670 /* We are going to specialize,
kono
parents:
diff changeset
671 so ignore nonspec truths. */
kono
parents:
diff changeset
672 NULL);
kono
parents:
diff changeset
673 known_vals.release ();
kono
parents:
diff changeset
674
kono
parents:
diff changeset
675 info->account_size_time (0, 0, true_pred, true_pred);
kono
parents:
diff changeset
676
kono
parents:
diff changeset
677 /* Remap size_time vectors.
kono
parents:
diff changeset
678 Simplify the predicate by prunning out alternatives that are known
kono
parents:
diff changeset
679 to be false.
kono
parents:
diff changeset
680 TODO: as on optimization, we can also eliminate conditions known
kono
parents:
diff changeset
681 to be true. */
kono
parents:
diff changeset
682 for (i = 0; vec_safe_iterate (entry, i, &e); i++)
kono
parents:
diff changeset
683 {
kono
parents:
diff changeset
684 predicate new_exec_pred;
kono
parents:
diff changeset
685 predicate new_nonconst_pred;
kono
parents:
diff changeset
686 new_exec_pred = e->exec_predicate.remap_after_duplication
kono
parents:
diff changeset
687 (possible_truths);
kono
parents:
diff changeset
688 new_nonconst_pred = e->nonconst_predicate.remap_after_duplication
kono
parents:
diff changeset
689 (possible_truths);
kono
parents:
diff changeset
690 if (new_exec_pred == false || new_nonconst_pred == false)
kono
parents:
diff changeset
691 optimized_out_size += e->size;
kono
parents:
diff changeset
692 else
kono
parents:
diff changeset
693 info->account_size_time (e->size, e->time, new_exec_pred,
kono
parents:
diff changeset
694 new_nonconst_pred);
kono
parents:
diff changeset
695 }
kono
parents:
diff changeset
696
kono
parents:
diff changeset
697 /* Remap edge predicates with the same simplification as above.
kono
parents:
diff changeset
698 Also copy constantness arrays. */
kono
parents:
diff changeset
699 for (edge = dst->callees; edge; edge = next)
kono
parents:
diff changeset
700 {
kono
parents:
diff changeset
701 predicate new_predicate;
kono
parents:
diff changeset
702 struct ipa_call_summary *es = ipa_call_summaries->get (edge);
kono
parents:
diff changeset
703 next = edge->next_callee;
kono
parents:
diff changeset
704
kono
parents:
diff changeset
705 if (!edge->inline_failed)
kono
parents:
diff changeset
706 inlined_to_p = true;
kono
parents:
diff changeset
707 if (!es->predicate)
kono
parents:
diff changeset
708 continue;
kono
parents:
diff changeset
709 new_predicate = es->predicate->remap_after_duplication
kono
parents:
diff changeset
710 (possible_truths);
kono
parents:
diff changeset
711 if (new_predicate == false && *es->predicate != false)
kono
parents:
diff changeset
712 optimized_out_size += es->call_stmt_size * ipa_fn_summary::size_scale;
kono
parents:
diff changeset
713 edge_set_predicate (edge, &new_predicate);
kono
parents:
diff changeset
714 }
kono
parents:
diff changeset
715
kono
parents:
diff changeset
716 /* Remap indirect edge predicates with the same simplificaiton as above.
kono
parents:
diff changeset
717 Also copy constantness arrays. */
kono
parents:
diff changeset
718 for (edge = dst->indirect_calls; edge; edge = next)
kono
parents:
diff changeset
719 {
kono
parents:
diff changeset
720 predicate new_predicate;
kono
parents:
diff changeset
721 struct ipa_call_summary *es = ipa_call_summaries->get (edge);
kono
parents:
diff changeset
722 next = edge->next_callee;
kono
parents:
diff changeset
723
kono
parents:
diff changeset
724 gcc_checking_assert (edge->inline_failed);
kono
parents:
diff changeset
725 if (!es->predicate)
kono
parents:
diff changeset
726 continue;
kono
parents:
diff changeset
727 new_predicate = es->predicate->remap_after_duplication
kono
parents:
diff changeset
728 (possible_truths);
kono
parents:
diff changeset
729 if (new_predicate == false && *es->predicate != false)
kono
parents:
diff changeset
730 optimized_out_size += es->call_stmt_size * ipa_fn_summary::size_scale;
kono
parents:
diff changeset
731 edge_set_predicate (edge, &new_predicate);
kono
parents:
diff changeset
732 }
kono
parents:
diff changeset
733 remap_hint_predicate_after_duplication (&info->loop_iterations,
kono
parents:
diff changeset
734 possible_truths);
kono
parents:
diff changeset
735 remap_hint_predicate_after_duplication (&info->loop_stride,
kono
parents:
diff changeset
736 possible_truths);
kono
parents:
diff changeset
737 remap_hint_predicate_after_duplication (&info->array_index,
kono
parents:
diff changeset
738 possible_truths);
kono
parents:
diff changeset
739
kono
parents:
diff changeset
740 /* If inliner or someone after inliner will ever start producing
kono
parents:
diff changeset
741 non-trivial clones, we will get trouble with lack of information
kono
parents:
diff changeset
742 about updating self sizes, because size vectors already contains
kono
parents:
diff changeset
743 sizes of the calees. */
kono
parents:
diff changeset
744 gcc_assert (!inlined_to_p || !optimized_out_size);
kono
parents:
diff changeset
745 }
kono
parents:
diff changeset
746 else
kono
parents:
diff changeset
747 {
kono
parents:
diff changeset
748 info->size_time_table = vec_safe_copy (info->size_time_table);
kono
parents:
diff changeset
749 if (info->loop_iterations)
kono
parents:
diff changeset
750 {
kono
parents:
diff changeset
751 predicate p = *info->loop_iterations;
kono
parents:
diff changeset
752 info->loop_iterations = NULL;
kono
parents:
diff changeset
753 set_hint_predicate (&info->loop_iterations, p);
kono
parents:
diff changeset
754 }
kono
parents:
diff changeset
755 if (info->loop_stride)
kono
parents:
diff changeset
756 {
kono
parents:
diff changeset
757 predicate p = *info->loop_stride;
kono
parents:
diff changeset
758 info->loop_stride = NULL;
kono
parents:
diff changeset
759 set_hint_predicate (&info->loop_stride, p);
kono
parents:
diff changeset
760 }
kono
parents:
diff changeset
761 if (info->array_index)
kono
parents:
diff changeset
762 {
kono
parents:
diff changeset
763 predicate p = *info->array_index;
kono
parents:
diff changeset
764 info->array_index = NULL;
kono
parents:
diff changeset
765 set_hint_predicate (&info->array_index, p);
kono
parents:
diff changeset
766 }
kono
parents:
diff changeset
767 }
kono
parents:
diff changeset
768 if (!dst->global.inlined_to)
kono
parents:
diff changeset
769 ipa_update_overall_fn_summary (dst);
kono
parents:
diff changeset
770 }
kono
parents:
diff changeset
771
kono
parents:
diff changeset
772
kono
parents:
diff changeset
773 /* Hook that is called by cgraph.c when a node is duplicated. */
kono
parents:
diff changeset
774
kono
parents:
diff changeset
775 void
kono
parents:
diff changeset
776 ipa_call_summary_t::duplicate (struct cgraph_edge *src,
kono
parents:
diff changeset
777 struct cgraph_edge *dst,
kono
parents:
diff changeset
778 struct ipa_call_summary *srcinfo,
kono
parents:
diff changeset
779 struct ipa_call_summary *info)
kono
parents:
diff changeset
780 {
kono
parents:
diff changeset
781 *info = *srcinfo;
kono
parents:
diff changeset
782 info->predicate = NULL;
kono
parents:
diff changeset
783 edge_set_predicate (dst, srcinfo->predicate);
kono
parents:
diff changeset
784 info->param = srcinfo->param.copy ();
kono
parents:
diff changeset
785 if (!dst->indirect_unknown_callee && src->indirect_unknown_callee)
kono
parents:
diff changeset
786 {
kono
parents:
diff changeset
787 info->call_stmt_size -= (eni_size_weights.indirect_call_cost
kono
parents:
diff changeset
788 - eni_size_weights.call_cost);
kono
parents:
diff changeset
789 info->call_stmt_time -= (eni_time_weights.indirect_call_cost
kono
parents:
diff changeset
790 - eni_time_weights.call_cost);
kono
parents:
diff changeset
791 }
kono
parents:
diff changeset
792 }
kono
parents:
diff changeset
793
kono
parents:
diff changeset
794
kono
parents:
diff changeset
795 /* Keep edge cache consistent across edge removal. */
kono
parents:
diff changeset
796
kono
parents:
diff changeset
797 void
kono
parents:
diff changeset
798 ipa_call_summary_t::remove (struct cgraph_edge *,
kono
parents:
diff changeset
799 struct ipa_call_summary *sum)
kono
parents:
diff changeset
800 {
kono
parents:
diff changeset
801 sum->reset ();
kono
parents:
diff changeset
802 }
kono
parents:
diff changeset
803
kono
parents:
diff changeset
804
kono
parents:
diff changeset
805 /* Dump edge summaries associated to NODE and recursively to all clones.
kono
parents:
diff changeset
806 Indent by INDENT. */
kono
parents:
diff changeset
807
kono
parents:
diff changeset
808 static void
kono
parents:
diff changeset
809 dump_ipa_call_summary (FILE *f, int indent, struct cgraph_node *node,
kono
parents:
diff changeset
810 struct ipa_fn_summary *info)
kono
parents:
diff changeset
811 {
kono
parents:
diff changeset
812 struct cgraph_edge *edge;
kono
parents:
diff changeset
813 for (edge = node->callees; edge; edge = edge->next_callee)
kono
parents:
diff changeset
814 {
kono
parents:
diff changeset
815 struct ipa_call_summary *es = ipa_call_summaries->get (edge);
kono
parents:
diff changeset
816 struct cgraph_node *callee = edge->callee->ultimate_alias_target ();
kono
parents:
diff changeset
817 int i;
kono
parents:
diff changeset
818
kono
parents:
diff changeset
819 fprintf (f,
kono
parents:
diff changeset
820 "%*s%s/%i %s\n%*s loop depth:%2i freq:%4i size:%2i"
kono
parents:
diff changeset
821 " time: %2i callee size:%2i stack:%2i",
kono
parents:
diff changeset
822 indent, "", callee->name (), callee->order,
kono
parents:
diff changeset
823 !edge->inline_failed
kono
parents:
diff changeset
824 ? "inlined" : cgraph_inline_failed_string (edge-> inline_failed),
kono
parents:
diff changeset
825 indent, "", es->loop_depth, edge->frequency,
kono
parents:
diff changeset
826 es->call_stmt_size, es->call_stmt_time,
kono
parents:
diff changeset
827 (int) ipa_fn_summaries->get (callee)->size / ipa_fn_summary::size_scale,
kono
parents:
diff changeset
828 (int) ipa_fn_summaries->get (callee)->estimated_stack_size);
kono
parents:
diff changeset
829
kono
parents:
diff changeset
830 if (es->predicate)
kono
parents:
diff changeset
831 {
kono
parents:
diff changeset
832 fprintf (f, " predicate: ");
kono
parents:
diff changeset
833 es->predicate->dump (f, info->conds);
kono
parents:
diff changeset
834 }
kono
parents:
diff changeset
835 else
kono
parents:
diff changeset
836 fprintf (f, "\n");
kono
parents:
diff changeset
837 if (es->param.exists ())
kono
parents:
diff changeset
838 for (i = 0; i < (int) es->param.length (); i++)
kono
parents:
diff changeset
839 {
kono
parents:
diff changeset
840 int prob = es->param[i].change_prob;
kono
parents:
diff changeset
841
kono
parents:
diff changeset
842 if (!prob)
kono
parents:
diff changeset
843 fprintf (f, "%*s op%i is compile time invariant\n",
kono
parents:
diff changeset
844 indent + 2, "", i);
kono
parents:
diff changeset
845 else if (prob != REG_BR_PROB_BASE)
kono
parents:
diff changeset
846 fprintf (f, "%*s op%i change %f%% of time\n", indent + 2, "", i,
kono
parents:
diff changeset
847 prob * 100.0 / REG_BR_PROB_BASE);
kono
parents:
diff changeset
848 }
kono
parents:
diff changeset
849 if (!edge->inline_failed)
kono
parents:
diff changeset
850 {
kono
parents:
diff changeset
851 fprintf (f, "%*sStack frame offset %i, callee self size %i,"
kono
parents:
diff changeset
852 " callee size %i\n",
kono
parents:
diff changeset
853 indent + 2, "",
kono
parents:
diff changeset
854 (int) ipa_fn_summaries->get (callee)->stack_frame_offset,
kono
parents:
diff changeset
855 (int) ipa_fn_summaries->get (callee)->estimated_self_stack_size,
kono
parents:
diff changeset
856 (int) ipa_fn_summaries->get (callee)->estimated_stack_size);
kono
parents:
diff changeset
857 dump_ipa_call_summary (f, indent + 2, callee, info);
kono
parents:
diff changeset
858 }
kono
parents:
diff changeset
859 }
kono
parents:
diff changeset
860 for (edge = node->indirect_calls; edge; edge = edge->next_callee)
kono
parents:
diff changeset
861 {
kono
parents:
diff changeset
862 struct ipa_call_summary *es = ipa_call_summaries->get (edge);
kono
parents:
diff changeset
863 fprintf (f, "%*sindirect call loop depth:%2i freq:%4i size:%2i"
kono
parents:
diff changeset
864 " time: %2i",
kono
parents:
diff changeset
865 indent, "",
kono
parents:
diff changeset
866 es->loop_depth,
kono
parents:
diff changeset
867 edge->frequency, es->call_stmt_size, es->call_stmt_time);
kono
parents:
diff changeset
868 if (es->predicate)
kono
parents:
diff changeset
869 {
kono
parents:
diff changeset
870 fprintf (f, "predicate: ");
kono
parents:
diff changeset
871 es->predicate->dump (f, info->conds);
kono
parents:
diff changeset
872 }
kono
parents:
diff changeset
873 else
kono
parents:
diff changeset
874 fprintf (f, "\n");
kono
parents:
diff changeset
875 }
kono
parents:
diff changeset
876 }
kono
parents:
diff changeset
877
kono
parents:
diff changeset
878
kono
parents:
diff changeset
879 void
kono
parents:
diff changeset
880 ipa_dump_fn_summary (FILE *f, struct cgraph_node *node)
kono
parents:
diff changeset
881 {
kono
parents:
diff changeset
882 if (node->definition)
kono
parents:
diff changeset
883 {
kono
parents:
diff changeset
884 struct ipa_fn_summary *s = ipa_fn_summaries->get (node);
kono
parents:
diff changeset
885 size_time_entry *e;
kono
parents:
diff changeset
886 int i;
kono
parents:
diff changeset
887 fprintf (f, "IPA function summary for %s/%i", node->name (),
kono
parents:
diff changeset
888 node->order);
kono
parents:
diff changeset
889 if (DECL_DISREGARD_INLINE_LIMITS (node->decl))
kono
parents:
diff changeset
890 fprintf (f, " always_inline");
kono
parents:
diff changeset
891 if (s->inlinable)
kono
parents:
diff changeset
892 fprintf (f, " inlinable");
kono
parents:
diff changeset
893 if (s->contains_cilk_spawn)
kono
parents:
diff changeset
894 fprintf (f, " contains_cilk_spawn");
kono
parents:
diff changeset
895 if (s->fp_expressions)
kono
parents:
diff changeset
896 fprintf (f, " fp_expression");
kono
parents:
diff changeset
897 fprintf (f, "\n global time: %f\n", s->time.to_double ());
kono
parents:
diff changeset
898 fprintf (f, " self size: %i\n", s->self_size);
kono
parents:
diff changeset
899 fprintf (f, " global size: %i\n", s->size);
kono
parents:
diff changeset
900 fprintf (f, " min size: %i\n", s->min_size);
kono
parents:
diff changeset
901 fprintf (f, " self stack: %i\n",
kono
parents:
diff changeset
902 (int) s->estimated_self_stack_size);
kono
parents:
diff changeset
903 fprintf (f, " global stack: %i\n", (int) s->estimated_stack_size);
kono
parents:
diff changeset
904 if (s->growth)
kono
parents:
diff changeset
905 fprintf (f, " estimated growth:%i\n", (int) s->growth);
kono
parents:
diff changeset
906 if (s->scc_no)
kono
parents:
diff changeset
907 fprintf (f, " In SCC: %i\n", (int) s->scc_no);
kono
parents:
diff changeset
908 for (i = 0; vec_safe_iterate (s->size_time_table, i, &e); i++)
kono
parents:
diff changeset
909 {
kono
parents:
diff changeset
910 fprintf (f, " size:%f, time:%f",
kono
parents:
diff changeset
911 (double) e->size / ipa_fn_summary::size_scale,
kono
parents:
diff changeset
912 e->time.to_double ());
kono
parents:
diff changeset
913 if (e->exec_predicate != true)
kono
parents:
diff changeset
914 {
kono
parents:
diff changeset
915 fprintf (f, ", executed if:");
kono
parents:
diff changeset
916 e->exec_predicate.dump (f, s->conds, 0);
kono
parents:
diff changeset
917 }
kono
parents:
diff changeset
918 if (e->exec_predicate != e->nonconst_predicate)
kono
parents:
diff changeset
919 {
kono
parents:
diff changeset
920 fprintf (f, ", nonconst if:");
kono
parents:
diff changeset
921 e->nonconst_predicate.dump (f, s->conds, 0);
kono
parents:
diff changeset
922 }
kono
parents:
diff changeset
923 fprintf (f, "\n");
kono
parents:
diff changeset
924 }
kono
parents:
diff changeset
925 if (s->loop_iterations)
kono
parents:
diff changeset
926 {
kono
parents:
diff changeset
927 fprintf (f, " loop iterations:");
kono
parents:
diff changeset
928 s->loop_iterations->dump (f, s->conds);
kono
parents:
diff changeset
929 }
kono
parents:
diff changeset
930 if (s->loop_stride)
kono
parents:
diff changeset
931 {
kono
parents:
diff changeset
932 fprintf (f, " loop stride:");
kono
parents:
diff changeset
933 s->loop_stride->dump (f, s->conds);
kono
parents:
diff changeset
934 }
kono
parents:
diff changeset
935 if (s->array_index)
kono
parents:
diff changeset
936 {
kono
parents:
diff changeset
937 fprintf (f, " array index:");
kono
parents:
diff changeset
938 s->array_index->dump (f, s->conds);
kono
parents:
diff changeset
939 }
kono
parents:
diff changeset
940 fprintf (f, " calls:\n");
kono
parents:
diff changeset
941 dump_ipa_call_summary (f, 4, node, s);
kono
parents:
diff changeset
942 fprintf (f, "\n");
kono
parents:
diff changeset
943 }
kono
parents:
diff changeset
944 }
kono
parents:
diff changeset
945
kono
parents:
diff changeset
946 DEBUG_FUNCTION void
kono
parents:
diff changeset
947 ipa_debug_fn_summary (struct cgraph_node *node)
kono
parents:
diff changeset
948 {
kono
parents:
diff changeset
949 ipa_dump_fn_summary (stderr, node);
kono
parents:
diff changeset
950 }
kono
parents:
diff changeset
951
kono
parents:
diff changeset
952 void
kono
parents:
diff changeset
953 ipa_dump_fn_summaries (FILE *f)
kono
parents:
diff changeset
954 {
kono
parents:
diff changeset
955 struct cgraph_node *node;
kono
parents:
diff changeset
956
kono
parents:
diff changeset
957 FOR_EACH_DEFINED_FUNCTION (node)
kono
parents:
diff changeset
958 if (!node->global.inlined_to)
kono
parents:
diff changeset
959 ipa_dump_fn_summary (f, node);
kono
parents:
diff changeset
960 }
kono
parents:
diff changeset
961
kono
parents:
diff changeset
962 /* Callback of walk_aliased_vdefs. Flags that it has been invoked to the
kono
parents:
diff changeset
963 boolean variable pointed to by DATA. */
kono
parents:
diff changeset
964
kono
parents:
diff changeset
965 static bool
kono
parents:
diff changeset
966 mark_modified (ao_ref *ao ATTRIBUTE_UNUSED, tree vdef ATTRIBUTE_UNUSED,
kono
parents:
diff changeset
967 void *data)
kono
parents:
diff changeset
968 {
kono
parents:
diff changeset
969 bool *b = (bool *) data;
kono
parents:
diff changeset
970 *b = true;
kono
parents:
diff changeset
971 return true;
kono
parents:
diff changeset
972 }
kono
parents:
diff changeset
973
kono
parents:
diff changeset
974 /* If OP refers to value of function parameter, return the corresponding
kono
parents:
diff changeset
975 parameter. If non-NULL, the size of the memory load (or the SSA_NAME of the
kono
parents:
diff changeset
976 PARM_DECL) will be stored to *SIZE_P in that case too. */
kono
parents:
diff changeset
977
kono
parents:
diff changeset
978 static tree
kono
parents:
diff changeset
979 unmodified_parm_1 (gimple *stmt, tree op, HOST_WIDE_INT *size_p)
kono
parents:
diff changeset
980 {
kono
parents:
diff changeset
981 /* SSA_NAME referring to parm default def? */
kono
parents:
diff changeset
982 if (TREE_CODE (op) == SSA_NAME
kono
parents:
diff changeset
983 && SSA_NAME_IS_DEFAULT_DEF (op)
kono
parents:
diff changeset
984 && TREE_CODE (SSA_NAME_VAR (op)) == PARM_DECL)
kono
parents:
diff changeset
985 {
kono
parents:
diff changeset
986 if (size_p)
kono
parents:
diff changeset
987 *size_p = tree_to_shwi (TYPE_SIZE (TREE_TYPE (op)));
kono
parents:
diff changeset
988 return SSA_NAME_VAR (op);
kono
parents:
diff changeset
989 }
kono
parents:
diff changeset
990 /* Non-SSA parm reference? */
kono
parents:
diff changeset
991 if (TREE_CODE (op) == PARM_DECL)
kono
parents:
diff changeset
992 {
kono
parents:
diff changeset
993 bool modified = false;
kono
parents:
diff changeset
994
kono
parents:
diff changeset
995 ao_ref refd;
kono
parents:
diff changeset
996 ao_ref_init (&refd, op);
kono
parents:
diff changeset
997 walk_aliased_vdefs (&refd, gimple_vuse (stmt), mark_modified, &modified,
kono
parents:
diff changeset
998 NULL);
kono
parents:
diff changeset
999 if (!modified)
kono
parents:
diff changeset
1000 {
kono
parents:
diff changeset
1001 if (size_p)
kono
parents:
diff changeset
1002 *size_p = tree_to_shwi (TYPE_SIZE (TREE_TYPE (op)));
kono
parents:
diff changeset
1003 return op;
kono
parents:
diff changeset
1004 }
kono
parents:
diff changeset
1005 }
kono
parents:
diff changeset
1006 return NULL_TREE;
kono
parents:
diff changeset
1007 }
kono
parents:
diff changeset
1008
kono
parents:
diff changeset
1009 /* If OP refers to value of function parameter, return the corresponding
kono
parents:
diff changeset
1010 parameter. Also traverse chains of SSA register assignments. If non-NULL,
kono
parents:
diff changeset
1011 the size of the memory load (or the SSA_NAME of the PARM_DECL) will be
kono
parents:
diff changeset
1012 stored to *SIZE_P in that case too. */
kono
parents:
diff changeset
1013
kono
parents:
diff changeset
1014 static tree
kono
parents:
diff changeset
1015 unmodified_parm (gimple *stmt, tree op, HOST_WIDE_INT *size_p)
kono
parents:
diff changeset
1016 {
kono
parents:
diff changeset
1017 tree res = unmodified_parm_1 (stmt, op, size_p);
kono
parents:
diff changeset
1018 if (res)
kono
parents:
diff changeset
1019 return res;
kono
parents:
diff changeset
1020
kono
parents:
diff changeset
1021 if (TREE_CODE (op) == SSA_NAME
kono
parents:
diff changeset
1022 && !SSA_NAME_IS_DEFAULT_DEF (op)
kono
parents:
diff changeset
1023 && gimple_assign_single_p (SSA_NAME_DEF_STMT (op)))
kono
parents:
diff changeset
1024 return unmodified_parm (SSA_NAME_DEF_STMT (op),
kono
parents:
diff changeset
1025 gimple_assign_rhs1 (SSA_NAME_DEF_STMT (op)),
kono
parents:
diff changeset
1026 size_p);
kono
parents:
diff changeset
1027 return NULL_TREE;
kono
parents:
diff changeset
1028 }
kono
parents:
diff changeset
1029
kono
parents:
diff changeset
1030 /* If OP refers to a value of a function parameter or value loaded from an
kono
parents:
diff changeset
1031 aggregate passed to a parameter (either by value or reference), return TRUE
kono
parents:
diff changeset
1032 and store the number of the parameter to *INDEX_P, the access size into
kono
parents:
diff changeset
1033 *SIZE_P, and information whether and how it has been loaded from an
kono
parents:
diff changeset
1034 aggregate into *AGGPOS. INFO describes the function parameters, STMT is the
kono
parents:
diff changeset
1035 statement in which OP is used or loaded. */
kono
parents:
diff changeset
1036
kono
parents:
diff changeset
1037 static bool
kono
parents:
diff changeset
1038 unmodified_parm_or_parm_agg_item (struct ipa_func_body_info *fbi,
kono
parents:
diff changeset
1039 gimple *stmt, tree op, int *index_p,
kono
parents:
diff changeset
1040 HOST_WIDE_INT *size_p,
kono
parents:
diff changeset
1041 struct agg_position_info *aggpos)
kono
parents:
diff changeset
1042 {
kono
parents:
diff changeset
1043 tree res = unmodified_parm_1 (stmt, op, size_p);
kono
parents:
diff changeset
1044
kono
parents:
diff changeset
1045 gcc_checking_assert (aggpos);
kono
parents:
diff changeset
1046 if (res)
kono
parents:
diff changeset
1047 {
kono
parents:
diff changeset
1048 *index_p = ipa_get_param_decl_index (fbi->info, res);
kono
parents:
diff changeset
1049 if (*index_p < 0)
kono
parents:
diff changeset
1050 return false;
kono
parents:
diff changeset
1051 aggpos->agg_contents = false;
kono
parents:
diff changeset
1052 aggpos->by_ref = false;
kono
parents:
diff changeset
1053 return true;
kono
parents:
diff changeset
1054 }
kono
parents:
diff changeset
1055
kono
parents:
diff changeset
1056 if (TREE_CODE (op) == SSA_NAME)
kono
parents:
diff changeset
1057 {
kono
parents:
diff changeset
1058 if (SSA_NAME_IS_DEFAULT_DEF (op)
kono
parents:
diff changeset
1059 || !gimple_assign_single_p (SSA_NAME_DEF_STMT (op)))
kono
parents:
diff changeset
1060 return false;
kono
parents:
diff changeset
1061 stmt = SSA_NAME_DEF_STMT (op);
kono
parents:
diff changeset
1062 op = gimple_assign_rhs1 (stmt);
kono
parents:
diff changeset
1063 if (!REFERENCE_CLASS_P (op))
kono
parents:
diff changeset
1064 return unmodified_parm_or_parm_agg_item (fbi, stmt, op, index_p, size_p,
kono
parents:
diff changeset
1065 aggpos);
kono
parents:
diff changeset
1066 }
kono
parents:
diff changeset
1067
kono
parents:
diff changeset
1068 aggpos->agg_contents = true;
kono
parents:
diff changeset
1069 return ipa_load_from_parm_agg (fbi, fbi->info->descriptors,
kono
parents:
diff changeset
1070 stmt, op, index_p, &aggpos->offset,
kono
parents:
diff changeset
1071 size_p, &aggpos->by_ref);
kono
parents:
diff changeset
1072 }
kono
parents:
diff changeset
1073
kono
parents:
diff changeset
1074 /* See if statement might disappear after inlining.
kono
parents:
diff changeset
1075 0 - means not eliminated
kono
parents:
diff changeset
1076 1 - half of statements goes away
kono
parents:
diff changeset
1077 2 - for sure it is eliminated.
kono
parents:
diff changeset
1078 We are not terribly sophisticated, basically looking for simple abstraction
kono
parents:
diff changeset
1079 penalty wrappers. */
kono
parents:
diff changeset
1080
kono
parents:
diff changeset
1081 static int
kono
parents:
diff changeset
1082 eliminated_by_inlining_prob (gimple *stmt)
kono
parents:
diff changeset
1083 {
kono
parents:
diff changeset
1084 enum gimple_code code = gimple_code (stmt);
kono
parents:
diff changeset
1085 enum tree_code rhs_code;
kono
parents:
diff changeset
1086
kono
parents:
diff changeset
1087 if (!optimize)
kono
parents:
diff changeset
1088 return 0;
kono
parents:
diff changeset
1089
kono
parents:
diff changeset
1090 switch (code)
kono
parents:
diff changeset
1091 {
kono
parents:
diff changeset
1092 case GIMPLE_RETURN:
kono
parents:
diff changeset
1093 return 2;
kono
parents:
diff changeset
1094 case GIMPLE_ASSIGN:
kono
parents:
diff changeset
1095 if (gimple_num_ops (stmt) != 2)
kono
parents:
diff changeset
1096 return 0;
kono
parents:
diff changeset
1097
kono
parents:
diff changeset
1098 rhs_code = gimple_assign_rhs_code (stmt);
kono
parents:
diff changeset
1099
kono
parents:
diff changeset
1100 /* Casts of parameters, loads from parameters passed by reference
kono
parents:
diff changeset
1101 and stores to return value or parameters are often free after
kono
parents:
diff changeset
1102 inlining dua to SRA and further combining.
kono
parents:
diff changeset
1103 Assume that half of statements goes away. */
kono
parents:
diff changeset
1104 if (CONVERT_EXPR_CODE_P (rhs_code)
kono
parents:
diff changeset
1105 || rhs_code == VIEW_CONVERT_EXPR
kono
parents:
diff changeset
1106 || rhs_code == ADDR_EXPR
kono
parents:
diff changeset
1107 || gimple_assign_rhs_class (stmt) == GIMPLE_SINGLE_RHS)
kono
parents:
diff changeset
1108 {
kono
parents:
diff changeset
1109 tree rhs = gimple_assign_rhs1 (stmt);
kono
parents:
diff changeset
1110 tree lhs = gimple_assign_lhs (stmt);
kono
parents:
diff changeset
1111 tree inner_rhs = get_base_address (rhs);
kono
parents:
diff changeset
1112 tree inner_lhs = get_base_address (lhs);
kono
parents:
diff changeset
1113 bool rhs_free = false;
kono
parents:
diff changeset
1114 bool lhs_free = false;
kono
parents:
diff changeset
1115
kono
parents:
diff changeset
1116 if (!inner_rhs)
kono
parents:
diff changeset
1117 inner_rhs = rhs;
kono
parents:
diff changeset
1118 if (!inner_lhs)
kono
parents:
diff changeset
1119 inner_lhs = lhs;
kono
parents:
diff changeset
1120
kono
parents:
diff changeset
1121 /* Reads of parameter are expected to be free. */
kono
parents:
diff changeset
1122 if (unmodified_parm (stmt, inner_rhs, NULL))
kono
parents:
diff changeset
1123 rhs_free = true;
kono
parents:
diff changeset
1124 /* Match expressions of form &this->field. Those will most likely
kono
parents:
diff changeset
1125 combine with something upstream after inlining. */
kono
parents:
diff changeset
1126 else if (TREE_CODE (inner_rhs) == ADDR_EXPR)
kono
parents:
diff changeset
1127 {
kono
parents:
diff changeset
1128 tree op = get_base_address (TREE_OPERAND (inner_rhs, 0));
kono
parents:
diff changeset
1129 if (TREE_CODE (op) == PARM_DECL)
kono
parents:
diff changeset
1130 rhs_free = true;
kono
parents:
diff changeset
1131 else if (TREE_CODE (op) == MEM_REF
kono
parents:
diff changeset
1132 && unmodified_parm (stmt, TREE_OPERAND (op, 0), NULL))
kono
parents:
diff changeset
1133 rhs_free = true;
kono
parents:
diff changeset
1134 }
kono
parents:
diff changeset
1135
kono
parents:
diff changeset
1136 /* When parameter is not SSA register because its address is taken
kono
parents:
diff changeset
1137 and it is just copied into one, the statement will be completely
kono
parents:
diff changeset
1138 free after inlining (we will copy propagate backward). */
kono
parents:
diff changeset
1139 if (rhs_free && is_gimple_reg (lhs))
kono
parents:
diff changeset
1140 return 2;
kono
parents:
diff changeset
1141
kono
parents:
diff changeset
1142 /* Reads of parameters passed by reference
kono
parents:
diff changeset
1143 expected to be free (i.e. optimized out after inlining). */
kono
parents:
diff changeset
1144 if (TREE_CODE (inner_rhs) == MEM_REF
kono
parents:
diff changeset
1145 && unmodified_parm (stmt, TREE_OPERAND (inner_rhs, 0), NULL))
kono
parents:
diff changeset
1146 rhs_free = true;
kono
parents:
diff changeset
1147
kono
parents:
diff changeset
1148 /* Copying parameter passed by reference into gimple register is
kono
parents:
diff changeset
1149 probably also going to copy propagate, but we can't be quite
kono
parents:
diff changeset
1150 sure. */
kono
parents:
diff changeset
1151 if (rhs_free && is_gimple_reg (lhs))
kono
parents:
diff changeset
1152 lhs_free = true;
kono
parents:
diff changeset
1153
kono
parents:
diff changeset
1154 /* Writes to parameters, parameters passed by value and return value
kono
parents:
diff changeset
1155 (either dirrectly or passed via invisible reference) are free.
kono
parents:
diff changeset
1156
kono
parents:
diff changeset
1157 TODO: We ought to handle testcase like
kono
parents:
diff changeset
1158 struct a {int a,b;};
kono
parents:
diff changeset
1159 struct a
kono
parents:
diff changeset
1160 retrurnsturct (void)
kono
parents:
diff changeset
1161 {
kono
parents:
diff changeset
1162 struct a a ={1,2};
kono
parents:
diff changeset
1163 return a;
kono
parents:
diff changeset
1164 }
kono
parents:
diff changeset
1165
kono
parents:
diff changeset
1166 This translate into:
kono
parents:
diff changeset
1167
kono
parents:
diff changeset
1168 retrurnsturct ()
kono
parents:
diff changeset
1169 {
kono
parents:
diff changeset
1170 int a$b;
kono
parents:
diff changeset
1171 int a$a;
kono
parents:
diff changeset
1172 struct a a;
kono
parents:
diff changeset
1173 struct a D.2739;
kono
parents:
diff changeset
1174
kono
parents:
diff changeset
1175 <bb 2>:
kono
parents:
diff changeset
1176 D.2739.a = 1;
kono
parents:
diff changeset
1177 D.2739.b = 2;
kono
parents:
diff changeset
1178 return D.2739;
kono
parents:
diff changeset
1179
kono
parents:
diff changeset
1180 }
kono
parents:
diff changeset
1181 For that we either need to copy ipa-split logic detecting writes
kono
parents:
diff changeset
1182 to return value. */
kono
parents:
diff changeset
1183 if (TREE_CODE (inner_lhs) == PARM_DECL
kono
parents:
diff changeset
1184 || TREE_CODE (inner_lhs) == RESULT_DECL
kono
parents:
diff changeset
1185 || (TREE_CODE (inner_lhs) == MEM_REF
kono
parents:
diff changeset
1186 && (unmodified_parm (stmt, TREE_OPERAND (inner_lhs, 0), NULL)
kono
parents:
diff changeset
1187 || (TREE_CODE (TREE_OPERAND (inner_lhs, 0)) == SSA_NAME
kono
parents:
diff changeset
1188 && SSA_NAME_VAR (TREE_OPERAND (inner_lhs, 0))
kono
parents:
diff changeset
1189 && TREE_CODE (SSA_NAME_VAR (TREE_OPERAND
kono
parents:
diff changeset
1190 (inner_lhs,
kono
parents:
diff changeset
1191 0))) == RESULT_DECL))))
kono
parents:
diff changeset
1192 lhs_free = true;
kono
parents:
diff changeset
1193 if (lhs_free
kono
parents:
diff changeset
1194 && (is_gimple_reg (rhs) || is_gimple_min_invariant (rhs)))
kono
parents:
diff changeset
1195 rhs_free = true;
kono
parents:
diff changeset
1196 if (lhs_free && rhs_free)
kono
parents:
diff changeset
1197 return 1;
kono
parents:
diff changeset
1198 }
kono
parents:
diff changeset
1199 return 0;
kono
parents:
diff changeset
1200 default:
kono
parents:
diff changeset
1201 return 0;
kono
parents:
diff changeset
1202 }
kono
parents:
diff changeset
1203 }
kono
parents:
diff changeset
1204
kono
parents:
diff changeset
1205
kono
parents:
diff changeset
1206 /* If BB ends by a conditional we can turn into predicates, attach corresponding
kono
parents:
diff changeset
1207 predicates to the CFG edges. */
kono
parents:
diff changeset
1208
kono
parents:
diff changeset
1209 static void
kono
parents:
diff changeset
1210 set_cond_stmt_execution_predicate (struct ipa_func_body_info *fbi,
kono
parents:
diff changeset
1211 struct ipa_fn_summary *summary,
kono
parents:
diff changeset
1212 basic_block bb)
kono
parents:
diff changeset
1213 {
kono
parents:
diff changeset
1214 gimple *last;
kono
parents:
diff changeset
1215 tree op;
kono
parents:
diff changeset
1216 int index;
kono
parents:
diff changeset
1217 HOST_WIDE_INT size;
kono
parents:
diff changeset
1218 struct agg_position_info aggpos;
kono
parents:
diff changeset
1219 enum tree_code code, inverted_code;
kono
parents:
diff changeset
1220 edge e;
kono
parents:
diff changeset
1221 edge_iterator ei;
kono
parents:
diff changeset
1222 gimple *set_stmt;
kono
parents:
diff changeset
1223 tree op2;
kono
parents:
diff changeset
1224
kono
parents:
diff changeset
1225 last = last_stmt (bb);
kono
parents:
diff changeset
1226 if (!last || gimple_code (last) != GIMPLE_COND)
kono
parents:
diff changeset
1227 return;
kono
parents:
diff changeset
1228 if (!is_gimple_ip_invariant (gimple_cond_rhs (last)))
kono
parents:
diff changeset
1229 return;
kono
parents:
diff changeset
1230 op = gimple_cond_lhs (last);
kono
parents:
diff changeset
1231 /* TODO: handle conditionals like
kono
parents:
diff changeset
1232 var = op0 < 4;
kono
parents:
diff changeset
1233 if (var != 0). */
kono
parents:
diff changeset
1234 if (unmodified_parm_or_parm_agg_item (fbi, last, op, &index, &size, &aggpos))
kono
parents:
diff changeset
1235 {
kono
parents:
diff changeset
1236 code = gimple_cond_code (last);
kono
parents:
diff changeset
1237 inverted_code = invert_tree_comparison (code, HONOR_NANS (op));
kono
parents:
diff changeset
1238
kono
parents:
diff changeset
1239 FOR_EACH_EDGE (e, ei, bb->succs)
kono
parents:
diff changeset
1240 {
kono
parents:
diff changeset
1241 enum tree_code this_code = (e->flags & EDGE_TRUE_VALUE
kono
parents:
diff changeset
1242 ? code : inverted_code);
kono
parents:
diff changeset
1243 /* invert_tree_comparison will return ERROR_MARK on FP
kono
parents:
diff changeset
1244 comparsions that are not EQ/NE instead of returning proper
kono
parents:
diff changeset
1245 unordered one. Be sure it is not confused with NON_CONSTANT. */
kono
parents:
diff changeset
1246 if (this_code != ERROR_MARK)
kono
parents:
diff changeset
1247 {
kono
parents:
diff changeset
1248 predicate p
kono
parents:
diff changeset
1249 = add_condition (summary, index, size, &aggpos, this_code,
kono
parents:
diff changeset
1250 unshare_expr_without_location
kono
parents:
diff changeset
1251 (gimple_cond_rhs (last)));
kono
parents:
diff changeset
1252 e->aux = edge_predicate_pool.allocate ();
kono
parents:
diff changeset
1253 *(predicate *) e->aux = p;
kono
parents:
diff changeset
1254 }
kono
parents:
diff changeset
1255 }
kono
parents:
diff changeset
1256 }
kono
parents:
diff changeset
1257
kono
parents:
diff changeset
1258 if (TREE_CODE (op) != SSA_NAME)
kono
parents:
diff changeset
1259 return;
kono
parents:
diff changeset
1260 /* Special case
kono
parents:
diff changeset
1261 if (builtin_constant_p (op))
kono
parents:
diff changeset
1262 constant_code
kono
parents:
diff changeset
1263 else
kono
parents:
diff changeset
1264 nonconstant_code.
kono
parents:
diff changeset
1265 Here we can predicate nonconstant_code. We can't
kono
parents:
diff changeset
1266 really handle constant_code since we have no predicate
kono
parents:
diff changeset
1267 for this and also the constant code is not known to be
kono
parents:
diff changeset
1268 optimized away when inliner doen't see operand is constant.
kono
parents:
diff changeset
1269 Other optimizers might think otherwise. */
kono
parents:
diff changeset
1270 if (gimple_cond_code (last) != NE_EXPR
kono
parents:
diff changeset
1271 || !integer_zerop (gimple_cond_rhs (last)))
kono
parents:
diff changeset
1272 return;
kono
parents:
diff changeset
1273 set_stmt = SSA_NAME_DEF_STMT (op);
kono
parents:
diff changeset
1274 if (!gimple_call_builtin_p (set_stmt, BUILT_IN_CONSTANT_P)
kono
parents:
diff changeset
1275 || gimple_call_num_args (set_stmt) != 1)
kono
parents:
diff changeset
1276 return;
kono
parents:
diff changeset
1277 op2 = gimple_call_arg (set_stmt, 0);
kono
parents:
diff changeset
1278 if (!unmodified_parm_or_parm_agg_item (fbi, set_stmt, op2, &index, &size,
kono
parents:
diff changeset
1279 &aggpos))
kono
parents:
diff changeset
1280 return;
kono
parents:
diff changeset
1281 FOR_EACH_EDGE (e, ei, bb->succs) if (e->flags & EDGE_FALSE_VALUE)
kono
parents:
diff changeset
1282 {
kono
parents:
diff changeset
1283 predicate p = add_condition (summary, index, size, &aggpos,
kono
parents:
diff changeset
1284 predicate::is_not_constant, NULL_TREE);
kono
parents:
diff changeset
1285 e->aux = edge_predicate_pool.allocate ();
kono
parents:
diff changeset
1286 *(predicate *) e->aux = p;
kono
parents:
diff changeset
1287 }
kono
parents:
diff changeset
1288 }
kono
parents:
diff changeset
1289
kono
parents:
diff changeset
1290
kono
parents:
diff changeset
1291 /* If BB ends by a switch we can turn into predicates, attach corresponding
kono
parents:
diff changeset
1292 predicates to the CFG edges. */
kono
parents:
diff changeset
1293
kono
parents:
diff changeset
1294 static void
kono
parents:
diff changeset
1295 set_switch_stmt_execution_predicate (struct ipa_func_body_info *fbi,
kono
parents:
diff changeset
1296 struct ipa_fn_summary *summary,
kono
parents:
diff changeset
1297 basic_block bb)
kono
parents:
diff changeset
1298 {
kono
parents:
diff changeset
1299 gimple *lastg;
kono
parents:
diff changeset
1300 tree op;
kono
parents:
diff changeset
1301 int index;
kono
parents:
diff changeset
1302 HOST_WIDE_INT size;
kono
parents:
diff changeset
1303 struct agg_position_info aggpos;
kono
parents:
diff changeset
1304 edge e;
kono
parents:
diff changeset
1305 edge_iterator ei;
kono
parents:
diff changeset
1306 size_t n;
kono
parents:
diff changeset
1307 size_t case_idx;
kono
parents:
diff changeset
1308
kono
parents:
diff changeset
1309 lastg = last_stmt (bb);
kono
parents:
diff changeset
1310 if (!lastg || gimple_code (lastg) != GIMPLE_SWITCH)
kono
parents:
diff changeset
1311 return;
kono
parents:
diff changeset
1312 gswitch *last = as_a <gswitch *> (lastg);
kono
parents:
diff changeset
1313 op = gimple_switch_index (last);
kono
parents:
diff changeset
1314 if (!unmodified_parm_or_parm_agg_item (fbi, last, op, &index, &size, &aggpos))
kono
parents:
diff changeset
1315 return;
kono
parents:
diff changeset
1316
kono
parents:
diff changeset
1317 FOR_EACH_EDGE (e, ei, bb->succs)
kono
parents:
diff changeset
1318 {
kono
parents:
diff changeset
1319 e->aux = edge_predicate_pool.allocate ();
kono
parents:
diff changeset
1320 *(predicate *) e->aux = false;
kono
parents:
diff changeset
1321 }
kono
parents:
diff changeset
1322 n = gimple_switch_num_labels (last);
kono
parents:
diff changeset
1323 for (case_idx = 0; case_idx < n; ++case_idx)
kono
parents:
diff changeset
1324 {
kono
parents:
diff changeset
1325 tree cl = gimple_switch_label (last, case_idx);
kono
parents:
diff changeset
1326 tree min, max;
kono
parents:
diff changeset
1327 predicate p;
kono
parents:
diff changeset
1328
kono
parents:
diff changeset
1329 e = find_edge (bb, label_to_block (CASE_LABEL (cl)));
kono
parents:
diff changeset
1330 min = CASE_LOW (cl);
kono
parents:
diff changeset
1331 max = CASE_HIGH (cl);
kono
parents:
diff changeset
1332
kono
parents:
diff changeset
1333 /* For default we might want to construct predicate that none
kono
parents:
diff changeset
1334 of cases is met, but it is bit hard to do not having negations
kono
parents:
diff changeset
1335 of conditionals handy. */
kono
parents:
diff changeset
1336 if (!min && !max)
kono
parents:
diff changeset
1337 p = true;
kono
parents:
diff changeset
1338 else if (!max)
kono
parents:
diff changeset
1339 p = add_condition (summary, index, size, &aggpos, EQ_EXPR,
kono
parents:
diff changeset
1340 unshare_expr_without_location (min));
kono
parents:
diff changeset
1341 else
kono
parents:
diff changeset
1342 {
kono
parents:
diff changeset
1343 predicate p1, p2;
kono
parents:
diff changeset
1344 p1 = add_condition (summary, index, size, &aggpos, GE_EXPR,
kono
parents:
diff changeset
1345 unshare_expr_without_location (min));
kono
parents:
diff changeset
1346 p2 = add_condition (summary, index, size, &aggpos, LE_EXPR,
kono
parents:
diff changeset
1347 unshare_expr_without_location (max));
kono
parents:
diff changeset
1348 p = p1 & p2;
kono
parents:
diff changeset
1349 }
kono
parents:
diff changeset
1350 *(struct predicate *) e->aux
kono
parents:
diff changeset
1351 = p.or_with (summary->conds, *(struct predicate *) e->aux);
kono
parents:
diff changeset
1352 }
kono
parents:
diff changeset
1353 }
kono
parents:
diff changeset
1354
kono
parents:
diff changeset
1355
kono
parents:
diff changeset
1356 /* For each BB in NODE attach to its AUX pointer predicate under
kono
parents:
diff changeset
1357 which it is executable. */
kono
parents:
diff changeset
1358
kono
parents:
diff changeset
1359 static void
kono
parents:
diff changeset
1360 compute_bb_predicates (struct ipa_func_body_info *fbi,
kono
parents:
diff changeset
1361 struct cgraph_node *node,
kono
parents:
diff changeset
1362 struct ipa_fn_summary *summary)
kono
parents:
diff changeset
1363 {
kono
parents:
diff changeset
1364 struct function *my_function = DECL_STRUCT_FUNCTION (node->decl);
kono
parents:
diff changeset
1365 bool done = false;
kono
parents:
diff changeset
1366 basic_block bb;
kono
parents:
diff changeset
1367
kono
parents:
diff changeset
1368 FOR_EACH_BB_FN (bb, my_function)
kono
parents:
diff changeset
1369 {
kono
parents:
diff changeset
1370 set_cond_stmt_execution_predicate (fbi, summary, bb);
kono
parents:
diff changeset
1371 set_switch_stmt_execution_predicate (fbi, summary, bb);
kono
parents:
diff changeset
1372 }
kono
parents:
diff changeset
1373
kono
parents:
diff changeset
1374 /* Entry block is always executable. */
kono
parents:
diff changeset
1375 ENTRY_BLOCK_PTR_FOR_FN (my_function)->aux
kono
parents:
diff changeset
1376 = edge_predicate_pool.allocate ();
kono
parents:
diff changeset
1377 *(predicate *) ENTRY_BLOCK_PTR_FOR_FN (my_function)->aux = true;
kono
parents:
diff changeset
1378
kono
parents:
diff changeset
1379 /* A simple dataflow propagation of predicates forward in the CFG.
kono
parents:
diff changeset
1380 TODO: work in reverse postorder. */
kono
parents:
diff changeset
1381 while (!done)
kono
parents:
diff changeset
1382 {
kono
parents:
diff changeset
1383 done = true;
kono
parents:
diff changeset
1384 FOR_EACH_BB_FN (bb, my_function)
kono
parents:
diff changeset
1385 {
kono
parents:
diff changeset
1386 predicate p = false;
kono
parents:
diff changeset
1387 edge e;
kono
parents:
diff changeset
1388 edge_iterator ei;
kono
parents:
diff changeset
1389 FOR_EACH_EDGE (e, ei, bb->preds)
kono
parents:
diff changeset
1390 {
kono
parents:
diff changeset
1391 if (e->src->aux)
kono
parents:
diff changeset
1392 {
kono
parents:
diff changeset
1393 predicate this_bb_predicate
kono
parents:
diff changeset
1394 = *(predicate *) e->src->aux;
kono
parents:
diff changeset
1395 if (e->aux)
kono
parents:
diff changeset
1396 this_bb_predicate &= (*(struct predicate *) e->aux);
kono
parents:
diff changeset
1397 p = p.or_with (summary->conds, this_bb_predicate);
kono
parents:
diff changeset
1398 if (p == true)
kono
parents:
diff changeset
1399 break;
kono
parents:
diff changeset
1400 }
kono
parents:
diff changeset
1401 }
kono
parents:
diff changeset
1402 if (p == false)
kono
parents:
diff changeset
1403 gcc_checking_assert (!bb->aux);
kono
parents:
diff changeset
1404 else
kono
parents:
diff changeset
1405 {
kono
parents:
diff changeset
1406 if (!bb->aux)
kono
parents:
diff changeset
1407 {
kono
parents:
diff changeset
1408 done = false;
kono
parents:
diff changeset
1409 bb->aux = edge_predicate_pool.allocate ();
kono
parents:
diff changeset
1410 *((predicate *) bb->aux) = p;
kono
parents:
diff changeset
1411 }
kono
parents:
diff changeset
1412 else if (p != *(predicate *) bb->aux)
kono
parents:
diff changeset
1413 {
kono
parents:
diff changeset
1414 /* This OR operation is needed to ensure monotonous data flow
kono
parents:
diff changeset
1415 in the case we hit the limit on number of clauses and the
kono
parents:
diff changeset
1416 and/or operations above give approximate answers. */
kono
parents:
diff changeset
1417 p = p.or_with (summary->conds, *(predicate *)bb->aux);
kono
parents:
diff changeset
1418 if (p != *(predicate *) bb->aux)
kono
parents:
diff changeset
1419 {
kono
parents:
diff changeset
1420 done = false;
kono
parents:
diff changeset
1421 *((predicate *) bb->aux) = p;
kono
parents:
diff changeset
1422 }
kono
parents:
diff changeset
1423 }
kono
parents:
diff changeset
1424 }
kono
parents:
diff changeset
1425 }
kono
parents:
diff changeset
1426 }
kono
parents:
diff changeset
1427 }
kono
parents:
diff changeset
1428
kono
parents:
diff changeset
1429
kono
parents:
diff changeset
1430 /* Return predicate specifying when the STMT might have result that is not
kono
parents:
diff changeset
1431 a compile time constant. */
kono
parents:
diff changeset
1432
kono
parents:
diff changeset
1433 static predicate
kono
parents:
diff changeset
1434 will_be_nonconstant_expr_predicate (struct ipa_node_params *info,
kono
parents:
diff changeset
1435 struct ipa_fn_summary *summary,
kono
parents:
diff changeset
1436 tree expr,
kono
parents:
diff changeset
1437 vec<predicate> nonconstant_names)
kono
parents:
diff changeset
1438 {
kono
parents:
diff changeset
1439 tree parm;
kono
parents:
diff changeset
1440 int index;
kono
parents:
diff changeset
1441 HOST_WIDE_INT size;
kono
parents:
diff changeset
1442
kono
parents:
diff changeset
1443 while (UNARY_CLASS_P (expr))
kono
parents:
diff changeset
1444 expr = TREE_OPERAND (expr, 0);
kono
parents:
diff changeset
1445
kono
parents:
diff changeset
1446 parm = unmodified_parm (NULL, expr, &size);
kono
parents:
diff changeset
1447 if (parm && (index = ipa_get_param_decl_index (info, parm)) >= 0)
kono
parents:
diff changeset
1448 return add_condition (summary, index, size, NULL, predicate::changed,
kono
parents:
diff changeset
1449 NULL_TREE);
kono
parents:
diff changeset
1450 if (is_gimple_min_invariant (expr))
kono
parents:
diff changeset
1451 return false;
kono
parents:
diff changeset
1452 if (TREE_CODE (expr) == SSA_NAME)
kono
parents:
diff changeset
1453 return nonconstant_names[SSA_NAME_VERSION (expr)];
kono
parents:
diff changeset
1454 if (BINARY_CLASS_P (expr) || COMPARISON_CLASS_P (expr))
kono
parents:
diff changeset
1455 {
kono
parents:
diff changeset
1456 predicate p1 = will_be_nonconstant_expr_predicate
kono
parents:
diff changeset
1457 (info, summary, TREE_OPERAND (expr, 0),
kono
parents:
diff changeset
1458 nonconstant_names);
kono
parents:
diff changeset
1459 if (p1 == true)
kono
parents:
diff changeset
1460 return p1;
kono
parents:
diff changeset
1461
kono
parents:
diff changeset
1462 predicate p2;
kono
parents:
diff changeset
1463 p2 = will_be_nonconstant_expr_predicate (info, summary,
kono
parents:
diff changeset
1464 TREE_OPERAND (expr, 1),
kono
parents:
diff changeset
1465 nonconstant_names);
kono
parents:
diff changeset
1466 return p1.or_with (summary->conds, p2);
kono
parents:
diff changeset
1467 }
kono
parents:
diff changeset
1468 else if (TREE_CODE (expr) == COND_EXPR)
kono
parents:
diff changeset
1469 {
kono
parents:
diff changeset
1470 predicate p1 = will_be_nonconstant_expr_predicate
kono
parents:
diff changeset
1471 (info, summary, TREE_OPERAND (expr, 0),
kono
parents:
diff changeset
1472 nonconstant_names);
kono
parents:
diff changeset
1473 if (p1 == true)
kono
parents:
diff changeset
1474 return p1;
kono
parents:
diff changeset
1475
kono
parents:
diff changeset
1476 predicate p2;
kono
parents:
diff changeset
1477 p2 = will_be_nonconstant_expr_predicate (info, summary,
kono
parents:
diff changeset
1478 TREE_OPERAND (expr, 1),
kono
parents:
diff changeset
1479 nonconstant_names);
kono
parents:
diff changeset
1480 if (p2 == true)
kono
parents:
diff changeset
1481 return p2;
kono
parents:
diff changeset
1482 p1 = p1.or_with (summary->conds, p2);
kono
parents:
diff changeset
1483 p2 = will_be_nonconstant_expr_predicate (info, summary,
kono
parents:
diff changeset
1484 TREE_OPERAND (expr, 2),
kono
parents:
diff changeset
1485 nonconstant_names);
kono
parents:
diff changeset
1486 return p2.or_with (summary->conds, p1);
kono
parents:
diff changeset
1487 }
kono
parents:
diff changeset
1488 else
kono
parents:
diff changeset
1489 {
kono
parents:
diff changeset
1490 debug_tree (expr);
kono
parents:
diff changeset
1491 gcc_unreachable ();
kono
parents:
diff changeset
1492 }
kono
parents:
diff changeset
1493 return false;
kono
parents:
diff changeset
1494 }
kono
parents:
diff changeset
1495
kono
parents:
diff changeset
1496
kono
parents:
diff changeset
1497 /* Return predicate specifying when the STMT might have result that is not
kono
parents:
diff changeset
1498 a compile time constant. */
kono
parents:
diff changeset
1499
kono
parents:
diff changeset
1500 static predicate
kono
parents:
diff changeset
1501 will_be_nonconstant_predicate (struct ipa_func_body_info *fbi,
kono
parents:
diff changeset
1502 struct ipa_fn_summary *summary,
kono
parents:
diff changeset
1503 gimple *stmt,
kono
parents:
diff changeset
1504 vec<predicate> nonconstant_names)
kono
parents:
diff changeset
1505 {
kono
parents:
diff changeset
1506 predicate p = true;
kono
parents:
diff changeset
1507 ssa_op_iter iter;
kono
parents:
diff changeset
1508 tree use;
kono
parents:
diff changeset
1509 predicate op_non_const;
kono
parents:
diff changeset
1510 bool is_load;
kono
parents:
diff changeset
1511 int base_index;
kono
parents:
diff changeset
1512 HOST_WIDE_INT size;
kono
parents:
diff changeset
1513 struct agg_position_info aggpos;
kono
parents:
diff changeset
1514
kono
parents:
diff changeset
1515 /* What statments might be optimized away
kono
parents:
diff changeset
1516 when their arguments are constant. */
kono
parents:
diff changeset
1517 if (gimple_code (stmt) != GIMPLE_ASSIGN
kono
parents:
diff changeset
1518 && gimple_code (stmt) != GIMPLE_COND
kono
parents:
diff changeset
1519 && gimple_code (stmt) != GIMPLE_SWITCH
kono
parents:
diff changeset
1520 && (gimple_code (stmt) != GIMPLE_CALL
kono
parents:
diff changeset
1521 || !(gimple_call_flags (stmt) & ECF_CONST)))
kono
parents:
diff changeset
1522 return p;
kono
parents:
diff changeset
1523
kono
parents:
diff changeset
1524 /* Stores will stay anyway. */
kono
parents:
diff changeset
1525 if (gimple_store_p (stmt))
kono
parents:
diff changeset
1526 return p;
kono
parents:
diff changeset
1527
kono
parents:
diff changeset
1528 is_load = gimple_assign_load_p (stmt);
kono
parents:
diff changeset
1529
kono
parents:
diff changeset
1530 /* Loads can be optimized when the value is known. */
kono
parents:
diff changeset
1531 if (is_load)
kono
parents:
diff changeset
1532 {
kono
parents:
diff changeset
1533 tree op;
kono
parents:
diff changeset
1534 gcc_assert (gimple_assign_single_p (stmt));
kono
parents:
diff changeset
1535 op = gimple_assign_rhs1 (stmt);
kono
parents:
diff changeset
1536 if (!unmodified_parm_or_parm_agg_item (fbi, stmt, op, &base_index, &size,
kono
parents:
diff changeset
1537 &aggpos))
kono
parents:
diff changeset
1538 return p;
kono
parents:
diff changeset
1539 }
kono
parents:
diff changeset
1540 else
kono
parents:
diff changeset
1541 base_index = -1;
kono
parents:
diff changeset
1542
kono
parents:
diff changeset
1543 /* See if we understand all operands before we start
kono
parents:
diff changeset
1544 adding conditionals. */
kono
parents:
diff changeset
1545 FOR_EACH_SSA_TREE_OPERAND (use, stmt, iter, SSA_OP_USE)
kono
parents:
diff changeset
1546 {
kono
parents:
diff changeset
1547 tree parm = unmodified_parm (stmt, use, NULL);
kono
parents:
diff changeset
1548 /* For arguments we can build a condition. */
kono
parents:
diff changeset
1549 if (parm && ipa_get_param_decl_index (fbi->info, parm) >= 0)
kono
parents:
diff changeset
1550 continue;
kono
parents:
diff changeset
1551 if (TREE_CODE (use) != SSA_NAME)
kono
parents:
diff changeset
1552 return p;
kono
parents:
diff changeset
1553 /* If we know when operand is constant,
kono
parents:
diff changeset
1554 we still can say something useful. */
kono
parents:
diff changeset
1555 if (nonconstant_names[SSA_NAME_VERSION (use)] != true)
kono
parents:
diff changeset
1556 continue;
kono
parents:
diff changeset
1557 return p;
kono
parents:
diff changeset
1558 }
kono
parents:
diff changeset
1559
kono
parents:
diff changeset
1560 if (is_load)
kono
parents:
diff changeset
1561 op_non_const =
kono
parents:
diff changeset
1562 add_condition (summary, base_index, size, &aggpos, predicate::changed,
kono
parents:
diff changeset
1563 NULL);
kono
parents:
diff changeset
1564 else
kono
parents:
diff changeset
1565 op_non_const = false;
kono
parents:
diff changeset
1566 FOR_EACH_SSA_TREE_OPERAND (use, stmt, iter, SSA_OP_USE)
kono
parents:
diff changeset
1567 {
kono
parents:
diff changeset
1568 HOST_WIDE_INT size;
kono
parents:
diff changeset
1569 tree parm = unmodified_parm (stmt, use, &size);
kono
parents:
diff changeset
1570 int index;
kono
parents:
diff changeset
1571
kono
parents:
diff changeset
1572 if (parm && (index = ipa_get_param_decl_index (fbi->info, parm)) >= 0)
kono
parents:
diff changeset
1573 {
kono
parents:
diff changeset
1574 if (index != base_index)
kono
parents:
diff changeset
1575 p = add_condition (summary, index, size, NULL, predicate::changed,
kono
parents:
diff changeset
1576 NULL_TREE);
kono
parents:
diff changeset
1577 else
kono
parents:
diff changeset
1578 continue;
kono
parents:
diff changeset
1579 }
kono
parents:
diff changeset
1580 else
kono
parents:
diff changeset
1581 p = nonconstant_names[SSA_NAME_VERSION (use)];
kono
parents:
diff changeset
1582 op_non_const = p.or_with (summary->conds, op_non_const);
kono
parents:
diff changeset
1583 }
kono
parents:
diff changeset
1584 if ((gimple_code (stmt) == GIMPLE_ASSIGN || gimple_code (stmt) == GIMPLE_CALL)
kono
parents:
diff changeset
1585 && gimple_op (stmt, 0)
kono
parents:
diff changeset
1586 && TREE_CODE (gimple_op (stmt, 0)) == SSA_NAME)
kono
parents:
diff changeset
1587 nonconstant_names[SSA_NAME_VERSION (gimple_op (stmt, 0))]
kono
parents:
diff changeset
1588 = op_non_const;
kono
parents:
diff changeset
1589 return op_non_const;
kono
parents:
diff changeset
1590 }
kono
parents:
diff changeset
1591
kono
parents:
diff changeset
1592 struct record_modified_bb_info
kono
parents:
diff changeset
1593 {
kono
parents:
diff changeset
1594 bitmap bb_set;
kono
parents:
diff changeset
1595 gimple *stmt;
kono
parents:
diff changeset
1596 };
kono
parents:
diff changeset
1597
kono
parents:
diff changeset
1598 /* Value is initialized in INIT_BB and used in USE_BB. We want to copute
kono
parents:
diff changeset
1599 probability how often it changes between USE_BB.
kono
parents:
diff changeset
1600 INIT_BB->frequency/USE_BB->frequency is an estimate, but if INIT_BB
kono
parents:
diff changeset
1601 is in different loop nest, we can do better.
kono
parents:
diff changeset
1602 This is all just estimate. In theory we look for minimal cut separating
kono
parents:
diff changeset
1603 INIT_BB and USE_BB, but we only want to anticipate loop invariant motion
kono
parents:
diff changeset
1604 anyway. */
kono
parents:
diff changeset
1605
kono
parents:
diff changeset
1606 static basic_block
kono
parents:
diff changeset
1607 get_minimal_bb (basic_block init_bb, basic_block use_bb)
kono
parents:
diff changeset
1608 {
kono
parents:
diff changeset
1609 struct loop *l = find_common_loop (init_bb->loop_father, use_bb->loop_father);
kono
parents:
diff changeset
1610 if (l && l->header->frequency < init_bb->frequency)
kono
parents:
diff changeset
1611 return l->header;
kono
parents:
diff changeset
1612 return init_bb;
kono
parents:
diff changeset
1613 }
kono
parents:
diff changeset
1614
kono
parents:
diff changeset
1615 /* Callback of walk_aliased_vdefs. Records basic blocks where the value may be
kono
parents:
diff changeset
1616 set except for info->stmt. */
kono
parents:
diff changeset
1617
kono
parents:
diff changeset
1618 static bool
kono
parents:
diff changeset
1619 record_modified (ao_ref *ao ATTRIBUTE_UNUSED, tree vdef, void *data)
kono
parents:
diff changeset
1620 {
kono
parents:
diff changeset
1621 struct record_modified_bb_info *info =
kono
parents:
diff changeset
1622 (struct record_modified_bb_info *) data;
kono
parents:
diff changeset
1623 if (SSA_NAME_DEF_STMT (vdef) == info->stmt)
kono
parents:
diff changeset
1624 return false;
kono
parents:
diff changeset
1625 bitmap_set_bit (info->bb_set,
kono
parents:
diff changeset
1626 SSA_NAME_IS_DEFAULT_DEF (vdef)
kono
parents:
diff changeset
1627 ? ENTRY_BLOCK_PTR_FOR_FN (cfun)->index
kono
parents:
diff changeset
1628 : get_minimal_bb
kono
parents:
diff changeset
1629 (gimple_bb (SSA_NAME_DEF_STMT (vdef)),
kono
parents:
diff changeset
1630 gimple_bb (info->stmt))->index);
kono
parents:
diff changeset
1631 return false;
kono
parents:
diff changeset
1632 }
kono
parents:
diff changeset
1633
kono
parents:
diff changeset
1634 /* Return probability (based on REG_BR_PROB_BASE) that I-th parameter of STMT
kono
parents:
diff changeset
1635 will change since last invocation of STMT.
kono
parents:
diff changeset
1636
kono
parents:
diff changeset
1637 Value 0 is reserved for compile time invariants.
kono
parents:
diff changeset
1638 For common parameters it is REG_BR_PROB_BASE. For loop invariants it
kono
parents:
diff changeset
1639 ought to be REG_BR_PROB_BASE / estimated_iters. */
kono
parents:
diff changeset
1640
kono
parents:
diff changeset
1641 static int
kono
parents:
diff changeset
1642 param_change_prob (gimple *stmt, int i)
kono
parents:
diff changeset
1643 {
kono
parents:
diff changeset
1644 tree op = gimple_call_arg (stmt, i);
kono
parents:
diff changeset
1645 basic_block bb = gimple_bb (stmt);
kono
parents:
diff changeset
1646
kono
parents:
diff changeset
1647 if (TREE_CODE (op) == WITH_SIZE_EXPR)
kono
parents:
diff changeset
1648 op = TREE_OPERAND (op, 0);
kono
parents:
diff changeset
1649
kono
parents:
diff changeset
1650 tree base = get_base_address (op);
kono
parents:
diff changeset
1651
kono
parents:
diff changeset
1652 /* Global invariants never change. */
kono
parents:
diff changeset
1653 if (is_gimple_min_invariant (base))
kono
parents:
diff changeset
1654 return 0;
kono
parents:
diff changeset
1655
kono
parents:
diff changeset
1656 /* We would have to do non-trivial analysis to really work out what
kono
parents:
diff changeset
1657 is the probability of value to change (i.e. when init statement
kono
parents:
diff changeset
1658 is in a sibling loop of the call).
kono
parents:
diff changeset
1659
kono
parents:
diff changeset
1660 We do an conservative estimate: when call is executed N times more often
kono
parents:
diff changeset
1661 than the statement defining value, we take the frequency 1/N. */
kono
parents:
diff changeset
1662 if (TREE_CODE (base) == SSA_NAME)
kono
parents:
diff changeset
1663 {
kono
parents:
diff changeset
1664 int init_freq;
kono
parents:
diff changeset
1665
kono
parents:
diff changeset
1666 if (!bb->frequency)
kono
parents:
diff changeset
1667 return REG_BR_PROB_BASE;
kono
parents:
diff changeset
1668
kono
parents:
diff changeset
1669 if (SSA_NAME_IS_DEFAULT_DEF (base))
kono
parents:
diff changeset
1670 init_freq = ENTRY_BLOCK_PTR_FOR_FN (cfun)->frequency;
kono
parents:
diff changeset
1671 else
kono
parents:
diff changeset
1672 init_freq = get_minimal_bb
kono
parents:
diff changeset
1673 (gimple_bb (SSA_NAME_DEF_STMT (base)),
kono
parents:
diff changeset
1674 gimple_bb (stmt))->frequency;
kono
parents:
diff changeset
1675
kono
parents:
diff changeset
1676 if (!init_freq)
kono
parents:
diff changeset
1677 init_freq = 1;
kono
parents:
diff changeset
1678 if (init_freq < bb->frequency)
kono
parents:
diff changeset
1679 return MAX (GCOV_COMPUTE_SCALE (init_freq, bb->frequency), 1);
kono
parents:
diff changeset
1680 else
kono
parents:
diff changeset
1681 return REG_BR_PROB_BASE;
kono
parents:
diff changeset
1682 }
kono
parents:
diff changeset
1683 else
kono
parents:
diff changeset
1684 {
kono
parents:
diff changeset
1685 ao_ref refd;
kono
parents:
diff changeset
1686 int max;
kono
parents:
diff changeset
1687 struct record_modified_bb_info info;
kono
parents:
diff changeset
1688 bitmap_iterator bi;
kono
parents:
diff changeset
1689 unsigned index;
kono
parents:
diff changeset
1690 tree init = ctor_for_folding (base);
kono
parents:
diff changeset
1691
kono
parents:
diff changeset
1692 if (init != error_mark_node)
kono
parents:
diff changeset
1693 return 0;
kono
parents:
diff changeset
1694 if (!bb->frequency)
kono
parents:
diff changeset
1695 return REG_BR_PROB_BASE;
kono
parents:
diff changeset
1696 ao_ref_init (&refd, op);
kono
parents:
diff changeset
1697 info.stmt = stmt;
kono
parents:
diff changeset
1698 info.bb_set = BITMAP_ALLOC (NULL);
kono
parents:
diff changeset
1699 walk_aliased_vdefs (&refd, gimple_vuse (stmt), record_modified, &info,
kono
parents:
diff changeset
1700 NULL);
kono
parents:
diff changeset
1701 if (bitmap_bit_p (info.bb_set, bb->index))
kono
parents:
diff changeset
1702 {
kono
parents:
diff changeset
1703 BITMAP_FREE (info.bb_set);
kono
parents:
diff changeset
1704 return REG_BR_PROB_BASE;
kono
parents:
diff changeset
1705 }
kono
parents:
diff changeset
1706
kono
parents:
diff changeset
1707 /* Assume that every memory is initialized at entry.
kono
parents:
diff changeset
1708 TODO: Can we easilly determine if value is always defined
kono
parents:
diff changeset
1709 and thus we may skip entry block? */
kono
parents:
diff changeset
1710 if (ENTRY_BLOCK_PTR_FOR_FN (cfun)->frequency)
kono
parents:
diff changeset
1711 max = ENTRY_BLOCK_PTR_FOR_FN (cfun)->frequency;
kono
parents:
diff changeset
1712 else
kono
parents:
diff changeset
1713 max = 1;
kono
parents:
diff changeset
1714
kono
parents:
diff changeset
1715 EXECUTE_IF_SET_IN_BITMAP (info.bb_set, 0, index, bi)
kono
parents:
diff changeset
1716 max = MIN (max, BASIC_BLOCK_FOR_FN (cfun, index)->frequency);
kono
parents:
diff changeset
1717
kono
parents:
diff changeset
1718 BITMAP_FREE (info.bb_set);
kono
parents:
diff changeset
1719 if (max < bb->frequency)
kono
parents:
diff changeset
1720 return MAX (GCOV_COMPUTE_SCALE (max, bb->frequency), 1);
kono
parents:
diff changeset
1721 else
kono
parents:
diff changeset
1722 return REG_BR_PROB_BASE;
kono
parents:
diff changeset
1723 }
kono
parents:
diff changeset
1724 }
kono
parents:
diff changeset
1725
kono
parents:
diff changeset
1726 /* Find whether a basic block BB is the final block of a (half) diamond CFG
kono
parents:
diff changeset
1727 sub-graph and if the predicate the condition depends on is known. If so,
kono
parents:
diff changeset
1728 return true and store the pointer the predicate in *P. */
kono
parents:
diff changeset
1729
kono
parents:
diff changeset
1730 static bool
kono
parents:
diff changeset
1731 phi_result_unknown_predicate (struct ipa_node_params *info,
kono
parents:
diff changeset
1732 ipa_fn_summary *summary, basic_block bb,
kono
parents:
diff changeset
1733 predicate *p,
kono
parents:
diff changeset
1734 vec<predicate> nonconstant_names)
kono
parents:
diff changeset
1735 {
kono
parents:
diff changeset
1736 edge e;
kono
parents:
diff changeset
1737 edge_iterator ei;
kono
parents:
diff changeset
1738 basic_block first_bb = NULL;
kono
parents:
diff changeset
1739 gimple *stmt;
kono
parents:
diff changeset
1740
kono
parents:
diff changeset
1741 if (single_pred_p (bb))
kono
parents:
diff changeset
1742 {
kono
parents:
diff changeset
1743 *p = false;
kono
parents:
diff changeset
1744 return true;
kono
parents:
diff changeset
1745 }
kono
parents:
diff changeset
1746
kono
parents:
diff changeset
1747 FOR_EACH_EDGE (e, ei, bb->preds)
kono
parents:
diff changeset
1748 {
kono
parents:
diff changeset
1749 if (single_succ_p (e->src))
kono
parents:
diff changeset
1750 {
kono
parents:
diff changeset
1751 if (!single_pred_p (e->src))
kono
parents:
diff changeset
1752 return false;
kono
parents:
diff changeset
1753 if (!first_bb)
kono
parents:
diff changeset
1754 first_bb = single_pred (e->src);
kono
parents:
diff changeset
1755 else if (single_pred (e->src) != first_bb)
kono
parents:
diff changeset
1756 return false;
kono
parents:
diff changeset
1757 }
kono
parents:
diff changeset
1758 else
kono
parents:
diff changeset
1759 {
kono
parents:
diff changeset
1760 if (!first_bb)
kono
parents:
diff changeset
1761 first_bb = e->src;
kono
parents:
diff changeset
1762 else if (e->src != first_bb)
kono
parents:
diff changeset
1763 return false;
kono
parents:
diff changeset
1764 }
kono
parents:
diff changeset
1765 }
kono
parents:
diff changeset
1766
kono
parents:
diff changeset
1767 if (!first_bb)
kono
parents:
diff changeset
1768 return false;
kono
parents:
diff changeset
1769
kono
parents:
diff changeset
1770 stmt = last_stmt (first_bb);
kono
parents:
diff changeset
1771 if (!stmt
kono
parents:
diff changeset
1772 || gimple_code (stmt) != GIMPLE_COND
kono
parents:
diff changeset
1773 || !is_gimple_ip_invariant (gimple_cond_rhs (stmt)))
kono
parents:
diff changeset
1774 return false;
kono
parents:
diff changeset
1775
kono
parents:
diff changeset
1776 *p = will_be_nonconstant_expr_predicate (info, summary,
kono
parents:
diff changeset
1777 gimple_cond_lhs (stmt),
kono
parents:
diff changeset
1778 nonconstant_names);
kono
parents:
diff changeset
1779 if (*p == true)
kono
parents:
diff changeset
1780 return false;
kono
parents:
diff changeset
1781 else
kono
parents:
diff changeset
1782 return true;
kono
parents:
diff changeset
1783 }
kono
parents:
diff changeset
1784
kono
parents:
diff changeset
1785 /* Given a PHI statement in a function described by inline properties SUMMARY
kono
parents:
diff changeset
1786 and *P being the predicate describing whether the selected PHI argument is
kono
parents:
diff changeset
1787 known, store a predicate for the result of the PHI statement into
kono
parents:
diff changeset
1788 NONCONSTANT_NAMES, if possible. */
kono
parents:
diff changeset
1789
kono
parents:
diff changeset
1790 static void
kono
parents:
diff changeset
1791 predicate_for_phi_result (struct ipa_fn_summary *summary, gphi *phi,
kono
parents:
diff changeset
1792 predicate *p,
kono
parents:
diff changeset
1793 vec<predicate> nonconstant_names)
kono
parents:
diff changeset
1794 {
kono
parents:
diff changeset
1795 unsigned i;
kono
parents:
diff changeset
1796
kono
parents:
diff changeset
1797 for (i = 0; i < gimple_phi_num_args (phi); i++)
kono
parents:
diff changeset
1798 {
kono
parents:
diff changeset
1799 tree arg = gimple_phi_arg (phi, i)->def;
kono
parents:
diff changeset
1800 if (!is_gimple_min_invariant (arg))
kono
parents:
diff changeset
1801 {
kono
parents:
diff changeset
1802 gcc_assert (TREE_CODE (arg) == SSA_NAME);
kono
parents:
diff changeset
1803 *p = p->or_with (summary->conds,
kono
parents:
diff changeset
1804 nonconstant_names[SSA_NAME_VERSION (arg)]);
kono
parents:
diff changeset
1805 if (*p == true)
kono
parents:
diff changeset
1806 return;
kono
parents:
diff changeset
1807 }
kono
parents:
diff changeset
1808 }
kono
parents:
diff changeset
1809
kono
parents:
diff changeset
1810 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
1811 {
kono
parents:
diff changeset
1812 fprintf (dump_file, "\t\tphi predicate: ");
kono
parents:
diff changeset
1813 p->dump (dump_file, summary->conds);
kono
parents:
diff changeset
1814 }
kono
parents:
diff changeset
1815 nonconstant_names[SSA_NAME_VERSION (gimple_phi_result (phi))] = *p;
kono
parents:
diff changeset
1816 }
kono
parents:
diff changeset
1817
kono
parents:
diff changeset
1818 /* Return predicate specifying when array index in access OP becomes non-constant. */
kono
parents:
diff changeset
1819
kono
parents:
diff changeset
1820 static predicate
kono
parents:
diff changeset
1821 array_index_predicate (ipa_fn_summary *info,
kono
parents:
diff changeset
1822 vec< predicate> nonconstant_names, tree op)
kono
parents:
diff changeset
1823 {
kono
parents:
diff changeset
1824 predicate p = false;
kono
parents:
diff changeset
1825 while (handled_component_p (op))
kono
parents:
diff changeset
1826 {
kono
parents:
diff changeset
1827 if (TREE_CODE (op) == ARRAY_REF || TREE_CODE (op) == ARRAY_RANGE_REF)
kono
parents:
diff changeset
1828 {
kono
parents:
diff changeset
1829 if (TREE_CODE (TREE_OPERAND (op, 1)) == SSA_NAME)
kono
parents:
diff changeset
1830 p = p.or_with (info->conds,
kono
parents:
diff changeset
1831 nonconstant_names[SSA_NAME_VERSION
kono
parents:
diff changeset
1832 (TREE_OPERAND (op, 1))]);
kono
parents:
diff changeset
1833 }
kono
parents:
diff changeset
1834 op = TREE_OPERAND (op, 0);
kono
parents:
diff changeset
1835 }
kono
parents:
diff changeset
1836 return p;
kono
parents:
diff changeset
1837 }
kono
parents:
diff changeset
1838
kono
parents:
diff changeset
1839 /* For a typical usage of __builtin_expect (a<b, 1), we
kono
parents:
diff changeset
1840 may introduce an extra relation stmt:
kono
parents:
diff changeset
1841 With the builtin, we have
kono
parents:
diff changeset
1842 t1 = a <= b;
kono
parents:
diff changeset
1843 t2 = (long int) t1;
kono
parents:
diff changeset
1844 t3 = __builtin_expect (t2, 1);
kono
parents:
diff changeset
1845 if (t3 != 0)
kono
parents:
diff changeset
1846 goto ...
kono
parents:
diff changeset
1847 Without the builtin, we have
kono
parents:
diff changeset
1848 if (a<=b)
kono
parents:
diff changeset
1849 goto...
kono
parents:
diff changeset
1850 This affects the size/time estimation and may have
kono
parents:
diff changeset
1851 an impact on the earlier inlining.
kono
parents:
diff changeset
1852 Here find this pattern and fix it up later. */
kono
parents:
diff changeset
1853
kono
parents:
diff changeset
1854 static gimple *
kono
parents:
diff changeset
1855 find_foldable_builtin_expect (basic_block bb)
kono
parents:
diff changeset
1856 {
kono
parents:
diff changeset
1857 gimple_stmt_iterator bsi;
kono
parents:
diff changeset
1858
kono
parents:
diff changeset
1859 for (bsi = gsi_start_bb (bb); !gsi_end_p (bsi); gsi_next (&bsi))
kono
parents:
diff changeset
1860 {
kono
parents:
diff changeset
1861 gimple *stmt = gsi_stmt (bsi);
kono
parents:
diff changeset
1862 if (gimple_call_builtin_p (stmt, BUILT_IN_EXPECT)
kono
parents:
diff changeset
1863 || gimple_call_internal_p (stmt, IFN_BUILTIN_EXPECT))
kono
parents:
diff changeset
1864 {
kono
parents:
diff changeset
1865 tree var = gimple_call_lhs (stmt);
kono
parents:
diff changeset
1866 tree arg = gimple_call_arg (stmt, 0);
kono
parents:
diff changeset
1867 use_operand_p use_p;
kono
parents:
diff changeset
1868 gimple *use_stmt;
kono
parents:
diff changeset
1869 bool match = false;
kono
parents:
diff changeset
1870 bool done = false;
kono
parents:
diff changeset
1871
kono
parents:
diff changeset
1872 if (!var || !arg)
kono
parents:
diff changeset
1873 continue;
kono
parents:
diff changeset
1874 gcc_assert (TREE_CODE (var) == SSA_NAME);
kono
parents:
diff changeset
1875
kono
parents:
diff changeset
1876 while (TREE_CODE (arg) == SSA_NAME)
kono
parents:
diff changeset
1877 {
kono
parents:
diff changeset
1878 gimple *stmt_tmp = SSA_NAME_DEF_STMT (arg);
kono
parents:
diff changeset
1879 if (!is_gimple_assign (stmt_tmp))
kono
parents:
diff changeset
1880 break;
kono
parents:
diff changeset
1881 switch (gimple_assign_rhs_code (stmt_tmp))
kono
parents:
diff changeset
1882 {
kono
parents:
diff changeset
1883 case LT_EXPR:
kono
parents:
diff changeset
1884 case LE_EXPR:
kono
parents:
diff changeset
1885 case GT_EXPR:
kono
parents:
diff changeset
1886 case GE_EXPR:
kono
parents:
diff changeset
1887 case EQ_EXPR:
kono
parents:
diff changeset
1888 case NE_EXPR:
kono
parents:
diff changeset
1889 match = true;
kono
parents:
diff changeset
1890 done = true;
kono
parents:
diff changeset
1891 break;
kono
parents:
diff changeset
1892 CASE_CONVERT:
kono
parents:
diff changeset
1893 break;
kono
parents:
diff changeset
1894 default:
kono
parents:
diff changeset
1895 done = true;
kono
parents:
diff changeset
1896 break;
kono
parents:
diff changeset
1897 }
kono
parents:
diff changeset
1898 if (done)
kono
parents:
diff changeset
1899 break;
kono
parents:
diff changeset
1900 arg = gimple_assign_rhs1 (stmt_tmp);
kono
parents:
diff changeset
1901 }
kono
parents:
diff changeset
1902
kono
parents:
diff changeset
1903 if (match && single_imm_use (var, &use_p, &use_stmt)
kono
parents:
diff changeset
1904 && gimple_code (use_stmt) == GIMPLE_COND)
kono
parents:
diff changeset
1905 return use_stmt;
kono
parents:
diff changeset
1906 }
kono
parents:
diff changeset
1907 }
kono
parents:
diff changeset
1908 return NULL;
kono
parents:
diff changeset
1909 }
kono
parents:
diff changeset
1910
kono
parents:
diff changeset
1911 /* Return true when the basic blocks contains only clobbers followed by RESX.
kono
parents:
diff changeset
1912 Such BBs are kept around to make removal of dead stores possible with
kono
parents:
diff changeset
1913 presence of EH and will be optimized out by optimize_clobbers later in the
kono
parents:
diff changeset
1914 game.
kono
parents:
diff changeset
1915
kono
parents:
diff changeset
1916 NEED_EH is used to recurse in case the clobber has non-EH predecestors
kono
parents:
diff changeset
1917 that can be clobber only, too.. When it is false, the RESX is not necessary
kono
parents:
diff changeset
1918 on the end of basic block. */
kono
parents:
diff changeset
1919
kono
parents:
diff changeset
1920 static bool
kono
parents:
diff changeset
1921 clobber_only_eh_bb_p (basic_block bb, bool need_eh = true)
kono
parents:
diff changeset
1922 {
kono
parents:
diff changeset
1923 gimple_stmt_iterator gsi = gsi_last_bb (bb);
kono
parents:
diff changeset
1924 edge_iterator ei;
kono
parents:
diff changeset
1925 edge e;
kono
parents:
diff changeset
1926
kono
parents:
diff changeset
1927 if (need_eh)
kono
parents:
diff changeset
1928 {
kono
parents:
diff changeset
1929 if (gsi_end_p (gsi))
kono
parents:
diff changeset
1930 return false;
kono
parents:
diff changeset
1931 if (gimple_code (gsi_stmt (gsi)) != GIMPLE_RESX)
kono
parents:
diff changeset
1932 return false;
kono
parents:
diff changeset
1933 gsi_prev (&gsi);
kono
parents:
diff changeset
1934 }
kono
parents:
diff changeset
1935 else if (!single_succ_p (bb))
kono
parents:
diff changeset
1936 return false;
kono
parents:
diff changeset
1937
kono
parents:
diff changeset
1938 for (; !gsi_end_p (gsi); gsi_prev (&gsi))
kono
parents:
diff changeset
1939 {
kono
parents:
diff changeset
1940 gimple *stmt = gsi_stmt (gsi);
kono
parents:
diff changeset
1941 if (is_gimple_debug (stmt))
kono
parents:
diff changeset
1942 continue;
kono
parents:
diff changeset
1943 if (gimple_clobber_p (stmt))
kono
parents:
diff changeset
1944 continue;
kono
parents:
diff changeset
1945 if (gimple_code (stmt) == GIMPLE_LABEL)
kono
parents:
diff changeset
1946 break;
kono
parents:
diff changeset
1947 return false;
kono
parents:
diff changeset
1948 }
kono
parents:
diff changeset
1949
kono
parents:
diff changeset
1950 /* See if all predecestors are either throws or clobber only BBs. */
kono
parents:
diff changeset
1951 FOR_EACH_EDGE (e, ei, bb->preds)
kono
parents:
diff changeset
1952 if (!(e->flags & EDGE_EH)
kono
parents:
diff changeset
1953 && !clobber_only_eh_bb_p (e->src, false))
kono
parents:
diff changeset
1954 return false;
kono
parents:
diff changeset
1955
kono
parents:
diff changeset
1956 return true;
kono
parents:
diff changeset
1957 }
kono
parents:
diff changeset
1958
kono
parents:
diff changeset
1959 /* Return true if STMT compute a floating point expression that may be affected
kono
parents:
diff changeset
1960 by -ffast-math and similar flags. */
kono
parents:
diff changeset
1961
kono
parents:
diff changeset
1962 static bool
kono
parents:
diff changeset
1963 fp_expression_p (gimple *stmt)
kono
parents:
diff changeset
1964 {
kono
parents:
diff changeset
1965 ssa_op_iter i;
kono
parents:
diff changeset
1966 tree op;
kono
parents:
diff changeset
1967
kono
parents:
diff changeset
1968 FOR_EACH_SSA_TREE_OPERAND (op, stmt, i, SSA_OP_DEF|SSA_OP_USE)
kono
parents:
diff changeset
1969 if (FLOAT_TYPE_P (TREE_TYPE (op)))
kono
parents:
diff changeset
1970 return true;
kono
parents:
diff changeset
1971 return false;
kono
parents:
diff changeset
1972 }
kono
parents:
diff changeset
1973
kono
parents:
diff changeset
1974 /* Analyze function body for NODE.
kono
parents:
diff changeset
1975 EARLY indicates run from early optimization pipeline. */
kono
parents:
diff changeset
1976
kono
parents:
diff changeset
1977 static void
kono
parents:
diff changeset
1978 analyze_function_body (struct cgraph_node *node, bool early)
kono
parents:
diff changeset
1979 {
kono
parents:
diff changeset
1980 sreal time = 0;
kono
parents:
diff changeset
1981 /* Estimate static overhead for function prologue/epilogue and alignment. */
kono
parents:
diff changeset
1982 int size = 2;
kono
parents:
diff changeset
1983 /* Benefits are scaled by probability of elimination that is in range
kono
parents:
diff changeset
1984 <0,2>. */
kono
parents:
diff changeset
1985 basic_block bb;
kono
parents:
diff changeset
1986 struct function *my_function = DECL_STRUCT_FUNCTION (node->decl);
kono
parents:
diff changeset
1987 int freq;
kono
parents:
diff changeset
1988 struct ipa_fn_summary *info = ipa_fn_summaries->get (node);
kono
parents:
diff changeset
1989 predicate bb_predicate;
kono
parents:
diff changeset
1990 struct ipa_func_body_info fbi;
kono
parents:
diff changeset
1991 vec<predicate> nonconstant_names = vNULL;
kono
parents:
diff changeset
1992 int nblocks, n;
kono
parents:
diff changeset
1993 int *order;
kono
parents:
diff changeset
1994 predicate array_index = true;
kono
parents:
diff changeset
1995 gimple *fix_builtin_expect_stmt;
kono
parents:
diff changeset
1996
kono
parents:
diff changeset
1997 gcc_assert (my_function && my_function->cfg);
kono
parents:
diff changeset
1998 gcc_assert (cfun == my_function);
kono
parents:
diff changeset
1999
kono
parents:
diff changeset
2000 memset(&fbi, 0, sizeof(fbi));
kono
parents:
diff changeset
2001 info->conds = NULL;
kono
parents:
diff changeset
2002 info->size_time_table = NULL;
kono
parents:
diff changeset
2003
kono
parents:
diff changeset
2004 /* When optimizing and analyzing for IPA inliner, initialize loop optimizer
kono
parents:
diff changeset
2005 so we can produce proper inline hints.
kono
parents:
diff changeset
2006
kono
parents:
diff changeset
2007 When optimizing and analyzing for early inliner, initialize node params
kono
parents:
diff changeset
2008 so we can produce correct BB predicates. */
kono
parents:
diff changeset
2009
kono
parents:
diff changeset
2010 if (opt_for_fn (node->decl, optimize))
kono
parents:
diff changeset
2011 {
kono
parents:
diff changeset
2012 calculate_dominance_info (CDI_DOMINATORS);
kono
parents:
diff changeset
2013 if (!early)
kono
parents:
diff changeset
2014 loop_optimizer_init (LOOPS_NORMAL | LOOPS_HAVE_RECORDED_EXITS);
kono
parents:
diff changeset
2015 else
kono
parents:
diff changeset
2016 {
kono
parents:
diff changeset
2017 ipa_check_create_node_params ();
kono
parents:
diff changeset
2018 ipa_initialize_node_params (node);
kono
parents:
diff changeset
2019 }
kono
parents:
diff changeset
2020
kono
parents:
diff changeset
2021 if (ipa_node_params_sum)
kono
parents:
diff changeset
2022 {
kono
parents:
diff changeset
2023 fbi.node = node;
kono
parents:
diff changeset
2024 fbi.info = IPA_NODE_REF (node);
kono
parents:
diff changeset
2025 fbi.bb_infos = vNULL;
kono
parents:
diff changeset
2026 fbi.bb_infos.safe_grow_cleared (last_basic_block_for_fn (cfun));
kono
parents:
diff changeset
2027 fbi.param_count = count_formal_params(node->decl);
kono
parents:
diff changeset
2028 nonconstant_names.safe_grow_cleared
kono
parents:
diff changeset
2029 (SSANAMES (my_function)->length ());
kono
parents:
diff changeset
2030 }
kono
parents:
diff changeset
2031 }
kono
parents:
diff changeset
2032
kono
parents:
diff changeset
2033 if (dump_file)
kono
parents:
diff changeset
2034 fprintf (dump_file, "\nAnalyzing function body size: %s\n",
kono
parents:
diff changeset
2035 node->name ());
kono
parents:
diff changeset
2036
kono
parents:
diff changeset
2037 /* When we run into maximal number of entries, we assign everything to the
kono
parents:
diff changeset
2038 constant truth case. Be sure to have it in list. */
kono
parents:
diff changeset
2039 bb_predicate = true;
kono
parents:
diff changeset
2040 info->account_size_time (0, 0, bb_predicate, bb_predicate);
kono
parents:
diff changeset
2041
kono
parents:
diff changeset
2042 bb_predicate = predicate::not_inlined ();
kono
parents:
diff changeset
2043 info->account_size_time (2 * ipa_fn_summary::size_scale, 0, bb_predicate,
kono
parents:
diff changeset
2044 bb_predicate);
kono
parents:
diff changeset
2045
kono
parents:
diff changeset
2046 if (fbi.info)
kono
parents:
diff changeset
2047 compute_bb_predicates (&fbi, node, info);
kono
parents:
diff changeset
2048 order = XNEWVEC (int, n_basic_blocks_for_fn (cfun));
kono
parents:
diff changeset
2049 nblocks = pre_and_rev_post_order_compute (NULL, order, false);
kono
parents:
diff changeset
2050 for (n = 0; n < nblocks; n++)
kono
parents:
diff changeset
2051 {
kono
parents:
diff changeset
2052 bb = BASIC_BLOCK_FOR_FN (cfun, order[n]);
kono
parents:
diff changeset
2053 freq = compute_call_stmt_bb_frequency (node->decl, bb);
kono
parents:
diff changeset
2054 if (clobber_only_eh_bb_p (bb))
kono
parents:
diff changeset
2055 {
kono
parents:
diff changeset
2056 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2057 fprintf (dump_file, "\n Ignoring BB %i;"
kono
parents:
diff changeset
2058 " it will be optimized away by cleanup_clobbers\n",
kono
parents:
diff changeset
2059 bb->index);
kono
parents:
diff changeset
2060 continue;
kono
parents:
diff changeset
2061 }
kono
parents:
diff changeset
2062
kono
parents:
diff changeset
2063 /* TODO: Obviously predicates can be propagated down across CFG. */
kono
parents:
diff changeset
2064 if (fbi.info)
kono
parents:
diff changeset
2065 {
kono
parents:
diff changeset
2066 if (bb->aux)
kono
parents:
diff changeset
2067 bb_predicate = *(predicate *) bb->aux;
kono
parents:
diff changeset
2068 else
kono
parents:
diff changeset
2069 bb_predicate = false;
kono
parents:
diff changeset
2070 }
kono
parents:
diff changeset
2071 else
kono
parents:
diff changeset
2072 bb_predicate = true;
kono
parents:
diff changeset
2073
kono
parents:
diff changeset
2074 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2075 {
kono
parents:
diff changeset
2076 fprintf (dump_file, "\n BB %i predicate:", bb->index);
kono
parents:
diff changeset
2077 bb_predicate.dump (dump_file, info->conds);
kono
parents:
diff changeset
2078 }
kono
parents:
diff changeset
2079
kono
parents:
diff changeset
2080 if (fbi.info && nonconstant_names.exists ())
kono
parents:
diff changeset
2081 {
kono
parents:
diff changeset
2082 predicate phi_predicate;
kono
parents:
diff changeset
2083 bool first_phi = true;
kono
parents:
diff changeset
2084
kono
parents:
diff changeset
2085 for (gphi_iterator bsi = gsi_start_phis (bb); !gsi_end_p (bsi);
kono
parents:
diff changeset
2086 gsi_next (&bsi))
kono
parents:
diff changeset
2087 {
kono
parents:
diff changeset
2088 if (first_phi
kono
parents:
diff changeset
2089 && !phi_result_unknown_predicate (fbi.info, info, bb,
kono
parents:
diff changeset
2090 &phi_predicate,
kono
parents:
diff changeset
2091 nonconstant_names))
kono
parents:
diff changeset
2092 break;
kono
parents:
diff changeset
2093 first_phi = false;
kono
parents:
diff changeset
2094 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2095 {
kono
parents:
diff changeset
2096 fprintf (dump_file, " ");
kono
parents:
diff changeset
2097 print_gimple_stmt (dump_file, gsi_stmt (bsi), 0);
kono
parents:
diff changeset
2098 }
kono
parents:
diff changeset
2099 predicate_for_phi_result (info, bsi.phi (), &phi_predicate,
kono
parents:
diff changeset
2100 nonconstant_names);
kono
parents:
diff changeset
2101 }
kono
parents:
diff changeset
2102 }
kono
parents:
diff changeset
2103
kono
parents:
diff changeset
2104 fix_builtin_expect_stmt = find_foldable_builtin_expect (bb);
kono
parents:
diff changeset
2105
kono
parents:
diff changeset
2106 for (gimple_stmt_iterator bsi = gsi_start_bb (bb); !gsi_end_p (bsi);
kono
parents:
diff changeset
2107 gsi_next (&bsi))
kono
parents:
diff changeset
2108 {
kono
parents:
diff changeset
2109 gimple *stmt = gsi_stmt (bsi);
kono
parents:
diff changeset
2110 int this_size = estimate_num_insns (stmt, &eni_size_weights);
kono
parents:
diff changeset
2111 int this_time = estimate_num_insns (stmt, &eni_time_weights);
kono
parents:
diff changeset
2112 int prob;
kono
parents:
diff changeset
2113 predicate will_be_nonconstant;
kono
parents:
diff changeset
2114
kono
parents:
diff changeset
2115 /* This relation stmt should be folded after we remove
kono
parents:
diff changeset
2116 buildin_expect call. Adjust the cost here. */
kono
parents:
diff changeset
2117 if (stmt == fix_builtin_expect_stmt)
kono
parents:
diff changeset
2118 {
kono
parents:
diff changeset
2119 this_size--;
kono
parents:
diff changeset
2120 this_time--;
kono
parents:
diff changeset
2121 }
kono
parents:
diff changeset
2122
kono
parents:
diff changeset
2123 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2124 {
kono
parents:
diff changeset
2125 fprintf (dump_file, " ");
kono
parents:
diff changeset
2126 print_gimple_stmt (dump_file, stmt, 0);
kono
parents:
diff changeset
2127 fprintf (dump_file, "\t\tfreq:%3.2f size:%3i time:%3i\n",
kono
parents:
diff changeset
2128 ((double) freq) / CGRAPH_FREQ_BASE, this_size,
kono
parents:
diff changeset
2129 this_time);
kono
parents:
diff changeset
2130 }
kono
parents:
diff changeset
2131
kono
parents:
diff changeset
2132 if (gimple_assign_load_p (stmt) && nonconstant_names.exists ())
kono
parents:
diff changeset
2133 {
kono
parents:
diff changeset
2134 predicate this_array_index;
kono
parents:
diff changeset
2135 this_array_index =
kono
parents:
diff changeset
2136 array_index_predicate (info, nonconstant_names,
kono
parents:
diff changeset
2137 gimple_assign_rhs1 (stmt));
kono
parents:
diff changeset
2138 if (this_array_index != false)
kono
parents:
diff changeset
2139 array_index &= this_array_index;
kono
parents:
diff changeset
2140 }
kono
parents:
diff changeset
2141 if (gimple_store_p (stmt) && nonconstant_names.exists ())
kono
parents:
diff changeset
2142 {
kono
parents:
diff changeset
2143 predicate this_array_index;
kono
parents:
diff changeset
2144 this_array_index =
kono
parents:
diff changeset
2145 array_index_predicate (info, nonconstant_names,
kono
parents:
diff changeset
2146 gimple_get_lhs (stmt));
kono
parents:
diff changeset
2147 if (this_array_index != false)
kono
parents:
diff changeset
2148 array_index &= this_array_index;
kono
parents:
diff changeset
2149 }
kono
parents:
diff changeset
2150
kono
parents:
diff changeset
2151
kono
parents:
diff changeset
2152 if (is_gimple_call (stmt)
kono
parents:
diff changeset
2153 && !gimple_call_internal_p (stmt))
kono
parents:
diff changeset
2154 {
kono
parents:
diff changeset
2155 struct cgraph_edge *edge = node->get_edge (stmt);
kono
parents:
diff changeset
2156 struct ipa_call_summary *es = ipa_call_summaries->get (edge);
kono
parents:
diff changeset
2157
kono
parents:
diff changeset
2158 /* Special case: results of BUILT_IN_CONSTANT_P will be always
kono
parents:
diff changeset
2159 resolved as constant. We however don't want to optimize
kono
parents:
diff changeset
2160 out the cgraph edges. */
kono
parents:
diff changeset
2161 if (nonconstant_names.exists ()
kono
parents:
diff changeset
2162 && gimple_call_builtin_p (stmt, BUILT_IN_CONSTANT_P)
kono
parents:
diff changeset
2163 && gimple_call_lhs (stmt)
kono
parents:
diff changeset
2164 && TREE_CODE (gimple_call_lhs (stmt)) == SSA_NAME)
kono
parents:
diff changeset
2165 {
kono
parents:
diff changeset
2166 predicate false_p = false;
kono
parents:
diff changeset
2167 nonconstant_names[SSA_NAME_VERSION (gimple_call_lhs (stmt))]
kono
parents:
diff changeset
2168 = false_p;
kono
parents:
diff changeset
2169 }
kono
parents:
diff changeset
2170 if (ipa_node_params_sum)
kono
parents:
diff changeset
2171 {
kono
parents:
diff changeset
2172 int count = gimple_call_num_args (stmt);
kono
parents:
diff changeset
2173 int i;
kono
parents:
diff changeset
2174
kono
parents:
diff changeset
2175 if (count)
kono
parents:
diff changeset
2176 es->param.safe_grow_cleared (count);
kono
parents:
diff changeset
2177 for (i = 0; i < count; i++)
kono
parents:
diff changeset
2178 {
kono
parents:
diff changeset
2179 int prob = param_change_prob (stmt, i);
kono
parents:
diff changeset
2180 gcc_assert (prob >= 0 && prob <= REG_BR_PROB_BASE);
kono
parents:
diff changeset
2181 es->param[i].change_prob = prob;
kono
parents:
diff changeset
2182 }
kono
parents:
diff changeset
2183 }
kono
parents:
diff changeset
2184
kono
parents:
diff changeset
2185 es->call_stmt_size = this_size;
kono
parents:
diff changeset
2186 es->call_stmt_time = this_time;
kono
parents:
diff changeset
2187 es->loop_depth = bb_loop_depth (bb);
kono
parents:
diff changeset
2188 edge_set_predicate (edge, &bb_predicate);
kono
parents:
diff changeset
2189 }
kono
parents:
diff changeset
2190
kono
parents:
diff changeset
2191 /* TODO: When conditional jump or swithc is known to be constant, but
kono
parents:
diff changeset
2192 we did not translate it into the predicates, we really can account
kono
parents:
diff changeset
2193 just maximum of the possible paths. */
kono
parents:
diff changeset
2194 if (fbi.info)
kono
parents:
diff changeset
2195 will_be_nonconstant
kono
parents:
diff changeset
2196 = will_be_nonconstant_predicate (&fbi, info,
kono
parents:
diff changeset
2197 stmt, nonconstant_names);
kono
parents:
diff changeset
2198 else
kono
parents:
diff changeset
2199 will_be_nonconstant = true;
kono
parents:
diff changeset
2200 if (this_time || this_size)
kono
parents:
diff changeset
2201 {
kono
parents:
diff changeset
2202 this_time *= freq;
kono
parents:
diff changeset
2203
kono
parents:
diff changeset
2204 prob = eliminated_by_inlining_prob (stmt);
kono
parents:
diff changeset
2205 if (prob == 1 && dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2206 fprintf (dump_file,
kono
parents:
diff changeset
2207 "\t\t50%% will be eliminated by inlining\n");
kono
parents:
diff changeset
2208 if (prob == 2 && dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2209 fprintf (dump_file, "\t\tWill be eliminated by inlining\n");
kono
parents:
diff changeset
2210
kono
parents:
diff changeset
2211 struct predicate p = bb_predicate & will_be_nonconstant;
kono
parents:
diff changeset
2212
kono
parents:
diff changeset
2213 /* We can ignore statement when we proved it is never going
kono
parents:
diff changeset
2214 to happen, but we can not do that for call statements
kono
parents:
diff changeset
2215 because edges are accounted specially. */
kono
parents:
diff changeset
2216
kono
parents:
diff changeset
2217 if (*(is_gimple_call (stmt) ? &bb_predicate : &p) != false)
kono
parents:
diff changeset
2218 {
kono
parents:
diff changeset
2219 time += this_time;
kono
parents:
diff changeset
2220 size += this_size;
kono
parents:
diff changeset
2221 }
kono
parents:
diff changeset
2222
kono
parents:
diff changeset
2223 /* We account everything but the calls. Calls have their own
kono
parents:
diff changeset
2224 size/time info attached to cgraph edges. This is necessary
kono
parents:
diff changeset
2225 in order to make the cost disappear after inlining. */
kono
parents:
diff changeset
2226 if (!is_gimple_call (stmt))
kono
parents:
diff changeset
2227 {
kono
parents:
diff changeset
2228 if (prob)
kono
parents:
diff changeset
2229 {
kono
parents:
diff changeset
2230 predicate ip = bb_predicate & predicate::not_inlined ();
kono
parents:
diff changeset
2231 info->account_size_time (this_size * prob,
kono
parents:
diff changeset
2232 (sreal)(this_time * prob)
kono
parents:
diff changeset
2233 / (CGRAPH_FREQ_BASE * 2), ip,
kono
parents:
diff changeset
2234 p);
kono
parents:
diff changeset
2235 }
kono
parents:
diff changeset
2236 if (prob != 2)
kono
parents:
diff changeset
2237 info->account_size_time (this_size * (2 - prob),
kono
parents:
diff changeset
2238 (sreal)(this_time * (2 - prob))
kono
parents:
diff changeset
2239 / (CGRAPH_FREQ_BASE * 2),
kono
parents:
diff changeset
2240 bb_predicate,
kono
parents:
diff changeset
2241 p);
kono
parents:
diff changeset
2242 }
kono
parents:
diff changeset
2243
kono
parents:
diff changeset
2244 if (!info->fp_expressions && fp_expression_p (stmt))
kono
parents:
diff changeset
2245 {
kono
parents:
diff changeset
2246 info->fp_expressions = true;
kono
parents:
diff changeset
2247 if (dump_file)
kono
parents:
diff changeset
2248 fprintf (dump_file, " fp_expression set\n");
kono
parents:
diff changeset
2249 }
kono
parents:
diff changeset
2250
kono
parents:
diff changeset
2251 gcc_assert (time >= 0);
kono
parents:
diff changeset
2252 gcc_assert (size >= 0);
kono
parents:
diff changeset
2253 }
kono
parents:
diff changeset
2254 }
kono
parents:
diff changeset
2255 }
kono
parents:
diff changeset
2256 set_hint_predicate (&ipa_fn_summaries->get (node)->array_index, array_index);
kono
parents:
diff changeset
2257 time = time / CGRAPH_FREQ_BASE;
kono
parents:
diff changeset
2258 free (order);
kono
parents:
diff changeset
2259
kono
parents:
diff changeset
2260 if (nonconstant_names.exists () && !early)
kono
parents:
diff changeset
2261 {
kono
parents:
diff changeset
2262 struct loop *loop;
kono
parents:
diff changeset
2263 predicate loop_iterations = true;
kono
parents:
diff changeset
2264 predicate loop_stride = true;
kono
parents:
diff changeset
2265
kono
parents:
diff changeset
2266 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2267 flow_loops_dump (dump_file, NULL, 0);
kono
parents:
diff changeset
2268 scev_initialize ();
kono
parents:
diff changeset
2269 FOR_EACH_LOOP (loop, 0)
kono
parents:
diff changeset
2270 {
kono
parents:
diff changeset
2271 vec<edge> exits;
kono
parents:
diff changeset
2272 edge ex;
kono
parents:
diff changeset
2273 unsigned int j;
kono
parents:
diff changeset
2274 struct tree_niter_desc niter_desc;
kono
parents:
diff changeset
2275 bb_predicate = *(predicate *) loop->header->aux;
kono
parents:
diff changeset
2276
kono
parents:
diff changeset
2277 exits = get_loop_exit_edges (loop);
kono
parents:
diff changeset
2278 FOR_EACH_VEC_ELT (exits, j, ex)
kono
parents:
diff changeset
2279 if (number_of_iterations_exit (loop, ex, &niter_desc, false)
kono
parents:
diff changeset
2280 && !is_gimple_min_invariant (niter_desc.niter))
kono
parents:
diff changeset
2281 {
kono
parents:
diff changeset
2282 predicate will_be_nonconstant
kono
parents:
diff changeset
2283 = will_be_nonconstant_expr_predicate (fbi.info, info,
kono
parents:
diff changeset
2284 niter_desc.niter,
kono
parents:
diff changeset
2285 nonconstant_names);
kono
parents:
diff changeset
2286 if (will_be_nonconstant != true)
kono
parents:
diff changeset
2287 will_be_nonconstant = bb_predicate & will_be_nonconstant;
kono
parents:
diff changeset
2288 if (will_be_nonconstant != true
kono
parents:
diff changeset
2289 && will_be_nonconstant != false)
kono
parents:
diff changeset
2290 /* This is slightly inprecise. We may want to represent each
kono
parents:
diff changeset
2291 loop with independent predicate. */
kono
parents:
diff changeset
2292 loop_iterations &= will_be_nonconstant;
kono
parents:
diff changeset
2293 }
kono
parents:
diff changeset
2294 exits.release ();
kono
parents:
diff changeset
2295 }
kono
parents:
diff changeset
2296
kono
parents:
diff changeset
2297 /* To avoid quadratic behavior we analyze stride predicates only
kono
parents:
diff changeset
2298 with respect to the containing loop. Thus we simply iterate
kono
parents:
diff changeset
2299 over all defs in the outermost loop body. */
kono
parents:
diff changeset
2300 for (loop = loops_for_fn (cfun)->tree_root->inner;
kono
parents:
diff changeset
2301 loop != NULL; loop = loop->next)
kono
parents:
diff changeset
2302 {
kono
parents:
diff changeset
2303 basic_block *body = get_loop_body (loop);
kono
parents:
diff changeset
2304 for (unsigned i = 0; i < loop->num_nodes; i++)
kono
parents:
diff changeset
2305 {
kono
parents:
diff changeset
2306 gimple_stmt_iterator gsi;
kono
parents:
diff changeset
2307 bb_predicate = *(predicate *) body[i]->aux;
kono
parents:
diff changeset
2308 for (gsi = gsi_start_bb (body[i]); !gsi_end_p (gsi);
kono
parents:
diff changeset
2309 gsi_next (&gsi))
kono
parents:
diff changeset
2310 {
kono
parents:
diff changeset
2311 gimple *stmt = gsi_stmt (gsi);
kono
parents:
diff changeset
2312
kono
parents:
diff changeset
2313 if (!is_gimple_assign (stmt))
kono
parents:
diff changeset
2314 continue;
kono
parents:
diff changeset
2315
kono
parents:
diff changeset
2316 tree def = gimple_assign_lhs (stmt);
kono
parents:
diff changeset
2317 if (TREE_CODE (def) != SSA_NAME)
kono
parents:
diff changeset
2318 continue;
kono
parents:
diff changeset
2319
kono
parents:
diff changeset
2320 affine_iv iv;
kono
parents:
diff changeset
2321 if (!simple_iv (loop_containing_stmt (stmt),
kono
parents:
diff changeset
2322 loop_containing_stmt (stmt),
kono
parents:
diff changeset
2323 def, &iv, true)
kono
parents:
diff changeset
2324 || is_gimple_min_invariant (iv.step))
kono
parents:
diff changeset
2325 continue;
kono
parents:
diff changeset
2326
kono
parents:
diff changeset
2327 predicate will_be_nonconstant
kono
parents:
diff changeset
2328 = will_be_nonconstant_expr_predicate (fbi.info, info,
kono
parents:
diff changeset
2329 iv.step,
kono
parents:
diff changeset
2330 nonconstant_names);
kono
parents:
diff changeset
2331 if (will_be_nonconstant != true)
kono
parents:
diff changeset
2332 will_be_nonconstant = bb_predicate & will_be_nonconstant;
kono
parents:
diff changeset
2333 if (will_be_nonconstant != true
kono
parents:
diff changeset
2334 && will_be_nonconstant != false)
kono
parents:
diff changeset
2335 /* This is slightly inprecise. We may want to represent
kono
parents:
diff changeset
2336 each loop with independent predicate. */
kono
parents:
diff changeset
2337 loop_stride = loop_stride & will_be_nonconstant;
kono
parents:
diff changeset
2338 }
kono
parents:
diff changeset
2339 }
kono
parents:
diff changeset
2340 free (body);
kono
parents:
diff changeset
2341 }
kono
parents:
diff changeset
2342 set_hint_predicate (&ipa_fn_summaries->get (node)->loop_iterations,
kono
parents:
diff changeset
2343 loop_iterations);
kono
parents:
diff changeset
2344 set_hint_predicate (&ipa_fn_summaries->get (node)->loop_stride,
kono
parents:
diff changeset
2345 loop_stride);
kono
parents:
diff changeset
2346 scev_finalize ();
kono
parents:
diff changeset
2347 }
kono
parents:
diff changeset
2348 FOR_ALL_BB_FN (bb, my_function)
kono
parents:
diff changeset
2349 {
kono
parents:
diff changeset
2350 edge e;
kono
parents:
diff changeset
2351 edge_iterator ei;
kono
parents:
diff changeset
2352
kono
parents:
diff changeset
2353 if (bb->aux)
kono
parents:
diff changeset
2354 edge_predicate_pool.remove ((predicate *)bb->aux);
kono
parents:
diff changeset
2355 bb->aux = NULL;
kono
parents:
diff changeset
2356 FOR_EACH_EDGE (e, ei, bb->succs)
kono
parents:
diff changeset
2357 {
kono
parents:
diff changeset
2358 if (e->aux)
kono
parents:
diff changeset
2359 edge_predicate_pool.remove ((predicate *) e->aux);
kono
parents:
diff changeset
2360 e->aux = NULL;
kono
parents:
diff changeset
2361 }
kono
parents:
diff changeset
2362 }
kono
parents:
diff changeset
2363 ipa_fn_summaries->get (node)->time = time;
kono
parents:
diff changeset
2364 ipa_fn_summaries->get (node)->self_size = size;
kono
parents:
diff changeset
2365 nonconstant_names.release ();
kono
parents:
diff changeset
2366 ipa_release_body_info (&fbi);
kono
parents:
diff changeset
2367 if (opt_for_fn (node->decl, optimize))
kono
parents:
diff changeset
2368 {
kono
parents:
diff changeset
2369 if (!early)
kono
parents:
diff changeset
2370 loop_optimizer_finalize ();
kono
parents:
diff changeset
2371 else if (!ipa_edge_args_sum)
kono
parents:
diff changeset
2372 ipa_free_all_node_params ();
kono
parents:
diff changeset
2373 free_dominance_info (CDI_DOMINATORS);
kono
parents:
diff changeset
2374 }
kono
parents:
diff changeset
2375 if (dump_file)
kono
parents:
diff changeset
2376 {
kono
parents:
diff changeset
2377 fprintf (dump_file, "\n");
kono
parents:
diff changeset
2378 ipa_dump_fn_summary (dump_file, node);
kono
parents:
diff changeset
2379 }
kono
parents:
diff changeset
2380 }
kono
parents:
diff changeset
2381
kono
parents:
diff changeset
2382
kono
parents:
diff changeset
2383 /* Compute function summary.
kono
parents:
diff changeset
2384 EARLY is true when we compute parameters during early opts. */
kono
parents:
diff changeset
2385
kono
parents:
diff changeset
2386 void
kono
parents:
diff changeset
2387 compute_fn_summary (struct cgraph_node *node, bool early)
kono
parents:
diff changeset
2388 {
kono
parents:
diff changeset
2389 HOST_WIDE_INT self_stack_size;
kono
parents:
diff changeset
2390 struct cgraph_edge *e;
kono
parents:
diff changeset
2391 struct ipa_fn_summary *info;
kono
parents:
diff changeset
2392
kono
parents:
diff changeset
2393 gcc_assert (!node->global.inlined_to);
kono
parents:
diff changeset
2394
kono
parents:
diff changeset
2395 if (!ipa_fn_summaries)
kono
parents:
diff changeset
2396 ipa_fn_summary_alloc ();
kono
parents:
diff changeset
2397
kono
parents:
diff changeset
2398 info = ipa_fn_summaries->get (node);
kono
parents:
diff changeset
2399 info->reset (node);
kono
parents:
diff changeset
2400
kono
parents:
diff changeset
2401 /* Estimate the stack size for the function if we're optimizing. */
kono
parents:
diff changeset
2402 self_stack_size = optimize && !node->thunk.thunk_p
kono
parents:
diff changeset
2403 ? estimated_stack_frame_size (node) : 0;
kono
parents:
diff changeset
2404 info->estimated_self_stack_size = self_stack_size;
kono
parents:
diff changeset
2405 info->estimated_stack_size = self_stack_size;
kono
parents:
diff changeset
2406 info->stack_frame_offset = 0;
kono
parents:
diff changeset
2407
kono
parents:
diff changeset
2408 if (node->thunk.thunk_p)
kono
parents:
diff changeset
2409 {
kono
parents:
diff changeset
2410 struct ipa_call_summary *es = ipa_call_summaries->get (node->callees);
kono
parents:
diff changeset
2411 predicate t = true;
kono
parents:
diff changeset
2412
kono
parents:
diff changeset
2413 node->local.can_change_signature = false;
kono
parents:
diff changeset
2414 es->call_stmt_size = eni_size_weights.call_cost;
kono
parents:
diff changeset
2415 es->call_stmt_time = eni_time_weights.call_cost;
kono
parents:
diff changeset
2416 info->account_size_time (ipa_fn_summary::size_scale * 2, 2, t, t);
kono
parents:
diff changeset
2417 t = predicate::not_inlined ();
kono
parents:
diff changeset
2418 info->account_size_time (2 * ipa_fn_summary::size_scale, 0, t, t);
kono
parents:
diff changeset
2419 ipa_update_overall_fn_summary (node);
kono
parents:
diff changeset
2420 info->self_size = info->size;
kono
parents:
diff changeset
2421 /* We can not inline instrumentation clones. */
kono
parents:
diff changeset
2422 if (node->thunk.add_pointer_bounds_args)
kono
parents:
diff changeset
2423 {
kono
parents:
diff changeset
2424 info->inlinable = false;
kono
parents:
diff changeset
2425 node->callees->inline_failed = CIF_CHKP;
kono
parents:
diff changeset
2426 }
kono
parents:
diff changeset
2427 else
kono
parents:
diff changeset
2428 info->inlinable = true;
kono
parents:
diff changeset
2429 }
kono
parents:
diff changeset
2430 else
kono
parents:
diff changeset
2431 {
kono
parents:
diff changeset
2432 /* Even is_gimple_min_invariant rely on current_function_decl. */
kono
parents:
diff changeset
2433 push_cfun (DECL_STRUCT_FUNCTION (node->decl));
kono
parents:
diff changeset
2434
kono
parents:
diff changeset
2435 /* Can this function be inlined at all? */
kono
parents:
diff changeset
2436 if (!opt_for_fn (node->decl, optimize)
kono
parents:
diff changeset
2437 && !lookup_attribute ("always_inline",
kono
parents:
diff changeset
2438 DECL_ATTRIBUTES (node->decl)))
kono
parents:
diff changeset
2439 info->inlinable = false;
kono
parents:
diff changeset
2440 else
kono
parents:
diff changeset
2441 info->inlinable = tree_inlinable_function_p (node->decl);
kono
parents:
diff changeset
2442
kono
parents:
diff changeset
2443 info->contains_cilk_spawn = fn_contains_cilk_spawn_p (cfun);
kono
parents:
diff changeset
2444
kono
parents:
diff changeset
2445 /* Type attributes can use parameter indices to describe them. */
kono
parents:
diff changeset
2446 if (TYPE_ATTRIBUTES (TREE_TYPE (node->decl)))
kono
parents:
diff changeset
2447 node->local.can_change_signature = false;
kono
parents:
diff changeset
2448 else
kono
parents:
diff changeset
2449 {
kono
parents:
diff changeset
2450 /* Otherwise, inlinable functions always can change signature. */
kono
parents:
diff changeset
2451 if (info->inlinable)
kono
parents:
diff changeset
2452 node->local.can_change_signature = true;
kono
parents:
diff changeset
2453 else
kono
parents:
diff changeset
2454 {
kono
parents:
diff changeset
2455 /* Functions calling builtin_apply can not change signature. */
kono
parents:
diff changeset
2456 for (e = node->callees; e; e = e->next_callee)
kono
parents:
diff changeset
2457 {
kono
parents:
diff changeset
2458 tree cdecl = e->callee->decl;
kono
parents:
diff changeset
2459 if (DECL_BUILT_IN (cdecl)
kono
parents:
diff changeset
2460 && DECL_BUILT_IN_CLASS (cdecl) == BUILT_IN_NORMAL
kono
parents:
diff changeset
2461 && (DECL_FUNCTION_CODE (cdecl) == BUILT_IN_APPLY_ARGS
kono
parents:
diff changeset
2462 || DECL_FUNCTION_CODE (cdecl) == BUILT_IN_VA_START))
kono
parents:
diff changeset
2463 break;
kono
parents:
diff changeset
2464 }
kono
parents:
diff changeset
2465 node->local.can_change_signature = !e;
kono
parents:
diff changeset
2466 }
kono
parents:
diff changeset
2467 }
kono
parents:
diff changeset
2468 /* Functions called by instrumentation thunk can't change signature
kono
parents:
diff changeset
2469 because instrumentation thunk modification is not supported. */
kono
parents:
diff changeset
2470 if (node->local.can_change_signature)
kono
parents:
diff changeset
2471 for (e = node->callers; e; e = e->next_caller)
kono
parents:
diff changeset
2472 if (e->caller->thunk.thunk_p
kono
parents:
diff changeset
2473 && e->caller->thunk.add_pointer_bounds_args)
kono
parents:
diff changeset
2474 {
kono
parents:
diff changeset
2475 node->local.can_change_signature = false;
kono
parents:
diff changeset
2476 break;
kono
parents:
diff changeset
2477 }
kono
parents:
diff changeset
2478 analyze_function_body (node, early);
kono
parents:
diff changeset
2479 pop_cfun ();
kono
parents:
diff changeset
2480 }
kono
parents:
diff changeset
2481 for (e = node->callees; e; e = e->next_callee)
kono
parents:
diff changeset
2482 if (e->callee->comdat_local_p ())
kono
parents:
diff changeset
2483 break;
kono
parents:
diff changeset
2484 node->calls_comdat_local = (e != NULL);
kono
parents:
diff changeset
2485
kono
parents:
diff changeset
2486 /* Inlining characteristics are maintained by the cgraph_mark_inline. */
kono
parents:
diff changeset
2487 info->size = info->self_size;
kono
parents:
diff changeset
2488 info->stack_frame_offset = 0;
kono
parents:
diff changeset
2489 info->estimated_stack_size = info->estimated_self_stack_size;
kono
parents:
diff changeset
2490
kono
parents:
diff changeset
2491 /* Code above should compute exactly the same result as
kono
parents:
diff changeset
2492 ipa_update_overall_fn_summary but because computation happens in
kono
parents:
diff changeset
2493 different order the roundoff errors result in slight changes. */
kono
parents:
diff changeset
2494 ipa_update_overall_fn_summary (node);
kono
parents:
diff changeset
2495 gcc_assert (info->size == info->self_size);
kono
parents:
diff changeset
2496 }
kono
parents:
diff changeset
2497
kono
parents:
diff changeset
2498
kono
parents:
diff changeset
2499 /* Compute parameters of functions used by inliner using
kono
parents:
diff changeset
2500 current_function_decl. */
kono
parents:
diff changeset
2501
kono
parents:
diff changeset
2502 static unsigned int
kono
parents:
diff changeset
2503 compute_fn_summary_for_current (void)
kono
parents:
diff changeset
2504 {
kono
parents:
diff changeset
2505 compute_fn_summary (cgraph_node::get (current_function_decl), true);
kono
parents:
diff changeset
2506 return 0;
kono
parents:
diff changeset
2507 }
kono
parents:
diff changeset
2508
kono
parents:
diff changeset
2509 /* Estimate benefit devirtualizing indirect edge IE, provided KNOWN_VALS,
kono
parents:
diff changeset
2510 KNOWN_CONTEXTS and KNOWN_AGGS. */
kono
parents:
diff changeset
2511
kono
parents:
diff changeset
2512 static bool
kono
parents:
diff changeset
2513 estimate_edge_devirt_benefit (struct cgraph_edge *ie,
kono
parents:
diff changeset
2514 int *size, int *time,
kono
parents:
diff changeset
2515 vec<tree> known_vals,
kono
parents:
diff changeset
2516 vec<ipa_polymorphic_call_context> known_contexts,
kono
parents:
diff changeset
2517 vec<ipa_agg_jump_function_p> known_aggs)
kono
parents:
diff changeset
2518 {
kono
parents:
diff changeset
2519 tree target;
kono
parents:
diff changeset
2520 struct cgraph_node *callee;
kono
parents:
diff changeset
2521 struct ipa_fn_summary *isummary;
kono
parents:
diff changeset
2522 enum availability avail;
kono
parents:
diff changeset
2523 bool speculative;
kono
parents:
diff changeset
2524
kono
parents:
diff changeset
2525 if (!known_vals.exists () && !known_contexts.exists ())
kono
parents:
diff changeset
2526 return false;
kono
parents:
diff changeset
2527 if (!opt_for_fn (ie->caller->decl, flag_indirect_inlining))
kono
parents:
diff changeset
2528 return false;
kono
parents:
diff changeset
2529
kono
parents:
diff changeset
2530 target = ipa_get_indirect_edge_target (ie, known_vals, known_contexts,
kono
parents:
diff changeset
2531 known_aggs, &speculative);
kono
parents:
diff changeset
2532 if (!target || speculative)
kono
parents:
diff changeset
2533 return false;
kono
parents:
diff changeset
2534
kono
parents:
diff changeset
2535 /* Account for difference in cost between indirect and direct calls. */
kono
parents:
diff changeset
2536 *size -= (eni_size_weights.indirect_call_cost - eni_size_weights.call_cost);
kono
parents:
diff changeset
2537 *time -= (eni_time_weights.indirect_call_cost - eni_time_weights.call_cost);
kono
parents:
diff changeset
2538 gcc_checking_assert (*time >= 0);
kono
parents:
diff changeset
2539 gcc_checking_assert (*size >= 0);
kono
parents:
diff changeset
2540
kono
parents:
diff changeset
2541 callee = cgraph_node::get (target);
kono
parents:
diff changeset
2542 if (!callee || !callee->definition)
kono
parents:
diff changeset
2543 return false;
kono
parents:
diff changeset
2544 callee = callee->function_symbol (&avail);
kono
parents:
diff changeset
2545 if (avail < AVAIL_AVAILABLE)
kono
parents:
diff changeset
2546 return false;
kono
parents:
diff changeset
2547 isummary = ipa_fn_summaries->get (callee);
kono
parents:
diff changeset
2548 return isummary->inlinable;
kono
parents:
diff changeset
2549 }
kono
parents:
diff changeset
2550
kono
parents:
diff changeset
2551 /* Increase SIZE, MIN_SIZE (if non-NULL) and TIME for size and time needed to
kono
parents:
diff changeset
2552 handle edge E with probability PROB.
kono
parents:
diff changeset
2553 Set HINTS if edge may be devirtualized.
kono
parents:
diff changeset
2554 KNOWN_VALS, KNOWN_AGGS and KNOWN_CONTEXTS describe context of the call
kono
parents:
diff changeset
2555 site. */
kono
parents:
diff changeset
2556
kono
parents:
diff changeset
2557 static inline void
kono
parents:
diff changeset
2558 estimate_edge_size_and_time (struct cgraph_edge *e, int *size, int *min_size,
kono
parents:
diff changeset
2559 sreal *time,
kono
parents:
diff changeset
2560 int prob,
kono
parents:
diff changeset
2561 vec<tree> known_vals,
kono
parents:
diff changeset
2562 vec<ipa_polymorphic_call_context> known_contexts,
kono
parents:
diff changeset
2563 vec<ipa_agg_jump_function_p> known_aggs,
kono
parents:
diff changeset
2564 ipa_hints *hints)
kono
parents:
diff changeset
2565 {
kono
parents:
diff changeset
2566 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
2567 int call_size = es->call_stmt_size;
kono
parents:
diff changeset
2568 int call_time = es->call_stmt_time;
kono
parents:
diff changeset
2569 int cur_size;
kono
parents:
diff changeset
2570 if (!e->callee
kono
parents:
diff changeset
2571 && estimate_edge_devirt_benefit (e, &call_size, &call_time,
kono
parents:
diff changeset
2572 known_vals, known_contexts, known_aggs)
kono
parents:
diff changeset
2573 && hints && e->maybe_hot_p ())
kono
parents:
diff changeset
2574 *hints |= INLINE_HINT_indirect_call;
kono
parents:
diff changeset
2575 cur_size = call_size * ipa_fn_summary::size_scale;
kono
parents:
diff changeset
2576 *size += cur_size;
kono
parents:
diff changeset
2577 if (min_size)
kono
parents:
diff changeset
2578 *min_size += cur_size;
kono
parents:
diff changeset
2579 if (prob == REG_BR_PROB_BASE)
kono
parents:
diff changeset
2580 *time += ((sreal)(call_time * e->frequency)) / CGRAPH_FREQ_BASE;
kono
parents:
diff changeset
2581 else
kono
parents:
diff changeset
2582 *time += ((sreal)call_time) * (prob * e->frequency)
kono
parents:
diff changeset
2583 / (CGRAPH_FREQ_BASE * REG_BR_PROB_BASE);
kono
parents:
diff changeset
2584 }
kono
parents:
diff changeset
2585
kono
parents:
diff changeset
2586
kono
parents:
diff changeset
2587
kono
parents:
diff changeset
2588 /* Increase SIZE, MIN_SIZE and TIME for size and time needed to handle all
kono
parents:
diff changeset
2589 calls in NODE. POSSIBLE_TRUTHS, KNOWN_VALS, KNOWN_AGGS and KNOWN_CONTEXTS
kono
parents:
diff changeset
2590 describe context of the call site. */
kono
parents:
diff changeset
2591
kono
parents:
diff changeset
2592 static void
kono
parents:
diff changeset
2593 estimate_calls_size_and_time (struct cgraph_node *node, int *size,
kono
parents:
diff changeset
2594 int *min_size, sreal *time,
kono
parents:
diff changeset
2595 ipa_hints *hints,
kono
parents:
diff changeset
2596 clause_t possible_truths,
kono
parents:
diff changeset
2597 vec<tree> known_vals,
kono
parents:
diff changeset
2598 vec<ipa_polymorphic_call_context> known_contexts,
kono
parents:
diff changeset
2599 vec<ipa_agg_jump_function_p> known_aggs)
kono
parents:
diff changeset
2600 {
kono
parents:
diff changeset
2601 struct cgraph_edge *e;
kono
parents:
diff changeset
2602 for (e = node->callees; e; e = e->next_callee)
kono
parents:
diff changeset
2603 {
kono
parents:
diff changeset
2604 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
2605
kono
parents:
diff changeset
2606 /* Do not care about zero sized builtins. */
kono
parents:
diff changeset
2607 if (e->inline_failed && !es->call_stmt_size)
kono
parents:
diff changeset
2608 {
kono
parents:
diff changeset
2609 gcc_checking_assert (!es->call_stmt_time);
kono
parents:
diff changeset
2610 continue;
kono
parents:
diff changeset
2611 }
kono
parents:
diff changeset
2612 if (!es->predicate
kono
parents:
diff changeset
2613 || es->predicate->evaluate (possible_truths))
kono
parents:
diff changeset
2614 {
kono
parents:
diff changeset
2615 if (e->inline_failed)
kono
parents:
diff changeset
2616 {
kono
parents:
diff changeset
2617 /* Predicates of calls shall not use NOT_CHANGED codes,
kono
parents:
diff changeset
2618 sowe do not need to compute probabilities. */
kono
parents:
diff changeset
2619 estimate_edge_size_and_time (e, size,
kono
parents:
diff changeset
2620 es->predicate ? NULL : min_size,
kono
parents:
diff changeset
2621 time, REG_BR_PROB_BASE,
kono
parents:
diff changeset
2622 known_vals, known_contexts,
kono
parents:
diff changeset
2623 known_aggs, hints);
kono
parents:
diff changeset
2624 }
kono
parents:
diff changeset
2625 else
kono
parents:
diff changeset
2626 estimate_calls_size_and_time (e->callee, size, min_size, time,
kono
parents:
diff changeset
2627 hints,
kono
parents:
diff changeset
2628 possible_truths,
kono
parents:
diff changeset
2629 known_vals, known_contexts,
kono
parents:
diff changeset
2630 known_aggs);
kono
parents:
diff changeset
2631 }
kono
parents:
diff changeset
2632 }
kono
parents:
diff changeset
2633 for (e = node->indirect_calls; e; e = e->next_callee)
kono
parents:
diff changeset
2634 {
kono
parents:
diff changeset
2635 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
2636 if (!es->predicate
kono
parents:
diff changeset
2637 || es->predicate->evaluate (possible_truths))
kono
parents:
diff changeset
2638 estimate_edge_size_and_time (e, size,
kono
parents:
diff changeset
2639 es->predicate ? NULL : min_size,
kono
parents:
diff changeset
2640 time, REG_BR_PROB_BASE,
kono
parents:
diff changeset
2641 known_vals, known_contexts, known_aggs,
kono
parents:
diff changeset
2642 hints);
kono
parents:
diff changeset
2643 }
kono
parents:
diff changeset
2644 }
kono
parents:
diff changeset
2645
kono
parents:
diff changeset
2646
kono
parents:
diff changeset
2647 /* Estimate size and time needed to execute NODE assuming
kono
parents:
diff changeset
2648 POSSIBLE_TRUTHS clause, and KNOWN_VALS, KNOWN_AGGS and KNOWN_CONTEXTS
kono
parents:
diff changeset
2649 information about NODE's arguments. If non-NULL use also probability
kono
parents:
diff changeset
2650 information present in INLINE_PARAM_SUMMARY vector.
kono
parents:
diff changeset
2651 Additionally detemine hints determined by the context. Finally compute
kono
parents:
diff changeset
2652 minimal size needed for the call that is independent on the call context and
kono
parents:
diff changeset
2653 can be used for fast estimates. Return the values in RET_SIZE,
kono
parents:
diff changeset
2654 RET_MIN_SIZE, RET_TIME and RET_HINTS. */
kono
parents:
diff changeset
2655
kono
parents:
diff changeset
2656 void
kono
parents:
diff changeset
2657 estimate_node_size_and_time (struct cgraph_node *node,
kono
parents:
diff changeset
2658 clause_t possible_truths,
kono
parents:
diff changeset
2659 clause_t nonspec_possible_truths,
kono
parents:
diff changeset
2660 vec<tree> known_vals,
kono
parents:
diff changeset
2661 vec<ipa_polymorphic_call_context> known_contexts,
kono
parents:
diff changeset
2662 vec<ipa_agg_jump_function_p> known_aggs,
kono
parents:
diff changeset
2663 int *ret_size, int *ret_min_size,
kono
parents:
diff changeset
2664 sreal *ret_time,
kono
parents:
diff changeset
2665 sreal *ret_nonspecialized_time,
kono
parents:
diff changeset
2666 ipa_hints *ret_hints,
kono
parents:
diff changeset
2667 vec<inline_param_summary>
kono
parents:
diff changeset
2668 inline_param_summary)
kono
parents:
diff changeset
2669 {
kono
parents:
diff changeset
2670 struct ipa_fn_summary *info = ipa_fn_summaries->get (node);
kono
parents:
diff changeset
2671 size_time_entry *e;
kono
parents:
diff changeset
2672 int size = 0;
kono
parents:
diff changeset
2673 sreal time = 0;
kono
parents:
diff changeset
2674 int min_size = 0;
kono
parents:
diff changeset
2675 ipa_hints hints = 0;
kono
parents:
diff changeset
2676 int i;
kono
parents:
diff changeset
2677
kono
parents:
diff changeset
2678 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2679 {
kono
parents:
diff changeset
2680 bool found = false;
kono
parents:
diff changeset
2681 fprintf (dump_file, " Estimating body: %s/%i\n"
kono
parents:
diff changeset
2682 " Known to be false: ", node->name (),
kono
parents:
diff changeset
2683 node->order);
kono
parents:
diff changeset
2684
kono
parents:
diff changeset
2685 for (i = predicate::not_inlined_condition;
kono
parents:
diff changeset
2686 i < (predicate::first_dynamic_condition
kono
parents:
diff changeset
2687 + (int) vec_safe_length (info->conds)); i++)
kono
parents:
diff changeset
2688 if (!(possible_truths & (1 << i)))
kono
parents:
diff changeset
2689 {
kono
parents:
diff changeset
2690 if (found)
kono
parents:
diff changeset
2691 fprintf (dump_file, ", ");
kono
parents:
diff changeset
2692 found = true;
kono
parents:
diff changeset
2693 dump_condition (dump_file, info->conds, i);
kono
parents:
diff changeset
2694 }
kono
parents:
diff changeset
2695 }
kono
parents:
diff changeset
2696
kono
parents:
diff changeset
2697 estimate_calls_size_and_time (node, &size, &min_size, &time, &hints, possible_truths,
kono
parents:
diff changeset
2698 known_vals, known_contexts, known_aggs);
kono
parents:
diff changeset
2699 sreal nonspecialized_time = time;
kono
parents:
diff changeset
2700
kono
parents:
diff changeset
2701 for (i = 0; vec_safe_iterate (info->size_time_table, i, &e); i++)
kono
parents:
diff changeset
2702 {
kono
parents:
diff changeset
2703 bool exec = e->exec_predicate.evaluate (nonspec_possible_truths);
kono
parents:
diff changeset
2704
kono
parents:
diff changeset
2705 /* Because predicates are conservative, it can happen that nonconst is 1
kono
parents:
diff changeset
2706 but exec is 0. */
kono
parents:
diff changeset
2707 if (exec)
kono
parents:
diff changeset
2708 {
kono
parents:
diff changeset
2709 bool nonconst = e->nonconst_predicate.evaluate (possible_truths);
kono
parents:
diff changeset
2710
kono
parents:
diff changeset
2711 gcc_checking_assert (e->time >= 0);
kono
parents:
diff changeset
2712 gcc_checking_assert (time >= 0);
kono
parents:
diff changeset
2713
kono
parents:
diff changeset
2714 /* We compute specialized size only because size of nonspecialized
kono
parents:
diff changeset
2715 copy is context independent.
kono
parents:
diff changeset
2716
kono
parents:
diff changeset
2717 The difference between nonspecialized execution and specialized is
kono
parents:
diff changeset
2718 that nonspecialized is not going to have optimized out computations
kono
parents:
diff changeset
2719 known to be constant in a specialized setting. */
kono
parents:
diff changeset
2720 if (nonconst)
kono
parents:
diff changeset
2721 size += e->size;
kono
parents:
diff changeset
2722 nonspecialized_time += e->time;
kono
parents:
diff changeset
2723 if (!nonconst)
kono
parents:
diff changeset
2724 ;
kono
parents:
diff changeset
2725 else if (!inline_param_summary.exists ())
kono
parents:
diff changeset
2726 {
kono
parents:
diff changeset
2727 if (nonconst)
kono
parents:
diff changeset
2728 time += e->time;
kono
parents:
diff changeset
2729 }
kono
parents:
diff changeset
2730 else
kono
parents:
diff changeset
2731 {
kono
parents:
diff changeset
2732 int prob = e->nonconst_predicate.probability
kono
parents:
diff changeset
2733 (info->conds, possible_truths,
kono
parents:
diff changeset
2734 inline_param_summary);
kono
parents:
diff changeset
2735 gcc_checking_assert (prob >= 0);
kono
parents:
diff changeset
2736 gcc_checking_assert (prob <= REG_BR_PROB_BASE);
kono
parents:
diff changeset
2737 time += e->time * prob / REG_BR_PROB_BASE;
kono
parents:
diff changeset
2738 }
kono
parents:
diff changeset
2739 gcc_checking_assert (time >= 0);
kono
parents:
diff changeset
2740 }
kono
parents:
diff changeset
2741 }
kono
parents:
diff changeset
2742 gcc_checking_assert ((*info->size_time_table)[0].exec_predicate == true);
kono
parents:
diff changeset
2743 gcc_checking_assert ((*info->size_time_table)[0].nonconst_predicate == true);
kono
parents:
diff changeset
2744 min_size = (*info->size_time_table)[0].size;
kono
parents:
diff changeset
2745 gcc_checking_assert (size >= 0);
kono
parents:
diff changeset
2746 gcc_checking_assert (time >= 0);
kono
parents:
diff changeset
2747 /* nonspecialized_time should be always bigger than specialized time.
kono
parents:
diff changeset
2748 Roundoff issues however may get into the way. */
kono
parents:
diff changeset
2749 gcc_checking_assert ((nonspecialized_time - time) >= -1);
kono
parents:
diff changeset
2750
kono
parents:
diff changeset
2751 /* Roundoff issues may make specialized time bigger than nonspecialized
kono
parents:
diff changeset
2752 time. We do not really want that to happen because some heurstics
kono
parents:
diff changeset
2753 may get confused by seeing negative speedups. */
kono
parents:
diff changeset
2754 if (time > nonspecialized_time)
kono
parents:
diff changeset
2755 time = nonspecialized_time;
kono
parents:
diff changeset
2756
kono
parents:
diff changeset
2757 if (info->loop_iterations
kono
parents:
diff changeset
2758 && !info->loop_iterations->evaluate (possible_truths))
kono
parents:
diff changeset
2759 hints |= INLINE_HINT_loop_iterations;
kono
parents:
diff changeset
2760 if (info->loop_stride
kono
parents:
diff changeset
2761 && !info->loop_stride->evaluate (possible_truths))
kono
parents:
diff changeset
2762 hints |= INLINE_HINT_loop_stride;
kono
parents:
diff changeset
2763 if (info->array_index
kono
parents:
diff changeset
2764 && !info->array_index->evaluate (possible_truths))
kono
parents:
diff changeset
2765 hints |= INLINE_HINT_array_index;
kono
parents:
diff changeset
2766 if (info->scc_no)
kono
parents:
diff changeset
2767 hints |= INLINE_HINT_in_scc;
kono
parents:
diff changeset
2768 if (DECL_DECLARED_INLINE_P (node->decl))
kono
parents:
diff changeset
2769 hints |= INLINE_HINT_declared_inline;
kono
parents:
diff changeset
2770
kono
parents:
diff changeset
2771 size = RDIV (size, ipa_fn_summary::size_scale);
kono
parents:
diff changeset
2772 min_size = RDIV (min_size, ipa_fn_summary::size_scale);
kono
parents:
diff changeset
2773
kono
parents:
diff changeset
2774 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
2775 fprintf (dump_file, "\n size:%i time:%f nonspec time:%f\n", (int) size,
kono
parents:
diff changeset
2776 time.to_double (), nonspecialized_time.to_double ());
kono
parents:
diff changeset
2777 if (ret_time)
kono
parents:
diff changeset
2778 *ret_time = time;
kono
parents:
diff changeset
2779 if (ret_nonspecialized_time)
kono
parents:
diff changeset
2780 *ret_nonspecialized_time = nonspecialized_time;
kono
parents:
diff changeset
2781 if (ret_size)
kono
parents:
diff changeset
2782 *ret_size = size;
kono
parents:
diff changeset
2783 if (ret_min_size)
kono
parents:
diff changeset
2784 *ret_min_size = min_size;
kono
parents:
diff changeset
2785 if (ret_hints)
kono
parents:
diff changeset
2786 *ret_hints = hints;
kono
parents:
diff changeset
2787 return;
kono
parents:
diff changeset
2788 }
kono
parents:
diff changeset
2789
kono
parents:
diff changeset
2790
kono
parents:
diff changeset
2791 /* Estimate size and time needed to execute callee of EDGE assuming that
kono
parents:
diff changeset
2792 parameters known to be constant at caller of EDGE are propagated.
kono
parents:
diff changeset
2793 KNOWN_VALS and KNOWN_CONTEXTS are vectors of assumed known constant values
kono
parents:
diff changeset
2794 and types for parameters. */
kono
parents:
diff changeset
2795
kono
parents:
diff changeset
2796 void
kono
parents:
diff changeset
2797 estimate_ipcp_clone_size_and_time (struct cgraph_node *node,
kono
parents:
diff changeset
2798 vec<tree> known_vals,
kono
parents:
diff changeset
2799 vec<ipa_polymorphic_call_context>
kono
parents:
diff changeset
2800 known_contexts,
kono
parents:
diff changeset
2801 vec<ipa_agg_jump_function_p> known_aggs,
kono
parents:
diff changeset
2802 int *ret_size, sreal *ret_time,
kono
parents:
diff changeset
2803 sreal *ret_nonspec_time,
kono
parents:
diff changeset
2804 ipa_hints *hints)
kono
parents:
diff changeset
2805 {
kono
parents:
diff changeset
2806 clause_t clause, nonspec_clause;
kono
parents:
diff changeset
2807
kono
parents:
diff changeset
2808 evaluate_conditions_for_known_args (node, false, known_vals, known_aggs,
kono
parents:
diff changeset
2809 &clause, &nonspec_clause);
kono
parents:
diff changeset
2810 estimate_node_size_and_time (node, clause, nonspec_clause,
kono
parents:
diff changeset
2811 known_vals, known_contexts,
kono
parents:
diff changeset
2812 known_aggs, ret_size, NULL, ret_time,
kono
parents:
diff changeset
2813 ret_nonspec_time, hints, vNULL);
kono
parents:
diff changeset
2814 }
kono
parents:
diff changeset
2815
kono
parents:
diff changeset
2816
kono
parents:
diff changeset
2817 /* Update summary information of inline clones after inlining.
kono
parents:
diff changeset
2818 Compute peak stack usage. */
kono
parents:
diff changeset
2819
kono
parents:
diff changeset
2820 static void
kono
parents:
diff changeset
2821 inline_update_callee_summaries (struct cgraph_node *node, int depth)
kono
parents:
diff changeset
2822 {
kono
parents:
diff changeset
2823 struct cgraph_edge *e;
kono
parents:
diff changeset
2824 struct ipa_fn_summary *callee_info = ipa_fn_summaries->get (node);
kono
parents:
diff changeset
2825 struct ipa_fn_summary *caller_info = ipa_fn_summaries->get (node->callers->caller);
kono
parents:
diff changeset
2826 HOST_WIDE_INT peak;
kono
parents:
diff changeset
2827
kono
parents:
diff changeset
2828 callee_info->stack_frame_offset
kono
parents:
diff changeset
2829 = caller_info->stack_frame_offset
kono
parents:
diff changeset
2830 + caller_info->estimated_self_stack_size;
kono
parents:
diff changeset
2831 peak = callee_info->stack_frame_offset
kono
parents:
diff changeset
2832 + callee_info->estimated_self_stack_size;
kono
parents:
diff changeset
2833 if (ipa_fn_summaries->get (node->global.inlined_to)->estimated_stack_size < peak)
kono
parents:
diff changeset
2834 ipa_fn_summaries->get (node->global.inlined_to)->estimated_stack_size = peak;
kono
parents:
diff changeset
2835 ipa_propagate_frequency (node);
kono
parents:
diff changeset
2836 for (e = node->callees; e; e = e->next_callee)
kono
parents:
diff changeset
2837 {
kono
parents:
diff changeset
2838 if (!e->inline_failed)
kono
parents:
diff changeset
2839 inline_update_callee_summaries (e->callee, depth);
kono
parents:
diff changeset
2840 ipa_call_summaries->get (e)->loop_depth += depth;
kono
parents:
diff changeset
2841 }
kono
parents:
diff changeset
2842 for (e = node->indirect_calls; e; e = e->next_callee)
kono
parents:
diff changeset
2843 ipa_call_summaries->get (e)->loop_depth += depth;
kono
parents:
diff changeset
2844 }
kono
parents:
diff changeset
2845
kono
parents:
diff changeset
2846 /* Update change_prob of EDGE after INLINED_EDGE has been inlined.
kono
parents:
diff changeset
2847 When functoin A is inlined in B and A calls C with parameter that
kono
parents:
diff changeset
2848 changes with probability PROB1 and C is known to be passthroug
kono
parents:
diff changeset
2849 of argument if B that change with probability PROB2, the probability
kono
parents:
diff changeset
2850 of change is now PROB1*PROB2. */
kono
parents:
diff changeset
2851
kono
parents:
diff changeset
2852 static void
kono
parents:
diff changeset
2853 remap_edge_change_prob (struct cgraph_edge *inlined_edge,
kono
parents:
diff changeset
2854 struct cgraph_edge *edge)
kono
parents:
diff changeset
2855 {
kono
parents:
diff changeset
2856 if (ipa_node_params_sum)
kono
parents:
diff changeset
2857 {
kono
parents:
diff changeset
2858 int i;
kono
parents:
diff changeset
2859 struct ipa_edge_args *args = IPA_EDGE_REF (edge);
kono
parents:
diff changeset
2860 struct ipa_call_summary *es = ipa_call_summaries->get (edge);
kono
parents:
diff changeset
2861 struct ipa_call_summary *inlined_es
kono
parents:
diff changeset
2862 = ipa_call_summaries->get (inlined_edge);
kono
parents:
diff changeset
2863
kono
parents:
diff changeset
2864 for (i = 0; i < ipa_get_cs_argument_count (args); i++)
kono
parents:
diff changeset
2865 {
kono
parents:
diff changeset
2866 struct ipa_jump_func *jfunc = ipa_get_ith_jump_func (args, i);
kono
parents:
diff changeset
2867 if (jfunc->type == IPA_JF_PASS_THROUGH
kono
parents:
diff changeset
2868 || jfunc->type == IPA_JF_ANCESTOR)
kono
parents:
diff changeset
2869 {
kono
parents:
diff changeset
2870 int id = jfunc->type == IPA_JF_PASS_THROUGH
kono
parents:
diff changeset
2871 ? ipa_get_jf_pass_through_formal_id (jfunc)
kono
parents:
diff changeset
2872 : ipa_get_jf_ancestor_formal_id (jfunc);
kono
parents:
diff changeset
2873 if (id < (int) inlined_es->param.length ())
kono
parents:
diff changeset
2874 {
kono
parents:
diff changeset
2875 int prob1 = es->param[i].change_prob;
kono
parents:
diff changeset
2876 int prob2 = inlined_es->param[id].change_prob;
kono
parents:
diff changeset
2877 int prob = combine_probabilities (prob1, prob2);
kono
parents:
diff changeset
2878
kono
parents:
diff changeset
2879 if (prob1 && prob2 && !prob)
kono
parents:
diff changeset
2880 prob = 1;
kono
parents:
diff changeset
2881
kono
parents:
diff changeset
2882 es->param[i].change_prob = prob;
kono
parents:
diff changeset
2883 }
kono
parents:
diff changeset
2884 }
kono
parents:
diff changeset
2885 }
kono
parents:
diff changeset
2886 }
kono
parents:
diff changeset
2887 }
kono
parents:
diff changeset
2888
kono
parents:
diff changeset
2889 /* Update edge summaries of NODE after INLINED_EDGE has been inlined.
kono
parents:
diff changeset
2890
kono
parents:
diff changeset
2891 Remap predicates of callees of NODE. Rest of arguments match
kono
parents:
diff changeset
2892 remap_predicate.
kono
parents:
diff changeset
2893
kono
parents:
diff changeset
2894 Also update change probabilities. */
kono
parents:
diff changeset
2895
kono
parents:
diff changeset
2896 static void
kono
parents:
diff changeset
2897 remap_edge_summaries (struct cgraph_edge *inlined_edge,
kono
parents:
diff changeset
2898 struct cgraph_node *node,
kono
parents:
diff changeset
2899 struct ipa_fn_summary *info,
kono
parents:
diff changeset
2900 struct ipa_fn_summary *callee_info,
kono
parents:
diff changeset
2901 vec<int> operand_map,
kono
parents:
diff changeset
2902 vec<int> offset_map,
kono
parents:
diff changeset
2903 clause_t possible_truths,
kono
parents:
diff changeset
2904 predicate *toplev_predicate)
kono
parents:
diff changeset
2905 {
kono
parents:
diff changeset
2906 struct cgraph_edge *e, *next;
kono
parents:
diff changeset
2907 for (e = node->callees; e; e = next)
kono
parents:
diff changeset
2908 {
kono
parents:
diff changeset
2909 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
2910 predicate p;
kono
parents:
diff changeset
2911 next = e->next_callee;
kono
parents:
diff changeset
2912
kono
parents:
diff changeset
2913 if (e->inline_failed)
kono
parents:
diff changeset
2914 {
kono
parents:
diff changeset
2915 remap_edge_change_prob (inlined_edge, e);
kono
parents:
diff changeset
2916
kono
parents:
diff changeset
2917 if (es->predicate)
kono
parents:
diff changeset
2918 {
kono
parents:
diff changeset
2919 p = es->predicate->remap_after_inlining
kono
parents:
diff changeset
2920 (info, callee_info, operand_map,
kono
parents:
diff changeset
2921 offset_map, possible_truths,
kono
parents:
diff changeset
2922 *toplev_predicate);
kono
parents:
diff changeset
2923 edge_set_predicate (e, &p);
kono
parents:
diff changeset
2924 }
kono
parents:
diff changeset
2925 else
kono
parents:
diff changeset
2926 edge_set_predicate (e, toplev_predicate);
kono
parents:
diff changeset
2927 }
kono
parents:
diff changeset
2928 else
kono
parents:
diff changeset
2929 remap_edge_summaries (inlined_edge, e->callee, info, callee_info,
kono
parents:
diff changeset
2930 operand_map, offset_map, possible_truths,
kono
parents:
diff changeset
2931 toplev_predicate);
kono
parents:
diff changeset
2932 }
kono
parents:
diff changeset
2933 for (e = node->indirect_calls; e; e = next)
kono
parents:
diff changeset
2934 {
kono
parents:
diff changeset
2935 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
2936 predicate p;
kono
parents:
diff changeset
2937 next = e->next_callee;
kono
parents:
diff changeset
2938
kono
parents:
diff changeset
2939 remap_edge_change_prob (inlined_edge, e);
kono
parents:
diff changeset
2940 if (es->predicate)
kono
parents:
diff changeset
2941 {
kono
parents:
diff changeset
2942 p = es->predicate->remap_after_inlining
kono
parents:
diff changeset
2943 (info, callee_info, operand_map, offset_map,
kono
parents:
diff changeset
2944 possible_truths, *toplev_predicate);
kono
parents:
diff changeset
2945 edge_set_predicate (e, &p);
kono
parents:
diff changeset
2946 }
kono
parents:
diff changeset
2947 else
kono
parents:
diff changeset
2948 edge_set_predicate (e, toplev_predicate);
kono
parents:
diff changeset
2949 }
kono
parents:
diff changeset
2950 }
kono
parents:
diff changeset
2951
kono
parents:
diff changeset
2952 /* Same as remap_predicate, but set result into hint *HINT. */
kono
parents:
diff changeset
2953
kono
parents:
diff changeset
2954 static void
kono
parents:
diff changeset
2955 remap_hint_predicate (struct ipa_fn_summary *info,
kono
parents:
diff changeset
2956 struct ipa_fn_summary *callee_info,
kono
parents:
diff changeset
2957 predicate **hint,
kono
parents:
diff changeset
2958 vec<int> operand_map,
kono
parents:
diff changeset
2959 vec<int> offset_map,
kono
parents:
diff changeset
2960 clause_t possible_truths,
kono
parents:
diff changeset
2961 predicate *toplev_predicate)
kono
parents:
diff changeset
2962 {
kono
parents:
diff changeset
2963 predicate p;
kono
parents:
diff changeset
2964
kono
parents:
diff changeset
2965 if (!*hint)
kono
parents:
diff changeset
2966 return;
kono
parents:
diff changeset
2967 p = (*hint)->remap_after_inlining
kono
parents:
diff changeset
2968 (info, callee_info,
kono
parents:
diff changeset
2969 operand_map, offset_map,
kono
parents:
diff changeset
2970 possible_truths, *toplev_predicate);
kono
parents:
diff changeset
2971 if (p != false && p != true)
kono
parents:
diff changeset
2972 {
kono
parents:
diff changeset
2973 if (!*hint)
kono
parents:
diff changeset
2974 set_hint_predicate (hint, p);
kono
parents:
diff changeset
2975 else
kono
parents:
diff changeset
2976 **hint &= p;
kono
parents:
diff changeset
2977 }
kono
parents:
diff changeset
2978 }
kono
parents:
diff changeset
2979
kono
parents:
diff changeset
2980 /* We inlined EDGE. Update summary of the function we inlined into. */
kono
parents:
diff changeset
2981
kono
parents:
diff changeset
2982 void
kono
parents:
diff changeset
2983 ipa_merge_fn_summary_after_inlining (struct cgraph_edge *edge)
kono
parents:
diff changeset
2984 {
kono
parents:
diff changeset
2985 struct ipa_fn_summary *callee_info = ipa_fn_summaries->get (edge->callee);
kono
parents:
diff changeset
2986 struct cgraph_node *to = (edge->caller->global.inlined_to
kono
parents:
diff changeset
2987 ? edge->caller->global.inlined_to : edge->caller);
kono
parents:
diff changeset
2988 struct ipa_fn_summary *info = ipa_fn_summaries->get (to);
kono
parents:
diff changeset
2989 clause_t clause = 0; /* not_inline is known to be false. */
kono
parents:
diff changeset
2990 size_time_entry *e;
kono
parents:
diff changeset
2991 vec<int> operand_map = vNULL;
kono
parents:
diff changeset
2992 vec<int> offset_map = vNULL;
kono
parents:
diff changeset
2993 int i;
kono
parents:
diff changeset
2994 predicate toplev_predicate;
kono
parents:
diff changeset
2995 predicate true_p = true;
kono
parents:
diff changeset
2996 struct ipa_call_summary *es = ipa_call_summaries->get (edge);
kono
parents:
diff changeset
2997
kono
parents:
diff changeset
2998 if (es->predicate)
kono
parents:
diff changeset
2999 toplev_predicate = *es->predicate;
kono
parents:
diff changeset
3000 else
kono
parents:
diff changeset
3001 toplev_predicate = true;
kono
parents:
diff changeset
3002
kono
parents:
diff changeset
3003 info->fp_expressions |= callee_info->fp_expressions;
kono
parents:
diff changeset
3004
kono
parents:
diff changeset
3005 if (callee_info->conds)
kono
parents:
diff changeset
3006 evaluate_properties_for_edge (edge, true, &clause, NULL, NULL, NULL, NULL);
kono
parents:
diff changeset
3007 if (ipa_node_params_sum && callee_info->conds)
kono
parents:
diff changeset
3008 {
kono
parents:
diff changeset
3009 struct ipa_edge_args *args = IPA_EDGE_REF (edge);
kono
parents:
diff changeset
3010 int count = ipa_get_cs_argument_count (args);
kono
parents:
diff changeset
3011 int i;
kono
parents:
diff changeset
3012
kono
parents:
diff changeset
3013 if (count)
kono
parents:
diff changeset
3014 {
kono
parents:
diff changeset
3015 operand_map.safe_grow_cleared (count);
kono
parents:
diff changeset
3016 offset_map.safe_grow_cleared (count);
kono
parents:
diff changeset
3017 }
kono
parents:
diff changeset
3018 for (i = 0; i < count; i++)
kono
parents:
diff changeset
3019 {
kono
parents:
diff changeset
3020 struct ipa_jump_func *jfunc = ipa_get_ith_jump_func (args, i);
kono
parents:
diff changeset
3021 int map = -1;
kono
parents:
diff changeset
3022
kono
parents:
diff changeset
3023 /* TODO: handle non-NOPs when merging. */
kono
parents:
diff changeset
3024 if (jfunc->type == IPA_JF_PASS_THROUGH)
kono
parents:
diff changeset
3025 {
kono
parents:
diff changeset
3026 if (ipa_get_jf_pass_through_operation (jfunc) == NOP_EXPR)
kono
parents:
diff changeset
3027 map = ipa_get_jf_pass_through_formal_id (jfunc);
kono
parents:
diff changeset
3028 if (!ipa_get_jf_pass_through_agg_preserved (jfunc))
kono
parents:
diff changeset
3029 offset_map[i] = -1;
kono
parents:
diff changeset
3030 }
kono
parents:
diff changeset
3031 else if (jfunc->type == IPA_JF_ANCESTOR)
kono
parents:
diff changeset
3032 {
kono
parents:
diff changeset
3033 HOST_WIDE_INT offset = ipa_get_jf_ancestor_offset (jfunc);
kono
parents:
diff changeset
3034 if (offset >= 0 && offset < INT_MAX)
kono
parents:
diff changeset
3035 {
kono
parents:
diff changeset
3036 map = ipa_get_jf_ancestor_formal_id (jfunc);
kono
parents:
diff changeset
3037 if (!ipa_get_jf_ancestor_agg_preserved (jfunc))
kono
parents:
diff changeset
3038 offset = -1;
kono
parents:
diff changeset
3039 offset_map[i] = offset;
kono
parents:
diff changeset
3040 }
kono
parents:
diff changeset
3041 }
kono
parents:
diff changeset
3042 operand_map[i] = map;
kono
parents:
diff changeset
3043 gcc_assert (map < ipa_get_param_count (IPA_NODE_REF (to)));
kono
parents:
diff changeset
3044 }
kono
parents:
diff changeset
3045 }
kono
parents:
diff changeset
3046 for (i = 0; vec_safe_iterate (callee_info->size_time_table, i, &e); i++)
kono
parents:
diff changeset
3047 {
kono
parents:
diff changeset
3048 predicate p;
kono
parents:
diff changeset
3049 p = e->exec_predicate.remap_after_inlining
kono
parents:
diff changeset
3050 (info, callee_info, operand_map,
kono
parents:
diff changeset
3051 offset_map, clause,
kono
parents:
diff changeset
3052 toplev_predicate);
kono
parents:
diff changeset
3053 predicate nonconstp;
kono
parents:
diff changeset
3054 nonconstp = e->nonconst_predicate.remap_after_inlining
kono
parents:
diff changeset
3055 (info, callee_info, operand_map,
kono
parents:
diff changeset
3056 offset_map, clause,
kono
parents:
diff changeset
3057 toplev_predicate);
kono
parents:
diff changeset
3058 if (p != false && nonconstp != false)
kono
parents:
diff changeset
3059 {
kono
parents:
diff changeset
3060 sreal add_time = ((sreal)e->time * edge->frequency) / CGRAPH_FREQ_BASE;
kono
parents:
diff changeset
3061 int prob = e->nonconst_predicate.probability (callee_info->conds,
kono
parents:
diff changeset
3062 clause, es->param);
kono
parents:
diff changeset
3063 add_time = add_time * prob / REG_BR_PROB_BASE;
kono
parents:
diff changeset
3064 if (prob != REG_BR_PROB_BASE
kono
parents:
diff changeset
3065 && dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
3066 {
kono
parents:
diff changeset
3067 fprintf (dump_file, "\t\tScaling time by probability:%f\n",
kono
parents:
diff changeset
3068 (double) prob / REG_BR_PROB_BASE);
kono
parents:
diff changeset
3069 }
kono
parents:
diff changeset
3070 info->account_size_time (e->size, add_time, p, nonconstp);
kono
parents:
diff changeset
3071 }
kono
parents:
diff changeset
3072 }
kono
parents:
diff changeset
3073 remap_edge_summaries (edge, edge->callee, info, callee_info, operand_map,
kono
parents:
diff changeset
3074 offset_map, clause, &toplev_predicate);
kono
parents:
diff changeset
3075 remap_hint_predicate (info, callee_info,
kono
parents:
diff changeset
3076 &callee_info->loop_iterations,
kono
parents:
diff changeset
3077 operand_map, offset_map, clause, &toplev_predicate);
kono
parents:
diff changeset
3078 remap_hint_predicate (info, callee_info,
kono
parents:
diff changeset
3079 &callee_info->loop_stride,
kono
parents:
diff changeset
3080 operand_map, offset_map, clause, &toplev_predicate);
kono
parents:
diff changeset
3081 remap_hint_predicate (info, callee_info,
kono
parents:
diff changeset
3082 &callee_info->array_index,
kono
parents:
diff changeset
3083 operand_map, offset_map, clause, &toplev_predicate);
kono
parents:
diff changeset
3084
kono
parents:
diff changeset
3085 inline_update_callee_summaries (edge->callee,
kono
parents:
diff changeset
3086 ipa_call_summaries->get (edge)->loop_depth);
kono
parents:
diff changeset
3087
kono
parents:
diff changeset
3088 /* We do not maintain predicates of inlined edges, free it. */
kono
parents:
diff changeset
3089 edge_set_predicate (edge, &true_p);
kono
parents:
diff changeset
3090 /* Similarly remove param summaries. */
kono
parents:
diff changeset
3091 es->param.release ();
kono
parents:
diff changeset
3092 operand_map.release ();
kono
parents:
diff changeset
3093 offset_map.release ();
kono
parents:
diff changeset
3094 }
kono
parents:
diff changeset
3095
kono
parents:
diff changeset
3096 /* For performance reasons ipa_merge_fn_summary_after_inlining is not updating overall size
kono
parents:
diff changeset
3097 and time. Recompute it. */
kono
parents:
diff changeset
3098
kono
parents:
diff changeset
3099 void
kono
parents:
diff changeset
3100 ipa_update_overall_fn_summary (struct cgraph_node *node)
kono
parents:
diff changeset
3101 {
kono
parents:
diff changeset
3102 struct ipa_fn_summary *info = ipa_fn_summaries->get (node);
kono
parents:
diff changeset
3103 size_time_entry *e;
kono
parents:
diff changeset
3104 int i;
kono
parents:
diff changeset
3105
kono
parents:
diff changeset
3106 info->size = 0;
kono
parents:
diff changeset
3107 info->time = 0;
kono
parents:
diff changeset
3108 for (i = 0; vec_safe_iterate (info->size_time_table, i, &e); i++)
kono
parents:
diff changeset
3109 {
kono
parents:
diff changeset
3110 info->size += e->size;
kono
parents:
diff changeset
3111 info->time += e->time;
kono
parents:
diff changeset
3112 }
kono
parents:
diff changeset
3113 estimate_calls_size_and_time (node, &info->size, &info->min_size,
kono
parents:
diff changeset
3114 &info->time, NULL,
kono
parents:
diff changeset
3115 ~(clause_t) (1 << predicate::false_condition),
kono
parents:
diff changeset
3116 vNULL, vNULL, vNULL);
kono
parents:
diff changeset
3117 info->size = (info->size + ipa_fn_summary::size_scale / 2) / ipa_fn_summary::size_scale;
kono
parents:
diff changeset
3118 }
kono
parents:
diff changeset
3119
kono
parents:
diff changeset
3120
kono
parents:
diff changeset
3121 /* This function performs intraprocedural analysis in NODE that is required to
kono
parents:
diff changeset
3122 inline indirect calls. */
kono
parents:
diff changeset
3123
kono
parents:
diff changeset
3124 static void
kono
parents:
diff changeset
3125 inline_indirect_intraprocedural_analysis (struct cgraph_node *node)
kono
parents:
diff changeset
3126 {
kono
parents:
diff changeset
3127 ipa_analyze_node (node);
kono
parents:
diff changeset
3128 if (dump_file && (dump_flags & TDF_DETAILS))
kono
parents:
diff changeset
3129 {
kono
parents:
diff changeset
3130 ipa_print_node_params (dump_file, node);
kono
parents:
diff changeset
3131 ipa_print_node_jump_functions (dump_file, node);
kono
parents:
diff changeset
3132 }
kono
parents:
diff changeset
3133 }
kono
parents:
diff changeset
3134
kono
parents:
diff changeset
3135
kono
parents:
diff changeset
3136 /* Note function body size. */
kono
parents:
diff changeset
3137
kono
parents:
diff changeset
3138 void
kono
parents:
diff changeset
3139 inline_analyze_function (struct cgraph_node *node)
kono
parents:
diff changeset
3140 {
kono
parents:
diff changeset
3141 push_cfun (DECL_STRUCT_FUNCTION (node->decl));
kono
parents:
diff changeset
3142
kono
parents:
diff changeset
3143 if (dump_file)
kono
parents:
diff changeset
3144 fprintf (dump_file, "\nAnalyzing function: %s/%u\n",
kono
parents:
diff changeset
3145 node->name (), node->order);
kono
parents:
diff changeset
3146 if (opt_for_fn (node->decl, optimize) && !node->thunk.thunk_p)
kono
parents:
diff changeset
3147 inline_indirect_intraprocedural_analysis (node);
kono
parents:
diff changeset
3148 compute_fn_summary (node, false);
kono
parents:
diff changeset
3149 if (!optimize)
kono
parents:
diff changeset
3150 {
kono
parents:
diff changeset
3151 struct cgraph_edge *e;
kono
parents:
diff changeset
3152 for (e = node->callees; e; e = e->next_callee)
kono
parents:
diff changeset
3153 e->inline_failed = CIF_FUNCTION_NOT_OPTIMIZED;
kono
parents:
diff changeset
3154 for (e = node->indirect_calls; e; e = e->next_callee)
kono
parents:
diff changeset
3155 e->inline_failed = CIF_FUNCTION_NOT_OPTIMIZED;
kono
parents:
diff changeset
3156 }
kono
parents:
diff changeset
3157
kono
parents:
diff changeset
3158 pop_cfun ();
kono
parents:
diff changeset
3159 }
kono
parents:
diff changeset
3160
kono
parents:
diff changeset
3161
kono
parents:
diff changeset
3162 /* Called when new function is inserted to callgraph late. */
kono
parents:
diff changeset
3163
kono
parents:
diff changeset
3164 void
kono
parents:
diff changeset
3165 ipa_fn_summary_t::insert (struct cgraph_node *node, ipa_fn_summary *)
kono
parents:
diff changeset
3166 {
kono
parents:
diff changeset
3167 inline_analyze_function (node);
kono
parents:
diff changeset
3168 }
kono
parents:
diff changeset
3169
kono
parents:
diff changeset
3170 /* Note function body size. */
kono
parents:
diff changeset
3171
kono
parents:
diff changeset
3172 static void
kono
parents:
diff changeset
3173 ipa_fn_summary_generate (void)
kono
parents:
diff changeset
3174 {
kono
parents:
diff changeset
3175 struct cgraph_node *node;
kono
parents:
diff changeset
3176
kono
parents:
diff changeset
3177 FOR_EACH_DEFINED_FUNCTION (node)
kono
parents:
diff changeset
3178 if (DECL_STRUCT_FUNCTION (node->decl))
kono
parents:
diff changeset
3179 node->local.versionable = tree_versionable_function_p (node->decl);
kono
parents:
diff changeset
3180
kono
parents:
diff changeset
3181 ipa_fn_summary_alloc ();
kono
parents:
diff changeset
3182
kono
parents:
diff changeset
3183 ipa_fn_summaries->enable_insertion_hook ();
kono
parents:
diff changeset
3184
kono
parents:
diff changeset
3185 ipa_register_cgraph_hooks ();
kono
parents:
diff changeset
3186
kono
parents:
diff changeset
3187 FOR_EACH_DEFINED_FUNCTION (node)
kono
parents:
diff changeset
3188 if (!node->alias
kono
parents:
diff changeset
3189 && (flag_generate_lto || flag_generate_offload|| flag_wpa
kono
parents:
diff changeset
3190 || opt_for_fn (node->decl, optimize)))
kono
parents:
diff changeset
3191 inline_analyze_function (node);
kono
parents:
diff changeset
3192 }
kono
parents:
diff changeset
3193
kono
parents:
diff changeset
3194
kono
parents:
diff changeset
3195 /* Write inline summary for edge E to OB. */
kono
parents:
diff changeset
3196
kono
parents:
diff changeset
3197 static void
kono
parents:
diff changeset
3198 read_ipa_call_summary (struct lto_input_block *ib, struct cgraph_edge *e)
kono
parents:
diff changeset
3199 {
kono
parents:
diff changeset
3200 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
3201 predicate p;
kono
parents:
diff changeset
3202 int length, i;
kono
parents:
diff changeset
3203
kono
parents:
diff changeset
3204 es->call_stmt_size = streamer_read_uhwi (ib);
kono
parents:
diff changeset
3205 es->call_stmt_time = streamer_read_uhwi (ib);
kono
parents:
diff changeset
3206 es->loop_depth = streamer_read_uhwi (ib);
kono
parents:
diff changeset
3207 p.stream_in (ib);
kono
parents:
diff changeset
3208 edge_set_predicate (e, &p);
kono
parents:
diff changeset
3209 length = streamer_read_uhwi (ib);
kono
parents:
diff changeset
3210 if (length)
kono
parents:
diff changeset
3211 {
kono
parents:
diff changeset
3212 es->param.safe_grow_cleared (length);
kono
parents:
diff changeset
3213 for (i = 0; i < length; i++)
kono
parents:
diff changeset
3214 es->param[i].change_prob = streamer_read_uhwi (ib);
kono
parents:
diff changeset
3215 }
kono
parents:
diff changeset
3216 }
kono
parents:
diff changeset
3217
kono
parents:
diff changeset
3218
kono
parents:
diff changeset
3219 /* Stream in inline summaries from the section. */
kono
parents:
diff changeset
3220
kono
parents:
diff changeset
3221 static void
kono
parents:
diff changeset
3222 inline_read_section (struct lto_file_decl_data *file_data, const char *data,
kono
parents:
diff changeset
3223 size_t len)
kono
parents:
diff changeset
3224 {
kono
parents:
diff changeset
3225 const struct lto_function_header *header =
kono
parents:
diff changeset
3226 (const struct lto_function_header *) data;
kono
parents:
diff changeset
3227 const int cfg_offset = sizeof (struct lto_function_header);
kono
parents:
diff changeset
3228 const int main_offset = cfg_offset + header->cfg_size;
kono
parents:
diff changeset
3229 const int string_offset = main_offset + header->main_size;
kono
parents:
diff changeset
3230 struct data_in *data_in;
kono
parents:
diff changeset
3231 unsigned int i, count2, j;
kono
parents:
diff changeset
3232 unsigned int f_count;
kono
parents:
diff changeset
3233
kono
parents:
diff changeset
3234 lto_input_block ib ((const char *) data + main_offset, header->main_size,
kono
parents:
diff changeset
3235 file_data->mode_table);
kono
parents:
diff changeset
3236
kono
parents:
diff changeset
3237 data_in =
kono
parents:
diff changeset
3238 lto_data_in_create (file_data, (const char *) data + string_offset,
kono
parents:
diff changeset
3239 header->string_size, vNULL);
kono
parents:
diff changeset
3240 f_count = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3241 for (i = 0; i < f_count; i++)
kono
parents:
diff changeset
3242 {
kono
parents:
diff changeset
3243 unsigned int index;
kono
parents:
diff changeset
3244 struct cgraph_node *node;
kono
parents:
diff changeset
3245 struct ipa_fn_summary *info;
kono
parents:
diff changeset
3246 lto_symtab_encoder_t encoder;
kono
parents:
diff changeset
3247 struct bitpack_d bp;
kono
parents:
diff changeset
3248 struct cgraph_edge *e;
kono
parents:
diff changeset
3249 predicate p;
kono
parents:
diff changeset
3250
kono
parents:
diff changeset
3251 index = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3252 encoder = file_data->symtab_node_encoder;
kono
parents:
diff changeset
3253 node = dyn_cast<cgraph_node *> (lto_symtab_encoder_deref (encoder,
kono
parents:
diff changeset
3254 index));
kono
parents:
diff changeset
3255 info = ipa_fn_summaries->get (node);
kono
parents:
diff changeset
3256
kono
parents:
diff changeset
3257 info->estimated_stack_size
kono
parents:
diff changeset
3258 = info->estimated_self_stack_size = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3259 info->size = info->self_size = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3260 info->time = sreal::stream_in (&ib);
kono
parents:
diff changeset
3261
kono
parents:
diff changeset
3262 bp = streamer_read_bitpack (&ib);
kono
parents:
diff changeset
3263 info->inlinable = bp_unpack_value (&bp, 1);
kono
parents:
diff changeset
3264 info->contains_cilk_spawn = bp_unpack_value (&bp, 1);
kono
parents:
diff changeset
3265 info->fp_expressions = bp_unpack_value (&bp, 1);
kono
parents:
diff changeset
3266
kono
parents:
diff changeset
3267 count2 = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3268 gcc_assert (!info->conds);
kono
parents:
diff changeset
3269 for (j = 0; j < count2; j++)
kono
parents:
diff changeset
3270 {
kono
parents:
diff changeset
3271 struct condition c;
kono
parents:
diff changeset
3272 c.operand_num = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3273 c.size = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3274 c.code = (enum tree_code) streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3275 c.val = stream_read_tree (&ib, data_in);
kono
parents:
diff changeset
3276 bp = streamer_read_bitpack (&ib);
kono
parents:
diff changeset
3277 c.agg_contents = bp_unpack_value (&bp, 1);
kono
parents:
diff changeset
3278 c.by_ref = bp_unpack_value (&bp, 1);
kono
parents:
diff changeset
3279 if (c.agg_contents)
kono
parents:
diff changeset
3280 c.offset = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3281 vec_safe_push (info->conds, c);
kono
parents:
diff changeset
3282 }
kono
parents:
diff changeset
3283 count2 = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3284 gcc_assert (!info->size_time_table);
kono
parents:
diff changeset
3285 for (j = 0; j < count2; j++)
kono
parents:
diff changeset
3286 {
kono
parents:
diff changeset
3287 struct size_time_entry e;
kono
parents:
diff changeset
3288
kono
parents:
diff changeset
3289 e.size = streamer_read_uhwi (&ib);
kono
parents:
diff changeset
3290 e.time = sreal::stream_in (&ib);
kono
parents:
diff changeset
3291 e.exec_predicate.stream_in (&ib);
kono
parents:
diff changeset
3292 e.nonconst_predicate.stream_in (&ib);
kono
parents:
diff changeset
3293
kono
parents:
diff changeset
3294 vec_safe_push (info->size_time_table, e);
kono
parents:
diff changeset
3295 }
kono
parents:
diff changeset
3296
kono
parents:
diff changeset
3297 p.stream_in (&ib);
kono
parents:
diff changeset
3298 set_hint_predicate (&info->loop_iterations, p);
kono
parents:
diff changeset
3299 p.stream_in (&ib);
kono
parents:
diff changeset
3300 set_hint_predicate (&info->loop_stride, p);
kono
parents:
diff changeset
3301 p.stream_in (&ib);
kono
parents:
diff changeset
3302 set_hint_predicate (&info->array_index, p);
kono
parents:
diff changeset
3303 for (e = node->callees; e; e = e->next_callee)
kono
parents:
diff changeset
3304 read_ipa_call_summary (&ib, e);
kono
parents:
diff changeset
3305 for (e = node->indirect_calls; e; e = e->next_callee)
kono
parents:
diff changeset
3306 read_ipa_call_summary (&ib, e);
kono
parents:
diff changeset
3307 }
kono
parents:
diff changeset
3308
kono
parents:
diff changeset
3309 lto_free_section_data (file_data, LTO_section_ipa_fn_summary, NULL, data,
kono
parents:
diff changeset
3310 len);
kono
parents:
diff changeset
3311 lto_data_in_delete (data_in);
kono
parents:
diff changeset
3312 }
kono
parents:
diff changeset
3313
kono
parents:
diff changeset
3314
kono
parents:
diff changeset
3315 /* Read inline summary. Jump functions are shared among ipa-cp
kono
parents:
diff changeset
3316 and inliner, so when ipa-cp is active, we don't need to write them
kono
parents:
diff changeset
3317 twice. */
kono
parents:
diff changeset
3318
kono
parents:
diff changeset
3319 static void
kono
parents:
diff changeset
3320 ipa_fn_summary_read (void)
kono
parents:
diff changeset
3321 {
kono
parents:
diff changeset
3322 struct lto_file_decl_data **file_data_vec = lto_get_file_decl_data ();
kono
parents:
diff changeset
3323 struct lto_file_decl_data *file_data;
kono
parents:
diff changeset
3324 unsigned int j = 0;
kono
parents:
diff changeset
3325
kono
parents:
diff changeset
3326 ipa_fn_summary_alloc ();
kono
parents:
diff changeset
3327
kono
parents:
diff changeset
3328 while ((file_data = file_data_vec[j++]))
kono
parents:
diff changeset
3329 {
kono
parents:
diff changeset
3330 size_t len;
kono
parents:
diff changeset
3331 const char *data = lto_get_section_data (file_data,
kono
parents:
diff changeset
3332 LTO_section_ipa_fn_summary,
kono
parents:
diff changeset
3333 NULL, &len);
kono
parents:
diff changeset
3334 if (data)
kono
parents:
diff changeset
3335 inline_read_section (file_data, data, len);
kono
parents:
diff changeset
3336 else
kono
parents:
diff changeset
3337 /* Fatal error here. We do not want to support compiling ltrans units
kono
parents:
diff changeset
3338 with different version of compiler or different flags than the WPA
kono
parents:
diff changeset
3339 unit, so this should never happen. */
kono
parents:
diff changeset
3340 fatal_error (input_location,
kono
parents:
diff changeset
3341 "ipa inline summary is missing in input file");
kono
parents:
diff changeset
3342 }
kono
parents:
diff changeset
3343 ipa_register_cgraph_hooks ();
kono
parents:
diff changeset
3344 if (!flag_ipa_cp)
kono
parents:
diff changeset
3345 ipa_prop_read_jump_functions ();
kono
parents:
diff changeset
3346
kono
parents:
diff changeset
3347 gcc_assert (ipa_fn_summaries);
kono
parents:
diff changeset
3348 ipa_fn_summaries->enable_insertion_hook ();
kono
parents:
diff changeset
3349 }
kono
parents:
diff changeset
3350
kono
parents:
diff changeset
3351
kono
parents:
diff changeset
3352 /* Write inline summary for edge E to OB. */
kono
parents:
diff changeset
3353
kono
parents:
diff changeset
3354 static void
kono
parents:
diff changeset
3355 write_ipa_call_summary (struct output_block *ob, struct cgraph_edge *e)
kono
parents:
diff changeset
3356 {
kono
parents:
diff changeset
3357 struct ipa_call_summary *es = ipa_call_summaries->get (e);
kono
parents:
diff changeset
3358 int i;
kono
parents:
diff changeset
3359
kono
parents:
diff changeset
3360 streamer_write_uhwi (ob, es->call_stmt_size);
kono
parents:
diff changeset
3361 streamer_write_uhwi (ob, es->call_stmt_time);
kono
parents:
diff changeset
3362 streamer_write_uhwi (ob, es->loop_depth);
kono
parents:
diff changeset
3363 if (es->predicate)
kono
parents:
diff changeset
3364 es->predicate->stream_out (ob);
kono
parents:
diff changeset
3365 else
kono
parents:
diff changeset
3366 streamer_write_uhwi (ob, 0);
kono
parents:
diff changeset
3367 streamer_write_uhwi (ob, es->param.length ());
kono
parents:
diff changeset
3368 for (i = 0; i < (int) es->param.length (); i++)
kono
parents:
diff changeset
3369 streamer_write_uhwi (ob, es->param[i].change_prob);
kono
parents:
diff changeset
3370 }
kono
parents:
diff changeset
3371
kono
parents:
diff changeset
3372
kono
parents:
diff changeset
3373 /* Write inline summary for node in SET.
kono
parents:
diff changeset
3374 Jump functions are shared among ipa-cp and inliner, so when ipa-cp is
kono
parents:
diff changeset
3375 active, we don't need to write them twice. */
kono
parents:
diff changeset
3376
kono
parents:
diff changeset
3377 static void
kono
parents:
diff changeset
3378 ipa_fn_summary_write (void)
kono
parents:
diff changeset
3379 {
kono
parents:
diff changeset
3380 struct output_block *ob = create_output_block (LTO_section_ipa_fn_summary);
kono
parents:
diff changeset
3381 lto_symtab_encoder_t encoder = ob->decl_state->symtab_node_encoder;
kono
parents:
diff changeset
3382 unsigned int count = 0;
kono
parents:
diff changeset
3383 int i;
kono
parents:
diff changeset
3384
kono
parents:
diff changeset
3385 for (i = 0; i < lto_symtab_encoder_size (encoder); i++)
kono
parents:
diff changeset
3386 {
kono
parents:
diff changeset
3387 symtab_node *snode = lto_symtab_encoder_deref (encoder, i);
kono
parents:
diff changeset
3388 cgraph_node *cnode = dyn_cast <cgraph_node *> (snode);
kono
parents:
diff changeset
3389 if (cnode && cnode->definition && !cnode->alias)
kono
parents:
diff changeset
3390 count++;
kono
parents:
diff changeset
3391 }
kono
parents:
diff changeset
3392 streamer_write_uhwi (ob, count);
kono
parents:
diff changeset
3393
kono
parents:
diff changeset
3394 for (i = 0; i < lto_symtab_encoder_size (encoder); i++)
kono
parents:
diff changeset
3395 {
kono
parents:
diff changeset
3396 symtab_node *snode = lto_symtab_encoder_deref (encoder, i);
kono
parents:
diff changeset
3397 cgraph_node *cnode = dyn_cast <cgraph_node *> (snode);
kono
parents:
diff changeset
3398 if (cnode && cnode->definition && !cnode->alias)
kono
parents:
diff changeset
3399 {
kono
parents:
diff changeset
3400 struct ipa_fn_summary *info = ipa_fn_summaries->get (cnode);
kono
parents:
diff changeset
3401 struct bitpack_d bp;
kono
parents:
diff changeset
3402 struct cgraph_edge *edge;
kono
parents:
diff changeset
3403 int i;
kono
parents:
diff changeset
3404 size_time_entry *e;
kono
parents:
diff changeset
3405 struct condition *c;
kono
parents:
diff changeset
3406
kono
parents:
diff changeset
3407 streamer_write_uhwi (ob, lto_symtab_encoder_encode (encoder, cnode));
kono
parents:
diff changeset
3408 streamer_write_hwi (ob, info->estimated_self_stack_size);
kono
parents:
diff changeset
3409 streamer_write_hwi (ob, info->self_size);
kono
parents:
diff changeset
3410 info->time.stream_out (ob);
kono
parents:
diff changeset
3411 bp = bitpack_create (ob->main_stream);
kono
parents:
diff changeset
3412 bp_pack_value (&bp, info->inlinable, 1);
kono
parents:
diff changeset
3413 bp_pack_value (&bp, info->contains_cilk_spawn, 1);
kono
parents:
diff changeset
3414 bp_pack_value (&bp, info->fp_expressions, 1);
kono
parents:
diff changeset
3415 streamer_write_bitpack (&bp);
kono
parents:
diff changeset
3416 streamer_write_uhwi (ob, vec_safe_length (info->conds));
kono
parents:
diff changeset
3417 for (i = 0; vec_safe_iterate (info->conds, i, &c); i++)
kono
parents:
diff changeset
3418 {
kono
parents:
diff changeset
3419 streamer_write_uhwi (ob, c->operand_num);
kono
parents:
diff changeset
3420 streamer_write_uhwi (ob, c->size);
kono
parents:
diff changeset
3421 streamer_write_uhwi (ob, c->code);
kono
parents:
diff changeset
3422 stream_write_tree (ob, c->val, true);
kono
parents:
diff changeset
3423 bp = bitpack_create (ob->main_stream);
kono
parents:
diff changeset
3424 bp_pack_value (&bp, c->agg_contents, 1);
kono
parents:
diff changeset
3425 bp_pack_value (&bp, c->by_ref, 1);
kono
parents:
diff changeset
3426 streamer_write_bitpack (&bp);
kono
parents:
diff changeset
3427 if (c->agg_contents)
kono
parents:
diff changeset
3428 streamer_write_uhwi (ob, c->offset);
kono
parents:
diff changeset
3429 }
kono
parents:
diff changeset
3430 streamer_write_uhwi (ob, vec_safe_length (info->size_time_table));
kono
parents:
diff changeset
3431 for (i = 0; vec_safe_iterate (info->size_time_table, i, &e); i++)
kono
parents:
diff changeset
3432 {
kono
parents:
diff changeset
3433 streamer_write_uhwi (ob, e->size);
kono
parents:
diff changeset
3434 e->time.stream_out (ob);
kono
parents:
diff changeset
3435 e->exec_predicate.stream_out (ob);
kono
parents:
diff changeset
3436 e->nonconst_predicate.stream_out (ob);
kono
parents:
diff changeset
3437 }
kono
parents:
diff changeset
3438 if (info->loop_iterations)
kono
parents:
diff changeset
3439 info->loop_iterations->stream_out (ob);
kono
parents:
diff changeset
3440 else
kono
parents:
diff changeset
3441 streamer_write_uhwi (ob, 0);
kono
parents:
diff changeset
3442 if (info->loop_stride)
kono
parents:
diff changeset
3443 info->loop_stride->stream_out (ob);
kono
parents:
diff changeset
3444 else
kono
parents:
diff changeset
3445 streamer_write_uhwi (ob, 0);
kono
parents:
diff changeset
3446 if (info->array_index)
kono
parents:
diff changeset
3447 info->array_index->stream_out (ob);
kono
parents:
diff changeset
3448 else
kono
parents:
diff changeset
3449 streamer_write_uhwi (ob, 0);
kono
parents:
diff changeset
3450 for (edge = cnode->callees; edge; edge = edge->next_callee)
kono
parents:
diff changeset
3451 write_ipa_call_summary (ob, edge);
kono
parents:
diff changeset
3452 for (edge = cnode->indirect_calls; edge; edge = edge->next_callee)
kono
parents:
diff changeset
3453 write_ipa_call_summary (ob, edge);
kono
parents:
diff changeset
3454 }
kono
parents:
diff changeset
3455 }
kono
parents:
diff changeset
3456 streamer_write_char_stream (ob->main_stream, 0);
kono
parents:
diff changeset
3457 produce_asm (ob, NULL);
kono
parents:
diff changeset
3458 destroy_output_block (ob);
kono
parents:
diff changeset
3459
kono
parents:
diff changeset
3460 if (!flag_ipa_cp)
kono
parents:
diff changeset
3461 ipa_prop_write_jump_functions ();
kono
parents:
diff changeset
3462 }
kono
parents:
diff changeset
3463
kono
parents:
diff changeset
3464
kono
parents:
diff changeset
3465 /* Release inline summary. */
kono
parents:
diff changeset
3466
kono
parents:
diff changeset
3467 void
kono
parents:
diff changeset
3468 ipa_free_fn_summary (void)
kono
parents:
diff changeset
3469 {
kono
parents:
diff changeset
3470 struct cgraph_node *node;
kono
parents:
diff changeset
3471 if (!ipa_call_summaries)
kono
parents:
diff changeset
3472 return;
kono
parents:
diff changeset
3473 FOR_EACH_DEFINED_FUNCTION (node)
kono
parents:
diff changeset
3474 if (!node->alias)
kono
parents:
diff changeset
3475 ipa_fn_summaries->get (node)->reset (node);
kono
parents:
diff changeset
3476 ipa_fn_summaries->release ();
kono
parents:
diff changeset
3477 ipa_fn_summaries = NULL;
kono
parents:
diff changeset
3478 ipa_call_summaries->release ();
kono
parents:
diff changeset
3479 delete ipa_call_summaries;
kono
parents:
diff changeset
3480 ipa_call_summaries = NULL;
kono
parents:
diff changeset
3481 edge_predicate_pool.release ();
kono
parents:
diff changeset
3482 }
kono
parents:
diff changeset
3483
kono
parents:
diff changeset
3484 namespace {
kono
parents:
diff changeset
3485
kono
parents:
diff changeset
3486 const pass_data pass_data_local_fn_summary =
kono
parents:
diff changeset
3487 {
kono
parents:
diff changeset
3488 GIMPLE_PASS, /* type */
kono
parents:
diff changeset
3489 "local-fnsummary", /* name */
kono
parents:
diff changeset
3490 OPTGROUP_INLINE, /* optinfo_flags */
kono
parents:
diff changeset
3491 TV_INLINE_PARAMETERS, /* tv_id */
kono
parents:
diff changeset
3492 0, /* properties_required */
kono
parents:
diff changeset
3493 0, /* properties_provided */
kono
parents:
diff changeset
3494 0, /* properties_destroyed */
kono
parents:
diff changeset
3495 0, /* todo_flags_start */
kono
parents:
diff changeset
3496 0, /* todo_flags_finish */
kono
parents:
diff changeset
3497 };
kono
parents:
diff changeset
3498
kono
parents:
diff changeset
3499 class pass_local_fn_summary : public gimple_opt_pass
kono
parents:
diff changeset
3500 {
kono
parents:
diff changeset
3501 public:
kono
parents:
diff changeset
3502 pass_local_fn_summary (gcc::context *ctxt)
kono
parents:
diff changeset
3503 : gimple_opt_pass (pass_data_local_fn_summary, ctxt)
kono
parents:
diff changeset
3504 {}
kono
parents:
diff changeset
3505
kono
parents:
diff changeset
3506 /* opt_pass methods: */
kono
parents:
diff changeset
3507 opt_pass * clone () { return new pass_local_fn_summary (m_ctxt); }
kono
parents:
diff changeset
3508 virtual unsigned int execute (function *)
kono
parents:
diff changeset
3509 {
kono
parents:
diff changeset
3510 return compute_fn_summary_for_current ();
kono
parents:
diff changeset
3511 }
kono
parents:
diff changeset
3512
kono
parents:
diff changeset
3513 }; // class pass_local_fn_summary
kono
parents:
diff changeset
3514
kono
parents:
diff changeset
3515 } // anon namespace
kono
parents:
diff changeset
3516
kono
parents:
diff changeset
3517 gimple_opt_pass *
kono
parents:
diff changeset
3518 make_pass_local_fn_summary (gcc::context *ctxt)
kono
parents:
diff changeset
3519 {
kono
parents:
diff changeset
3520 return new pass_local_fn_summary (ctxt);
kono
parents:
diff changeset
3521 }
kono
parents:
diff changeset
3522
kono
parents:
diff changeset
3523
kono
parents:
diff changeset
3524 /* Free inline summary. */
kono
parents:
diff changeset
3525
kono
parents:
diff changeset
3526 namespace {
kono
parents:
diff changeset
3527
kono
parents:
diff changeset
3528 const pass_data pass_data_ipa_free_fn_summary =
kono
parents:
diff changeset
3529 {
kono
parents:
diff changeset
3530 SIMPLE_IPA_PASS, /* type */
kono
parents:
diff changeset
3531 "free-fnsummary", /* name */
kono
parents:
diff changeset
3532 OPTGROUP_NONE, /* optinfo_flags */
kono
parents:
diff changeset
3533 TV_IPA_FREE_INLINE_SUMMARY, /* tv_id */
kono
parents:
diff changeset
3534 0, /* properties_required */
kono
parents:
diff changeset
3535 0, /* properties_provided */
kono
parents:
diff changeset
3536 0, /* properties_destroyed */
kono
parents:
diff changeset
3537 0, /* todo_flags_start */
kono
parents:
diff changeset
3538 /* Early optimizations may make function unreachable. We can not
kono
parents:
diff changeset
3539 remove unreachable functions as part of the ealry opts pass because
kono
parents:
diff changeset
3540 TODOs are run before subpasses. Do it here. */
kono
parents:
diff changeset
3541 ( TODO_remove_functions | TODO_dump_symtab ), /* todo_flags_finish */
kono
parents:
diff changeset
3542 };
kono
parents:
diff changeset
3543
kono
parents:
diff changeset
3544 class pass_ipa_free_fn_summary : public simple_ipa_opt_pass
kono
parents:
diff changeset
3545 {
kono
parents:
diff changeset
3546 public:
kono
parents:
diff changeset
3547 pass_ipa_free_fn_summary (gcc::context *ctxt)
kono
parents:
diff changeset
3548 : simple_ipa_opt_pass (pass_data_ipa_free_fn_summary, ctxt)
kono
parents:
diff changeset
3549 {}
kono
parents:
diff changeset
3550
kono
parents:
diff changeset
3551 /* opt_pass methods: */
kono
parents:
diff changeset
3552 virtual unsigned int execute (function *)
kono
parents:
diff changeset
3553 {
kono
parents:
diff changeset
3554 ipa_free_fn_summary ();
kono
parents:
diff changeset
3555 return 0;
kono
parents:
diff changeset
3556 }
kono
parents:
diff changeset
3557
kono
parents:
diff changeset
3558 }; // class pass_ipa_free_fn_summary
kono
parents:
diff changeset
3559
kono
parents:
diff changeset
3560 } // anon namespace
kono
parents:
diff changeset
3561
kono
parents:
diff changeset
3562 simple_ipa_opt_pass *
kono
parents:
diff changeset
3563 make_pass_ipa_free_fn_summary (gcc::context *ctxt)
kono
parents:
diff changeset
3564 {
kono
parents:
diff changeset
3565 return new pass_ipa_free_fn_summary (ctxt);
kono
parents:
diff changeset
3566 }
kono
parents:
diff changeset
3567
kono
parents:
diff changeset
3568 namespace {
kono
parents:
diff changeset
3569
kono
parents:
diff changeset
3570 const pass_data pass_data_ipa_fn_summary =
kono
parents:
diff changeset
3571 {
kono
parents:
diff changeset
3572 IPA_PASS, /* type */
kono
parents:
diff changeset
3573 "fnsummary", /* name */
kono
parents:
diff changeset
3574 OPTGROUP_INLINE, /* optinfo_flags */
kono
parents:
diff changeset
3575 TV_IPA_FNSUMMARY, /* tv_id */
kono
parents:
diff changeset
3576 0, /* properties_required */
kono
parents:
diff changeset
3577 0, /* properties_provided */
kono
parents:
diff changeset
3578 0, /* properties_destroyed */
kono
parents:
diff changeset
3579 0, /* todo_flags_start */
kono
parents:
diff changeset
3580 ( TODO_dump_symtab ), /* todo_flags_finish */
kono
parents:
diff changeset
3581 };
kono
parents:
diff changeset
3582
kono
parents:
diff changeset
3583 class pass_ipa_fn_summary : public ipa_opt_pass_d
kono
parents:
diff changeset
3584 {
kono
parents:
diff changeset
3585 public:
kono
parents:
diff changeset
3586 pass_ipa_fn_summary (gcc::context *ctxt)
kono
parents:
diff changeset
3587 : ipa_opt_pass_d (pass_data_ipa_fn_summary, ctxt,
kono
parents:
diff changeset
3588 ipa_fn_summary_generate, /* generate_summary */
kono
parents:
diff changeset
3589 ipa_fn_summary_write, /* write_summary */
kono
parents:
diff changeset
3590 ipa_fn_summary_read, /* read_summary */
kono
parents:
diff changeset
3591 NULL, /* write_optimization_summary */
kono
parents:
diff changeset
3592 NULL, /* read_optimization_summary */
kono
parents:
diff changeset
3593 NULL, /* stmt_fixup */
kono
parents:
diff changeset
3594 0, /* function_transform_todo_flags_start */
kono
parents:
diff changeset
3595 NULL, /* function_transform */
kono
parents:
diff changeset
3596 NULL) /* variable_transform */
kono
parents:
diff changeset
3597 {}
kono
parents:
diff changeset
3598
kono
parents:
diff changeset
3599 /* opt_pass methods: */
kono
parents:
diff changeset
3600 virtual unsigned int execute (function *) { return 0; }
kono
parents:
diff changeset
3601
kono
parents:
diff changeset
3602 }; // class pass_ipa_fn_summary
kono
parents:
diff changeset
3603
kono
parents:
diff changeset
3604 } // anon namespace
kono
parents:
diff changeset
3605
kono
parents:
diff changeset
3606 ipa_opt_pass_d *
kono
parents:
diff changeset
3607 make_pass_ipa_fn_summary (gcc::context *ctxt)
kono
parents:
diff changeset
3608 {
kono
parents:
diff changeset
3609 return new pass_ipa_fn_summary (ctxt);
kono
parents:
diff changeset
3610 }