hist.c 16.2 KB
Newer Older
1
#include <math.h>
2
#include <linux/compiler.h>
3 4 5 6

#include "../util/hist.h"
#include "../util/util.h"
#include "../util/sort.h"
7
#include "../util/evsel.h"
8 9 10

/* hist period print (hpp) functions */

11 12 13 14 15 16 17
#define hpp__call_print_fn(hpp, fn, fmt, ...)			\
({								\
	int __ret = fn(hpp, fmt, ##__VA_ARGS__);		\
	advance_hpp(hpp, __ret);				\
	__ret;							\
})

18 19 20
static int __hpp__fmt(struct perf_hpp *hpp, struct hist_entry *he,
		      hpp_field_fn get_field, const char *fmt, int len,
		      hpp_snprint_fn print_fn, bool fmt_percent)
21
{
22
	int ret;
23
	struct hists *hists = he->hists;
24
	struct perf_evsel *evsel = hists_to_evsel(hists);
25 26
	char *buf = hpp->buf;
	size_t size = hpp->size;
27

28 29
	if (fmt_percent) {
		double percent = 0.0;
30
		u64 total = hists__total_period(hists);
31

32 33
		if (total)
			percent = 100.0 * get_field(he) / total;
34

35
		ret = hpp__call_print_fn(hpp, print_fn, fmt, len, percent);
36
	} else
37
		ret = hpp__call_print_fn(hpp, print_fn, fmt, len, get_field(he));
38

39
	if (perf_evsel__is_group_event(evsel)) {
40 41 42 43 44 45 46 47
		int prev_idx, idx_delta;
		struct hist_entry *pair;
		int nr_members = evsel->nr_members;

		prev_idx = perf_evsel__group_idx(evsel);

		list_for_each_entry(pair, &he->pairs.head, pairs.node) {
			u64 period = get_field(pair);
48
			u64 total = hists__total_period(pair->hists);
49 50 51 52 53 54 55 56 57 58 59 60

			if (!total)
				continue;

			evsel = hists_to_evsel(pair->hists);
			idx_delta = perf_evsel__group_idx(evsel) - prev_idx - 1;

			while (idx_delta--) {
				/*
				 * zero-fill group members in the middle which
				 * have no sample
				 */
61
				if (fmt_percent) {
62
					ret += hpp__call_print_fn(hpp, print_fn,
63
								  fmt, len, 0.0);
64
				} else {
65
					ret += hpp__call_print_fn(hpp, print_fn,
66
								  fmt, len, 0ULL);
67
				}
68 69
			}

70
			if (fmt_percent) {
71
				ret += hpp__call_print_fn(hpp, print_fn, fmt, len,
72 73 74
							  100.0 * period / total);
			} else {
				ret += hpp__call_print_fn(hpp, print_fn, fmt,
75
							  len, period);
76
			}
77 78 79 80 81 82 83 84 85 86

			prev_idx = perf_evsel__group_idx(evsel);
		}

		idx_delta = nr_members - prev_idx - 1;

		while (idx_delta--) {
			/*
			 * zero-fill group members at last which have no sample
			 */
87
			if (fmt_percent) {
88
				ret += hpp__call_print_fn(hpp, print_fn,
89
							  fmt, len, 0.0);
90
			} else {
91
				ret += hpp__call_print_fn(hpp, print_fn,
92
							  fmt, len, 0ULL);
93
			}
94 95
		}
	}
96 97 98 99 100 101 102 103

	/*
	 * Restore original buf and size as it's where caller expects
	 * the result will be saved.
	 */
	hpp->buf = buf;
	hpp->size = size;

104
	return ret;
105 106
}

107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128
int hpp__fmt(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
	     struct hist_entry *he, hpp_field_fn get_field,
	     const char *fmtstr, hpp_snprint_fn print_fn, bool fmt_percent)
{
	int len = fmt->user_len ?: fmt->len;

	if (symbol_conf.field_sep) {
		return __hpp__fmt(hpp, he, get_field, fmtstr, 1,
				  print_fn, fmt_percent);
	}

	if (fmt_percent)
		len -= 2; /* 2 for a space and a % sign */
	else
		len -= 1;

	return  __hpp__fmt(hpp, he, get_field, fmtstr, len, print_fn, fmt_percent);
}

int hpp__fmt_acc(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
		 struct hist_entry *he, hpp_field_fn get_field,
		 const char *fmtstr, hpp_snprint_fn print_fn, bool fmt_percent)
129 130
{
	if (!symbol_conf.cumulate_callchain) {
131 132
		int len = fmt->user_len ?: fmt->len;
		return snprintf(hpp->buf, hpp->size, " %*s", len - 1, "N/A");
133 134
	}

135
	return hpp__fmt(fmt, hpp, he, get_field, fmtstr, print_fn, fmt_percent);
136 137
}

138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164
static int field_cmp(u64 field_a, u64 field_b)
{
	if (field_a > field_b)
		return 1;
	if (field_a < field_b)
		return -1;
	return 0;
}

static int __hpp__sort(struct hist_entry *a, struct hist_entry *b,
		       hpp_field_fn get_field)
{
	s64 ret;
	int i, nr_members;
	struct perf_evsel *evsel;
	struct hist_entry *pair;
	u64 *fields_a, *fields_b;

	ret = field_cmp(get_field(a), get_field(b));
	if (ret || !symbol_conf.event_group)
		return ret;

	evsel = hists_to_evsel(a->hists);
	if (!perf_evsel__is_group_event(evsel))
		return ret;

	nr_members = evsel->nr_members;
165 166
	fields_a = calloc(nr_members, sizeof(*fields_a));
	fields_b = calloc(nr_members, sizeof(*fields_b));
167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193

	if (!fields_a || !fields_b)
		goto out;

	list_for_each_entry(pair, &a->pairs.head, pairs.node) {
		evsel = hists_to_evsel(pair->hists);
		fields_a[perf_evsel__group_idx(evsel)] = get_field(pair);
	}

	list_for_each_entry(pair, &b->pairs.head, pairs.node) {
		evsel = hists_to_evsel(pair->hists);
		fields_b[perf_evsel__group_idx(evsel)] = get_field(pair);
	}

	for (i = 1; i < nr_members; i++) {
		ret = field_cmp(fields_a[i], fields_b[i]);
		if (ret)
			break;
	}

out:
	free(fields_a);
	free(fields_b);

	return ret;
}

194 195 196 197 198 199 200 201 202 203 204 205 206
static int __hpp__sort_acc(struct hist_entry *a, struct hist_entry *b,
			   hpp_field_fn get_field)
{
	s64 ret = 0;

	if (symbol_conf.cumulate_callchain) {
		/*
		 * Put caller above callee when they have equal period.
		 */
		ret = field_cmp(get_field(a), get_field(b));
		if (ret)
			return ret;

207 208 209
		if (a->thread != b->thread || !symbol_conf.use_callchain)
			return 0;

210 211 212 213 214
		ret = b->callchain->max_depth - a->callchain->max_depth;
	}
	return ret;
}

215 216 217 218 219 220 221 222 223 224 225 226 227
static int hpp__width_fn(struct perf_hpp_fmt *fmt,
			 struct perf_hpp *hpp __maybe_unused,
			 struct perf_evsel *evsel)
{
	int len = fmt->user_len ?: fmt->len;

	if (symbol_conf.event_group)
		len = max(len, evsel->nr_members * fmt->len);

	if (len < (int)strlen(fmt->name))
		len = strlen(fmt->name);

	return len;
228 229
}

230 231 232 233 234
static int hpp__header_fn(struct perf_hpp_fmt *fmt, struct perf_hpp *hpp,
			  struct perf_evsel *evsel)
{
	int len = hpp__width_fn(fmt, hpp, evsel);
	return scnprintf(hpp->buf, hpp->size, "%*s", len, fmt->name);
235 236
}

237 238 239 240 241
static int hpp_color_scnprintf(struct perf_hpp *hpp, const char *fmt, ...)
{
	va_list args;
	ssize_t ssize = hpp->size;
	double percent;
242
	int ret, len;
243 244

	va_start(args, fmt);
245
	len = va_arg(args, int);
246
	percent = va_arg(args, double);
247
	ret = percent_color_len_snprintf(hpp->buf, hpp->size, fmt, len, percent);
248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265
	va_end(args);

	return (ret >= ssize) ? (ssize - 1) : ret;
}

static int hpp_entry_scnprintf(struct perf_hpp *hpp, const char *fmt, ...)
{
	va_list args;
	ssize_t ssize = hpp->size;
	int ret;

	va_start(args, fmt);
	ret = vsnprintf(hpp->buf, hpp->size, fmt, args);
	va_end(args);

	return (ret >= ssize) ? (ssize - 1) : ret;
}

266 267 268 269 270 271
#define __HPP_COLOR_PERCENT_FN(_type, _field)					\
static u64 he_get_##_field(struct hist_entry *he)				\
{										\
	return he->stat._field;							\
}										\
										\
272
static int hpp__color_##_type(struct perf_hpp_fmt *fmt,				\
273
			      struct perf_hpp *hpp, struct hist_entry *he) 	\
274
{										\
275 276
	return hpp__fmt(fmt, hpp, he, he_get_##_field, " %*.2f%%",		\
			hpp_color_scnprintf, true);				\
277 278
}

279
#define __HPP_ENTRY_PERCENT_FN(_type, _field)					\
280
static int hpp__entry_##_type(struct perf_hpp_fmt *fmt,				\
281
			      struct perf_hpp *hpp, struct hist_entry *he) 	\
282
{										\
283 284
	return hpp__fmt(fmt, hpp, he, he_get_##_field, " %*.2f%%",		\
			hpp_entry_scnprintf, true);				\
285 286
}

287
#define __HPP_SORT_FN(_type, _field)						\
288 289
static int64_t hpp__sort_##_type(struct perf_hpp_fmt *fmt __maybe_unused, 	\
				 struct hist_entry *a, struct hist_entry *b) 	\
290
{										\
291
	return __hpp__sort(a, b, he_get_##_field);				\
292 293
}

294 295 296 297 298 299
#define __HPP_COLOR_ACC_PERCENT_FN(_type, _field)				\
static u64 he_get_acc_##_field(struct hist_entry *he)				\
{										\
	return he->stat_acc->_field;						\
}										\
										\
300
static int hpp__color_##_type(struct perf_hpp_fmt *fmt,				\
301 302
			      struct perf_hpp *hpp, struct hist_entry *he) 	\
{										\
303 304
	return hpp__fmt_acc(fmt, hpp, he, he_get_acc_##_field, " %*.2f%%", 	\
			    hpp_color_scnprintf, true);				\
305 306 307
}

#define __HPP_ENTRY_ACC_PERCENT_FN(_type, _field)				\
308
static int hpp__entry_##_type(struct perf_hpp_fmt *fmt,				\
309 310
			      struct perf_hpp *hpp, struct hist_entry *he) 	\
{										\
311
	return hpp__fmt_acc(fmt, hpp, he, he_get_acc_##_field, " %*.2f%%",	\
312
			    hpp_entry_scnprintf, true);				\
313 314 315
}

#define __HPP_SORT_ACC_FN(_type, _field)					\
316 317
static int64_t hpp__sort_##_type(struct perf_hpp_fmt *fmt __maybe_unused, 	\
				 struct hist_entry *a, struct hist_entry *b) 	\
318 319 320 321
{										\
	return __hpp__sort_acc(a, b, he_get_acc_##_field);			\
}

322 323 324 325 326 327
#define __HPP_ENTRY_RAW_FN(_type, _field)					\
static u64 he_get_raw_##_field(struct hist_entry *he)				\
{										\
	return he->stat._field;							\
}										\
										\
328
static int hpp__entry_##_type(struct perf_hpp_fmt *fmt,				\
329
			      struct perf_hpp *hpp, struct hist_entry *he) 	\
330
{										\
331 332
	return hpp__fmt(fmt, hpp, he, he_get_raw_##_field, " %*"PRIu64, 	\
			hpp_entry_scnprintf, false);				\
333 334
}

335
#define __HPP_SORT_RAW_FN(_type, _field)					\
336 337
static int64_t hpp__sort_##_type(struct perf_hpp_fmt *fmt __maybe_unused, 	\
				 struct hist_entry *a, struct hist_entry *b) 	\
338
{										\
339
	return __hpp__sort(a, b, he_get_raw_##_field);				\
340 341 342
}


343
#define HPP_PERCENT_FNS(_type, _field)					\
344
__HPP_COLOR_PERCENT_FN(_type, _field)					\
345 346
__HPP_ENTRY_PERCENT_FN(_type, _field)					\
__HPP_SORT_FN(_type, _field)
347

348
#define HPP_PERCENT_ACC_FNS(_type, _field)				\
349 350 351 352
__HPP_COLOR_ACC_PERCENT_FN(_type, _field)				\
__HPP_ENTRY_ACC_PERCENT_FN(_type, _field)				\
__HPP_SORT_ACC_FN(_type, _field)

353
#define HPP_RAW_FNS(_type, _field)					\
354 355
__HPP_ENTRY_RAW_FN(_type, _field)					\
__HPP_SORT_RAW_FN(_type, _field)
356

357 358 359 360 361 362
HPP_PERCENT_FNS(overhead, period)
HPP_PERCENT_FNS(overhead_sys, period_sys)
HPP_PERCENT_FNS(overhead_us, period_us)
HPP_PERCENT_FNS(overhead_guest_sys, period_guest_sys)
HPP_PERCENT_FNS(overhead_guest_us, period_guest_us)
HPP_PERCENT_ACC_FNS(overhead_acc, period)
363

364 365
HPP_RAW_FNS(samples, nr_events)
HPP_RAW_FNS(period, period)
366

367 368
static int64_t hpp__nop_cmp(struct perf_hpp_fmt *fmt __maybe_unused,
			    struct hist_entry *a __maybe_unused,
369 370 371 372 373
			    struct hist_entry *b __maybe_unused)
{
	return 0;
}

374 375 376 377 378 379 380 381 382 383 384 385 386
static bool perf_hpp__is_hpp_entry(struct perf_hpp_fmt *a)
{
	return a->header == hpp__header_fn;
}

static bool hpp__equal(struct perf_hpp_fmt *a, struct perf_hpp_fmt *b)
{
	if (!perf_hpp__is_hpp_entry(a) || !perf_hpp__is_hpp_entry(b))
		return false;

	return a->idx == b->idx;
}

387
#define HPP__COLOR_PRINT_FNS(_name, _fn, _idx)		\
388
	{						\
389 390 391 392 393
		.name   = _name,			\
		.header	= hpp__header_fn,		\
		.width	= hpp__width_fn,		\
		.color	= hpp__color_ ## _fn,		\
		.entry	= hpp__entry_ ## _fn,		\
394 395
		.cmp	= hpp__nop_cmp,			\
		.collapse = hpp__nop_cmp,		\
396
		.sort	= hpp__sort_ ## _fn,		\
397
		.idx	= PERF_HPP__ ## _idx,		\
398
		.equal	= hpp__equal,			\
399
	}
400

401
#define HPP__COLOR_ACC_PRINT_FNS(_name, _fn, _idx)	\
402
	{						\
403 404 405 406 407
		.name   = _name,			\
		.header	= hpp__header_fn,		\
		.width	= hpp__width_fn,		\
		.color	= hpp__color_ ## _fn,		\
		.entry	= hpp__entry_ ## _fn,		\
408 409
		.cmp	= hpp__nop_cmp,			\
		.collapse = hpp__nop_cmp,		\
410
		.sort	= hpp__sort_ ## _fn,		\
411
		.idx	= PERF_HPP__ ## _idx,		\
412
		.equal	= hpp__equal,			\
413 414
	}

415
#define HPP__PRINT_FNS(_name, _fn, _idx)		\
416
	{						\
417 418 419 420
		.name   = _name,			\
		.header	= hpp__header_fn,		\
		.width	= hpp__width_fn,		\
		.entry	= hpp__entry_ ## _fn,		\
421 422
		.cmp	= hpp__nop_cmp,			\
		.collapse = hpp__nop_cmp,		\
423
		.sort	= hpp__sort_ ## _fn,		\
424
		.idx	= PERF_HPP__ ## _idx,		\
425
		.equal	= hpp__equal,			\
426
	}
427 428

struct perf_hpp_fmt perf_hpp__format[] = {
429 430 431 432 433 434 435 436
	HPP__COLOR_PRINT_FNS("Overhead", overhead, OVERHEAD),
	HPP__COLOR_PRINT_FNS("sys", overhead_sys, OVERHEAD_SYS),
	HPP__COLOR_PRINT_FNS("usr", overhead_us, OVERHEAD_US),
	HPP__COLOR_PRINT_FNS("guest sys", overhead_guest_sys, OVERHEAD_GUEST_SYS),
	HPP__COLOR_PRINT_FNS("guest usr", overhead_guest_us, OVERHEAD_GUEST_US),
	HPP__COLOR_ACC_PRINT_FNS("Children", overhead_acc, OVERHEAD_ACC),
	HPP__PRINT_FNS("Samples", samples, SAMPLES),
	HPP__PRINT_FNS("Period", period, PERIOD)
437 438
};

439 440 441 442
struct perf_hpp_list perf_hpp_list = {
	.fields	= LIST_HEAD_INIT(perf_hpp_list.fields),
	.sorts	= LIST_HEAD_INIT(perf_hpp_list.sorts),
};
443

444
#undef HPP__COLOR_PRINT_FNS
445
#undef HPP__COLOR_ACC_PRINT_FNS
446 447
#undef HPP__PRINT_FNS

448
#undef HPP_PERCENT_FNS
449
#undef HPP_PERCENT_ACC_FNS
450 451 452 453 454 455
#undef HPP_RAW_FNS

#undef __HPP_HEADER_FN
#undef __HPP_WIDTH_FN
#undef __HPP_COLOR_PERCENT_FN
#undef __HPP_ENTRY_PERCENT_FN
456 457
#undef __HPP_COLOR_ACC_PERCENT_FN
#undef __HPP_ENTRY_ACC_PERCENT_FN
458
#undef __HPP_ENTRY_RAW_FN
459 460 461
#undef __HPP_SORT_FN
#undef __HPP_SORT_ACC_FN
#undef __HPP_SORT_RAW_FN
462 463


464
void perf_hpp__init(void)
465
{
466 467 468
	int i;

	for (i = 0; i < PERF_HPP__MAX_INDEX; i++) {
469 470 471 472 473 474 475
		struct perf_hpp_fmt *fmt = &perf_hpp__format[i];

		INIT_LIST_HEAD(&fmt->list);

		/* sort_list may be linked by setup_sorting() */
		if (fmt->sort_list.next == NULL)
			INIT_LIST_HEAD(&fmt->sort_list);
476 477
	}

478 479 480
	/*
	 * If user specified field order, no need to setup default fields.
	 */
481
	if (is_strict_order(field_order))
482 483
		return;

484
	if (symbol_conf.cumulate_callchain) {
485
		hpp_dimension__add_output(PERF_HPP__OVERHEAD_ACC);
486
		perf_hpp__format[PERF_HPP__OVERHEAD].name = "Self";
487 488
	}

489
	hpp_dimension__add_output(PERF_HPP__OVERHEAD);
490

491
	if (symbol_conf.show_cpu_utilization) {
492 493
		hpp_dimension__add_output(PERF_HPP__OVERHEAD_SYS);
		hpp_dimension__add_output(PERF_HPP__OVERHEAD_US);
494 495

		if (perf_guest) {
496 497
			hpp_dimension__add_output(PERF_HPP__OVERHEAD_GUEST_SYS);
			hpp_dimension__add_output(PERF_HPP__OVERHEAD_GUEST_US);
498 499 500 501
		}
	}

	if (symbol_conf.show_nr_samples)
502
		hpp_dimension__add_output(PERF_HPP__SAMPLES);
503 504

	if (symbol_conf.show_total_period)
505
		hpp_dimension__add_output(PERF_HPP__PERIOD);
506
}
507

508 509
void perf_hpp_list__column_register(struct perf_hpp_list *list,
				    struct perf_hpp_fmt *format)
510
{
511
	list_add_tail(&format->list, &list->fields);
512 513
}

514 515
void perf_hpp_list__register_sort_field(struct perf_hpp_list *list,
					struct perf_hpp_fmt *format)
516
{
517
	list_add_tail(&format->sort_list, &list->sorts);
518 519
}

520
void perf_hpp__column_unregister(struct perf_hpp_fmt *format)
521
{
522
	list_del(&format->list);
523 524
}

525 526
void perf_hpp__cancel_cumulate(void)
{
527 528
	struct perf_hpp_fmt *fmt, *acc, *ovh, *tmp;

529
	if (is_strict_order(field_order))
530 531
		return;

532 533 534 535 536 537 538 539 540 541 542 543
	ovh = &perf_hpp__format[PERF_HPP__OVERHEAD];
	acc = &perf_hpp__format[PERF_HPP__OVERHEAD_ACC];

	perf_hpp__for_each_format_safe(fmt, tmp) {
		if (acc->equal(acc, fmt)) {
			perf_hpp__column_unregister(fmt);
			continue;
		}

		if (ovh->equal(ovh, fmt))
			fmt->name = "Overhead";
	}
544 545
}

546 547 548 549 550
static bool fmt_equal(struct perf_hpp_fmt *a, struct perf_hpp_fmt *b)
{
	return a->equal && a->equal(a, b);
}

551 552 553 554 555 556
void perf_hpp__setup_output_field(void)
{
	struct perf_hpp_fmt *fmt;

	/* append sort keys to output field */
	perf_hpp__for_each_sort_list(fmt) {
557
		struct perf_hpp_fmt *pos;
558

559 560 561
		perf_hpp__for_each_format(pos) {
			if (fmt_equal(fmt, pos))
				goto next;
562 563 564 565 566 567 568 569 570 571 572 573 574 575
		}

		perf_hpp__column_register(fmt);
next:
		continue;
	}
}

void perf_hpp__append_sort_keys(void)
{
	struct perf_hpp_fmt *fmt;

	/* append output fields to sort keys */
	perf_hpp__for_each_format(fmt) {
576
		struct perf_hpp_fmt *pos;
577

578 579 580
		perf_hpp__for_each_sort_list(pos) {
			if (fmt_equal(fmt, pos))
				goto next;
581 582 583 584 585
		}

		perf_hpp__register_sort_field(fmt);
next:
		continue;
586 587 588
	}
}

589 590 591 592 593 594
static void fmt_free(struct perf_hpp_fmt *fmt)
{
	if (fmt->free)
		fmt->free(fmt);
}

595 596 597 598 599 600 601 602
void perf_hpp__reset_output_field(void)
{
	struct perf_hpp_fmt *fmt, *tmp;

	/* reset output fields */
	perf_hpp__for_each_format_safe(fmt, tmp) {
		list_del_init(&fmt->list);
		list_del_init(&fmt->sort_list);
603
		fmt_free(fmt);
604 605 606 607 608 609
	}

	/* reset sort keys */
	perf_hpp__for_each_sort_list_safe(fmt, tmp) {
		list_del_init(&fmt->list);
		list_del_init(&fmt->sort_list);
610
		fmt_free(fmt);
611 612 613
	}
}

614 615 616 617 618
/*
 * See hists__fprintf to match the column widths
 */
unsigned int hists__sort_list_width(struct hists *hists)
{
619
	struct perf_hpp_fmt *fmt;
620 621
	int ret = 0;
	bool first = true;
622
	struct perf_hpp dummy_hpp;
623

624
	perf_hpp__for_each_format(fmt) {
625
		if (perf_hpp__should_skip(fmt, hists))
626 627 628 629 630
			continue;

		if (first)
			first = false;
		else
631 632
			ret += 2;

633
		ret += fmt->width(fmt, &dummy_hpp, hists_to_evsel(hists));
634 635
	}

636
	if (verbose && sort__has_sym) /* Addr + origin */
637 638 639 640
		ret += 3 + BITS_PER_LONG / 4;

	return ret;
}
641 642 643 644 645 646

void perf_hpp__reset_width(struct perf_hpp_fmt *fmt, struct hists *hists)
{
	if (perf_hpp__is_sort_entry(fmt))
		return perf_hpp__reset_sort_width(fmt, hists);

647
	BUG_ON(fmt->idx >= PERF_HPP__MAX_INDEX);
648

649
	switch (fmt->idx) {
650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670
	case PERF_HPP__OVERHEAD:
	case PERF_HPP__OVERHEAD_SYS:
	case PERF_HPP__OVERHEAD_US:
	case PERF_HPP__OVERHEAD_ACC:
		fmt->len = 8;
		break;

	case PERF_HPP__OVERHEAD_GUEST_SYS:
	case PERF_HPP__OVERHEAD_GUEST_US:
		fmt->len = 9;
		break;

	case PERF_HPP__SAMPLES:
	case PERF_HPP__PERIOD:
		fmt->len = 12;
		break;

	default:
		break;
	}
}
671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688

void perf_hpp__set_user_width(const char *width_list_str)
{
	struct perf_hpp_fmt *fmt;
	const char *ptr = width_list_str;

	perf_hpp__for_each_format(fmt) {
		char *p;

		int len = strtol(ptr, &p, 10);
		fmt->user_len = len;

		if (*p == ',')
			ptr = p + 1;
		else
			break;
	}
}