1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun #include "parse-events.h"
3*4882a593Smuzhiyun #include "pmu.h"
4*4882a593Smuzhiyun #include "tests.h"
5*4882a593Smuzhiyun #include <errno.h>
6*4882a593Smuzhiyun #include <stdio.h>
7*4882a593Smuzhiyun #include <linux/kernel.h>
8*4882a593Smuzhiyun #include <linux/limits.h>
9*4882a593Smuzhiyun
10*4882a593Smuzhiyun /* Simulated format definitions. */
11*4882a593Smuzhiyun static struct test_format {
12*4882a593Smuzhiyun const char *name;
13*4882a593Smuzhiyun const char *value;
14*4882a593Smuzhiyun } test_formats[] = {
15*4882a593Smuzhiyun { "krava01", "config:0-1,62-63\n", },
16*4882a593Smuzhiyun { "krava02", "config:10-17\n", },
17*4882a593Smuzhiyun { "krava03", "config:5\n", },
18*4882a593Smuzhiyun { "krava11", "config1:0,2,4,6,8,20-28\n", },
19*4882a593Smuzhiyun { "krava12", "config1:63\n", },
20*4882a593Smuzhiyun { "krava13", "config1:45-47\n", },
21*4882a593Smuzhiyun { "krava21", "config2:0-3,10-13,20-23,30-33,40-43,50-53,60-63\n", },
22*4882a593Smuzhiyun { "krava22", "config2:8,18,48,58\n", },
23*4882a593Smuzhiyun { "krava23", "config2:28-29,38\n", },
24*4882a593Smuzhiyun };
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun /* Simulated users input. */
27*4882a593Smuzhiyun static struct parse_events_term test_terms[] = {
28*4882a593Smuzhiyun {
29*4882a593Smuzhiyun .config = (char *) "krava01",
30*4882a593Smuzhiyun .val.num = 15,
31*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
32*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
33*4882a593Smuzhiyun },
34*4882a593Smuzhiyun {
35*4882a593Smuzhiyun .config = (char *) "krava02",
36*4882a593Smuzhiyun .val.num = 170,
37*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
38*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
39*4882a593Smuzhiyun },
40*4882a593Smuzhiyun {
41*4882a593Smuzhiyun .config = (char *) "krava03",
42*4882a593Smuzhiyun .val.num = 1,
43*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
44*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
45*4882a593Smuzhiyun },
46*4882a593Smuzhiyun {
47*4882a593Smuzhiyun .config = (char *) "krava11",
48*4882a593Smuzhiyun .val.num = 27,
49*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
50*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
51*4882a593Smuzhiyun },
52*4882a593Smuzhiyun {
53*4882a593Smuzhiyun .config = (char *) "krava12",
54*4882a593Smuzhiyun .val.num = 1,
55*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
56*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
57*4882a593Smuzhiyun },
58*4882a593Smuzhiyun {
59*4882a593Smuzhiyun .config = (char *) "krava13",
60*4882a593Smuzhiyun .val.num = 2,
61*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
62*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
63*4882a593Smuzhiyun },
64*4882a593Smuzhiyun {
65*4882a593Smuzhiyun .config = (char *) "krava21",
66*4882a593Smuzhiyun .val.num = 119,
67*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
68*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
69*4882a593Smuzhiyun },
70*4882a593Smuzhiyun {
71*4882a593Smuzhiyun .config = (char *) "krava22",
72*4882a593Smuzhiyun .val.num = 11,
73*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
74*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
75*4882a593Smuzhiyun },
76*4882a593Smuzhiyun {
77*4882a593Smuzhiyun .config = (char *) "krava23",
78*4882a593Smuzhiyun .val.num = 2,
79*4882a593Smuzhiyun .type_val = PARSE_EVENTS__TERM_TYPE_NUM,
80*4882a593Smuzhiyun .type_term = PARSE_EVENTS__TERM_TYPE_USER,
81*4882a593Smuzhiyun },
82*4882a593Smuzhiyun };
83*4882a593Smuzhiyun
84*4882a593Smuzhiyun /*
85*4882a593Smuzhiyun * Prepare format directory data, exported by kernel
86*4882a593Smuzhiyun * at /sys/bus/event_source/devices/<dev>/format.
87*4882a593Smuzhiyun */
test_format_dir_get(void)88*4882a593Smuzhiyun static char *test_format_dir_get(void)
89*4882a593Smuzhiyun {
90*4882a593Smuzhiyun static char dir[PATH_MAX];
91*4882a593Smuzhiyun unsigned int i;
92*4882a593Smuzhiyun
93*4882a593Smuzhiyun snprintf(dir, PATH_MAX, "/tmp/perf-pmu-test-format-XXXXXX");
94*4882a593Smuzhiyun if (!mkdtemp(dir))
95*4882a593Smuzhiyun return NULL;
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(test_formats); i++) {
98*4882a593Smuzhiyun static char name[PATH_MAX];
99*4882a593Smuzhiyun struct test_format *format = &test_formats[i];
100*4882a593Smuzhiyun FILE *file;
101*4882a593Smuzhiyun
102*4882a593Smuzhiyun scnprintf(name, PATH_MAX, "%s/%s", dir, format->name);
103*4882a593Smuzhiyun
104*4882a593Smuzhiyun file = fopen(name, "w");
105*4882a593Smuzhiyun if (!file)
106*4882a593Smuzhiyun return NULL;
107*4882a593Smuzhiyun
108*4882a593Smuzhiyun if (1 != fwrite(format->value, strlen(format->value), 1, file))
109*4882a593Smuzhiyun break;
110*4882a593Smuzhiyun
111*4882a593Smuzhiyun fclose(file);
112*4882a593Smuzhiyun }
113*4882a593Smuzhiyun
114*4882a593Smuzhiyun return dir;
115*4882a593Smuzhiyun }
116*4882a593Smuzhiyun
117*4882a593Smuzhiyun /* Cleanup format directory. */
test_format_dir_put(char * dir)118*4882a593Smuzhiyun static int test_format_dir_put(char *dir)
119*4882a593Smuzhiyun {
120*4882a593Smuzhiyun char buf[PATH_MAX];
121*4882a593Smuzhiyun snprintf(buf, PATH_MAX, "rm -f %s/*\n", dir);
122*4882a593Smuzhiyun if (system(buf))
123*4882a593Smuzhiyun return -1;
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun snprintf(buf, PATH_MAX, "rmdir %s\n", dir);
126*4882a593Smuzhiyun return system(buf);
127*4882a593Smuzhiyun }
128*4882a593Smuzhiyun
test_terms_list(void)129*4882a593Smuzhiyun static struct list_head *test_terms_list(void)
130*4882a593Smuzhiyun {
131*4882a593Smuzhiyun static LIST_HEAD(terms);
132*4882a593Smuzhiyun unsigned int i;
133*4882a593Smuzhiyun
134*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(test_terms); i++)
135*4882a593Smuzhiyun list_add_tail(&test_terms[i].list, &terms);
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun return &terms;
138*4882a593Smuzhiyun }
139*4882a593Smuzhiyun
test__pmu(struct test * test __maybe_unused,int subtest __maybe_unused)140*4882a593Smuzhiyun int test__pmu(struct test *test __maybe_unused, int subtest __maybe_unused)
141*4882a593Smuzhiyun {
142*4882a593Smuzhiyun char *format = test_format_dir_get();
143*4882a593Smuzhiyun LIST_HEAD(formats);
144*4882a593Smuzhiyun struct list_head *terms = test_terms_list();
145*4882a593Smuzhiyun int ret;
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun if (!format)
148*4882a593Smuzhiyun return -EINVAL;
149*4882a593Smuzhiyun
150*4882a593Smuzhiyun do {
151*4882a593Smuzhiyun struct perf_event_attr attr;
152*4882a593Smuzhiyun
153*4882a593Smuzhiyun memset(&attr, 0, sizeof(attr));
154*4882a593Smuzhiyun
155*4882a593Smuzhiyun ret = perf_pmu__format_parse(format, &formats);
156*4882a593Smuzhiyun if (ret)
157*4882a593Smuzhiyun break;
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun ret = perf_pmu__config_terms("perf-pmu-test", &formats, &attr,
160*4882a593Smuzhiyun terms, false, NULL);
161*4882a593Smuzhiyun if (ret)
162*4882a593Smuzhiyun break;
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun ret = -EINVAL;
165*4882a593Smuzhiyun
166*4882a593Smuzhiyun if (attr.config != 0xc00000000002a823)
167*4882a593Smuzhiyun break;
168*4882a593Smuzhiyun if (attr.config1 != 0x8000400000000145)
169*4882a593Smuzhiyun break;
170*4882a593Smuzhiyun if (attr.config2 != 0x0400000020041d07)
171*4882a593Smuzhiyun break;
172*4882a593Smuzhiyun
173*4882a593Smuzhiyun ret = 0;
174*4882a593Smuzhiyun } while (0);
175*4882a593Smuzhiyun
176*4882a593Smuzhiyun perf_pmu__del_formats(&formats);
177*4882a593Smuzhiyun test_format_dir_put(format);
178*4882a593Smuzhiyun return ret;
179*4882a593Smuzhiyun }
180