annotate gcc/ipa-fnsummary.c @ 136:4627f235cf2a

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