hist.c 18.5 KB
Newer Older
1
#include "annotate.h"
2
#include "util.h"
3
#include "build-id.h"
4
#include "hist.h"
5 6
#include "session.h"
#include "sort.h"
7
#include <math.h>
8

9 10 11 12
static bool hists__filter_entry_by_dso(struct hists *hists,
				       struct hist_entry *he);
static bool hists__filter_entry_by_thread(struct hists *hists,
					  struct hist_entry *he);
13 14
static bool hists__filter_entry_by_symbol(struct hists *hists,
					  struct hist_entry *he);
15

16 17 18 19
enum hist_filter {
	HIST_FILTER__DSO,
	HIST_FILTER__THREAD,
	HIST_FILTER__PARENT,
20
	HIST_FILTER__SYMBOL,
21 22
};

23 24
struct callchain_param	callchain_param = {
	.mode	= CHAIN_GRAPH_REL,
25 26
	.min_percent = 0.5,
	.order  = ORDER_CALLEE
27 28
};

29
u16 hists__col_len(struct hists *hists, enum hist_column col)
30
{
31
	return hists->col_len[col];
32 33
}

34
void hists__set_col_len(struct hists *hists, enum hist_column col, u16 len)
35
{
36
	hists->col_len[col] = len;
37 38
}

39
bool hists__new_col_len(struct hists *hists, enum hist_column col, u16 len)
40
{
41 42
	if (len > hists__col_len(hists, col)) {
		hists__set_col_len(hists, col, len);
43 44 45 46 47
		return true;
	}
	return false;
}

48
void hists__reset_col_len(struct hists *hists)
49 50 51 52
{
	enum hist_column col;

	for (col = 0; col < HISTC_NR_COLS; ++col)
53
		hists__set_col_len(hists, col, 0);
54 55
}

56 57 58 59 60 61 62 63 64 65
static void hists__set_unres_dso_col_len(struct hists *hists, int dso)
{
	const unsigned int unresolved_col_width = BITS_PER_LONG / 4;

	if (hists__col_len(hists, dso) < unresolved_col_width &&
	    !symbol_conf.col_width_list_str && !symbol_conf.field_sep &&
	    !symbol_conf.dso_list)
		hists__set_col_len(hists, dso, unresolved_col_width);
}

66
void hists__calc_col_len(struct hists *hists, struct hist_entry *h)
67
{
68
	const unsigned int unresolved_col_width = BITS_PER_LONG / 4;
69 70 71
	u16 len;

	if (h->ms.sym)
72 73 74
		hists__new_col_len(hists, HISTC_SYMBOL, h->ms.sym->namelen + 4);
	else
		hists__set_unres_dso_col_len(hists, HISTC_DSO);
75 76

	len = thread__comm_len(h->thread);
77 78
	if (hists__new_col_len(hists, HISTC_COMM, len))
		hists__set_col_len(hists, HISTC_THREAD, len + 6);
79 80 81

	if (h->ms.map) {
		len = dso__name_len(h->ms.map->dso);
82
		hists__new_col_len(hists, HISTC_DSO, len);
83
	}
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114

	if (h->branch_info) {
		int symlen;
		/*
		 * +4 accounts for '[x] ' priv level info
		 * +2 account of 0x prefix on raw addresses
		 */
		if (h->branch_info->from.sym) {
			symlen = (int)h->branch_info->from.sym->namelen + 4;
			hists__new_col_len(hists, HISTC_SYMBOL_FROM, symlen);

			symlen = dso__name_len(h->branch_info->from.map->dso);
			hists__new_col_len(hists, HISTC_DSO_FROM, symlen);
		} else {
			symlen = unresolved_col_width + 4 + 2;
			hists__new_col_len(hists, HISTC_SYMBOL_FROM, symlen);
			hists__set_unres_dso_col_len(hists, HISTC_DSO_FROM);
		}

		if (h->branch_info->to.sym) {
			symlen = (int)h->branch_info->to.sym->namelen + 4;
			hists__new_col_len(hists, HISTC_SYMBOL_TO, symlen);

			symlen = dso__name_len(h->branch_info->to.map->dso);
			hists__new_col_len(hists, HISTC_DSO_TO, symlen);
		} else {
			symlen = unresolved_col_width + 4 + 2;
			hists__new_col_len(hists, HISTC_SYMBOL_TO, symlen);
			hists__set_unres_dso_col_len(hists, HISTC_DSO_TO);
		}
	}
115 116
}

117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132
void hists__output_recalc_col_len(struct hists *hists, int max_rows)
{
	struct rb_node *next = rb_first(&hists->entries);
	struct hist_entry *n;
	int row = 0;

	hists__reset_col_len(hists);

	while (next && row++ < max_rows) {
		n = rb_entry(next, struct hist_entry, rb_node);
		if (!n->filtered)
			hists__calc_col_len(hists, n);
		next = rb_next(&n->rb_node);
	}
}

133
static void hist_entry__add_cpumode_period(struct hist_entry *he,
134
					   unsigned int cpumode, u64 period)
135
{
136
	switch (cpumode) {
137
	case PERF_RECORD_MISC_KERNEL:
138
		he->stat.period_sys += period;
139 140
		break;
	case PERF_RECORD_MISC_USER:
141
		he->stat.period_us += period;
142 143
		break;
	case PERF_RECORD_MISC_GUEST_KERNEL:
144
		he->stat.period_guest_sys += period;
145 146
		break;
	case PERF_RECORD_MISC_GUEST_USER:
147
		he->stat.period_guest_us += period;
148 149 150 151 152 153
		break;
	default:
		break;
	}
}

154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169
static void he_stat__add_period(struct he_stat *he_stat, u64 period)
{
	he_stat->period		+= period;
	he_stat->nr_events	+= 1;
}

static void he_stat__add_stat(struct he_stat *dest, struct he_stat *src)
{
	dest->period		+= src->period;
	dest->period_sys	+= src->period_sys;
	dest->period_us		+= src->period_us;
	dest->period_guest_sys	+= src->period_guest_sys;
	dest->period_guest_us	+= src->period_guest_us;
	dest->nr_events		+= src->nr_events;
}

170 171
static void hist_entry__decay(struct hist_entry *he)
{
172 173
	he->stat.period = (he->stat.period * 7) / 8;
	he->stat.nr_events = (he->stat.nr_events * 7) / 8;
174 175 176 177
}

static bool hists__decay_entry(struct hists *hists, struct hist_entry *he)
{
178
	u64 prev_period = he->stat.period;
179 180

	if (prev_period == 0)
181
		return true;
182

183
	hist_entry__decay(he);
184 185

	if (!he->filtered)
186
		hists->stats.total_period -= prev_period - he->stat.period;
187

188
	return he->stat.period == 0;
189 190
}

191 192
static void __hists__decay_entries(struct hists *hists, bool zap_user,
				   bool zap_kernel, bool threaded)
193 194 195 196 197 198 199
{
	struct rb_node *next = rb_first(&hists->entries);
	struct hist_entry *n;

	while (next) {
		n = rb_entry(next, struct hist_entry, rb_node);
		next = rb_next(&n->rb_node);
200 201 202 203 204
		/*
		 * We may be annotating this, for instance, so keep it here in
		 * case some it gets new samples, we'll eventually free it when
		 * the user stops browsing and it agains gets fully decayed.
		 */
205 206 207 208
		if (((zap_user && n->level == '.') ||
		     (zap_kernel && n->level != '.') ||
		     hists__decay_entry(hists, n)) &&
		    !n->used) {
209 210
			rb_erase(&n->rb_node, &hists->entries);

211
			if (sort__need_collapse || threaded)
212 213 214 215 216 217 218 219
				rb_erase(&n->rb_node_in, &hists->entries_collapsed);

			hist_entry__free(n);
			--hists->nr_entries;
		}
	}
}

220
void hists__decay_entries(struct hists *hists, bool zap_user, bool zap_kernel)
221
{
222
	return __hists__decay_entries(hists, zap_user, zap_kernel, false);
223 224
}

225 226
void hists__decay_entries_threaded(struct hists *hists,
				   bool zap_user, bool zap_kernel)
227
{
228
	return __hists__decay_entries(hists, zap_user, zap_kernel, true);
229 230
}

231
/*
232
 * histogram, sorted on item, collects periods
233 234
 */

235 236
static struct hist_entry *hist_entry__new(struct hist_entry *template)
{
237
	size_t callchain_size = symbol_conf.use_callchain ? sizeof(struct callchain_root) : 0;
238
	struct hist_entry *he = malloc(sizeof(*he) + callchain_size);
239

240 241
	if (he != NULL) {
		*he = *template;
242

243 244
		if (he->ms.map)
			he->ms.map->referenced = true;
245
		if (symbol_conf.use_callchain)
246
			callchain_init(he->callchain);
247 248

		INIT_LIST_HEAD(&he->pairs.node);
249 250
	}

251
	return he;
252 253
}

254
static void hists__inc_nr_entries(struct hists *hists, struct hist_entry *h)
255
{
256
	if (!h->filtered) {
257 258
		hists__calc_col_len(hists, h);
		++hists->nr_entries;
259
		hists->stats.total_period += h->stat.period;
260
	}
261 262
}

263 264 265 266 267 268 269
static u8 symbol__parent_filter(const struct symbol *parent)
{
	if (symbol_conf.exclude_other && parent == NULL)
		return 1 << HIST_FILTER__PARENT;
	return 0;
}

270 271
static struct hist_entry *add_hist_entry(struct hists *hists,
				      struct hist_entry *entry,
272
				      struct addr_location *al,
273
				      u64 period)
274
{
275
	struct rb_node **p;
276 277 278 279
	struct rb_node *parent = NULL;
	struct hist_entry *he;
	int cmp;

280 281 282 283
	pthread_mutex_lock(&hists->lock);

	p = &hists->entries_in->rb_node;

284 285
	while (*p != NULL) {
		parent = *p;
286
		he = rb_entry(parent, struct hist_entry, rb_node_in);
287

288
		cmp = hist_entry__cmp(entry, he);
289 290

		if (!cmp) {
291
			he_stat__add_period(&he->stat, period);
292 293 294 295 296 297 298 299 300 301 302 303

			/* If the map of an existing hist_entry has
			 * become out-of-date due to an exec() or
			 * similar, update it.  Otherwise we will
			 * mis-adjust symbol addresses when computing
			 * the history counter to increment.
			 */
			if (he->ms.map != entry->ms.map) {
				he->ms.map = entry->ms.map;
				if (he->ms.map)
					he->ms.map->referenced = true;
			}
304
			goto out;
305 306 307 308 309 310 311 312
		}

		if (cmp < 0)
			p = &(*p)->rb_left;
		else
			p = &(*p)->rb_right;
	}

313
	he = hist_entry__new(entry);
314
	if (!he)
315 316 317 318
		goto out_unlock;

	rb_link_node(&he->rb_node_in, parent, p);
	rb_insert_color(&he->rb_node_in, hists->entries_in);
319
out:
320
	hist_entry__add_cpumode_period(he, al->cpumode, period);
321 322
out_unlock:
	pthread_mutex_unlock(&hists->lock);
323 324 325
	return he;
}

326 327 328 329 330 331 332 333 334 335 336 337 338 339 340
struct hist_entry *__hists__add_branch_entry(struct hists *self,
					     struct addr_location *al,
					     struct symbol *sym_parent,
					     struct branch_info *bi,
					     u64 period)
{
	struct hist_entry entry = {
		.thread	= al->thread,
		.ms = {
			.map	= bi->to.map,
			.sym	= bi->to.sym,
		},
		.cpu	= al->cpu,
		.ip	= bi->to.addr,
		.level	= al->level,
341 342
		.stat = {
			.period	= period,
343
			.nr_events = 1,
344
		},
345 346 347
		.parent = sym_parent,
		.filtered = symbol__parent_filter(sym_parent),
		.branch_info = bi,
348
		.hists	= self,
349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366
	};

	return add_hist_entry(self, &entry, al, period);
}

struct hist_entry *__hists__add_entry(struct hists *self,
				      struct addr_location *al,
				      struct symbol *sym_parent, u64 period)
{
	struct hist_entry entry = {
		.thread	= al->thread,
		.ms = {
			.map	= al->map,
			.sym	= al->sym,
		},
		.cpu	= al->cpu,
		.ip	= al->addr,
		.level	= al->level,
367 368
		.stat = {
			.period	= period,
369
			.nr_events = 1,
370
		},
371 372
		.parent = sym_parent,
		.filtered = symbol__parent_filter(sym_parent),
373
		.hists	= self,
374 375 376 377 378
	};

	return add_hist_entry(self, &entry, al, period);
}

379 380 381 382 383 384 385
int64_t
hist_entry__cmp(struct hist_entry *left, struct hist_entry *right)
{
	struct sort_entry *se;
	int64_t cmp = 0;

	list_for_each_entry(se, &hist_entry__sort_list, list) {
386
		cmp = se->se_cmp(left, right);
387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402
		if (cmp)
			break;
	}

	return cmp;
}

int64_t
hist_entry__collapse(struct hist_entry *left, struct hist_entry *right)
{
	struct sort_entry *se;
	int64_t cmp = 0;

	list_for_each_entry(se, &hist_entry__sort_list, list) {
		int64_t (*f)(struct hist_entry *, struct hist_entry *);

403
		f = se->se_collapse ?: se->se_cmp;
404 405 406 407 408 409 410 411 412 413 414

		cmp = f(left, right);
		if (cmp)
			break;
	}

	return cmp;
}

void hist_entry__free(struct hist_entry *he)
{
415
	free(he->branch_info);
416 417 418 419 420 421 422
	free(he);
}

/*
 * collapse the histogram
 */

423
static bool hists__collapse_insert_entry(struct hists *hists __maybe_unused,
424 425
					 struct rb_root *root,
					 struct hist_entry *he)
426
{
427
	struct rb_node **p = &root->rb_node;
428 429 430 431 432 433
	struct rb_node *parent = NULL;
	struct hist_entry *iter;
	int64_t cmp;

	while (*p != NULL) {
		parent = *p;
434
		iter = rb_entry(parent, struct hist_entry, rb_node_in);
435 436 437 438

		cmp = hist_entry__collapse(iter, he);

		if (!cmp) {
439
			he_stat__add_stat(&iter->stat, &he->stat);
440

441
			if (symbol_conf.use_callchain) {
442 443 444
				callchain_cursor_reset(&callchain_cursor);
				callchain_merge(&callchain_cursor,
						iter->callchain,
445 446
						he->callchain);
			}
447
			hist_entry__free(he);
448
			return false;
449 450 451 452 453 454 455 456
		}

		if (cmp < 0)
			p = &(*p)->rb_left;
		else
			p = &(*p)->rb_right;
	}

457 458
	rb_link_node(&he->rb_node_in, parent, p);
	rb_insert_color(&he->rb_node_in, root);
459
	return true;
460 461
}

462
static struct rb_root *hists__get_rotate_entries_in(struct hists *hists)
463
{
464 465 466 467 468 469 470 471 472 473 474 475 476
	struct rb_root *root;

	pthread_mutex_lock(&hists->lock);

	root = hists->entries_in;
	if (++hists->entries_in > &hists->entries_in_array[1])
		hists->entries_in = &hists->entries_in_array[0];

	pthread_mutex_unlock(&hists->lock);

	return root;
}

477 478 479 480
static void hists__apply_filters(struct hists *hists, struct hist_entry *he)
{
	hists__filter_entry_by_dso(hists, he);
	hists__filter_entry_by_thread(hists, he);
481
	hists__filter_entry_by_symbol(hists, he);
482 483
}

484 485 486
static void __hists__collapse_resort(struct hists *hists, bool threaded)
{
	struct rb_root *root;
487 488 489
	struct rb_node *next;
	struct hist_entry *n;

490
	if (!sort__need_collapse && !threaded)
491 492
		return;

493 494
	root = hists__get_rotate_entries_in(hists);
	next = rb_first(root);
495

496
	while (next) {
497 498
		n = rb_entry(next, struct hist_entry, rb_node_in);
		next = rb_next(&n->rb_node_in);
499

500
		rb_erase(&n->rb_node_in, root);
501 502 503 504 505 506 507 508
		if (hists__collapse_insert_entry(hists, &hists->entries_collapsed, n)) {
			/*
			 * If it wasn't combined with one of the entries already
			 * collapsed, we need to apply the filters that may have
			 * been set by, say, the hist_browser.
			 */
			hists__apply_filters(hists, n);
		}
509
	}
510
}
511

512 513 514 515 516 517 518 519
void hists__collapse_resort(struct hists *hists)
{
	return __hists__collapse_resort(hists, false);
}

void hists__collapse_resort_threaded(struct hists *hists)
{
	return __hists__collapse_resort(hists, true);
520 521 522
}

/*
523
 * reverse the map, sort on period.
524 525
 */

526 527 528
static void __hists__insert_output_entry(struct rb_root *entries,
					 struct hist_entry *he,
					 u64 min_callchain_hits)
529
{
530
	struct rb_node **p = &entries->rb_node;
531 532 533
	struct rb_node *parent = NULL;
	struct hist_entry *iter;

534
	if (symbol_conf.use_callchain)
535
		callchain_param.sort(&he->sorted_chain, he->callchain,
536 537 538 539 540 541
				      min_callchain_hits, &callchain_param);

	while (*p != NULL) {
		parent = *p;
		iter = rb_entry(parent, struct hist_entry, rb_node);

542
		if (he->stat.period > iter->stat.period)
543 544 545 546 547 548
			p = &(*p)->rb_left;
		else
			p = &(*p)->rb_right;
	}

	rb_link_node(&he->rb_node, parent, p);
549
	rb_insert_color(&he->rb_node, entries);
550 551
}

552
static void __hists__output_resort(struct hists *hists, bool threaded)
553
{
554
	struct rb_root *root;
555 556 557 558
	struct rb_node *next;
	struct hist_entry *n;
	u64 min_callchain_hits;

559
	min_callchain_hits = hists->stats.total_period * (callchain_param.min_percent / 100);
560

561 562 563 564 565 566 567
	if (sort__need_collapse || threaded)
		root = &hists->entries_collapsed;
	else
		root = hists->entries_in;

	next = rb_first(root);
	hists->entries = RB_ROOT;
568

569
	hists->nr_entries = 0;
570
	hists->stats.total_period = 0;
571
	hists__reset_col_len(hists);
572

573
	while (next) {
574 575
		n = rb_entry(next, struct hist_entry, rb_node_in);
		next = rb_next(&n->rb_node_in);
576

577
		__hists__insert_output_entry(&hists->entries, n, min_callchain_hits);
578
		hists__inc_nr_entries(hists, n);
579
	}
580
}
581

582 583 584 585 586 587 588 589
void hists__output_resort(struct hists *hists)
{
	return __hists__output_resort(hists, false);
}

void hists__output_resort_threaded(struct hists *hists)
{
	return __hists__output_resort(hists, true);
590
}
591

592
static void hists__remove_entry_filter(struct hists *hists, struct hist_entry *h,
593 594 595 596 597 598
				       enum hist_filter filter)
{
	h->filtered &= ~(1 << filter);
	if (h->filtered)
		return;

599
	++hists->nr_entries;
600
	if (h->ms.unfolded)
601
		hists->nr_entries += h->nr_rows;
602
	h->row_offset = 0;
603 604
	hists->stats.total_period += h->stat.period;
	hists->stats.nr_events[PERF_RECORD_SAMPLE] += h->stat.nr_events;
605

606
	hists__calc_col_len(hists, h);
607 608
}

609 610 611 612 613 614 615 616 617 618 619 620 621

static bool hists__filter_entry_by_dso(struct hists *hists,
				       struct hist_entry *he)
{
	if (hists->dso_filter != NULL &&
	    (he->ms.map == NULL || he->ms.map->dso != hists->dso_filter)) {
		he->filtered |= (1 << HIST_FILTER__DSO);
		return true;
	}

	return false;
}

622
void hists__filter_by_dso(struct hists *hists)
623 624 625
{
	struct rb_node *nd;

626 627 628
	hists->nr_entries = hists->stats.total_period = 0;
	hists->stats.nr_events[PERF_RECORD_SAMPLE] = 0;
	hists__reset_col_len(hists);
629

630
	for (nd = rb_first(&hists->entries); nd; nd = rb_next(nd)) {
631 632 633 634 635
		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);

		if (symbol_conf.exclude_other && !h->parent)
			continue;

636
		if (hists__filter_entry_by_dso(hists, h))
637 638
			continue;

639
		hists__remove_entry_filter(hists, h, HIST_FILTER__DSO);
640 641 642
	}
}

643 644 645 646 647 648 649 650 651 652 653 654
static bool hists__filter_entry_by_thread(struct hists *hists,
					  struct hist_entry *he)
{
	if (hists->thread_filter != NULL &&
	    he->thread != hists->thread_filter) {
		he->filtered |= (1 << HIST_FILTER__THREAD);
		return true;
	}

	return false;
}

655
void hists__filter_by_thread(struct hists *hists)
656 657 658
{
	struct rb_node *nd;

659 660 661
	hists->nr_entries = hists->stats.total_period = 0;
	hists->stats.nr_events[PERF_RECORD_SAMPLE] = 0;
	hists__reset_col_len(hists);
662

663
	for (nd = rb_first(&hists->entries); nd; nd = rb_next(nd)) {
664 665
		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);

666
		if (hists__filter_entry_by_thread(hists, h))
667
			continue;
668

669
		hists__remove_entry_filter(hists, h, HIST_FILTER__THREAD);
670 671
	}
}
672

673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703
static bool hists__filter_entry_by_symbol(struct hists *hists,
					  struct hist_entry *he)
{
	if (hists->symbol_filter_str != NULL &&
	    (!he->ms.sym || strstr(he->ms.sym->name,
				   hists->symbol_filter_str) == NULL)) {
		he->filtered |= (1 << HIST_FILTER__SYMBOL);
		return true;
	}

	return false;
}

void hists__filter_by_symbol(struct hists *hists)
{
	struct rb_node *nd;

	hists->nr_entries = hists->stats.total_period = 0;
	hists->stats.nr_events[PERF_RECORD_SAMPLE] = 0;
	hists__reset_col_len(hists);

	for (nd = rb_first(&hists->entries); nd; nd = rb_next(nd)) {
		struct hist_entry *h = rb_entry(nd, struct hist_entry, rb_node);

		if (hists__filter_entry_by_symbol(hists, h))
			continue;

		hists__remove_entry_filter(hists, h, HIST_FILTER__SYMBOL);
	}
}

704
int hist_entry__inc_addr_samples(struct hist_entry *he, int evidx, u64 ip)
705
{
706
	return symbol__inc_addr_samples(he->ms.sym, he->ms.map, evidx, ip);
707 708
}

709
int hist_entry__annotate(struct hist_entry *he, size_t privsize)
710
{
711
	return symbol__annotate(he->ms.sym, he->ms.map, privsize);
712
}
713

714
void hists__inc_nr_events(struct hists *hists, u32 type)
715
{
716 717
	++hists->stats.nr_events[0];
	++hists->stats.nr_events[type];
718
}
719

720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744
static struct hist_entry *hists__add_dummy_entry(struct hists *hists,
						 struct hist_entry *pair)
{
	struct rb_node **p = &hists->entries.rb_node;
	struct rb_node *parent = NULL;
	struct hist_entry *he;
	int cmp;

	while (*p != NULL) {
		parent = *p;
		he = rb_entry(parent, struct hist_entry, rb_node);

		cmp = hist_entry__cmp(pair, he);

		if (!cmp)
			goto out;

		if (cmp < 0)
			p = &(*p)->rb_left;
		else
			p = &(*p)->rb_right;
	}

	he = hist_entry__new(pair);
	if (he) {
745 746
		memset(&he->stat, 0, sizeof(he->stat));
		he->hists = hists;
747 748 749 750 751 752 753 754
		rb_link_node(&he->rb_node, parent, p);
		rb_insert_color(&he->rb_node, &hists->entries);
		hists__inc_nr_entries(hists, he);
	}
out:
	return he;
}

755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790
static struct hist_entry *hists__find_entry(struct hists *hists,
					    struct hist_entry *he)
{
	struct rb_node *n = hists->entries.rb_node;

	while (n) {
		struct hist_entry *iter = rb_entry(n, struct hist_entry, rb_node);
		int64_t cmp = hist_entry__cmp(he, iter);

		if (cmp < 0)
			n = n->rb_left;
		else if (cmp > 0)
			n = n->rb_right;
		else
			return iter;
	}

	return NULL;
}

/*
 * Look for pairs to link to the leader buckets (hist_entries):
 */
void hists__match(struct hists *leader, struct hists *other)
{
	struct rb_node *nd;
	struct hist_entry *pos, *pair;

	for (nd = rb_first(&leader->entries); nd; nd = rb_next(nd)) {
		pos  = rb_entry(nd, struct hist_entry, rb_node);
		pair = hists__find_entry(other, pos);

		if (pair)
			hist__entry_add_pair(pos, pair);
	}
}
791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814

/*
 * Look for entries in the other hists that are not present in the leader, if
 * we find them, just add a dummy entry on the leader hists, with period=0,
 * nr_events=0, to serve as the list header.
 */
int hists__link(struct hists *leader, struct hists *other)
{
	struct rb_node *nd;
	struct hist_entry *pos, *pair;

	for (nd = rb_first(&other->entries); nd; nd = rb_next(nd)) {
		pos = rb_entry(nd, struct hist_entry, rb_node);

		if (!hist_entry__has_pairs(pos)) {
			pair = hists__add_dummy_entry(leader, pos);
			if (pair == NULL)
				return -1;
			hist__entry_add_pair(pair, pos);
		}
	}

	return 0;
}