1 /* Profile counter container type.
2 Copyright (C) 2017-2020 Free Software Foundation, Inc.
3 Contributed by Jan Hubicka
5 This file is part of GCC.
7 GCC is free software; you can redistribute it and/or modify it under
8 the terms of the GNU General Public License as published by the Free
9 Software Foundation; either version 3, or (at your option) any later
12 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
13 WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
17 You should have received a copy of the GNU General Public License
18 along with GCC; see the file COPYING3. If not see
19 <http://www.gnu.org/licenses/>. */
23 #include "coretypes.h"
24 #include "profile-count.h"
27 #include "basic-block.h"
31 #include "data-streamer.h"
36 /* Names from profile_quality enum values. */
38 const char *profile_quality_names
[] =
43 "guessed_global0adjusted",
50 /* Get a string describing QUALITY. */
53 profile_quality_as_string (enum profile_quality quality
)
55 return profile_quality_names
[quality
];
58 /* Parse VALUE as profile quality and return true when a valid QUALITY. */
61 parse_profile_quality (const char *value
, profile_quality
*quality
)
63 for (unsigned i
= 0; i
< ARRAY_SIZE (profile_quality_names
); i
++)
64 if (strcmp (profile_quality_names
[i
], value
) == 0)
66 *quality
= (profile_quality
)i
;
73 /* Display names from profile_quality enum values. */
75 const char *profile_quality_display_names
[] =
79 "estimated locally, globally 0",
80 "estimated locally, globally 0 adjusted",
90 profile_count::dump (FILE *f
) const
92 if (!initialized_p ())
93 fprintf (f
, "uninitialized");
95 fprintf (f
, "%" PRId64
" (%s)", m_val
,
96 profile_quality_display_names
[m_quality
]);
99 /* Dump THIS to stderr. */
102 profile_count::debug () const
105 fprintf (stderr
, "\n");
108 /* Return true if THIS differs from OTHER; tolerate small differences. */
111 profile_count::differs_from_p (profile_count other
) const
113 gcc_checking_assert (compatible_p (other
));
114 if (!initialized_p () || !other
.initialized_p ())
116 if ((uint64_t)m_val
- (uint64_t)other
.m_val
< 100
117 || (uint64_t)other
.m_val
- (uint64_t)m_val
< 100)
121 int64_t ratio
= (int64_t)m_val
* 100 / other
.m_val
;
122 return ratio
< 99 || ratio
> 101;
125 /* Stream THIS from IB. */
128 profile_count::stream_in (class lto_input_block
*ib
)
131 ret
.m_val
= streamer_read_gcov_count (ib
);
132 ret
.m_quality
= (profile_quality
) streamer_read_uhwi (ib
);
136 /* Stream THIS to OB. */
139 profile_count::stream_out (struct output_block
*ob
)
141 streamer_write_gcov_count (ob
, m_val
);
142 streamer_write_uhwi (ob
, m_quality
);
145 /* Stream THIS to OB. */
148 profile_count::stream_out (struct lto_output_stream
*ob
)
150 streamer_write_gcov_count_stream (ob
, m_val
);
151 streamer_write_uhwi_stream (ob
, m_quality
);
154 /* Dump THIS to F. */
157 profile_probability::dump (FILE *f
) const
159 if (!initialized_p ())
160 fprintf (f
, "uninitialized");
163 /* Make difference between 0.00 as a roundoff error and actual 0.
166 fprintf (f
, "never");
167 else if (m_val
== max_probability
)
168 fprintf (f
, "always");
170 fprintf (f
, "%3.1f%%", (double)m_val
* 100 / max_probability
);
171 if (m_quality
== ADJUSTED
)
172 fprintf (f
, " (adjusted)");
173 else if (m_quality
== AFDO
)
174 fprintf (f
, " (auto FDO)");
175 else if (m_quality
== GUESSED
)
176 fprintf (f
, " (guessed)");
180 /* Dump THIS to stderr. */
183 profile_probability::debug () const
186 fprintf (stderr
, "\n");
189 /* Return true if THIS differs from OTHER; tolerate small differences. */
192 profile_probability::differs_from_p (profile_probability other
) const
194 if (!initialized_p () || !other
.initialized_p ())
196 if ((uint64_t)m_val
- (uint64_t)other
.m_val
< max_probability
/ 1000
197 || (uint64_t)other
.m_val
- (uint64_t)max_probability
< 1000)
201 int64_t ratio
= (int64_t)m_val
* 100 / other
.m_val
;
202 return ratio
< 99 || ratio
> 101;
205 /* Return true if THIS differs significantly from OTHER. */
208 profile_probability::differs_lot_from_p (profile_probability other
) const
210 if (!initialized_p () || !other
.initialized_p ())
212 uint32_t d
= m_val
> other
.m_val
? m_val
- other
.m_val
: other
.m_val
- m_val
;
213 return d
> max_probability
/ 2;
216 /* Stream THIS from IB. */
219 profile_probability::stream_in (class lto_input_block
*ib
)
221 profile_probability ret
;
222 ret
.m_val
= streamer_read_uhwi (ib
);
223 ret
.m_quality
= (profile_quality
) streamer_read_uhwi (ib
);
227 /* Stream THIS to OB. */
230 profile_probability::stream_out (struct output_block
*ob
)
232 streamer_write_uhwi (ob
, m_val
);
233 streamer_write_uhwi (ob
, m_quality
);
236 /* Stream THIS to OB. */
239 profile_probability::stream_out (struct lto_output_stream
*ob
)
241 streamer_write_uhwi_stream (ob
, m_val
);
242 streamer_write_uhwi_stream (ob
, m_quality
);
245 /* Compute RES=(a*b + c/2)/c capping and return false if overflow happened. */
248 slow_safe_scale_64bit (uint64_t a
, uint64_t b
, uint64_t c
, uint64_t *res
)
250 FIXED_WIDE_INT (128) tmp
= a
;
251 wi::overflow_type overflow
;
252 tmp
= wi::udiv_floor (wi::umul (tmp
, b
, &overflow
) + (c
/ 2), c
);
253 gcc_checking_assert (!overflow
);
254 if (wi::fits_uhwi_p (tmp
))
256 *res
= tmp
.to_uhwi ();
259 *res
= (uint64_t) -1;
263 /* Return count as frequency within FUN scaled in range 0 to REG_FREQ_MAX
264 Used for legacy code and should not be used anymore. */
267 profile_count::to_frequency (struct function
*fun
) const
269 if (!initialized_p ())
271 if (*this == zero ())
273 gcc_assert (REG_BR_PROB_BASE
== BB_FREQ_MAX
274 && fun
->cfg
->count_max
.initialized_p ());
275 profile_probability prob
= probability_in (fun
->cfg
->count_max
);
276 if (!prob
.initialized_p ())
277 return REG_BR_PROB_BASE
;
278 return prob
.to_reg_br_prob_base ();
281 /* Return count as frequency within FUN scaled in range 0 to CGRAPH_FREQ_MAX
282 where CGRAPH_FREQ_BASE means that count equals to entry block count.
283 Used for legacy code and should not be used anymore. */
286 profile_count::to_cgraph_frequency (profile_count entry_bb_count
) const
288 if (!initialized_p () || !entry_bb_count
.initialized_p ())
289 return CGRAPH_FREQ_BASE
;
290 if (*this == zero ())
292 gcc_checking_assert (entry_bb_count
.initialized_p ());
294 gcc_checking_assert (compatible_p (entry_bb_count
));
295 if (!safe_scale_64bit (!entry_bb_count
.m_val
? m_val
+ 1 : m_val
,
296 CGRAPH_FREQ_BASE
, MAX (1, entry_bb_count
.m_val
), &scale
))
297 return CGRAPH_FREQ_MAX
;
298 return MIN (scale
, CGRAPH_FREQ_MAX
);
301 /* Return THIS/IN as sreal value. */
304 profile_count::to_sreal_scale (profile_count in
, bool *known
) const
306 if (!initialized_p () || !in
.initialized_p ())
314 /* Watch for cases where one count is IPA and other is not. */
315 if (in
.ipa ().initialized_p ())
317 gcc_checking_assert (ipa ().initialized_p ());
318 /* If current count is inter-procedurally 0 and IN is inter-procedurally
319 non-zero, return 0. */
320 if (in
.ipa ().nonzero_p ()
321 && !ipa().nonzero_p ())
325 /* We can handle correctly 0 IPA count within locally estimated
326 profile, but otherwise we are lost and this should not happen. */
327 gcc_checking_assert (!ipa ().initialized_p () || !ipa ().nonzero_p ());
328 if (*this == zero ())
330 if (m_val
== in
.m_val
)
332 gcc_checking_assert (compatible_p (in
));
340 return (sreal
)m_val
/ (sreal
)in
.m_val
;
343 /* We want to scale profile across function boundary from NUM to DEN.
344 Take care of the side case when DEN is zeros. We still want to behave
345 sanely here which means
346 - scale to profile_count::zero () if NUM is profile_count::zero
347 - do not affect anything if NUM == DEN
348 - preserve counter value but adjust quality in other cases. */
351 profile_count::adjust_for_ipa_scaling (profile_count
*num
,
354 /* Scaling is no-op if NUM and DEN are the same. */
357 /* Scaling to zero is always zero. */
360 /* If den is non-zero we are safe. */
361 if (den
->force_nonzero () == *den
)
363 /* Force both to non-zero so we do not push profiles to 0 when
364 both num == 0 and den == 0. */
365 *den
= den
->force_nonzero ();
366 *num
= num
->force_nonzero ();
369 /* THIS is a count of bb which is known to be executed IPA times.
370 Combine this information into bb counter. This means returning IPA
371 if it is nonzero, not changing anything if IPA is uninitialized
372 and if IPA is zero, turning THIS into corresponding local profile with
376 profile_count::combine_with_ipa_count (profile_count ipa
)
378 if (!initialized_p ())
381 if (ipa
.nonzero_p ())
383 if (!ipa
.initialized_p () || *this == zero ())
386 return this->global0 ();
387 return this->global0adjusted ();
390 /* Sae as profile_count::combine_with_ipa_count but within function with count
393 profile_count::combine_with_ipa_count_within (profile_count ipa
,
397 if (!initialized_p ())
399 if (ipa2
.ipa () == ipa2
&& ipa
.initialized_p ())
402 ret
= combine_with_ipa_count (ipa
);
403 gcc_checking_assert (ret
.compatible_p (ipa2
));
407 /* The profiling runtime uses gcov_type, which is usually 64bit integer.
408 Conversions back and forth are used to read the coverage and get it
409 into internal representation. */
412 profile_count::from_gcov_type (gcov_type v
, profile_quality quality
)
415 gcc_checking_assert (v
>= 0);
416 if (dump_file
&& v
>= (gcov_type
)max_count
)
418 "Capping gcov count %" PRId64
" to max_count %" PRId64
"\n",
419 (int64_t) v
, (int64_t) max_count
);
420 ret
.m_val
= MIN (v
, (gcov_type
)max_count
);
421 ret
.m_quality
= quality
;
425 /* COUNT1 times event happens with *THIS probability, COUNT2 times OTHER
426 happens with COUNT2 probability. Return probability that either *THIS or
430 profile_probability::combine_with_count (profile_count count1
,
431 profile_probability other
,
432 profile_count count2
) const
434 /* If probabilities are same, we are done.
435 If counts are nonzero we can distribute accordingly. In remaining
436 cases just average the values and hope for the best. */
437 if (*this == other
|| count1
== count2
438 || (count2
== profile_count::zero ()
439 && !(count1
== profile_count::zero ())))
441 if (count1
== profile_count::zero () && !(count2
== profile_count::zero ()))
443 else if (count1
.nonzero_p () || count2
.nonzero_p ())
444 return *this * count1
.probability_in (count1
+ count2
)
445 + other
* count2
.probability_in (count1
+ count2
);
447 return *this * even () + other
* even ();