blob: 06f5a3a4295c52cac138ac2526afdac04ffbf0dc [file] [log] [blame]
Jiri Olsace1e22b2016-02-15 09:34:35 +01001#include <stddef.h>
2#include <stdlib.h>
3#include <string.h>
4#include <errno.h>
Jiri Olsa54fbad52016-02-24 09:46:42 +01005#include <sys/types.h>
6#include <sys/stat.h>
7#include <unistd.h>
8#include <api/fs/fs.h>
Arnaldo Carvalho de Melo877a7a12017-04-17 11:39:06 -03009#include <linux/kernel.h>
Jiri Olsaacbe6132016-02-15 09:34:34 +010010#include "mem-events.h"
Jiri Olsace1e22b2016-02-15 09:34:35 +010011#include "debug.h"
Jiri Olsa0c877d72016-02-24 09:46:46 +010012#include "symbol.h"
Jiri Olsaaadddd62016-09-22 17:36:30 +020013#include "sort.h"
Jiri Olsaacbe6132016-02-15 09:34:34 +010014
Jiri Olsab0d745b2016-06-14 20:19:11 +020015unsigned int perf_mem_events__loads_ldlat = 30;
16
Jiri Olsa54fbad52016-02-24 09:46:42 +010017#define E(t, n, s) { .tag = t, .name = n, .sysfs_name = s }
Jiri Olsaacbe6132016-02-15 09:34:34 +010018
19struct perf_mem_event perf_mem_events[PERF_MEM_EVENTS__MAX] = {
Jiri Olsab0d745b2016-06-14 20:19:11 +020020 E("ldlat-loads", "cpu/mem-loads,ldlat=%u/P", "mem-loads"),
Jiri Olsa54fbad52016-02-24 09:46:42 +010021 E("ldlat-stores", "cpu/mem-stores/P", "mem-stores"),
Jiri Olsaacbe6132016-02-15 09:34:34 +010022};
Jiri Olsa54fbad52016-02-24 09:46:42 +010023#undef E
Jiri Olsaacbe6132016-02-15 09:34:34 +010024
25#undef E
Jiri Olsace1e22b2016-02-15 09:34:35 +010026
Jiri Olsab0d745b2016-06-14 20:19:11 +020027static char mem_loads_name[100];
28static bool mem_loads_name__init;
29
Jiri Olsa2ba7ac52016-02-24 09:46:43 +010030char *perf_mem_events__name(int i)
31{
Jiri Olsab0d745b2016-06-14 20:19:11 +020032 if (i == PERF_MEM_EVENTS__LOAD) {
33 if (!mem_loads_name__init) {
34 mem_loads_name__init = true;
35 scnprintf(mem_loads_name, sizeof(mem_loads_name),
36 perf_mem_events[i].name,
37 perf_mem_events__loads_ldlat);
38 }
39 return mem_loads_name;
40 }
41
Jiri Olsa2ba7ac52016-02-24 09:46:43 +010042 return (char *)perf_mem_events[i].name;
43}
44
Jiri Olsace1e22b2016-02-15 09:34:35 +010045int perf_mem_events__parse(const char *str)
46{
47 char *tok, *saveptr = NULL;
48 bool found = false;
49 char *buf;
50 int j;
51
52 /* We need buffer that we know we can write to. */
53 buf = malloc(strlen(str) + 1);
54 if (!buf)
55 return -ENOMEM;
56
57 strcpy(buf, str);
58
59 tok = strtok_r((char *)buf, ",", &saveptr);
60
61 while (tok) {
62 for (j = 0; j < PERF_MEM_EVENTS__MAX; j++) {
63 struct perf_mem_event *e = &perf_mem_events[j];
64
65 if (strstr(e->tag, tok))
66 e->record = found = true;
67 }
68
69 tok = strtok_r(NULL, ",", &saveptr);
70 }
71
72 free(buf);
73
74 if (found)
75 return 0;
76
77 pr_err("failed: event '%s' not found, use '-e list' to get list of available events\n", str);
78 return -1;
79}
Jiri Olsa54fbad52016-02-24 09:46:42 +010080
81int perf_mem_events__init(void)
82{
83 const char *mnt = sysfs__mount();
84 bool found = false;
85 int j;
86
87 if (!mnt)
88 return -ENOENT;
89
90 for (j = 0; j < PERF_MEM_EVENTS__MAX; j++) {
91 char path[PATH_MAX];
92 struct perf_mem_event *e = &perf_mem_events[j];
93 struct stat st;
94
95 scnprintf(path, PATH_MAX, "%s/devices/cpu/events/%s",
96 mnt, e->sysfs_name);
97
98 if (!stat(path, &st))
99 e->supported = found = true;
100 }
101
102 return found ? 0 : -ENOENT;
103}
Jiri Olsa0c877d72016-02-24 09:46:46 +0100104
105static const char * const tlb_access[] = {
106 "N/A",
107 "HIT",
108 "MISS",
109 "L1",
110 "L2",
111 "Walker",
112 "Fault",
113};
114
Jiri Olsab1a5fbea2016-02-24 09:46:50 +0100115int perf_mem__tlb_scnprintf(char *out, size_t sz, struct mem_info *mem_info)
Jiri Olsa0c877d72016-02-24 09:46:46 +0100116{
117 size_t l = 0, i;
118 u64 m = PERF_MEM_TLB_NA;
119 u64 hit, miss;
120
121 sz -= 1; /* -1 for null termination */
122 out[0] = '\0';
123
124 if (mem_info)
125 m = mem_info->data_src.mem_dtlb;
126
127 hit = m & PERF_MEM_TLB_HIT;
128 miss = m & PERF_MEM_TLB_MISS;
129
130 /* already taken care of */
131 m &= ~(PERF_MEM_TLB_HIT|PERF_MEM_TLB_MISS);
132
133 for (i = 0; m && i < ARRAY_SIZE(tlb_access); i++, m >>= 1) {
134 if (!(m & 0x1))
135 continue;
136 if (l) {
137 strcat(out, " or ");
138 l += 4;
139 }
Jiri Olsab1a5fbea2016-02-24 09:46:50 +0100140 l += scnprintf(out + l, sz - l, tlb_access[i]);
Jiri Olsa0c877d72016-02-24 09:46:46 +0100141 }
142 if (*out == '\0')
Jiri Olsab1a5fbea2016-02-24 09:46:50 +0100143 l += scnprintf(out, sz - l, "N/A");
Jiri Olsa0c877d72016-02-24 09:46:46 +0100144 if (hit)
Jiri Olsab1a5fbea2016-02-24 09:46:50 +0100145 l += scnprintf(out + l, sz - l, " hit");
Jiri Olsa0c877d72016-02-24 09:46:46 +0100146 if (miss)
Jiri Olsab1a5fbea2016-02-24 09:46:50 +0100147 l += scnprintf(out + l, sz - l, " miss");
148
149 return l;
Jiri Olsa0c877d72016-02-24 09:46:46 +0100150}
Jiri Olsa071e9a12016-02-24 09:46:47 +0100151
152static const char * const mem_lvl[] = {
153 "N/A",
154 "HIT",
155 "MISS",
156 "L1",
157 "LFB",
158 "L2",
159 "L3",
160 "Local RAM",
161 "Remote RAM (1 hop)",
162 "Remote RAM (2 hops)",
163 "Remote Cache (1 hop)",
164 "Remote Cache (2 hops)",
165 "I/O",
166 "Uncached",
167};
168
Jiri Olsa96907562016-02-24 09:46:51 +0100169int perf_mem__lvl_scnprintf(char *out, size_t sz, struct mem_info *mem_info)
Jiri Olsa071e9a12016-02-24 09:46:47 +0100170{
171 size_t i, l = 0;
172 u64 m = PERF_MEM_LVL_NA;
173 u64 hit, miss;
174
175 if (mem_info)
176 m = mem_info->data_src.mem_lvl;
177
178 sz -= 1; /* -1 for null termination */
179 out[0] = '\0';
180
181 hit = m & PERF_MEM_LVL_HIT;
182 miss = m & PERF_MEM_LVL_MISS;
183
184 /* already taken care of */
185 m &= ~(PERF_MEM_LVL_HIT|PERF_MEM_LVL_MISS);
186
187 for (i = 0; m && i < ARRAY_SIZE(mem_lvl); i++, m >>= 1) {
188 if (!(m & 0x1))
189 continue;
190 if (l) {
191 strcat(out, " or ");
192 l += 4;
193 }
Jiri Olsa96907562016-02-24 09:46:51 +0100194 l += scnprintf(out + l, sz - l, mem_lvl[i]);
Jiri Olsa071e9a12016-02-24 09:46:47 +0100195 }
196 if (*out == '\0')
Jiri Olsa96907562016-02-24 09:46:51 +0100197 l += scnprintf(out, sz - l, "N/A");
Jiri Olsa071e9a12016-02-24 09:46:47 +0100198 if (hit)
Jiri Olsa96907562016-02-24 09:46:51 +0100199 l += scnprintf(out + l, sz - l, " hit");
Jiri Olsa071e9a12016-02-24 09:46:47 +0100200 if (miss)
Jiri Olsa96907562016-02-24 09:46:51 +0100201 l += scnprintf(out + l, sz - l, " miss");
202
203 return l;
Jiri Olsa071e9a12016-02-24 09:46:47 +0100204}
Jiri Olsa2c07af12016-02-24 09:46:48 +0100205
206static const char * const snoop_access[] = {
207 "N/A",
208 "None",
Jiri Olsa2c07af12016-02-24 09:46:48 +0100209 "Hit",
Andi Kleen166ebdd2017-04-19 10:49:40 -0700210 "Miss",
Jiri Olsa2c07af12016-02-24 09:46:48 +0100211 "HitM",
212};
213
Jiri Olsa149d7502016-02-24 09:46:52 +0100214int perf_mem__snp_scnprintf(char *out, size_t sz, struct mem_info *mem_info)
Jiri Olsa2c07af12016-02-24 09:46:48 +0100215{
216 size_t i, l = 0;
217 u64 m = PERF_MEM_SNOOP_NA;
218
219 sz -= 1; /* -1 for null termination */
220 out[0] = '\0';
221
222 if (mem_info)
223 m = mem_info->data_src.mem_snoop;
224
225 for (i = 0; m && i < ARRAY_SIZE(snoop_access); i++, m >>= 1) {
226 if (!(m & 0x1))
227 continue;
228 if (l) {
229 strcat(out, " or ");
230 l += 4;
231 }
Jiri Olsa149d7502016-02-24 09:46:52 +0100232 l += scnprintf(out + l, sz - l, snoop_access[i]);
Jiri Olsa2c07af12016-02-24 09:46:48 +0100233 }
234
235 if (*out == '\0')
Jiri Olsa149d7502016-02-24 09:46:52 +0100236 l += scnprintf(out, sz - l, "N/A");
237
238 return l;
Jiri Olsa2c07af12016-02-24 09:46:48 +0100239}
Jiri Olsa69a77272016-02-24 09:46:49 +0100240
Jiri Olsa8b0819c2016-02-24 09:46:53 +0100241int perf_mem__lck_scnprintf(char *out, size_t sz, struct mem_info *mem_info)
Jiri Olsa69a77272016-02-24 09:46:49 +0100242{
243 u64 mask = PERF_MEM_LOCK_NA;
Jiri Olsa8b0819c2016-02-24 09:46:53 +0100244 int l;
Jiri Olsa69a77272016-02-24 09:46:49 +0100245
246 if (mem_info)
247 mask = mem_info->data_src.mem_lock;
248
249 if (mask & PERF_MEM_LOCK_NA)
Jiri Olsa8b0819c2016-02-24 09:46:53 +0100250 l = scnprintf(out, sz, "N/A");
Jiri Olsa69a77272016-02-24 09:46:49 +0100251 else if (mask & PERF_MEM_LOCK_LOCKED)
Jiri Olsa8b0819c2016-02-24 09:46:53 +0100252 l = scnprintf(out, sz, "Yes");
Jiri Olsa69a77272016-02-24 09:46:49 +0100253 else
Jiri Olsa8b0819c2016-02-24 09:46:53 +0100254 l = scnprintf(out, sz, "No");
255
256 return l;
Jiri Olsa69a77272016-02-24 09:46:49 +0100257}
Jiri Olsac19ac912016-02-24 09:46:54 +0100258
259int perf_script__meminfo_scnprintf(char *out, size_t sz, struct mem_info *mem_info)
260{
261 int i = 0;
262
263 i += perf_mem__lvl_scnprintf(out, sz, mem_info);
264 i += scnprintf(out + i, sz - i, "|SNP ");
265 i += perf_mem__snp_scnprintf(out + i, sz - i, mem_info);
266 i += scnprintf(out + i, sz - i, "|TLB ");
267 i += perf_mem__tlb_scnprintf(out + i, sz - i, mem_info);
268 i += scnprintf(out + i, sz - i, "|LCK ");
269 i += perf_mem__lck_scnprintf(out + i, sz - i, mem_info);
270
271 return i;
272}
Jiri Olsaaadddd62016-09-22 17:36:30 +0200273
274int c2c_decode_stats(struct c2c_stats *stats, struct mem_info *mi)
275{
276 union perf_mem_data_src *data_src = &mi->data_src;
277 u64 daddr = mi->daddr.addr;
278 u64 op = data_src->mem_op;
279 u64 lvl = data_src->mem_lvl;
280 u64 snoop = data_src->mem_snoop;
281 u64 lock = data_src->mem_lock;
282 int err = 0;
283
Jiri Olsadba8ab92016-11-21 22:33:29 +0100284#define HITM_INC(__f) \
285do { \
286 stats->__f++; \
287 stats->tot_hitm++; \
288} while (0)
289
Jiri Olsaaadddd62016-09-22 17:36:30 +0200290#define P(a, b) PERF_MEM_##a##_##b
291
292 stats->nr_entries++;
293
294 if (lock & P(LOCK, LOCKED)) stats->locks++;
295
296 if (op & P(OP, LOAD)) {
297 /* load */
298 stats->load++;
299
300 if (!daddr) {
301 stats->ld_noadrs++;
302 return -1;
303 }
304
305 if (lvl & P(LVL, HIT)) {
306 if (lvl & P(LVL, UNC)) stats->ld_uncache++;
307 if (lvl & P(LVL, IO)) stats->ld_io++;
308 if (lvl & P(LVL, LFB)) stats->ld_fbhit++;
309 if (lvl & P(LVL, L1 )) stats->ld_l1hit++;
310 if (lvl & P(LVL, L2 )) stats->ld_l2hit++;
311 if (lvl & P(LVL, L3 )) {
312 if (snoop & P(SNOOP, HITM))
Jiri Olsadba8ab92016-11-21 22:33:29 +0100313 HITM_INC(lcl_hitm);
Jiri Olsaaadddd62016-09-22 17:36:30 +0200314 else
315 stats->ld_llchit++;
316 }
317
318 if (lvl & P(LVL, LOC_RAM)) {
319 stats->lcl_dram++;
320 if (snoop & P(SNOOP, HIT))
321 stats->ld_shared++;
322 else
323 stats->ld_excl++;
324 }
325
326 if ((lvl & P(LVL, REM_RAM1)) ||
327 (lvl & P(LVL, REM_RAM2))) {
328 stats->rmt_dram++;
329 if (snoop & P(SNOOP, HIT))
330 stats->ld_shared++;
331 else
332 stats->ld_excl++;
333 }
334 }
335
336 if ((lvl & P(LVL, REM_CCE1)) ||
337 (lvl & P(LVL, REM_CCE2))) {
338 if (snoop & P(SNOOP, HIT))
339 stats->rmt_hit++;
340 else if (snoop & P(SNOOP, HITM))
Jiri Olsadba8ab92016-11-21 22:33:29 +0100341 HITM_INC(rmt_hitm);
Jiri Olsaaadddd62016-09-22 17:36:30 +0200342 }
343
344 if ((lvl & P(LVL, MISS)))
345 stats->ld_miss++;
346
347 } else if (op & P(OP, STORE)) {
348 /* store */
349 stats->store++;
350
351 if (!daddr) {
352 stats->st_noadrs++;
353 return -1;
354 }
355
356 if (lvl & P(LVL, HIT)) {
357 if (lvl & P(LVL, UNC)) stats->st_uncache++;
358 if (lvl & P(LVL, L1 )) stats->st_l1hit++;
359 }
360 if (lvl & P(LVL, MISS))
361 if (lvl & P(LVL, L1)) stats->st_l1miss++;
362 } else {
363 /* unparsable data_src? */
364 stats->noparse++;
365 return -1;
366 }
367
368 if (!mi->daddr.map || !mi->iaddr.map) {
369 stats->nomap++;
370 return -1;
371 }
372
373#undef P
Jiri Olsadba8ab92016-11-21 22:33:29 +0100374#undef HITM_INC
Jiri Olsaaadddd62016-09-22 17:36:30 +0200375 return err;
376}
Jiri Olsa0a9a24c2016-09-22 17:36:31 +0200377
378void c2c_add_stats(struct c2c_stats *stats, struct c2c_stats *add)
379{
380 stats->nr_entries += add->nr_entries;
381
382 stats->locks += add->locks;
383 stats->store += add->store;
384 stats->st_uncache += add->st_uncache;
385 stats->st_noadrs += add->st_noadrs;
386 stats->st_l1hit += add->st_l1hit;
387 stats->st_l1miss += add->st_l1miss;
388 stats->load += add->load;
389 stats->ld_excl += add->ld_excl;
390 stats->ld_shared += add->ld_shared;
391 stats->ld_uncache += add->ld_uncache;
392 stats->ld_io += add->ld_io;
393 stats->ld_miss += add->ld_miss;
394 stats->ld_noadrs += add->ld_noadrs;
395 stats->ld_fbhit += add->ld_fbhit;
396 stats->ld_l1hit += add->ld_l1hit;
397 stats->ld_l2hit += add->ld_l2hit;
398 stats->ld_llchit += add->ld_llchit;
399 stats->lcl_hitm += add->lcl_hitm;
400 stats->rmt_hitm += add->rmt_hitm;
Jiri Olsadba8ab92016-11-21 22:33:29 +0100401 stats->tot_hitm += add->tot_hitm;
Jiri Olsa0a9a24c2016-09-22 17:36:31 +0200402 stats->rmt_hit += add->rmt_hit;
403 stats->lcl_dram += add->lcl_dram;
404 stats->rmt_dram += add->rmt_dram;
405 stats->nomap += add->nomap;
406 stats->noparse += add->noparse;
407}