hist.c 16.1 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
#define HPP__COLOR_PRINT_FNS(_name, _fn, _idx)		\
375
	{						\
376 377 378 379 380
		.name   = _name,			\
		.header	= hpp__header_fn,		\
		.width	= hpp__width_fn,		\
		.color	= hpp__color_ ## _fn,		\
		.entry	= hpp__entry_ ## _fn,		\
381 382
		.cmp	= hpp__nop_cmp,			\
		.collapse = hpp__nop_cmp,		\
383
		.sort	= hpp__sort_ ## _fn,		\
384
		.idx	= PERF_HPP__ ## _idx,		\
385
	}
386

387
#define HPP__COLOR_ACC_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 399
	}

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

struct perf_hpp_fmt perf_hpp__format[] = {
413 414 415 416 417 418 419 420
	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)
421 422
};

423
LIST_HEAD(perf_hpp__list);
424
LIST_HEAD(perf_hpp__sort_list);
425

426

427
#undef HPP__COLOR_PRINT_FNS
428
#undef HPP__COLOR_ACC_PRINT_FNS
429 430
#undef HPP__PRINT_FNS

431
#undef HPP_PERCENT_FNS
432
#undef HPP_PERCENT_ACC_FNS
433 434 435 436 437 438
#undef HPP_RAW_FNS

#undef __HPP_HEADER_FN
#undef __HPP_WIDTH_FN
#undef __HPP_COLOR_PERCENT_FN
#undef __HPP_ENTRY_PERCENT_FN
439 440
#undef __HPP_COLOR_ACC_PERCENT_FN
#undef __HPP_ENTRY_ACC_PERCENT_FN
441
#undef __HPP_ENTRY_RAW_FN
442 443 444
#undef __HPP_SORT_FN
#undef __HPP_SORT_ACC_FN
#undef __HPP_SORT_RAW_FN
445 446


447
void perf_hpp__init(void)
448
{
449 450 451
	int i;

	for (i = 0; i < PERF_HPP__MAX_INDEX; i++) {
452 453 454 455 456 457 458
		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);
459 460
	}

461 462 463
	/*
	 * If user specified field order, no need to setup default fields.
	 */
464
	if (is_strict_order(field_order))
465 466
		return;

467
	if (symbol_conf.cumulate_callchain) {
468
		hpp_dimension__add_output(PERF_HPP__OVERHEAD_ACC);
469
		perf_hpp__format[PERF_HPP__OVERHEAD].name = "Self";
470 471
	}

472
	hpp_dimension__add_output(PERF_HPP__OVERHEAD);
473

474
	if (symbol_conf.show_cpu_utilization) {
475 476
		hpp_dimension__add_output(PERF_HPP__OVERHEAD_SYS);
		hpp_dimension__add_output(PERF_HPP__OVERHEAD_US);
477 478

		if (perf_guest) {
479 480
			hpp_dimension__add_output(PERF_HPP__OVERHEAD_GUEST_SYS);
			hpp_dimension__add_output(PERF_HPP__OVERHEAD_GUEST_US);
481 482 483 484
		}
	}

	if (symbol_conf.show_nr_samples)
485
		hpp_dimension__add_output(PERF_HPP__SAMPLES);
486 487

	if (symbol_conf.show_total_period)
488
		hpp_dimension__add_output(PERF_HPP__PERIOD);
489
}
490

491 492 493 494 495
void perf_hpp__column_register(struct perf_hpp_fmt *format)
{
	list_add_tail(&format->list, &perf_hpp__list);
}

496 497 498 499 500
void perf_hpp__column_unregister(struct perf_hpp_fmt *format)
{
	list_del(&format->list);
}

501 502 503 504 505
void perf_hpp__register_sort_field(struct perf_hpp_fmt *format)
{
	list_add_tail(&format->sort_list, &perf_hpp__sort_list);
}

506
void perf_hpp__column_enable(unsigned col)
507 508
{
	BUG_ON(col >= PERF_HPP__MAX_INDEX);
509
	perf_hpp__column_register(&perf_hpp__format[col]);
510 511
}

512 513 514 515 516 517 518 519
void perf_hpp__column_disable(unsigned col)
{
	BUG_ON(col >= PERF_HPP__MAX_INDEX);
	perf_hpp__column_unregister(&perf_hpp__format[col]);
}

void perf_hpp__cancel_cumulate(void)
{
520
	if (is_strict_order(field_order))
521 522
		return;

523
	perf_hpp__column_disable(PERF_HPP__OVERHEAD_ACC);
524
	perf_hpp__format[PERF_HPP__OVERHEAD].name = "Overhead";
525 526
}

527 528 529 530 531
static bool fmt_equal(struct perf_hpp_fmt *a, struct perf_hpp_fmt *b)
{
	return a->equal && a->equal(a, b);
}

532 533 534 535 536 537
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) {
538 539 540 541 542 543 544 545 546 547 548 549
		if (!list_empty(&fmt->list))
			continue;

		/*
		 * sort entry fields are dynamically created,
		 * so they can share a same sort key even though
		 * the list is empty.
		 */
		if (perf_hpp__is_sort_entry(fmt)) {
			struct perf_hpp_fmt *pos;

			perf_hpp__for_each_format(pos) {
550
				if (fmt_equal(fmt, pos))
551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578
					goto next;
			}
		}

		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) {
		if (!list_empty(&fmt->sort_list))
			continue;

		/*
		 * sort entry fields are dynamically created,
		 * so they can share a same sort key even though
		 * the list is empty.
		 */
		if (perf_hpp__is_sort_entry(fmt)) {
			struct perf_hpp_fmt *pos;

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

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

590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606
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);
	}

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

607 608 609 610 611
/*
 * See hists__fprintf to match the column widths
 */
unsigned int hists__sort_list_width(struct hists *hists)
{
612
	struct perf_hpp_fmt *fmt;
613 614
	int ret = 0;
	bool first = true;
615
	struct perf_hpp dummy_hpp;
616

617
	perf_hpp__for_each_format(fmt) {
618
		if (perf_hpp__should_skip(fmt, hists))
619 620 621 622 623
			continue;

		if (first)
			first = false;
		else
624 625
			ret += 2;

626
		ret += fmt->width(fmt, &dummy_hpp, hists_to_evsel(hists));
627 628
	}

629
	if (verbose && sort__has_sym) /* Addr + origin */
630 631 632 633
		ret += 3 + BITS_PER_LONG / 4;

	return ret;
}
634 635 636 637 638 639

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);

640
	BUG_ON(fmt->idx >= PERF_HPP__MAX_INDEX);
641

642
	switch (fmt->idx) {
643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663
	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;
	}
}
664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681

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;
	}
}