File: find_event.c

package info (click to toggle)
ltt-control 2.13.15-2
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 17,656 kB
  • sloc: ansic: 167,287; sh: 27,018; makefile: 2,828; python: 1,380; yacc: 692; lex: 129; java: 109; perl: 99; cpp: 35; xml: 23
file content (111 lines) | stat: -rw-r--r-- 2,760 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
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
/*
 * Copyright (C) 2016 Julien Desfossez <jdesfossez@efficios.com>
 *
 * SPDX-License-Identifier: GPL-2.0-only
 *
 */

#include <errno.h>
#include <stdio.h>
#include <string.h>

#include <linux/perf_event.h>
#include <perfmon/perf_event.h>
#include <perfmon/pfmlib_perf_event.h>

int main(int argc, char **argv)
{
	int ret, fd;

	/* pfm query objects */
	pfm_perf_encode_arg_t pencoder;
	pfm_event_info_t info;

	/* Perf event object to be populated by libpfm */
	struct perf_event_attr attr;

	if (argc != 2) {
		fprintf(stderr, "Usage: %s <pmu counter to find>\n"
				"ex: %s UNHALTED_REFERENCE_CYCLES\n"
				"Returns the event raw number if found and actionable with"
				"return code 0.\n"
				"If not found returns 1,"
				"If not actionable return 2,"
				"on error returns 255\n",
				argv[0], argv[0]);
		ret = -1;
		goto end;
	}

	/* Initialize perf_event_attr. */
	memset(&attr, 0, sizeof(struct perf_event_attr));

	/* Initialize libpfm encoder structure. */
	memset(&pencoder, 0, sizeof(pencoder));
	pencoder.size = sizeof(pfm_perf_encode_arg_t);

	/* Initialize libpfm event info structure. */
	memset(&info, 0, sizeof(info));
	info.size = sizeof(info);

	/* Prepare the encoder for query. */
	pencoder.attr = &attr; /* Set the perf_event_attr pointer. */
	pencoder.fstr = NULL; /* Not interested by the fully qualified event string. */

	ret = pfm_initialize();
	if (ret != PFM_SUCCESS) {
		fprintf(stderr, "Failed to initialise libpfm: %s",
				pfm_strerror(ret));
		ret = 255;
		goto end;
	}

	ret = pfm_get_os_event_encoding(argv[1],
			PFM_PLM0 | PFM_PLM1 | PFM_PLM2 | PFM_PLM3,
			PFM_OS_PERF_EVENT, &pencoder);
	if (ret != PFM_SUCCESS) {
		fprintf(stderr, "libpfm: error pfm_get_os_event_encoding: %s\n",
				pfm_strerror(ret));
		ret = 1;
		goto end;
	}

	/*
	 * Query the raw code for later use. Do it now to simplify error
	 * management.
	 */
	ret = pfm_get_event_info(pencoder.idx, PFM_OS_NONE, &info);
	if (ret != PFM_SUCCESS) {
		fprintf(stderr, "libpfm: error pfm_get_event_info: %s\n", pfm_strerror(ret));
		ret = 1;
		goto end;
	}

	/*
	 * Now that the event is found, try to use it to validate that
	 * the current user has access to it and that it can be used on that
	 * host.
	 */

	/* Set the event to disabled to prevent unnecessary side effects. */
	pencoder.attr->disabled = 1;

	/* perf_event_open is provided by perfmon/perf_event.h. */
	fd = perf_event_open(pencoder.attr, 0, -1, -1, 0);
	if (fd == -1) {
		fprintf(stderr, "perf: error perf_event_open: %d: %s\n", errno,
				strerror(errno));
		ret = 2;
		goto end;
	}

	/* We close the fd immediately since the event is actionable. */
	close(fd);

	/* Output the raw code for the event */
	fprintf(stdout, "r%" PRIx64 "\n", info.code);
	ret = 0;

end:
	return ret;
}