blob: d7be552b21c8443670707cc1c542249fb9b0f951 [file] [log] [blame]
Arnaldo Carvalho de Meloa43783a2017-04-18 10:46:11 -03001#include <errno.h>
Arnaldo Carvalho de Melofd20e812017-04-17 15:23:08 -03002#include <inttypes.h>
Arnaldo Carvalho de Meloa9072bc2011-10-26 12:41:38 -02003#include "util.h"
Arnaldo Carvalho de Meloa0675582017-04-17 16:51:59 -03004#include "string2.h"
Arnaldo Carvalho de Melo391e4202017-04-19 18:51:14 -03005#include <sys/param.h>
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02006#include <sys/types.h>
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02007#include <byteswap.h>
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02008#include <unistd.h>
9#include <stdio.h>
10#include <stdlib.h>
Arnaldo Carvalho de Melo03536312017-06-16 12:18:27 -030011#include <linux/compiler.h>
Frederic Weisbecker8671dab2009-11-11 04:51:03 +010012#include <linux/list.h>
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -020013#include <linux/kernel.h>
Robert Richterb1e5a9b2011-12-07 10:02:57 +010014#include <linux/bitops.h>
David Carrillo-Cisnerosa4d8c982017-07-17 21:25:46 -070015#include <linux/stringify.h>
Arnaldo Carvalho de Melo7a8ef4c2017-04-19 20:57:47 -030016#include <sys/stat.h>
17#include <sys/types.h>
Stephane Eranianfbe96f22011-09-30 15:40:40 +020018#include <sys/utsname.h>
Arnaldo Carvalho de Melo7a8ef4c2017-04-19 20:57:47 -030019#include <unistd.h>
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +020020
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -020021#include "evlist.h"
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -030022#include "evsel.h"
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +020023#include "header.h"
Arnaldo Carvalho de Melo98521b32017-04-25 15:45:35 -030024#include "memswap.h"
Frederic Weisbecker03456a12009-10-06 23:36:47 +020025#include "../perf.h"
26#include "trace-event.h"
Arnaldo Carvalho de Melo301a0b02009-12-13 19:50:25 -020027#include "session.h"
Frederic Weisbecker8671dab2009-11-11 04:51:03 +010028#include "symbol.h"
Frederic Weisbecker4778d2e2009-11-11 04:51:05 +010029#include "debug.h"
Stephane Eranianfbe96f22011-09-30 15:40:40 +020030#include "cpumap.h"
Robert Richter50a96672012-08-16 21:10:24 +020031#include "pmu.h"
Jiri Olsa7dbf4dc2012-09-10 18:50:19 +020032#include "vdso.h"
Namhyung Kima1ae5652012-09-24 17:14:59 +090033#include "strbuf.h"
Jiri Olsaebb296c2012-10-27 23:18:28 +020034#include "build-id.h"
Jiri Olsacc9784bd2013-10-15 16:27:34 +020035#include "data.h"
Jiri Olsa720e98b2016-02-16 16:01:43 +010036#include <api/fs/fs.h>
37#include "asm/bug.h"
David Carrillo-Cisnerose9def1b2017-07-17 21:25:48 -070038#include "tool.h"
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +020039
Arnaldo Carvalho de Melo3d689ed2017-04-17 16:10:49 -030040#include "sane_ctype.h"
41
Stephane Eranian73323f52012-02-02 13:54:44 +010042/*
43 * magic2 = "PERFILE2"
44 * must be a numerical value to let the endianness
45 * determine the memory layout. That way we are able
46 * to detect endianness when reading the perf.data file
47 * back.
48 *
49 * we check for legacy (PERFFILE) format.
50 */
51static const char *__perf_magic1 = "PERFFILE";
52static const u64 __perf_magic2 = 0x32454c4946524550ULL;
53static const u64 __perf_magic2_sw = 0x50455246494c4532ULL;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +020054
Stephane Eranian73323f52012-02-02 13:54:44 +010055#define PERF_MAGIC __perf_magic2
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +020056
Soramichi AKIYAMAd25ed5d2017-01-17 00:22:37 +090057const char perf_version_string[] = PERF_VERSION;
58
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +020059struct perf_file_attr {
Ingo Molnarcdd6c482009-09-21 12:02:48 +020060 struct perf_event_attr attr;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +020061 struct perf_file_section ids;
62};
63
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -070064struct feat_fd {
65 struct perf_header *ph;
66 int fd;
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -070067 void *buf; /* Either buf != NULL or fd >= 0 */
David Carrillo-Cisneros62552452017-07-17 21:25:42 -070068 ssize_t offset;
69 size_t size;
David Carrillo-Cisnerosf9ebdcc2017-07-17 21:25:49 -070070 struct perf_evsel *events;
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -070071};
72
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -030073void perf_header__set_feat(struct perf_header *header, int feat)
Arnaldo Carvalho de Melo8d063672009-11-04 18:50:43 -020074{
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -030075 set_bit(feat, header->adds_features);
Arnaldo Carvalho de Melo8d063672009-11-04 18:50:43 -020076}
77
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -030078void perf_header__clear_feat(struct perf_header *header, int feat)
Arnaldo Carvalho de Melobaa2f6c2010-11-26 19:39:15 -020079{
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -030080 clear_bit(feat, header->adds_features);
Arnaldo Carvalho de Melobaa2f6c2010-11-26 19:39:15 -020081}
82
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -030083bool perf_header__has_feat(const struct perf_header *header, int feat)
Arnaldo Carvalho de Melo8d063672009-11-04 18:50:43 -020084{
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -030085 return test_bit(feat, header->adds_features);
Arnaldo Carvalho de Melo8d063672009-11-04 18:50:43 -020086}
87
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -070088static int __do_write_fd(struct feat_fd *ff, const void *buf, size_t size)
89{
90 ssize_t ret = writen(ff->fd, buf, size);
91
92 if (ret != (ssize_t)size)
93 return ret < 0 ? (int)ret : -1;
94 return 0;
95}
96
97static int __do_write_buf(struct feat_fd *ff, const void *buf, size_t size)
98{
99 /* struct perf_event_header::size is u16 */
100 const size_t max_size = 0xffff - sizeof(struct perf_event_header);
101 size_t new_size = ff->size;
102 void *addr;
103
104 if (size + ff->offset > max_size)
105 return -E2BIG;
106
107 while (size > (new_size - ff->offset))
108 new_size <<= 1;
109 new_size = min(max_size, new_size);
110
111 if (ff->size < new_size) {
112 addr = realloc(ff->buf, new_size);
113 if (!addr)
114 return -ENOMEM;
115 ff->buf = addr;
116 ff->size = new_size;
117 }
118
119 memcpy(ff->buf + ff->offset, buf, size);
120 ff->offset += size;
121
122 return 0;
123}
124
David Carrillo-Cisneros2ff53652017-07-17 21:25:36 -0700125/* Return: 0 if succeded, -ERR if failed. */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700126int do_write(struct feat_fd *ff, const void *buf, size_t size)
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +0200127{
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -0700128 if (!ff->buf)
129 return __do_write_fd(ff, buf, size);
130 return __do_write_buf(ff, buf, size);
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +0200131}
132
David Carrillo-Cisneros2ff53652017-07-17 21:25:36 -0700133/* Return: 0 if succeded, -ERR if failed. */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700134int write_padded(struct feat_fd *ff, const void *bf,
135 size_t count, size_t count_aligned)
Arnaldo Carvalho de Melof92cb242010-01-04 16:19:28 -0200136{
137 static const char zero_buf[NAME_ALIGN];
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700138 int err = do_write(ff, bf, count);
Arnaldo Carvalho de Melof92cb242010-01-04 16:19:28 -0200139
140 if (!err)
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700141 err = do_write(ff, zero_buf, count_aligned - count);
Arnaldo Carvalho de Melof92cb242010-01-04 16:19:28 -0200142
143 return err;
144}
145
Kan Liang2bb00d22015-09-01 09:58:12 -0400146#define string_size(str) \
147 (PERF_ALIGN((strlen(str) + 1), NAME_ALIGN) + sizeof(u32))
148
David Carrillo-Cisneros2ff53652017-07-17 21:25:36 -0700149/* Return: 0 if succeded, -ERR if failed. */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700150static int do_write_string(struct feat_fd *ff, const char *str)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200151{
152 u32 len, olen;
153 int ret;
154
155 olen = strlen(str) + 1;
Irina Tirdea9ac3e482012-09-11 01:15:01 +0300156 len = PERF_ALIGN(olen, NAME_ALIGN);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200157
158 /* write len, incl. \0 */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700159 ret = do_write(ff, &len, sizeof(len));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200160 if (ret < 0)
161 return ret;
162
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700163 return write_padded(ff, str, olen, len);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200164}
165
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -0700166static int __do_read_fd(struct feat_fd *ff, void *addr, ssize_t size)
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -0700167{
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700168 ssize_t ret = readn(ff->fd, addr, size);
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -0700169
170 if (ret != size)
171 return ret < 0 ? (int)ret : -1;
172 return 0;
173}
174
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -0700175static int __do_read_buf(struct feat_fd *ff, void *addr, ssize_t size)
176{
177 if (size > (ssize_t)ff->size - ff->offset)
178 return -1;
179
180 memcpy(addr, ff->buf + ff->offset, size);
181 ff->offset += size;
182
183 return 0;
184
185}
186
187static int __do_read(struct feat_fd *ff, void *addr, ssize_t size)
188{
189 if (!ff->buf)
190 return __do_read_fd(ff, addr, size);
191 return __do_read_buf(ff, addr, size);
192}
193
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700194static int do_read_u32(struct feat_fd *ff, u32 *addr)
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -0700195{
196 int ret;
197
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700198 ret = __do_read(ff, addr, sizeof(*addr));
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -0700199 if (ret)
200 return ret;
201
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700202 if (ff->ph->needs_swap)
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -0700203 *addr = bswap_32(*addr);
204 return 0;
205}
206
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700207static int do_read_u64(struct feat_fd *ff, u64 *addr)
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -0700208{
209 int ret;
210
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700211 ret = __do_read(ff, addr, sizeof(*addr));
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -0700212 if (ret)
213 return ret;
214
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700215 if (ff->ph->needs_swap)
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -0700216 *addr = bswap_64(*addr);
217 return 0;
218}
219
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700220static char *do_read_string(struct feat_fd *ff)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200221{
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200222 u32 len;
223 char *buf;
224
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700225 if (do_read_u32(ff, &len))
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200226 return NULL;
227
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200228 buf = malloc(len);
229 if (!buf)
230 return NULL;
231
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -0700232 if (!__do_read(ff, buf, len)) {
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200233 /*
234 * strings are padded by zeroes
235 * thus the actual strlen of buf
236 * may be less than len
237 */
238 return buf;
239 }
240
241 free(buf);
242 return NULL;
243}
244
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700245static int write_tracing_data(struct feat_fd *ff,
246 struct perf_evlist *evlist)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200247{
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -0700248 if (WARN(ff->buf, "Error: calling %s in pipe-mode.\n", __func__))
249 return -1;
250
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700251 return read_tracing_data(ff->fd, &evlist->entries);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200252}
253
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700254static int write_build_id(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300255 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200256{
257 struct perf_session *session;
258 int err;
259
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700260 session = container_of(ff->ph, struct perf_session, header);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200261
Robert Richtere20960c2011-12-07 10:02:55 +0100262 if (!perf_session__read_build_ids(session, true))
263 return -1;
264
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -0700265 if (WARN(ff->buf, "Error: calling %s in pipe-mode.\n", __func__))
266 return -1;
267
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700268 err = perf_session__write_buildid_table(session, ff);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200269 if (err < 0) {
270 pr_debug("failed to write buildid table\n");
271 return err;
272 }
Namhyung Kim73c5d222014-11-07 22:57:56 +0900273 perf_session__cache_build_ids(session);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200274
275 return 0;
276}
277
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700278static int write_hostname(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300279 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200280{
281 struct utsname uts;
282 int ret;
283
284 ret = uname(&uts);
285 if (ret < 0)
286 return -1;
287
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700288 return do_write_string(ff, uts.nodename);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200289}
290
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700291static int write_osrelease(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300292 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200293{
294 struct utsname uts;
295 int ret;
296
297 ret = uname(&uts);
298 if (ret < 0)
299 return -1;
300
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700301 return do_write_string(ff, uts.release);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200302}
303
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700304static int write_arch(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300305 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200306{
307 struct utsname uts;
308 int ret;
309
310 ret = uname(&uts);
311 if (ret < 0)
312 return -1;
313
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700314 return do_write_string(ff, uts.machine);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200315}
316
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700317static int write_version(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300318 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200319{
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700320 return do_write_string(ff, perf_version_string);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200321}
322
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700323static int __write_cpudesc(struct feat_fd *ff, const char *cpuinfo_proc)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200324{
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200325 FILE *file;
326 char *buf = NULL;
327 char *s, *p;
Wang Nan493c3032014-10-24 09:45:26 +0800328 const char *search = cpuinfo_proc;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200329 size_t len = 0;
330 int ret = -1;
331
332 if (!search)
333 return -1;
334
335 file = fopen("/proc/cpuinfo", "r");
336 if (!file)
337 return -1;
338
339 while (getline(&buf, &len, file) > 0) {
340 ret = strncmp(buf, search, strlen(search));
341 if (!ret)
342 break;
343 }
344
Wang Naned307752014-10-16 11:08:29 +0800345 if (ret) {
346 ret = -1;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200347 goto done;
Wang Naned307752014-10-16 11:08:29 +0800348 }
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200349
350 s = buf;
351
352 p = strchr(buf, ':');
353 if (p && *(p+1) == ' ' && *(p+2))
354 s = p + 2;
355 p = strchr(s, '\n');
356 if (p)
357 *p = '\0';
358
359 /* squash extra space characters (branding string) */
360 p = s;
361 while (*p) {
362 if (isspace(*p)) {
363 char *r = p + 1;
364 char *q = r;
365 *p = ' ';
366 while (*q && isspace(*q))
367 q++;
368 if (q != (p+1))
369 while ((*r++ = *q++));
370 }
371 p++;
372 }
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700373 ret = do_write_string(ff, s);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200374done:
375 free(buf);
376 fclose(file);
377 return ret;
378}
379
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700380static int write_cpudesc(struct feat_fd *ff,
Wang Nan493c3032014-10-24 09:45:26 +0800381 struct perf_evlist *evlist __maybe_unused)
382{
Wang Nan493c3032014-10-24 09:45:26 +0800383 const char *cpuinfo_procs[] = CPUINFO_PROC;
384 unsigned int i;
385
386 for (i = 0; i < ARRAY_SIZE(cpuinfo_procs); i++) {
387 int ret;
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700388 ret = __write_cpudesc(ff, cpuinfo_procs[i]);
Wang Nan493c3032014-10-24 09:45:26 +0800389 if (ret >= 0)
390 return ret;
391 }
392 return -1;
393}
394
395
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700396static int write_nrcpus(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300397 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200398{
399 long nr;
400 u32 nrc, nra;
401 int ret;
402
Jan Stancekda8a58b2017-02-17 12:10:26 +0100403 nrc = cpu__max_present_cpu();
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200404
405 nr = sysconf(_SC_NPROCESSORS_ONLN);
406 if (nr < 0)
407 return -1;
408
409 nra = (u32)(nr & UINT_MAX);
410
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700411 ret = do_write(ff, &nrc, sizeof(nrc));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200412 if (ret < 0)
413 return ret;
414
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700415 return do_write(ff, &nra, sizeof(nra));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200416}
417
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700418static int write_event_desc(struct feat_fd *ff,
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200419 struct perf_evlist *evlist)
420{
Robert Richter6606f872012-08-16 21:10:19 +0200421 struct perf_evsel *evsel;
Namhyung Kim74ba9e12012-09-05 14:02:47 +0900422 u32 nre, nri, sz;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200423 int ret;
424
Namhyung Kim74ba9e12012-09-05 14:02:47 +0900425 nre = evlist->nr_entries;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200426
427 /*
428 * write number of events
429 */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700430 ret = do_write(ff, &nre, sizeof(nre));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200431 if (ret < 0)
432 return ret;
433
434 /*
435 * size of perf_event_attr struct
436 */
Robert Richter6606f872012-08-16 21:10:19 +0200437 sz = (u32)sizeof(evsel->attr);
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700438 ret = do_write(ff, &sz, sizeof(sz));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200439 if (ret < 0)
440 return ret;
441
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -0300442 evlist__for_each_entry(evlist, evsel) {
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700443 ret = do_write(ff, &evsel->attr, sz);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200444 if (ret < 0)
445 return ret;
446 /*
447 * write number of unique id per event
448 * there is one id per instance of an event
449 *
450 * copy into an nri to be independent of the
451 * type of ids,
452 */
Robert Richter6606f872012-08-16 21:10:19 +0200453 nri = evsel->ids;
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700454 ret = do_write(ff, &nri, sizeof(nri));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200455 if (ret < 0)
456 return ret;
457
458 /*
459 * write event string as passed on cmdline
460 */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700461 ret = do_write_string(ff, perf_evsel__name(evsel));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200462 if (ret < 0)
463 return ret;
464 /*
465 * write unique ids for this event
466 */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700467 ret = do_write(ff, evsel->id, evsel->ids * sizeof(u64));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200468 if (ret < 0)
469 return ret;
470 }
471 return 0;
472}
473
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700474static int write_cmdline(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300475 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200476{
477 char buf[MAXPATHLEN];
Arnaldo Carvalho de Melob6998692015-09-08 16:58:20 -0300478 u32 n;
479 int i, ret;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200480
Tommi Rantala55f771282017-03-22 15:06:24 +0200481 /* actual path to perf binary */
482 ret = readlink("/proc/self/exe", buf, sizeof(buf) - 1);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200483 if (ret <= 0)
484 return -1;
485
486 /* readlink() does not add null termination */
487 buf[ret] = '\0';
488
489 /* account for binary path */
Arnaldo Carvalho de Melob6998692015-09-08 16:58:20 -0300490 n = perf_env.nr_cmdline + 1;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200491
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700492 ret = do_write(ff, &n, sizeof(n));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200493 if (ret < 0)
494 return ret;
495
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700496 ret = do_write_string(ff, buf);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200497 if (ret < 0)
498 return ret;
499
Arnaldo Carvalho de Melob6998692015-09-08 16:58:20 -0300500 for (i = 0 ; i < perf_env.nr_cmdline; i++) {
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700501 ret = do_write_string(ff, perf_env.cmdline_argv[i]);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200502 if (ret < 0)
503 return ret;
504 }
505 return 0;
506}
507
508#define CORE_SIB_FMT \
509 "/sys/devices/system/cpu/cpu%d/topology/core_siblings_list"
510#define THRD_SIB_FMT \
511 "/sys/devices/system/cpu/cpu%d/topology/thread_siblings_list"
512
513struct cpu_topo {
Kan Liang2bb00d22015-09-01 09:58:12 -0400514 u32 cpu_nr;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200515 u32 core_sib;
516 u32 thread_sib;
517 char **core_siblings;
518 char **thread_siblings;
519};
520
521static int build_cpu_topo(struct cpu_topo *tp, int cpu)
522{
523 FILE *fp;
524 char filename[MAXPATHLEN];
525 char *buf = NULL, *p;
526 size_t len = 0;
Stephane Eranianc5885742013-08-14 12:04:26 +0200527 ssize_t sret;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200528 u32 i = 0;
529 int ret = -1;
530
531 sprintf(filename, CORE_SIB_FMT, cpu);
532 fp = fopen(filename, "r");
533 if (!fp)
Stephane Eranianc5885742013-08-14 12:04:26 +0200534 goto try_threads;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200535
Stephane Eranianc5885742013-08-14 12:04:26 +0200536 sret = getline(&buf, &len, fp);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200537 fclose(fp);
Stephane Eranianc5885742013-08-14 12:04:26 +0200538 if (sret <= 0)
539 goto try_threads;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200540
541 p = strchr(buf, '\n');
542 if (p)
543 *p = '\0';
544
545 for (i = 0; i < tp->core_sib; i++) {
546 if (!strcmp(buf, tp->core_siblings[i]))
547 break;
548 }
549 if (i == tp->core_sib) {
550 tp->core_siblings[i] = buf;
551 tp->core_sib++;
552 buf = NULL;
553 len = 0;
554 }
Stephane Eranianc5885742013-08-14 12:04:26 +0200555 ret = 0;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200556
Stephane Eranianc5885742013-08-14 12:04:26 +0200557try_threads:
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200558 sprintf(filename, THRD_SIB_FMT, cpu);
559 fp = fopen(filename, "r");
560 if (!fp)
561 goto done;
562
563 if (getline(&buf, &len, fp) <= 0)
564 goto done;
565
566 p = strchr(buf, '\n');
567 if (p)
568 *p = '\0';
569
570 for (i = 0; i < tp->thread_sib; i++) {
571 if (!strcmp(buf, tp->thread_siblings[i]))
572 break;
573 }
574 if (i == tp->thread_sib) {
575 tp->thread_siblings[i] = buf;
576 tp->thread_sib++;
577 buf = NULL;
578 }
579 ret = 0;
580done:
581 if(fp)
582 fclose(fp);
583 free(buf);
584 return ret;
585}
586
587static void free_cpu_topo(struct cpu_topo *tp)
588{
589 u32 i;
590
591 if (!tp)
592 return;
593
594 for (i = 0 ; i < tp->core_sib; i++)
Arnaldo Carvalho de Melo74cf2492013-12-27 16:55:14 -0300595 zfree(&tp->core_siblings[i]);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200596
597 for (i = 0 ; i < tp->thread_sib; i++)
Arnaldo Carvalho de Melo74cf2492013-12-27 16:55:14 -0300598 zfree(&tp->thread_siblings[i]);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200599
600 free(tp);
601}
602
603static struct cpu_topo *build_cpu_topology(void)
604{
Jan Stancek43db2842017-02-17 12:10:25 +0100605 struct cpu_topo *tp = NULL;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200606 void *addr;
607 u32 nr, i;
Arnaldo Carvalho de Meloaa36ddd2015-09-09 10:37:01 -0300608 size_t sz;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200609 long ncpus;
610 int ret = -1;
Jan Stancek43db2842017-02-17 12:10:25 +0100611 struct cpu_map *map;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200612
Jan Stancekda8a58b2017-02-17 12:10:26 +0100613 ncpus = cpu__max_present_cpu();
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200614
Jan Stancek43db2842017-02-17 12:10:25 +0100615 /* build online CPU map */
616 map = cpu_map__new(NULL);
617 if (map == NULL) {
618 pr_debug("failed to get system cpumap\n");
619 return NULL;
620 }
621
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200622 nr = (u32)(ncpus & UINT_MAX);
623
624 sz = nr * sizeof(char *);
Arnaldo Carvalho de Meloaa36ddd2015-09-09 10:37:01 -0300625 addr = calloc(1, sizeof(*tp) + 2 * sz);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200626 if (!addr)
Jan Stancek43db2842017-02-17 12:10:25 +0100627 goto out_free;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200628
629 tp = addr;
Kan Liang2bb00d22015-09-01 09:58:12 -0400630 tp->cpu_nr = nr;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200631 addr += sizeof(*tp);
632 tp->core_siblings = addr;
633 addr += sz;
634 tp->thread_siblings = addr;
635
636 for (i = 0; i < nr; i++) {
Jan Stancek43db2842017-02-17 12:10:25 +0100637 if (!cpu_map__has(map, i))
638 continue;
639
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200640 ret = build_cpu_topo(tp, i);
641 if (ret < 0)
642 break;
643 }
Jan Stancek43db2842017-02-17 12:10:25 +0100644
645out_free:
646 cpu_map__put(map);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200647 if (ret) {
648 free_cpu_topo(tp);
649 tp = NULL;
650 }
651 return tp;
652}
653
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700654static int write_cpu_topology(struct feat_fd *ff,
655 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200656{
657 struct cpu_topo *tp;
658 u32 i;
Arnaldo Carvalho de Meloaa36ddd2015-09-09 10:37:01 -0300659 int ret, j;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200660
661 tp = build_cpu_topology();
662 if (!tp)
663 return -1;
664
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700665 ret = do_write(ff, &tp->core_sib, sizeof(tp->core_sib));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200666 if (ret < 0)
667 goto done;
668
669 for (i = 0; i < tp->core_sib; i++) {
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700670 ret = do_write_string(ff, tp->core_siblings[i]);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200671 if (ret < 0)
672 goto done;
673 }
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700674 ret = do_write(ff, &tp->thread_sib, sizeof(tp->thread_sib));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200675 if (ret < 0)
676 goto done;
677
678 for (i = 0; i < tp->thread_sib; i++) {
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700679 ret = do_write_string(ff, tp->thread_siblings[i]);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200680 if (ret < 0)
681 break;
682 }
Kan Liang2bb00d22015-09-01 09:58:12 -0400683
Arnaldo Carvalho de Meloaa36ddd2015-09-09 10:37:01 -0300684 ret = perf_env__read_cpu_topology_map(&perf_env);
685 if (ret < 0)
686 goto done;
687
688 for (j = 0; j < perf_env.nr_cpus_avail; j++) {
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700689 ret = do_write(ff, &perf_env.cpu[j].core_id,
Arnaldo Carvalho de Meloaa36ddd2015-09-09 10:37:01 -0300690 sizeof(perf_env.cpu[j].core_id));
Kan Liang2bb00d22015-09-01 09:58:12 -0400691 if (ret < 0)
692 return ret;
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700693 ret = do_write(ff, &perf_env.cpu[j].socket_id,
Arnaldo Carvalho de Meloaa36ddd2015-09-09 10:37:01 -0300694 sizeof(perf_env.cpu[j].socket_id));
Kan Liang2bb00d22015-09-01 09:58:12 -0400695 if (ret < 0)
696 return ret;
697 }
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200698done:
699 free_cpu_topo(tp);
700 return ret;
701}
702
703
704
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700705static int write_total_mem(struct feat_fd *ff,
706 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200707{
708 char *buf = NULL;
709 FILE *fp;
710 size_t len = 0;
711 int ret = -1, n;
712 uint64_t mem;
713
714 fp = fopen("/proc/meminfo", "r");
715 if (!fp)
716 return -1;
717
718 while (getline(&buf, &len, fp) > 0) {
719 ret = strncmp(buf, "MemTotal:", 9);
720 if (!ret)
721 break;
722 }
723 if (!ret) {
724 n = sscanf(buf, "%*s %"PRIu64, &mem);
725 if (n == 1)
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700726 ret = do_write(ff, &mem, sizeof(mem));
Wang Naned307752014-10-16 11:08:29 +0800727 } else
728 ret = -1;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200729 free(buf);
730 fclose(fp);
731 return ret;
732}
733
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700734static int write_topo_node(struct feat_fd *ff, int node)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200735{
736 char str[MAXPATHLEN];
737 char field[32];
738 char *buf = NULL, *p;
739 size_t len = 0;
740 FILE *fp;
741 u64 mem_total, mem_free, mem;
742 int ret = -1;
743
744 sprintf(str, "/sys/devices/system/node/node%d/meminfo", node);
745 fp = fopen(str, "r");
746 if (!fp)
747 return -1;
748
749 while (getline(&buf, &len, fp) > 0) {
750 /* skip over invalid lines */
751 if (!strchr(buf, ':'))
752 continue;
Alan Coxa761a2d2014-01-20 19:10:11 +0100753 if (sscanf(buf, "%*s %*d %31s %"PRIu64, field, &mem) != 2)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200754 goto done;
755 if (!strcmp(field, "MemTotal:"))
756 mem_total = mem;
757 if (!strcmp(field, "MemFree:"))
758 mem_free = mem;
759 }
760
761 fclose(fp);
Thomas Jarosch5809fde2013-01-28 10:21:14 +0100762 fp = NULL;
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200763
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700764 ret = do_write(ff, &mem_total, sizeof(u64));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200765 if (ret)
766 goto done;
767
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700768 ret = do_write(ff, &mem_free, sizeof(u64));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200769 if (ret)
770 goto done;
771
772 ret = -1;
773 sprintf(str, "/sys/devices/system/node/node%d/cpulist", node);
774
775 fp = fopen(str, "r");
776 if (!fp)
777 goto done;
778
779 if (getline(&buf, &len, fp) <= 0)
780 goto done;
781
782 p = strchr(buf, '\n');
783 if (p)
784 *p = '\0';
785
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700786 ret = do_write_string(ff, buf);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200787done:
788 free(buf);
Thomas Jarosch5809fde2013-01-28 10:21:14 +0100789 if (fp)
790 fclose(fp);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200791 return ret;
792}
793
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700794static int write_numa_topology(struct feat_fd *ff,
795 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200796{
797 char *buf = NULL;
798 size_t len = 0;
799 FILE *fp;
800 struct cpu_map *node_map = NULL;
801 char *c;
802 u32 nr, i, j;
803 int ret = -1;
804
805 fp = fopen("/sys/devices/system/node/online", "r");
806 if (!fp)
807 return -1;
808
809 if (getline(&buf, &len, fp) <= 0)
810 goto done;
811
812 c = strchr(buf, '\n');
813 if (c)
814 *c = '\0';
815
816 node_map = cpu_map__new(buf);
817 if (!node_map)
818 goto done;
819
820 nr = (u32)node_map->nr;
821
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700822 ret = do_write(ff, &nr, sizeof(nr));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200823 if (ret < 0)
824 goto done;
825
826 for (i = 0; i < nr; i++) {
827 j = (u32)node_map->map[i];
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700828 ret = do_write(ff, &j, sizeof(j));
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200829 if (ret < 0)
830 break;
831
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700832 ret = write_topo_node(ff, i);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200833 if (ret < 0)
834 break;
835 }
836done:
837 free(buf);
838 fclose(fp);
Masami Hiramatsu5191d8872015-12-09 11:11:35 +0900839 cpu_map__put(node_map);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200840 return ret;
841}
842
843/*
Robert Richter50a96672012-08-16 21:10:24 +0200844 * File format:
845 *
846 * struct pmu_mappings {
847 * u32 pmu_num;
848 * struct pmu_map {
849 * u32 type;
850 * char name[];
851 * }[pmu_num];
852 * };
853 */
854
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700855static int write_pmu_mappings(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300856 struct perf_evlist *evlist __maybe_unused)
Robert Richter50a96672012-08-16 21:10:24 +0200857{
858 struct perf_pmu *pmu = NULL;
David Carrillo-Cisnerosa02c3952017-07-17 21:25:44 -0700859 u32 pmu_num = 0;
Namhyung Kim5323f602012-12-17 15:38:54 +0900860 int ret;
Robert Richter50a96672012-08-16 21:10:24 +0200861
David Carrillo-Cisnerosa02c3952017-07-17 21:25:44 -0700862 /*
863 * Do a first pass to count number of pmu to avoid lseek so this
864 * works in pipe mode as well.
865 */
866 while ((pmu = perf_pmu__scan(pmu))) {
867 if (!pmu->name)
868 continue;
869 pmu_num++;
870 }
871
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700872 ret = do_write(ff, &pmu_num, sizeof(pmu_num));
Namhyung Kim5323f602012-12-17 15:38:54 +0900873 if (ret < 0)
874 return ret;
Robert Richter50a96672012-08-16 21:10:24 +0200875
876 while ((pmu = perf_pmu__scan(pmu))) {
877 if (!pmu->name)
878 continue;
Namhyung Kim5323f602012-12-17 15:38:54 +0900879
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700880 ret = do_write(ff, &pmu->type, sizeof(pmu->type));
Namhyung Kim5323f602012-12-17 15:38:54 +0900881 if (ret < 0)
882 return ret;
883
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700884 ret = do_write_string(ff, pmu->name);
Namhyung Kim5323f602012-12-17 15:38:54 +0900885 if (ret < 0)
886 return ret;
Robert Richter50a96672012-08-16 21:10:24 +0200887 }
888
Robert Richter50a96672012-08-16 21:10:24 +0200889 return 0;
890}
891
892/*
Namhyung Kima8bb5592013-01-22 18:09:31 +0900893 * File format:
894 *
895 * struct group_descs {
896 * u32 nr_groups;
897 * struct group_desc {
898 * char name[];
899 * u32 leader_idx;
900 * u32 nr_members;
901 * }[nr_groups];
902 * };
903 */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700904static int write_group_desc(struct feat_fd *ff,
Namhyung Kima8bb5592013-01-22 18:09:31 +0900905 struct perf_evlist *evlist)
906{
907 u32 nr_groups = evlist->nr_groups;
908 struct perf_evsel *evsel;
909 int ret;
910
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700911 ret = do_write(ff, &nr_groups, sizeof(nr_groups));
Namhyung Kima8bb5592013-01-22 18:09:31 +0900912 if (ret < 0)
913 return ret;
914
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -0300915 evlist__for_each_entry(evlist, evsel) {
Namhyung Kima8bb5592013-01-22 18:09:31 +0900916 if (perf_evsel__is_group_leader(evsel) &&
917 evsel->nr_members > 1) {
918 const char *name = evsel->group_name ?: "{anon_group}";
919 u32 leader_idx = evsel->idx;
920 u32 nr_members = evsel->nr_members;
921
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700922 ret = do_write_string(ff, name);
Namhyung Kima8bb5592013-01-22 18:09:31 +0900923 if (ret < 0)
924 return ret;
925
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700926 ret = do_write(ff, &leader_idx, sizeof(leader_idx));
Namhyung Kima8bb5592013-01-22 18:09:31 +0900927 if (ret < 0)
928 return ret;
929
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700930 ret = do_write(ff, &nr_members, sizeof(nr_members));
Namhyung Kima8bb5592013-01-22 18:09:31 +0900931 if (ret < 0)
932 return ret;
933 }
934 }
935 return 0;
936}
937
938/*
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200939 * default get_cpuid(): nothing gets recorded
Jiada Wang7a759cd2017-04-09 20:02:37 -0700940 * actual implementation must be in arch/$(SRCARCH)/util/header.c
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200941 */
Rui Teng11d8f872016-07-28 10:05:57 +0800942int __weak get_cpuid(char *buffer __maybe_unused, size_t sz __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200943{
944 return -1;
945}
946
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700947static int write_cpuid(struct feat_fd *ff,
Irina Tirdea1d037ca2012-09-11 01:15:03 +0300948 struct perf_evlist *evlist __maybe_unused)
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200949{
950 char buffer[64];
951 int ret;
952
953 ret = get_cpuid(buffer, sizeof(buffer));
954 if (!ret)
955 goto write_it;
956
957 return -1;
958write_it:
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700959 return do_write_string(ff, buffer);
Stephane Eranianfbe96f22011-09-30 15:40:40 +0200960}
961
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700962static int write_branch_stack(struct feat_fd *ff __maybe_unused,
963 struct perf_evlist *evlist __maybe_unused)
Stephane Eranian330aa672012-03-08 23:47:46 +0100964{
965 return 0;
966}
967
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700968static int write_auxtrace(struct feat_fd *ff,
Adrian Hunter4025ea42015-04-09 18:53:41 +0300969 struct perf_evlist *evlist __maybe_unused)
970{
Adrian Hunter99fa2982015-04-30 17:37:25 +0300971 struct perf_session *session;
972 int err;
973
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -0700974 if (WARN(ff->buf, "Error: calling %s in pipe-mode.\n", __func__))
975 return -1;
976
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700977 session = container_of(ff->ph, struct perf_session, header);
Adrian Hunter99fa2982015-04-30 17:37:25 +0300978
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -0700979 err = auxtrace_index__write(ff->fd, &session->auxtrace_index);
Adrian Hunter99fa2982015-04-30 17:37:25 +0300980 if (err < 0)
981 pr_err("Failed to write auxtrace index\n");
982 return err;
Adrian Hunter4025ea42015-04-09 18:53:41 +0300983}
984
Jiri Olsa720e98b2016-02-16 16:01:43 +0100985static int cpu_cache_level__sort(const void *a, const void *b)
986{
987 struct cpu_cache_level *cache_a = (struct cpu_cache_level *)a;
988 struct cpu_cache_level *cache_b = (struct cpu_cache_level *)b;
989
990 return cache_a->level - cache_b->level;
991}
992
993static bool cpu_cache_level__cmp(struct cpu_cache_level *a, struct cpu_cache_level *b)
994{
995 if (a->level != b->level)
996 return false;
997
998 if (a->line_size != b->line_size)
999 return false;
1000
1001 if (a->sets != b->sets)
1002 return false;
1003
1004 if (a->ways != b->ways)
1005 return false;
1006
1007 if (strcmp(a->type, b->type))
1008 return false;
1009
1010 if (strcmp(a->size, b->size))
1011 return false;
1012
1013 if (strcmp(a->map, b->map))
1014 return false;
1015
1016 return true;
1017}
1018
1019static int cpu_cache_level__read(struct cpu_cache_level *cache, u32 cpu, u16 level)
1020{
1021 char path[PATH_MAX], file[PATH_MAX];
1022 struct stat st;
1023 size_t len;
1024
1025 scnprintf(path, PATH_MAX, "devices/system/cpu/cpu%d/cache/index%d/", cpu, level);
1026 scnprintf(file, PATH_MAX, "%s/%s", sysfs__mountpoint(), path);
1027
1028 if (stat(file, &st))
1029 return 1;
1030
1031 scnprintf(file, PATH_MAX, "%s/level", path);
1032 if (sysfs__read_int(file, (int *) &cache->level))
1033 return -1;
1034
1035 scnprintf(file, PATH_MAX, "%s/coherency_line_size", path);
1036 if (sysfs__read_int(file, (int *) &cache->line_size))
1037 return -1;
1038
1039 scnprintf(file, PATH_MAX, "%s/number_of_sets", path);
1040 if (sysfs__read_int(file, (int *) &cache->sets))
1041 return -1;
1042
1043 scnprintf(file, PATH_MAX, "%s/ways_of_associativity", path);
1044 if (sysfs__read_int(file, (int *) &cache->ways))
1045 return -1;
1046
1047 scnprintf(file, PATH_MAX, "%s/type", path);
1048 if (sysfs__read_str(file, &cache->type, &len))
1049 return -1;
1050
1051 cache->type[len] = 0;
1052 cache->type = rtrim(cache->type);
1053
1054 scnprintf(file, PATH_MAX, "%s/size", path);
1055 if (sysfs__read_str(file, &cache->size, &len)) {
1056 free(cache->type);
1057 return -1;
1058 }
1059
1060 cache->size[len] = 0;
1061 cache->size = rtrim(cache->size);
1062
1063 scnprintf(file, PATH_MAX, "%s/shared_cpu_list", path);
1064 if (sysfs__read_str(file, &cache->map, &len)) {
1065 free(cache->map);
1066 free(cache->type);
1067 return -1;
1068 }
1069
1070 cache->map[len] = 0;
1071 cache->map = rtrim(cache->map);
1072 return 0;
1073}
1074
1075static void cpu_cache_level__fprintf(FILE *out, struct cpu_cache_level *c)
1076{
1077 fprintf(out, "L%d %-15s %8s [%s]\n", c->level, c->type, c->size, c->map);
1078}
1079
1080static int build_caches(struct cpu_cache_level caches[], u32 size, u32 *cntp)
1081{
1082 u32 i, cnt = 0;
1083 long ncpus;
1084 u32 nr, cpu;
1085 u16 level;
1086
1087 ncpus = sysconf(_SC_NPROCESSORS_CONF);
1088 if (ncpus < 0)
1089 return -1;
1090
1091 nr = (u32)(ncpus & UINT_MAX);
1092
1093 for (cpu = 0; cpu < nr; cpu++) {
1094 for (level = 0; level < 10; level++) {
1095 struct cpu_cache_level c;
1096 int err;
1097
1098 err = cpu_cache_level__read(&c, cpu, level);
1099 if (err < 0)
1100 return err;
1101
1102 if (err == 1)
1103 break;
1104
1105 for (i = 0; i < cnt; i++) {
1106 if (cpu_cache_level__cmp(&c, &caches[i]))
1107 break;
1108 }
1109
1110 if (i == cnt)
1111 caches[cnt++] = c;
1112 else
1113 cpu_cache_level__free(&c);
1114
1115 if (WARN_ONCE(cnt == size, "way too many cpu caches.."))
1116 goto out;
1117 }
1118 }
1119 out:
1120 *cntp = cnt;
1121 return 0;
1122}
1123
1124#define MAX_CACHES 2000
1125
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07001126static int write_cache(struct feat_fd *ff,
1127 struct perf_evlist *evlist __maybe_unused)
Jiri Olsa720e98b2016-02-16 16:01:43 +01001128{
1129 struct cpu_cache_level caches[MAX_CACHES];
1130 u32 cnt = 0, i, version = 1;
1131 int ret;
1132
1133 ret = build_caches(caches, MAX_CACHES, &cnt);
1134 if (ret)
1135 goto out;
1136
1137 qsort(&caches, cnt, sizeof(struct cpu_cache_level), cpu_cache_level__sort);
1138
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07001139 ret = do_write(ff, &version, sizeof(u32));
Jiri Olsa720e98b2016-02-16 16:01:43 +01001140 if (ret < 0)
1141 goto out;
1142
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07001143 ret = do_write(ff, &cnt, sizeof(u32));
Jiri Olsa720e98b2016-02-16 16:01:43 +01001144 if (ret < 0)
1145 goto out;
1146
1147 for (i = 0; i < cnt; i++) {
1148 struct cpu_cache_level *c = &caches[i];
1149
1150 #define _W(v) \
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07001151 ret = do_write(ff, &c->v, sizeof(u32)); \
Jiri Olsa720e98b2016-02-16 16:01:43 +01001152 if (ret < 0) \
1153 goto out;
1154
1155 _W(level)
1156 _W(line_size)
1157 _W(sets)
1158 _W(ways)
1159 #undef _W
1160
1161 #define _W(v) \
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07001162 ret = do_write_string(ff, (const char *) c->v); \
Jiri Olsa720e98b2016-02-16 16:01:43 +01001163 if (ret < 0) \
1164 goto out;
1165
1166 _W(type)
1167 _W(size)
1168 _W(map)
1169 #undef _W
1170 }
1171
1172out:
1173 for (i = 0; i < cnt; i++)
1174 cpu_cache_level__free(&caches[i]);
1175 return ret;
1176}
1177
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07001178static int write_stat(struct feat_fd *ff __maybe_unused,
Jiri Olsaffa517a2015-10-25 15:51:43 +01001179 struct perf_evlist *evlist __maybe_unused)
1180{
1181 return 0;
1182}
1183
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001184static void print_hostname(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001185{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001186 fprintf(fp, "# hostname : %s\n", ff->ph->env.hostname);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001187}
1188
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001189static void print_osrelease(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001190{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001191 fprintf(fp, "# os release : %s\n", ff->ph->env.os_release);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001192}
1193
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001194static void print_arch(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001195{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001196 fprintf(fp, "# arch : %s\n", ff->ph->env.arch);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001197}
1198
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001199static void print_cpudesc(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001200{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001201 fprintf(fp, "# cpudesc : %s\n", ff->ph->env.cpu_desc);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001202}
1203
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001204static void print_nrcpus(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001205{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001206 fprintf(fp, "# nrcpus online : %u\n", ff->ph->env.nr_cpus_online);
1207 fprintf(fp, "# nrcpus avail : %u\n", ff->ph->env.nr_cpus_avail);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001208}
1209
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001210static void print_version(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001211{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001212 fprintf(fp, "# perf version : %s\n", ff->ph->env.version);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001213}
1214
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001215static void print_cmdline(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001216{
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001217 int nr, i;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001218
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001219 nr = ff->ph->env.nr_cmdline;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001220
1221 fprintf(fp, "# cmdline : ");
1222
Jiri Olsa768dd3f2015-07-21 14:31:31 +02001223 for (i = 0; i < nr; i++)
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001224 fprintf(fp, "%s ", ff->ph->env.cmdline_argv[i]);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001225 fputc('\n', fp);
1226}
1227
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001228static void print_cpu_topology(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001229{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001230 struct perf_header *ph = ff->ph;
1231 int cpu_nr = ph->env.nr_cpus_avail;
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001232 int nr, i;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001233 char *str;
1234
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001235 nr = ph->env.nr_sibling_cores;
1236 str = ph->env.sibling_cores;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001237
1238 for (i = 0; i < nr; i++) {
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001239 fprintf(fp, "# sibling cores : %s\n", str);
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001240 str += strlen(str) + 1;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001241 }
1242
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001243 nr = ph->env.nr_sibling_threads;
1244 str = ph->env.sibling_threads;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001245
1246 for (i = 0; i < nr; i++) {
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001247 fprintf(fp, "# sibling threads : %s\n", str);
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001248 str += strlen(str) + 1;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001249 }
Kan Liang2bb00d22015-09-01 09:58:12 -04001250
1251 if (ph->env.cpu != NULL) {
1252 for (i = 0; i < cpu_nr; i++)
1253 fprintf(fp, "# CPU %d: Core ID %d, Socket ID %d\n", i,
1254 ph->env.cpu[i].core_id, ph->env.cpu[i].socket_id);
1255 } else
1256 fprintf(fp, "# Core ID and Socket ID information is not available\n");
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001257}
1258
Robert Richter4e1b9c62012-08-16 21:10:22 +02001259static void free_event_desc(struct perf_evsel *events)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001260{
Robert Richter4e1b9c62012-08-16 21:10:22 +02001261 struct perf_evsel *evsel;
1262
1263 if (!events)
1264 return;
1265
1266 for (evsel = events; evsel->attr.size; evsel++) {
Arnaldo Carvalho de Melo74cf2492013-12-27 16:55:14 -03001267 zfree(&evsel->name);
1268 zfree(&evsel->id);
Robert Richter4e1b9c62012-08-16 21:10:22 +02001269 }
1270
1271 free(events);
1272}
1273
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001274static struct perf_evsel *read_event_desc(struct feat_fd *ff)
Robert Richter4e1b9c62012-08-16 21:10:22 +02001275{
1276 struct perf_evsel *evsel, *events = NULL;
1277 u64 *id;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001278 void *buf = NULL;
Stephane Eranian62db9062012-02-09 23:21:07 +01001279 u32 nre, sz, nr, i, j;
Stephane Eranian62db9062012-02-09 23:21:07 +01001280 size_t msz;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001281
1282 /* number of events */
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001283 if (do_read_u32(ff, &nre))
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001284 goto error;
1285
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001286 if (do_read_u32(ff, &sz))
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001287 goto error;
1288
Stephane Eranian62db9062012-02-09 23:21:07 +01001289 /* buffer to hold on file attr struct */
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001290 buf = malloc(sz);
1291 if (!buf)
1292 goto error;
1293
Robert Richter4e1b9c62012-08-16 21:10:22 +02001294 /* the last event terminates with evsel->attr.size == 0: */
1295 events = calloc(nre + 1, sizeof(*events));
1296 if (!events)
1297 goto error;
1298
1299 msz = sizeof(evsel->attr);
Jiri Olsa9fafd982012-03-20 19:15:39 +01001300 if (sz < msz)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001301 msz = sz;
1302
Robert Richter4e1b9c62012-08-16 21:10:22 +02001303 for (i = 0, evsel = events; i < nre; evsel++, i++) {
1304 evsel->idx = i;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001305
Stephane Eranian62db9062012-02-09 23:21:07 +01001306 /*
1307 * must read entire on-file attr struct to
1308 * sync up with layout.
1309 */
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001310 if (__do_read(ff, buf, sz))
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001311 goto error;
1312
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001313 if (ff->ph->needs_swap)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001314 perf_event__attr_swap(buf);
1315
Robert Richter4e1b9c62012-08-16 21:10:22 +02001316 memcpy(&evsel->attr, buf, msz);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001317
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001318 if (do_read_u32(ff, &nr))
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001319 goto error;
1320
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001321 if (ff->ph->needs_swap)
Arnaldo Carvalho de Melo0807d2d2012-09-26 12:48:18 -03001322 evsel->needs_swap = true;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001323
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001324 evsel->name = do_read_string(ff);
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -07001325 if (!evsel->name)
1326 goto error;
Robert Richter4e1b9c62012-08-16 21:10:22 +02001327
1328 if (!nr)
1329 continue;
1330
1331 id = calloc(nr, sizeof(*id));
1332 if (!id)
1333 goto error;
1334 evsel->ids = nr;
1335 evsel->id = id;
1336
1337 for (j = 0 ; j < nr; j++) {
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001338 if (do_read_u64(ff, id))
Robert Richter4e1b9c62012-08-16 21:10:22 +02001339 goto error;
Robert Richter4e1b9c62012-08-16 21:10:22 +02001340 id++;
1341 }
1342 }
1343out:
Arnaldo Carvalho de Melo04662522013-12-26 17:41:15 -03001344 free(buf);
Robert Richter4e1b9c62012-08-16 21:10:22 +02001345 return events;
1346error:
Markus Elfring4cc97612015-06-25 17:12:32 +02001347 free_event_desc(events);
Robert Richter4e1b9c62012-08-16 21:10:22 +02001348 events = NULL;
1349 goto out;
1350}
1351
Peter Zijlstra2c5e8c52015-04-07 11:09:54 +02001352static int __desc_attr__fprintf(FILE *fp, const char *name, const char *val,
Arnaldo Carvalho de Melo03536312017-06-16 12:18:27 -03001353 void *priv __maybe_unused)
Peter Zijlstra2c5e8c52015-04-07 11:09:54 +02001354{
1355 return fprintf(fp, ", %s = %s", name, val);
1356}
1357
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001358static void print_event_desc(struct feat_fd *ff, FILE *fp)
Robert Richter4e1b9c62012-08-16 21:10:22 +02001359{
David Carrillo-Cisnerosf9ebdcc2017-07-17 21:25:49 -07001360 struct perf_evsel *evsel, *events;
Robert Richter4e1b9c62012-08-16 21:10:22 +02001361 u32 j;
1362 u64 *id;
1363
David Carrillo-Cisnerosf9ebdcc2017-07-17 21:25:49 -07001364 if (ff->events)
1365 events = ff->events;
1366 else
1367 events = read_event_desc(ff);
1368
Robert Richter4e1b9c62012-08-16 21:10:22 +02001369 if (!events) {
1370 fprintf(fp, "# event desc: not available or unable to read\n");
1371 return;
1372 }
1373
1374 for (evsel = events; evsel->attr.size; evsel++) {
1375 fprintf(fp, "# event : name = %s, ", evsel->name);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001376
Robert Richter4e1b9c62012-08-16 21:10:22 +02001377 if (evsel->ids) {
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001378 fprintf(fp, ", id = {");
Robert Richter4e1b9c62012-08-16 21:10:22 +02001379 for (j = 0, id = evsel->id; j < evsel->ids; j++, id++) {
1380 if (j)
1381 fputc(',', fp);
1382 fprintf(fp, " %"PRIu64, *id);
1383 }
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001384 fprintf(fp, " }");
Robert Richter4e1b9c62012-08-16 21:10:22 +02001385 }
Peter Zijlstra814c8c32015-03-31 00:19:31 +02001386
Peter Zijlstra2c5e8c52015-04-07 11:09:54 +02001387 perf_event_attr__fprintf(fp, &evsel->attr, __desc_attr__fprintf, NULL);
Robert Richter4e1b9c62012-08-16 21:10:22 +02001388
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001389 fputc('\n', fp);
1390 }
Robert Richter4e1b9c62012-08-16 21:10:22 +02001391
1392 free_event_desc(events);
David Carrillo-Cisnerosf9ebdcc2017-07-17 21:25:49 -07001393 ff->events = NULL;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001394}
1395
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001396static void print_total_mem(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001397{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001398 fprintf(fp, "# total memory : %llu kB\n", ff->ph->env.total_mem);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001399}
1400
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001401static void print_numa_topology(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001402{
Jiri Olsac60da222016-07-04 14:16:20 +02001403 int i;
1404 struct numa_node *n;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001405
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001406 for (i = 0; i < ff->ph->env.nr_numa_nodes; i++) {
1407 n = &ff->ph->env.numa_nodes[i];
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001408
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001409 fprintf(fp, "# node%u meminfo : total = %"PRIu64" kB,"
1410 " free = %"PRIu64" kB\n",
Jiri Olsac60da222016-07-04 14:16:20 +02001411 n->node, n->mem_total, n->mem_free);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001412
Jiri Olsac60da222016-07-04 14:16:20 +02001413 fprintf(fp, "# node%u cpu list : ", n->node);
1414 cpu_map__fprintf(n->map, fp);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001415 }
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001416}
1417
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001418static void print_cpuid(struct feat_fd *ff, FILE *fp)
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001419{
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001420 fprintf(fp, "# cpuid : %s\n", ff->ph->env.cpuid);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02001421}
1422
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001423static void print_branch_stack(struct feat_fd *ff __maybe_unused, FILE *fp)
Stephane Eranian330aa672012-03-08 23:47:46 +01001424{
1425 fprintf(fp, "# contains samples with branch stack\n");
1426}
1427
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001428static void print_auxtrace(struct feat_fd *ff __maybe_unused, FILE *fp)
Adrian Hunter4025ea42015-04-09 18:53:41 +03001429{
1430 fprintf(fp, "# contains AUX area data (e.g. instruction trace)\n");
1431}
1432
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001433static void print_stat(struct feat_fd *ff __maybe_unused, FILE *fp)
Jiri Olsaffa517a2015-10-25 15:51:43 +01001434{
1435 fprintf(fp, "# contains stat data\n");
1436}
1437
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001438static void print_cache(struct feat_fd *ff, FILE *fp __maybe_unused)
Jiri Olsa720e98b2016-02-16 16:01:43 +01001439{
1440 int i;
1441
1442 fprintf(fp, "# CPU cache info:\n");
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001443 for (i = 0; i < ff->ph->env.caches_cnt; i++) {
Jiri Olsa720e98b2016-02-16 16:01:43 +01001444 fprintf(fp, "# ");
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001445 cpu_cache_level__fprintf(fp, &ff->ph->env.caches[i]);
Jiri Olsa720e98b2016-02-16 16:01:43 +01001446 }
1447}
1448
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001449static void print_pmu_mappings(struct feat_fd *ff, FILE *fp)
Robert Richter50a96672012-08-16 21:10:24 +02001450{
1451 const char *delimiter = "# pmu mappings: ";
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001452 char *str, *tmp;
Robert Richter50a96672012-08-16 21:10:24 +02001453 u32 pmu_num;
1454 u32 type;
1455
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001456 pmu_num = ff->ph->env.nr_pmu_mappings;
Robert Richter50a96672012-08-16 21:10:24 +02001457 if (!pmu_num) {
1458 fprintf(fp, "# pmu mappings: not available\n");
1459 return;
1460 }
1461
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001462 str = ff->ph->env.pmu_mappings;
Namhyung Kimbe4a2de2012-09-05 14:02:49 +09001463
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001464 while (pmu_num) {
1465 type = strtoul(str, &tmp, 0);
1466 if (*tmp != ':')
1467 goto error;
1468
1469 str = tmp + 1;
1470 fprintf(fp, "%s%s = %" PRIu32, delimiter, str, type);
1471
Robert Richter50a96672012-08-16 21:10:24 +02001472 delimiter = ", ";
Namhyung Kim7e94cfc2012-09-24 17:15:00 +09001473 str += strlen(str) + 1;
1474 pmu_num--;
Robert Richter50a96672012-08-16 21:10:24 +02001475 }
1476
1477 fprintf(fp, "\n");
1478
1479 if (!pmu_num)
1480 return;
1481error:
1482 fprintf(fp, "# pmu mappings: unable to read\n");
1483}
1484
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001485static void print_group_desc(struct feat_fd *ff, FILE *fp)
Namhyung Kima8bb5592013-01-22 18:09:31 +09001486{
1487 struct perf_session *session;
1488 struct perf_evsel *evsel;
1489 u32 nr = 0;
1490
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07001491 session = container_of(ff->ph, struct perf_session, header);
Namhyung Kima8bb5592013-01-22 18:09:31 +09001492
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -03001493 evlist__for_each_entry(session->evlist, evsel) {
Namhyung Kima8bb5592013-01-22 18:09:31 +09001494 if (perf_evsel__is_group_leader(evsel) &&
1495 evsel->nr_members > 1) {
1496 fprintf(fp, "# group: %s{%s", evsel->group_name ?: "",
1497 perf_evsel__name(evsel));
1498
1499 nr = evsel->nr_members - 1;
1500 } else if (nr) {
1501 fprintf(fp, ",%s", perf_evsel__name(evsel));
1502
1503 if (--nr == 0)
1504 fprintf(fp, "}\n");
1505 }
1506 }
1507}
1508
Robert Richter08d95bd2012-02-10 15:41:55 +01001509static int __event_process_build_id(struct build_id_event *bev,
1510 char *filename,
1511 struct perf_session *session)
1512{
1513 int err = -1;
Robert Richter08d95bd2012-02-10 15:41:55 +01001514 struct machine *machine;
Wang Nan1f121b02015-06-03 08:52:21 +00001515 u16 cpumode;
Robert Richter08d95bd2012-02-10 15:41:55 +01001516 struct dso *dso;
1517 enum dso_kernel_type dso_type;
1518
1519 machine = perf_session__findnew_machine(session, bev->pid);
1520 if (!machine)
1521 goto out;
1522
Wang Nan1f121b02015-06-03 08:52:21 +00001523 cpumode = bev->header.misc & PERF_RECORD_MISC_CPUMODE_MASK;
Robert Richter08d95bd2012-02-10 15:41:55 +01001524
Wang Nan1f121b02015-06-03 08:52:21 +00001525 switch (cpumode) {
Robert Richter08d95bd2012-02-10 15:41:55 +01001526 case PERF_RECORD_MISC_KERNEL:
1527 dso_type = DSO_TYPE_KERNEL;
Robert Richter08d95bd2012-02-10 15:41:55 +01001528 break;
1529 case PERF_RECORD_MISC_GUEST_KERNEL:
1530 dso_type = DSO_TYPE_GUEST_KERNEL;
Robert Richter08d95bd2012-02-10 15:41:55 +01001531 break;
1532 case PERF_RECORD_MISC_USER:
1533 case PERF_RECORD_MISC_GUEST_USER:
1534 dso_type = DSO_TYPE_USER;
Robert Richter08d95bd2012-02-10 15:41:55 +01001535 break;
1536 default:
1537 goto out;
1538 }
1539
Arnaldo Carvalho de Meloaa7cc2a2015-05-29 11:31:12 -03001540 dso = machine__findnew_dso(machine, filename);
Robert Richter08d95bd2012-02-10 15:41:55 +01001541 if (dso != NULL) {
Masami Hiramatsub5d8bbe2016-05-11 22:51:59 +09001542 char sbuild_id[SBUILD_ID_SIZE];
Robert Richter08d95bd2012-02-10 15:41:55 +01001543
1544 dso__set_build_id(dso, &bev->build_id);
1545
Namhyung Kim1deec1b2017-05-31 21:01:03 +09001546 if (dso_type != DSO_TYPE_USER) {
1547 struct kmod_path m = { .name = NULL, };
1548
1549 if (!kmod_path__parse_name(&m, filename) && m.kmod)
Namhyung Kim6b335e82017-05-31 21:01:04 +09001550 dso__set_module_info(dso, &m, machine);
Namhyung Kim1deec1b2017-05-31 21:01:03 +09001551 else
1552 dso->kernel = dso_type;
1553
1554 free(m.name);
1555 }
Robert Richter08d95bd2012-02-10 15:41:55 +01001556
1557 build_id__sprintf(dso->build_id, sizeof(dso->build_id),
1558 sbuild_id);
1559 pr_debug("build id event received for %s: %s\n",
1560 dso->long_name, sbuild_id);
Arnaldo Carvalho de Melod3a7c482015-06-02 11:53:26 -03001561 dso__put(dso);
Robert Richter08d95bd2012-02-10 15:41:55 +01001562 }
1563
1564 err = 0;
1565out:
1566 return err;
1567}
1568
1569static int perf_header__read_build_ids_abi_quirk(struct perf_header *header,
1570 int input, u64 offset, u64 size)
1571{
1572 struct perf_session *session = container_of(header, struct perf_session, header);
1573 struct {
1574 struct perf_event_header header;
Irina Tirdea9ac3e482012-09-11 01:15:01 +03001575 u8 build_id[PERF_ALIGN(BUILD_ID_SIZE, sizeof(u64))];
Robert Richter08d95bd2012-02-10 15:41:55 +01001576 char filename[0];
1577 } old_bev;
1578 struct build_id_event bev;
1579 char filename[PATH_MAX];
1580 u64 limit = offset + size;
1581
1582 while (offset < limit) {
1583 ssize_t len;
1584
Namhyung Kim5323f602012-12-17 15:38:54 +09001585 if (readn(input, &old_bev, sizeof(old_bev)) != sizeof(old_bev))
Robert Richter08d95bd2012-02-10 15:41:55 +01001586 return -1;
1587
1588 if (header->needs_swap)
1589 perf_event_header__bswap(&old_bev.header);
1590
1591 len = old_bev.header.size - sizeof(old_bev);
Namhyung Kim5323f602012-12-17 15:38:54 +09001592 if (readn(input, filename, len) != len)
Robert Richter08d95bd2012-02-10 15:41:55 +01001593 return -1;
1594
1595 bev.header = old_bev.header;
1596
1597 /*
1598 * As the pid is the missing value, we need to fill
1599 * it properly. The header.misc value give us nice hint.
1600 */
1601 bev.pid = HOST_KERNEL_ID;
1602 if (bev.header.misc == PERF_RECORD_MISC_GUEST_USER ||
1603 bev.header.misc == PERF_RECORD_MISC_GUEST_KERNEL)
1604 bev.pid = DEFAULT_GUEST_KERNEL_ID;
1605
1606 memcpy(bev.build_id, old_bev.build_id, sizeof(bev.build_id));
1607 __event_process_build_id(&bev, filename, session);
1608
1609 offset += bev.header.size;
1610 }
1611
1612 return 0;
1613}
1614
1615static int perf_header__read_build_ids(struct perf_header *header,
1616 int input, u64 offset, u64 size)
1617{
1618 struct perf_session *session = container_of(header, struct perf_session, header);
1619 struct build_id_event bev;
1620 char filename[PATH_MAX];
1621 u64 limit = offset + size, orig_offset = offset;
1622 int err = -1;
1623
1624 while (offset < limit) {
1625 ssize_t len;
1626
Namhyung Kim5323f602012-12-17 15:38:54 +09001627 if (readn(input, &bev, sizeof(bev)) != sizeof(bev))
Robert Richter08d95bd2012-02-10 15:41:55 +01001628 goto out;
1629
1630 if (header->needs_swap)
1631 perf_event_header__bswap(&bev.header);
1632
1633 len = bev.header.size - sizeof(bev);
Namhyung Kim5323f602012-12-17 15:38:54 +09001634 if (readn(input, filename, len) != len)
Robert Richter08d95bd2012-02-10 15:41:55 +01001635 goto out;
1636 /*
1637 * The a1645ce1 changeset:
1638 *
1639 * "perf: 'perf kvm' tool for monitoring guest performance from host"
1640 *
1641 * Added a field to struct build_id_event that broke the file
1642 * format.
1643 *
1644 * Since the kernel build-id is the first entry, process the
1645 * table using the old format if the well known
1646 * '[kernel.kallsyms]' string for the kernel build-id has the
1647 * first 4 characters chopped off (where the pid_t sits).
1648 */
1649 if (memcmp(filename, "nel.kallsyms]", 13) == 0) {
1650 if (lseek(input, orig_offset, SEEK_SET) == (off_t)-1)
1651 return -1;
1652 return perf_header__read_build_ids_abi_quirk(header, input, offset, size);
1653 }
1654
1655 __event_process_build_id(&bev, filename, session);
1656
1657 offset += bev.header.size;
1658 }
1659 err = 0;
1660out:
1661 return err;
1662}
1663
David Carrillo-Cisnerosdfaa1582017-07-17 21:25:35 -07001664/* Macro for features that simply need to read and store a string. */
1665#define FEAT_PROCESS_STR_FUN(__feat, __feat_env) \
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001666static int process_##__feat(struct feat_fd *ff, void *data __maybe_unused) \
David Carrillo-Cisnerosdfaa1582017-07-17 21:25:35 -07001667{\
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001668 ff->ph->env.__feat_env = do_read_string(ff); \
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001669 return ff->ph->env.__feat_env ? 0 : -ENOMEM; \
David Carrillo-Cisnerosdfaa1582017-07-17 21:25:35 -07001670}
1671
1672FEAT_PROCESS_STR_FUN(hostname, hostname);
1673FEAT_PROCESS_STR_FUN(osrelease, os_release);
1674FEAT_PROCESS_STR_FUN(version, version);
1675FEAT_PROCESS_STR_FUN(arch, arch);
1676FEAT_PROCESS_STR_FUN(cpudesc, cpu_desc);
1677FEAT_PROCESS_STR_FUN(cpuid, cpuid);
1678
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001679static int process_tracing_data(struct feat_fd *ff, void *data)
Robert Richterf1c67db2012-02-10 15:41:56 +01001680{
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001681 ssize_t ret = trace_report(ff->fd, data, false);
1682
Namhyung Kim3dce2ce2013-03-21 16:18:48 +09001683 return ret < 0 ? -1 : 0;
Robert Richterf1c67db2012-02-10 15:41:56 +01001684}
1685
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001686static int process_build_id(struct feat_fd *ff, void *data __maybe_unused)
Robert Richterf1c67db2012-02-10 15:41:56 +01001687{
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001688 if (perf_header__read_build_ids(ff->ph, ff->fd, ff->offset, ff->size))
Robert Richterf1c67db2012-02-10 15:41:56 +01001689 pr_debug("Failed to read buildids, continuing...\n");
1690 return 0;
1691}
1692
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001693static int process_nrcpus(struct feat_fd *ff, void *data __maybe_unused)
Namhyung Kima1ae5652012-09-24 17:14:59 +09001694{
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -07001695 int ret;
1696 u32 nr_cpus_avail, nr_cpus_online;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001697
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001698 ret = do_read_u32(ff, &nr_cpus_avail);
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -07001699 if (ret)
1700 return ret;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001701
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001702 ret = do_read_u32(ff, &nr_cpus_online);
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -07001703 if (ret)
1704 return ret;
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001705 ff->ph->env.nr_cpus_avail = (int)nr_cpus_avail;
1706 ff->ph->env.nr_cpus_online = (int)nr_cpus_online;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001707 return 0;
1708}
1709
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001710static int process_total_mem(struct feat_fd *ff, void *data __maybe_unused)
Namhyung Kima1ae5652012-09-24 17:14:59 +09001711{
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -07001712 u64 total_mem;
1713 int ret;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001714
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001715 ret = do_read_u64(ff, &total_mem);
David Carrillo-Cisneros6200e4942017-07-17 21:25:34 -07001716 if (ret)
Namhyung Kima1ae5652012-09-24 17:14:59 +09001717 return -1;
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001718 ff->ph->env.total_mem = (unsigned long long)total_mem;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001719 return 0;
1720}
1721
Robert Richter7c2f7af2012-08-16 21:10:23 +02001722static struct perf_evsel *
1723perf_evlist__find_by_index(struct perf_evlist *evlist, int idx)
1724{
1725 struct perf_evsel *evsel;
1726
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -03001727 evlist__for_each_entry(evlist, evsel) {
Robert Richter7c2f7af2012-08-16 21:10:23 +02001728 if (evsel->idx == idx)
1729 return evsel;
1730 }
1731
1732 return NULL;
1733}
1734
1735static void
Namhyung Kim3d7eb862012-09-24 17:15:01 +09001736perf_evlist__set_event_name(struct perf_evlist *evlist,
1737 struct perf_evsel *event)
Robert Richter7c2f7af2012-08-16 21:10:23 +02001738{
1739 struct perf_evsel *evsel;
1740
1741 if (!event->name)
1742 return;
1743
1744 evsel = perf_evlist__find_by_index(evlist, event->idx);
1745 if (!evsel)
1746 return;
1747
1748 if (evsel->name)
1749 return;
1750
1751 evsel->name = strdup(event->name);
1752}
1753
1754static int
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001755process_event_desc(struct feat_fd *ff, void *data __maybe_unused)
Robert Richter7c2f7af2012-08-16 21:10:23 +02001756{
Namhyung Kim3d7eb862012-09-24 17:15:01 +09001757 struct perf_session *session;
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001758 struct perf_evsel *evsel, *events = read_event_desc(ff);
Robert Richter7c2f7af2012-08-16 21:10:23 +02001759
1760 if (!events)
1761 return 0;
1762
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001763 session = container_of(ff->ph, struct perf_session, header);
David Carrillo-Cisnerosf9ebdcc2017-07-17 21:25:49 -07001764
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01001765 if (session->data->is_pipe) {
David Carrillo-Cisnerosf9ebdcc2017-07-17 21:25:49 -07001766 /* Save events for reading later by print_event_desc,
1767 * since they can't be read again in pipe mode. */
1768 ff->events = events;
1769 }
1770
Robert Richter7c2f7af2012-08-16 21:10:23 +02001771 for (evsel = events; evsel->attr.size; evsel++)
1772 perf_evlist__set_event_name(session->evlist, evsel);
1773
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01001774 if (!session->data->is_pipe)
David Carrillo-Cisnerosf9ebdcc2017-07-17 21:25:49 -07001775 free_event_desc(events);
Robert Richter7c2f7af2012-08-16 21:10:23 +02001776
1777 return 0;
1778}
1779
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001780static int process_cmdline(struct feat_fd *ff, void *data __maybe_unused)
Namhyung Kima1ae5652012-09-24 17:14:59 +09001781{
Jiri Olsa768dd3f2015-07-21 14:31:31 +02001782 char *str, *cmdline = NULL, **argv = NULL;
1783 u32 nr, i, len = 0;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001784
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001785 if (do_read_u32(ff, &nr))
Namhyung Kima1ae5652012-09-24 17:14:59 +09001786 return -1;
1787
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001788 ff->ph->env.nr_cmdline = nr;
Jiri Olsa768dd3f2015-07-21 14:31:31 +02001789
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001790 cmdline = zalloc(ff->size + nr + 1);
Jiri Olsa768dd3f2015-07-21 14:31:31 +02001791 if (!cmdline)
1792 return -1;
1793
1794 argv = zalloc(sizeof(char *) * (nr + 1));
1795 if (!argv)
1796 goto error;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001797
1798 for (i = 0; i < nr; i++) {
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001799 str = do_read_string(ff);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001800 if (!str)
1801 goto error;
1802
Jiri Olsa768dd3f2015-07-21 14:31:31 +02001803 argv[i] = cmdline + len;
1804 memcpy(argv[i], str, strlen(str) + 1);
1805 len += strlen(str) + 1;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001806 free(str);
1807 }
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001808 ff->ph->env.cmdline = cmdline;
1809 ff->ph->env.cmdline_argv = (const char **) argv;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001810 return 0;
1811
1812error:
Jiri Olsa768dd3f2015-07-21 14:31:31 +02001813 free(argv);
1814 free(cmdline);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001815 return -1;
1816}
1817
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001818static int process_cpu_topology(struct feat_fd *ff, void *data __maybe_unused)
Namhyung Kima1ae5652012-09-24 17:14:59 +09001819{
Namhyung Kima1ae5652012-09-24 17:14:59 +09001820 u32 nr, i;
1821 char *str;
1822 struct strbuf sb;
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001823 int cpu_nr = ff->ph->env.nr_cpus_avail;
Kan Liang2bb00d22015-09-01 09:58:12 -04001824 u64 size = 0;
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001825 struct perf_header *ph = ff->ph;
Kan Liang2bb00d22015-09-01 09:58:12 -04001826
1827 ph->env.cpu = calloc(cpu_nr, sizeof(*ph->env.cpu));
1828 if (!ph->env.cpu)
1829 return -1;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001830
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001831 if (do_read_u32(ff, &nr))
Kan Liang2bb00d22015-09-01 09:58:12 -04001832 goto free_cpu;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001833
Namhyung Kima1ae5652012-09-24 17:14:59 +09001834 ph->env.nr_sibling_cores = nr;
Kan Liang2bb00d22015-09-01 09:58:12 -04001835 size += sizeof(u32);
Masami Hiramatsu642aada2016-05-10 14:47:35 +09001836 if (strbuf_init(&sb, 128) < 0)
1837 goto free_cpu;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001838
1839 for (i = 0; i < nr; i++) {
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001840 str = do_read_string(ff);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001841 if (!str)
1842 goto error;
1843
1844 /* include a NULL character at the end */
Masami Hiramatsu642aada2016-05-10 14:47:35 +09001845 if (strbuf_add(&sb, str, strlen(str) + 1) < 0)
1846 goto error;
Kan Liang2bb00d22015-09-01 09:58:12 -04001847 size += string_size(str);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001848 free(str);
1849 }
1850 ph->env.sibling_cores = strbuf_detach(&sb, NULL);
1851
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001852 if (do_read_u32(ff, &nr))
Namhyung Kima1ae5652012-09-24 17:14:59 +09001853 return -1;
1854
Namhyung Kima1ae5652012-09-24 17:14:59 +09001855 ph->env.nr_sibling_threads = nr;
Kan Liang2bb00d22015-09-01 09:58:12 -04001856 size += sizeof(u32);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001857
1858 for (i = 0; i < nr; i++) {
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001859 str = do_read_string(ff);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001860 if (!str)
1861 goto error;
1862
1863 /* include a NULL character at the end */
Masami Hiramatsu642aada2016-05-10 14:47:35 +09001864 if (strbuf_add(&sb, str, strlen(str) + 1) < 0)
1865 goto error;
Kan Liang2bb00d22015-09-01 09:58:12 -04001866 size += string_size(str);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001867 free(str);
1868 }
1869 ph->env.sibling_threads = strbuf_detach(&sb, NULL);
Kan Liang2bb00d22015-09-01 09:58:12 -04001870
1871 /*
1872 * The header may be from old perf,
1873 * which doesn't include core id and socket id information.
1874 */
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001875 if (ff->size <= size) {
Kan Liang2bb00d22015-09-01 09:58:12 -04001876 zfree(&ph->env.cpu);
1877 return 0;
1878 }
1879
1880 for (i = 0; i < (u32)cpu_nr; i++) {
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001881 if (do_read_u32(ff, &nr))
Kan Liang2bb00d22015-09-01 09:58:12 -04001882 goto free_cpu;
1883
Kan Liang2bb00d22015-09-01 09:58:12 -04001884 ph->env.cpu[i].core_id = nr;
1885
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001886 if (do_read_u32(ff, &nr))
Kan Liang2bb00d22015-09-01 09:58:12 -04001887 goto free_cpu;
1888
Jan Stancekda8a58b2017-02-17 12:10:26 +01001889 if (nr != (u32)-1 && nr > (u32)cpu_nr) {
Kan Liang2bb00d22015-09-01 09:58:12 -04001890 pr_debug("socket_id number is too big."
1891 "You may need to upgrade the perf tool.\n");
1892 goto free_cpu;
1893 }
1894
1895 ph->env.cpu[i].socket_id = nr;
1896 }
1897
Namhyung Kima1ae5652012-09-24 17:14:59 +09001898 return 0;
1899
1900error:
1901 strbuf_release(&sb);
Kan Liang2bb00d22015-09-01 09:58:12 -04001902free_cpu:
1903 zfree(&ph->env.cpu);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001904 return -1;
1905}
1906
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001907static int process_numa_topology(struct feat_fd *ff, void *data __maybe_unused)
Namhyung Kima1ae5652012-09-24 17:14:59 +09001908{
Jiri Olsac60da222016-07-04 14:16:20 +02001909 struct numa_node *nodes, *n;
Jiri Olsac60da222016-07-04 14:16:20 +02001910 u32 nr, i;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001911 char *str;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001912
1913 /* nr nodes */
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001914 if (do_read_u32(ff, &nr))
Masami Hiramatsu642aada2016-05-10 14:47:35 +09001915 return -1;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001916
Jiri Olsac60da222016-07-04 14:16:20 +02001917 nodes = zalloc(sizeof(*nodes) * nr);
1918 if (!nodes)
1919 return -ENOMEM;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001920
1921 for (i = 0; i < nr; i++) {
Jiri Olsac60da222016-07-04 14:16:20 +02001922 n = &nodes[i];
1923
Namhyung Kima1ae5652012-09-24 17:14:59 +09001924 /* node number */
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001925 if (do_read_u32(ff, &n->node))
Namhyung Kima1ae5652012-09-24 17:14:59 +09001926 goto error;
1927
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001928 if (do_read_u64(ff, &n->mem_total))
Namhyung Kima1ae5652012-09-24 17:14:59 +09001929 goto error;
1930
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001931 if (do_read_u64(ff, &n->mem_free))
Namhyung Kima1ae5652012-09-24 17:14:59 +09001932 goto error;
1933
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001934 str = do_read_string(ff);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001935 if (!str)
1936 goto error;
1937
Jiri Olsac60da222016-07-04 14:16:20 +02001938 n->map = cpu_map__new(str);
1939 if (!n->map)
Masami Hiramatsu642aada2016-05-10 14:47:35 +09001940 goto error;
Jiri Olsac60da222016-07-04 14:16:20 +02001941
Namhyung Kima1ae5652012-09-24 17:14:59 +09001942 free(str);
1943 }
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001944 ff->ph->env.nr_numa_nodes = nr;
1945 ff->ph->env.numa_nodes = nodes;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001946 return 0;
1947
1948error:
Jiri Olsac60da222016-07-04 14:16:20 +02001949 free(nodes);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001950 return -1;
1951}
1952
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07001953static int process_pmu_mappings(struct feat_fd *ff, void *data __maybe_unused)
Namhyung Kima1ae5652012-09-24 17:14:59 +09001954{
Namhyung Kima1ae5652012-09-24 17:14:59 +09001955 char *name;
1956 u32 pmu_num;
1957 u32 type;
1958 struct strbuf sb;
1959
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001960 if (do_read_u32(ff, &pmu_num))
Namhyung Kima1ae5652012-09-24 17:14:59 +09001961 return -1;
1962
Namhyung Kima1ae5652012-09-24 17:14:59 +09001963 if (!pmu_num) {
1964 pr_debug("pmu mappings not available\n");
1965 return 0;
1966 }
1967
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001968 ff->ph->env.nr_pmu_mappings = pmu_num;
Masami Hiramatsu642aada2016-05-10 14:47:35 +09001969 if (strbuf_init(&sb, 128) < 0)
1970 return -1;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001971
1972 while (pmu_num) {
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001973 if (do_read_u32(ff, &type))
Namhyung Kima1ae5652012-09-24 17:14:59 +09001974 goto error;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001975
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07001976 name = do_read_string(ff);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001977 if (!name)
1978 goto error;
1979
Masami Hiramatsu642aada2016-05-10 14:47:35 +09001980 if (strbuf_addf(&sb, "%u:%s", type, name) < 0)
1981 goto error;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001982 /* include a NULL character at the end */
Masami Hiramatsu642aada2016-05-10 14:47:35 +09001983 if (strbuf_add(&sb, "", 1) < 0)
1984 goto error;
Namhyung Kima1ae5652012-09-24 17:14:59 +09001985
Kan Liange0838e02015-09-10 11:03:05 -03001986 if (!strcmp(name, "msr"))
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001987 ff->ph->env.msr_pmu_type = type;
Kan Liange0838e02015-09-10 11:03:05 -03001988
Namhyung Kima1ae5652012-09-24 17:14:59 +09001989 free(name);
1990 pmu_num--;
1991 }
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07001992 ff->ph->env.pmu_mappings = strbuf_detach(&sb, NULL);
Namhyung Kima1ae5652012-09-24 17:14:59 +09001993 return 0;
1994
1995error:
1996 strbuf_release(&sb);
1997 return -1;
1998}
1999
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07002000static int process_group_desc(struct feat_fd *ff, void *data __maybe_unused)
Namhyung Kima8bb5592013-01-22 18:09:31 +09002001{
2002 size_t ret = -1;
2003 u32 i, nr, nr_groups;
2004 struct perf_session *session;
2005 struct perf_evsel *evsel, *leader = NULL;
2006 struct group_desc {
2007 char *name;
2008 u32 leader_idx;
2009 u32 nr_members;
2010 } *desc;
2011
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07002012 if (do_read_u32(ff, &nr_groups))
Namhyung Kima8bb5592013-01-22 18:09:31 +09002013 return -1;
2014
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002015 ff->ph->env.nr_groups = nr_groups;
Namhyung Kima8bb5592013-01-22 18:09:31 +09002016 if (!nr_groups) {
2017 pr_debug("group desc not available\n");
2018 return 0;
2019 }
2020
2021 desc = calloc(nr_groups, sizeof(*desc));
2022 if (!desc)
2023 return -1;
2024
2025 for (i = 0; i < nr_groups; i++) {
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07002026 desc[i].name = do_read_string(ff);
Namhyung Kima8bb5592013-01-22 18:09:31 +09002027 if (!desc[i].name)
2028 goto out_free;
2029
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07002030 if (do_read_u32(ff, &desc[i].leader_idx))
Namhyung Kima8bb5592013-01-22 18:09:31 +09002031 goto out_free;
2032
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07002033 if (do_read_u32(ff, &desc[i].nr_members))
Namhyung Kima8bb5592013-01-22 18:09:31 +09002034 goto out_free;
Namhyung Kima8bb5592013-01-22 18:09:31 +09002035 }
2036
2037 /*
2038 * Rebuild group relationship based on the group_desc
2039 */
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002040 session = container_of(ff->ph, struct perf_session, header);
Namhyung Kima8bb5592013-01-22 18:09:31 +09002041 session->evlist->nr_groups = nr_groups;
2042
2043 i = nr = 0;
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -03002044 evlist__for_each_entry(session->evlist, evsel) {
Namhyung Kima8bb5592013-01-22 18:09:31 +09002045 if (evsel->idx == (int) desc[i].leader_idx) {
2046 evsel->leader = evsel;
2047 /* {anon_group} is a dummy name */
Namhyung Kim210e8122013-11-18 11:20:43 +09002048 if (strcmp(desc[i].name, "{anon_group}")) {
Namhyung Kima8bb5592013-01-22 18:09:31 +09002049 evsel->group_name = desc[i].name;
Namhyung Kim210e8122013-11-18 11:20:43 +09002050 desc[i].name = NULL;
2051 }
Namhyung Kima8bb5592013-01-22 18:09:31 +09002052 evsel->nr_members = desc[i].nr_members;
2053
2054 if (i >= nr_groups || nr > 0) {
2055 pr_debug("invalid group desc\n");
2056 goto out_free;
2057 }
2058
2059 leader = evsel;
2060 nr = evsel->nr_members - 1;
2061 i++;
2062 } else if (nr) {
2063 /* This is a group member */
2064 evsel->leader = leader;
2065
2066 nr--;
2067 }
2068 }
2069
2070 if (i != nr_groups || nr != 0) {
2071 pr_debug("invalid group desc\n");
2072 goto out_free;
2073 }
2074
2075 ret = 0;
2076out_free:
Namhyung Kim50a27402013-11-18 11:20:44 +09002077 for (i = 0; i < nr_groups; i++)
Arnaldo Carvalho de Melo74cf2492013-12-27 16:55:14 -03002078 zfree(&desc[i].name);
Namhyung Kima8bb5592013-01-22 18:09:31 +09002079 free(desc);
2080
2081 return ret;
2082}
2083
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07002084static int process_auxtrace(struct feat_fd *ff, void *data __maybe_unused)
Adrian Hunter99fa2982015-04-30 17:37:25 +03002085{
2086 struct perf_session *session;
2087 int err;
2088
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002089 session = container_of(ff->ph, struct perf_session, header);
Adrian Hunter99fa2982015-04-30 17:37:25 +03002090
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07002091 err = auxtrace_index__process(ff->fd, ff->size, session,
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002092 ff->ph->needs_swap);
Adrian Hunter99fa2982015-04-30 17:37:25 +03002093 if (err < 0)
2094 pr_err("Failed to process auxtrace index\n");
2095 return err;
2096}
2097
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07002098static int process_cache(struct feat_fd *ff, void *data __maybe_unused)
Jiri Olsa720e98b2016-02-16 16:01:43 +01002099{
2100 struct cpu_cache_level *caches;
2101 u32 cnt, i, version;
2102
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07002103 if (do_read_u32(ff, &version))
Jiri Olsa720e98b2016-02-16 16:01:43 +01002104 return -1;
2105
Jiri Olsa720e98b2016-02-16 16:01:43 +01002106 if (version != 1)
2107 return -1;
2108
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07002109 if (do_read_u32(ff, &cnt))
Jiri Olsa720e98b2016-02-16 16:01:43 +01002110 return -1;
2111
Jiri Olsa720e98b2016-02-16 16:01:43 +01002112 caches = zalloc(sizeof(*caches) * cnt);
2113 if (!caches)
2114 return -1;
2115
2116 for (i = 0; i < cnt; i++) {
2117 struct cpu_cache_level c;
2118
2119 #define _R(v) \
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07002120 if (do_read_u32(ff, &c.v))\
Jiri Olsa720e98b2016-02-16 16:01:43 +01002121 goto out_free_caches; \
Jiri Olsa720e98b2016-02-16 16:01:43 +01002122
2123 _R(level)
2124 _R(line_size)
2125 _R(sets)
2126 _R(ways)
2127 #undef _R
2128
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002129 #define _R(v) \
David Carrillo-Cisneros48e5fce2017-07-17 21:25:43 -07002130 c.v = do_read_string(ff); \
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002131 if (!c.v) \
Jiri Olsa720e98b2016-02-16 16:01:43 +01002132 goto out_free_caches;
2133
2134 _R(type)
2135 _R(size)
2136 _R(map)
2137 #undef _R
2138
2139 caches[i] = c;
2140 }
2141
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002142 ff->ph->env.caches = caches;
2143 ff->ph->env.caches_cnt = cnt;
Jiri Olsa720e98b2016-02-16 16:01:43 +01002144 return 0;
2145out_free_caches:
2146 free(caches);
2147 return -1;
2148}
2149
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002150struct feature_ops {
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002151 int (*write)(struct feat_fd *ff, struct perf_evlist *evlist);
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07002152 void (*print)(struct feat_fd *ff, FILE *fp);
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07002153 int (*process)(struct feat_fd *ff, void *data);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002154 const char *name;
2155 bool full_only;
David Carrillo-Cisnerosa4d8c982017-07-17 21:25:46 -07002156 bool synthesize;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002157};
2158
David Carrillo-Cisnerosa4d8c982017-07-17 21:25:46 -07002159#define FEAT_OPR(n, func, __full_only) \
2160 [HEADER_##n] = { \
2161 .name = __stringify(n), \
2162 .write = write_##func, \
2163 .print = print_##func, \
2164 .full_only = __full_only, \
2165 .process = process_##func, \
2166 .synthesize = true \
2167 }
2168
2169#define FEAT_OPN(n, func, __full_only) \
2170 [HEADER_##n] = { \
2171 .name = __stringify(n), \
2172 .write = write_##func, \
2173 .print = print_##func, \
2174 .full_only = __full_only, \
2175 .process = process_##func \
2176 }
Robert Richter8cdfa782011-12-07 10:02:56 +01002177
2178/* feature_ops not implemented: */
Stephane Eranian2eeaaa02012-05-15 13:28:13 +02002179#define print_tracing_data NULL
2180#define print_build_id NULL
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002181
David Carrillo-Cisnerosa4d8c982017-07-17 21:25:46 -07002182#define process_branch_stack NULL
2183#define process_stat NULL
2184
2185
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002186static const struct feature_ops feat_ops[HEADER_LAST_FEATURE] = {
David Carrillo-Cisnerosa4d8c982017-07-17 21:25:46 -07002187 FEAT_OPN(TRACING_DATA, tracing_data, false),
2188 FEAT_OPN(BUILD_ID, build_id, false),
2189 FEAT_OPR(HOSTNAME, hostname, false),
2190 FEAT_OPR(OSRELEASE, osrelease, false),
2191 FEAT_OPR(VERSION, version, false),
2192 FEAT_OPR(ARCH, arch, false),
2193 FEAT_OPR(NRCPUS, nrcpus, false),
2194 FEAT_OPR(CPUDESC, cpudesc, false),
2195 FEAT_OPR(CPUID, cpuid, false),
2196 FEAT_OPR(TOTAL_MEM, total_mem, false),
2197 FEAT_OPR(EVENT_DESC, event_desc, false),
2198 FEAT_OPR(CMDLINE, cmdline, false),
2199 FEAT_OPR(CPU_TOPOLOGY, cpu_topology, true),
2200 FEAT_OPR(NUMA_TOPOLOGY, numa_topology, true),
2201 FEAT_OPN(BRANCH_STACK, branch_stack, false),
2202 FEAT_OPR(PMU_MAPPINGS, pmu_mappings, false),
2203 FEAT_OPN(GROUP_DESC, group_desc, false),
2204 FEAT_OPN(AUXTRACE, auxtrace, false),
2205 FEAT_OPN(STAT, stat, false),
2206 FEAT_OPN(CACHE, cache, true),
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002207};
2208
2209struct header_print_data {
2210 FILE *fp;
2211 bool full; /* extended list of headers */
2212};
2213
2214static int perf_file_section__fprintf_info(struct perf_file_section *section,
2215 struct perf_header *ph,
2216 int feat, int fd, void *data)
2217{
2218 struct header_print_data *hd = data;
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07002219 struct feat_fd ff;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002220
2221 if (lseek(fd, section->offset, SEEK_SET) == (off_t)-1) {
2222 pr_debug("Failed to lseek to %" PRIu64 " offset for feature "
2223 "%d, continuing...\n", section->offset, feat);
2224 return 0;
2225 }
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002226 if (feat >= HEADER_LAST_FEATURE) {
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002227 pr_warning("unknown feature %d\n", feat);
Robert Richterf7a8a132011-12-07 10:02:51 +01002228 return 0;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002229 }
2230 if (!feat_ops[feat].print)
2231 return 0;
2232
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07002233 ff = (struct feat_fd) {
2234 .fd = fd,
2235 .ph = ph,
2236 };
2237
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002238 if (!feat_ops[feat].full_only || hd->full)
David Carrillo-Cisneroscfc65422017-07-17 21:25:40 -07002239 feat_ops[feat].print(&ff, hd->fp);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002240 else
2241 fprintf(hd->fp, "# %s info available, use -I to display\n",
2242 feat_ops[feat].name);
2243
2244 return 0;
2245}
2246
2247int perf_header__fprintf_info(struct perf_session *session, FILE *fp, bool full)
2248{
2249 struct header_print_data hd;
2250 struct perf_header *header = &session->header;
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01002251 int fd = perf_data__fd(session->data);
Jiri Olsaf45f5612016-10-10 09:03:07 +02002252 struct stat st;
Jiri Olsaaabae162016-10-10 09:35:50 +02002253 int ret, bit;
Jiri Olsaf45f5612016-10-10 09:03:07 +02002254
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002255 hd.fp = fp;
2256 hd.full = full;
2257
Jiri Olsaf45f5612016-10-10 09:03:07 +02002258 ret = fstat(fd, &st);
2259 if (ret == -1)
2260 return -1;
2261
2262 fprintf(fp, "# captured on: %s", ctime(&st.st_ctime));
2263
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002264 perf_header__process_sections(header, fd, &hd,
2265 perf_file_section__fprintf_info);
Jiri Olsaaabae162016-10-10 09:35:50 +02002266
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01002267 if (session->data->is_pipe)
David Carrillo-Cisnerosc9d1c932017-04-10 13:14:32 -07002268 return 0;
2269
Jiri Olsaaabae162016-10-10 09:35:50 +02002270 fprintf(fp, "# missing features: ");
2271 for_each_clear_bit(bit, header->adds_features, HEADER_LAST_FEATURE) {
2272 if (bit)
2273 fprintf(fp, "%s ", feat_ops[bit].name);
2274 }
2275
2276 fprintf(fp, "\n");
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002277 return 0;
2278}
2279
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002280static int do_write_feat(struct feat_fd *ff, int type,
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002281 struct perf_file_section **p,
2282 struct perf_evlist *evlist)
2283{
2284 int err;
2285 int ret = 0;
2286
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002287 if (perf_header__has_feat(ff->ph, type)) {
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002288 if (!feat_ops[type].write)
2289 return -1;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002290
David Carrillo-Cisneros0b3d3412017-07-17 21:25:45 -07002291 if (WARN(ff->buf, "Error: calling %s in pipe-mode.\n", __func__))
2292 return -1;
2293
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002294 (*p)->offset = lseek(ff->fd, 0, SEEK_CUR);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002295
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002296 err = feat_ops[type].write(ff, evlist);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002297 if (err < 0) {
Jiri Olsa0c2aff42016-10-10 09:38:02 +02002298 pr_debug("failed to write feature %s\n", feat_ops[type].name);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002299
2300 /* undo anything written */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002301 lseek(ff->fd, (*p)->offset, SEEK_SET);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002302
2303 return -1;
2304 }
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002305 (*p)->size = lseek(ff->fd, 0, SEEK_CUR) - (*p)->offset;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002306 (*p)++;
2307 }
2308 return ret;
2309}
2310
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002311static int perf_header__adds_write(struct perf_header *header,
Arnaldo Carvalho de Melo361c99a2011-01-11 20:56:53 -02002312 struct perf_evlist *evlist, int fd)
Frederic Weisbecker2ba08252009-10-17 17:12:34 +02002313{
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002314 int nr_sections;
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002315 struct feat_fd ff;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002316 struct perf_file_section *feat_sec, *p;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002317 int sec_size;
2318 u64 sec_start;
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002319 int feat;
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002320 int err;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002321
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002322 ff = (struct feat_fd){
2323 .fd = fd,
2324 .ph = header,
2325 };
2326
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002327 nr_sections = bitmap_weight(header->adds_features, HEADER_FEAT_BITS);
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002328 if (!nr_sections)
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002329 return 0;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002330
Paul Gortmaker91b98802013-01-30 20:05:49 -05002331 feat_sec = p = calloc(nr_sections, sizeof(*feat_sec));
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002332 if (feat_sec == NULL)
2333 return -ENOMEM;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002334
2335 sec_size = sizeof(*feat_sec) * nr_sections;
2336
Jiri Olsa8d541e92013-07-17 19:49:44 +02002337 sec_start = header->feat_offset;
Xiao Guangrongf887f302010-02-04 16:46:42 +08002338 lseek(fd, sec_start + sec_size, SEEK_SET);
Frederic Weisbecker2ba08252009-10-17 17:12:34 +02002339
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002340 for_each_set_bit(feat, header->adds_features, HEADER_FEAT_BITS) {
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002341 if (do_write_feat(&ff, feat, &p, evlist))
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002342 perf_header__clear_feat(header, feat);
2343 }
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002344
Xiao Guangrongf887f302010-02-04 16:46:42 +08002345 lseek(fd, sec_start, SEEK_SET);
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002346 /*
2347 * may write more than needed due to dropped feature, but
2348 * this is okay, reader will skip the mising entries
2349 */
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002350 err = do_write(&ff, feat_sec, sec_size);
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002351 if (err < 0)
2352 pr_debug("failed to write feature section\n");
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002353 free(feat_sec);
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002354 return err;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002355}
Frederic Weisbecker2ba08252009-10-17 17:12:34 +02002356
Tom Zanussi8dc58102010-04-01 23:59:15 -05002357int perf_header__write_pipe(int fd)
2358{
2359 struct perf_pipe_file_header f_header;
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002360 struct feat_fd ff;
Tom Zanussi8dc58102010-04-01 23:59:15 -05002361 int err;
2362
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002363 ff = (struct feat_fd){ .fd = fd };
2364
Tom Zanussi8dc58102010-04-01 23:59:15 -05002365 f_header = (struct perf_pipe_file_header){
2366 .magic = PERF_MAGIC,
2367 .size = sizeof(f_header),
2368 };
2369
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002370 err = do_write(&ff, &f_header, sizeof(f_header));
Tom Zanussi8dc58102010-04-01 23:59:15 -05002371 if (err < 0) {
2372 pr_debug("failed to write perf pipe header\n");
2373 return err;
2374 }
2375
2376 return 0;
2377}
2378
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002379int perf_session__write_header(struct perf_session *session,
2380 struct perf_evlist *evlist,
2381 int fd, bool at_exit)
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002382{
2383 struct perf_file_header f_header;
2384 struct perf_file_attr f_attr;
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002385 struct perf_header *header = &session->header;
Jiri Olsa563aecb2013-06-05 13:35:06 +02002386 struct perf_evsel *evsel;
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002387 struct feat_fd ff;
Jiri Olsa944d62b2013-07-17 19:49:43 +02002388 u64 attr_offset;
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002389 int err;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002390
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002391 ff = (struct feat_fd){ .fd = fd};
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002392 lseek(fd, sizeof(f_header), SEEK_SET);
2393
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -03002394 evlist__for_each_entry(session->evlist, evsel) {
Robert Richter6606f872012-08-16 21:10:19 +02002395 evsel->id_offset = lseek(fd, 0, SEEK_CUR);
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002396 err = do_write(&ff, evsel->id, evsel->ids * sizeof(u64));
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002397 if (err < 0) {
2398 pr_debug("failed to write perf header\n");
2399 return err;
2400 }
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002401 }
2402
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002403 attr_offset = lseek(ff.fd, 0, SEEK_CUR);
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002404
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -03002405 evlist__for_each_entry(evlist, evsel) {
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002406 f_attr = (struct perf_file_attr){
Robert Richter6606f872012-08-16 21:10:19 +02002407 .attr = evsel->attr,
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002408 .ids = {
Robert Richter6606f872012-08-16 21:10:19 +02002409 .offset = evsel->id_offset,
2410 .size = evsel->ids * sizeof(u64),
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002411 }
2412 };
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002413 err = do_write(&ff, &f_attr, sizeof(f_attr));
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002414 if (err < 0) {
2415 pr_debug("failed to write perf header attribute\n");
2416 return err;
2417 }
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002418 }
2419
Adrian Hunterd645c442013-12-11 14:36:28 +02002420 if (!header->data_offset)
2421 header->data_offset = lseek(fd, 0, SEEK_CUR);
Jiri Olsa8d541e92013-07-17 19:49:44 +02002422 header->feat_offset = header->data_offset + header->data_size;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002423
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002424 if (at_exit) {
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002425 err = perf_header__adds_write(header, evlist, fd);
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002426 if (err < 0)
2427 return err;
2428 }
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002429
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002430 f_header = (struct perf_file_header){
2431 .magic = PERF_MAGIC,
2432 .size = sizeof(f_header),
2433 .attr_size = sizeof(f_attr),
2434 .attrs = {
Jiri Olsa944d62b2013-07-17 19:49:43 +02002435 .offset = attr_offset,
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002436 .size = evlist->nr_entries * sizeof(f_attr),
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002437 },
2438 .data = {
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002439 .offset = header->data_offset,
2440 .size = header->data_size,
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002441 },
Jiri Olsa44b3c572013-07-11 17:28:31 +02002442 /* event_types is ignored, store zeros */
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002443 };
2444
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002445 memcpy(&f_header.adds_features, &header->adds_features, sizeof(header->adds_features));
Frederic Weisbecker2ba08252009-10-17 17:12:34 +02002446
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002447 lseek(fd, 0, SEEK_SET);
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002448 err = do_write(&ff, &f_header, sizeof(f_header));
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002449 if (err < 0) {
2450 pr_debug("failed to write perf header\n");
2451 return err;
2452 }
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002453 lseek(fd, header->data_offset + header->data_size, SEEK_SET);
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002454
Arnaldo Carvalho de Melod5eed902009-11-19 14:55:56 -02002455 return 0;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002456}
2457
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002458static int perf_header__getbuffer64(struct perf_header *header,
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002459 int fd, void *buf, size_t size)
2460{
Arnaldo Carvalho de Melo1e7972c2011-01-03 16:50:55 -02002461 if (readn(fd, buf, size) <= 0)
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002462 return -1;
2463
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002464 if (header->needs_swap)
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002465 mem_bswap_64(buf, size);
2466
2467 return 0;
2468}
2469
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002470int perf_header__process_sections(struct perf_header *header, int fd,
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002471 void *data,
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002472 int (*process)(struct perf_file_section *section,
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002473 struct perf_header *ph,
2474 int feat, int fd, void *data))
Frederic Weisbecker2ba08252009-10-17 17:12:34 +02002475{
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002476 struct perf_file_section *feat_sec, *sec;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002477 int nr_sections;
2478 int sec_size;
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002479 int feat;
2480 int err;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002481
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002482 nr_sections = bitmap_weight(header->adds_features, HEADER_FEAT_BITS);
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002483 if (!nr_sections)
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002484 return 0;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002485
Paul Gortmaker91b98802013-01-30 20:05:49 -05002486 feat_sec = sec = calloc(nr_sections, sizeof(*feat_sec));
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002487 if (!feat_sec)
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002488 return -1;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002489
2490 sec_size = sizeof(*feat_sec) * nr_sections;
2491
Jiri Olsa8d541e92013-07-17 19:49:44 +02002492 lseek(fd, header->feat_offset, SEEK_SET);
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002493
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002494 err = perf_header__getbuffer64(header, fd, feat_sec, sec_size);
2495 if (err < 0)
Arnaldo Carvalho de Melo769885f2009-12-28 22:48:32 -02002496 goto out_free;
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002497
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002498 for_each_set_bit(feat, header->adds_features, HEADER_LAST_FEATURE) {
2499 err = process(sec++, header, feat, fd, data);
2500 if (err < 0)
2501 goto out_free;
Frederic Weisbecker4778d2e2009-11-11 04:51:05 +01002502 }
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002503 err = 0;
Arnaldo Carvalho de Melo769885f2009-12-28 22:48:32 -02002504out_free:
Frederic Weisbecker9e827dd2009-11-11 04:51:07 +01002505 free(feat_sec);
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002506 return err;
Arnaldo Carvalho de Melo769885f2009-12-28 22:48:32 -02002507}
Frederic Weisbecker2ba08252009-10-17 17:12:34 +02002508
Stephane Eranian114382a2012-02-09 23:21:08 +01002509static const int attr_file_abi_sizes[] = {
2510 [0] = PERF_ATTR_SIZE_VER0,
2511 [1] = PERF_ATTR_SIZE_VER1,
Jiri Olsa239cc472012-08-07 15:20:42 +02002512 [2] = PERF_ATTR_SIZE_VER2,
Jiri Olsa0f6a3012012-08-07 15:20:45 +02002513 [3] = PERF_ATTR_SIZE_VER3,
Stephane Eranian6a21c0b2014-09-24 13:48:39 +02002514 [4] = PERF_ATTR_SIZE_VER4,
Stephane Eranian114382a2012-02-09 23:21:08 +01002515 0,
2516};
2517
2518/*
2519 * In the legacy file format, the magic number is not used to encode endianness.
2520 * hdr_sz was used to encode endianness. But given that hdr_sz can vary based
2521 * on ABI revisions, we need to try all combinations for all endianness to
2522 * detect the endianness.
2523 */
2524static int try_all_file_abis(uint64_t hdr_sz, struct perf_header *ph)
2525{
2526 uint64_t ref_size, attr_size;
2527 int i;
2528
2529 for (i = 0 ; attr_file_abi_sizes[i]; i++) {
2530 ref_size = attr_file_abi_sizes[i]
2531 + sizeof(struct perf_file_section);
2532 if (hdr_sz != ref_size) {
2533 attr_size = bswap_64(hdr_sz);
2534 if (attr_size != ref_size)
2535 continue;
2536
2537 ph->needs_swap = true;
2538 }
2539 pr_debug("ABI%d perf.data file detected, need_swap=%d\n",
2540 i,
2541 ph->needs_swap);
2542 return 0;
2543 }
2544 /* could not determine endianness */
2545 return -1;
2546}
2547
2548#define PERF_PIPE_HDR_VER0 16
2549
2550static const size_t attr_pipe_abi_sizes[] = {
2551 [0] = PERF_PIPE_HDR_VER0,
2552 0,
2553};
2554
2555/*
2556 * In the legacy pipe format, there is an implicit assumption that endiannesss
2557 * between host recording the samples, and host parsing the samples is the
2558 * same. This is not always the case given that the pipe output may always be
2559 * redirected into a file and analyzed on a different machine with possibly a
2560 * different endianness and perf_event ABI revsions in the perf tool itself.
2561 */
2562static int try_all_pipe_abis(uint64_t hdr_sz, struct perf_header *ph)
2563{
2564 u64 attr_size;
2565 int i;
2566
2567 for (i = 0 ; attr_pipe_abi_sizes[i]; i++) {
2568 if (hdr_sz != attr_pipe_abi_sizes[i]) {
2569 attr_size = bswap_64(hdr_sz);
2570 if (attr_size != hdr_sz)
2571 continue;
2572
2573 ph->needs_swap = true;
2574 }
2575 pr_debug("Pipe ABI%d perf.data file detected\n", i);
2576 return 0;
2577 }
2578 return -1;
2579}
2580
Feng Tange84ba4e2012-10-30 11:56:07 +08002581bool is_perf_magic(u64 magic)
2582{
2583 if (!memcmp(&magic, __perf_magic1, sizeof(magic))
2584 || magic == __perf_magic2
2585 || magic == __perf_magic2_sw)
2586 return true;
2587
2588 return false;
2589}
2590
Stephane Eranian114382a2012-02-09 23:21:08 +01002591static int check_magic_endian(u64 magic, uint64_t hdr_sz,
2592 bool is_pipe, struct perf_header *ph)
Stephane Eranian73323f52012-02-02 13:54:44 +01002593{
2594 int ret;
2595
2596 /* check for legacy format */
Stephane Eranian114382a2012-02-09 23:21:08 +01002597 ret = memcmp(&magic, __perf_magic1, sizeof(magic));
Stephane Eranian73323f52012-02-02 13:54:44 +01002598 if (ret == 0) {
Jiri Olsa2a08c3e2013-07-17 19:49:47 +02002599 ph->version = PERF_HEADER_VERSION_1;
Stephane Eranian73323f52012-02-02 13:54:44 +01002600 pr_debug("legacy perf.data format\n");
Stephane Eranian114382a2012-02-09 23:21:08 +01002601 if (is_pipe)
2602 return try_all_pipe_abis(hdr_sz, ph);
Stephane Eranian73323f52012-02-02 13:54:44 +01002603
Stephane Eranian114382a2012-02-09 23:21:08 +01002604 return try_all_file_abis(hdr_sz, ph);
Stephane Eranian73323f52012-02-02 13:54:44 +01002605 }
Stephane Eranian114382a2012-02-09 23:21:08 +01002606 /*
2607 * the new magic number serves two purposes:
2608 * - unique number to identify actual perf.data files
2609 * - encode endianness of file
2610 */
Namhyung Kimf7913972015-01-29 17:06:45 +09002611 ph->version = PERF_HEADER_VERSION_2;
Stephane Eranian73323f52012-02-02 13:54:44 +01002612
Stephane Eranian114382a2012-02-09 23:21:08 +01002613 /* check magic number with one endianness */
2614 if (magic == __perf_magic2)
Stephane Eranian73323f52012-02-02 13:54:44 +01002615 return 0;
2616
Stephane Eranian114382a2012-02-09 23:21:08 +01002617 /* check magic number with opposite endianness */
2618 if (magic != __perf_magic2_sw)
Stephane Eranian73323f52012-02-02 13:54:44 +01002619 return -1;
2620
2621 ph->needs_swap = true;
2622
2623 return 0;
2624}
2625
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002626int perf_file_header__read(struct perf_file_header *header,
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002627 struct perf_header *ph, int fd)
2628{
Jiri Olsa727ebd52013-11-28 11:30:14 +01002629 ssize_t ret;
Stephane Eranian73323f52012-02-02 13:54:44 +01002630
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002631 lseek(fd, 0, SEEK_SET);
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002632
Stephane Eranian73323f52012-02-02 13:54:44 +01002633 ret = readn(fd, header, sizeof(*header));
2634 if (ret <= 0)
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002635 return -1;
2636
Stephane Eranian114382a2012-02-09 23:21:08 +01002637 if (check_magic_endian(header->magic,
2638 header->attr_size, false, ph) < 0) {
2639 pr_debug("magic/endian check failed\n");
Stephane Eranian73323f52012-02-02 13:54:44 +01002640 return -1;
Stephane Eranian114382a2012-02-09 23:21:08 +01002641 }
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002642
Stephane Eranian73323f52012-02-02 13:54:44 +01002643 if (ph->needs_swap) {
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002644 mem_bswap_64(header, offsetof(struct perf_file_header,
Stephane Eranian73323f52012-02-02 13:54:44 +01002645 adds_features));
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002646 }
2647
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002648 if (header->size != sizeof(*header)) {
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002649 /* Support the previous format */
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002650 if (header->size == offsetof(typeof(*header), adds_features))
2651 bitmap_zero(header->adds_features, HEADER_FEAT_BITS);
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002652 else
2653 return -1;
David Ahernd327fa42011-10-18 17:34:01 -06002654 } else if (ph->needs_swap) {
David Ahernd327fa42011-10-18 17:34:01 -06002655 /*
2656 * feature bitmap is declared as an array of unsigned longs --
2657 * not good since its size can differ between the host that
2658 * generated the data file and the host analyzing the file.
2659 *
2660 * We need to handle endianness, but we don't know the size of
2661 * the unsigned long where the file was generated. Take a best
2662 * guess at determining it: try 64-bit swap first (ie., file
2663 * created on a 64-bit host), and check if the hostname feature
2664 * bit is set (this feature bit is forced on as of fbe96f2).
2665 * If the bit is not, undo the 64-bit swap and try a 32-bit
2666 * swap. If the hostname bit is still not set (e.g., older data
2667 * file), punt and fallback to the original behavior --
2668 * clearing all feature bits and setting buildid.
2669 */
David Ahern80c01202012-06-08 11:47:51 -03002670 mem_bswap_64(&header->adds_features,
2671 BITS_TO_U64(HEADER_FEAT_BITS));
David Ahernd327fa42011-10-18 17:34:01 -06002672
2673 if (!test_bit(HEADER_HOSTNAME, header->adds_features)) {
David Ahern80c01202012-06-08 11:47:51 -03002674 /* unswap as u64 */
2675 mem_bswap_64(&header->adds_features,
2676 BITS_TO_U64(HEADER_FEAT_BITS));
2677
2678 /* unswap as u32 */
2679 mem_bswap_32(&header->adds_features,
2680 BITS_TO_U32(HEADER_FEAT_BITS));
David Ahernd327fa42011-10-18 17:34:01 -06002681 }
2682
2683 if (!test_bit(HEADER_HOSTNAME, header->adds_features)) {
2684 bitmap_zero(header->adds_features, HEADER_FEAT_BITS);
2685 set_bit(HEADER_BUILD_ID, header->adds_features);
2686 }
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002687 }
2688
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002689 memcpy(&ph->adds_features, &header->adds_features,
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002690 sizeof(ph->adds_features));
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002691
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002692 ph->data_offset = header->data.offset;
2693 ph->data_size = header->data.size;
Jiri Olsa8d541e92013-07-17 19:49:44 +02002694 ph->feat_offset = header->data.offset + header->data.size;
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002695 return 0;
2696}
2697
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002698static int perf_file_section__process(struct perf_file_section *section,
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002699 struct perf_header *ph,
Arnaldo Carvalho de Meloda378962012-06-27 13:08:42 -03002700 int feat, int fd, void *data)
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002701{
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07002702 struct feat_fd fdd = {
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002703 .fd = fd,
2704 .ph = ph,
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07002705 .size = section->size,
2706 .offset = section->offset,
David Carrillo-Cisneros1a222752017-07-17 21:25:41 -07002707 };
2708
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002709 if (lseek(fd, section->offset, SEEK_SET) == (off_t)-1) {
Arnaldo Carvalho de Melo9486aa32011-01-22 20:37:02 -02002710 pr_debug("Failed to lseek to %" PRIu64 " offset for feature "
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002711 "%d, continuing...\n", section->offset, feat);
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002712 return 0;
2713 }
2714
Robert Richterb1e5a9b2011-12-07 10:02:57 +01002715 if (feat >= HEADER_LAST_FEATURE) {
2716 pr_debug("unknown feature %d, continuing...\n", feat);
2717 return 0;
2718 }
2719
Robert Richterf1c67db2012-02-10 15:41:56 +01002720 if (!feat_ops[feat].process)
2721 return 0;
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002722
David Carrillo-Cisneros62552452017-07-17 21:25:42 -07002723 return feat_ops[feat].process(&fdd, data);
Arnaldo Carvalho de Melo37562ea2009-11-16 16:32:43 -02002724}
2725
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002726static int perf_file_header__read_pipe(struct perf_pipe_file_header *header,
Tom Zanussi454c4072010-05-01 01:41:20 -05002727 struct perf_header *ph, int fd,
2728 bool repipe)
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002729{
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002730 struct feat_fd ff = {
2731 .fd = STDOUT_FILENO,
2732 .ph = ph,
2733 };
Jiri Olsa727ebd52013-11-28 11:30:14 +01002734 ssize_t ret;
Stephane Eranian73323f52012-02-02 13:54:44 +01002735
2736 ret = readn(fd, header, sizeof(*header));
2737 if (ret <= 0)
2738 return -1;
2739
Stephane Eranian114382a2012-02-09 23:21:08 +01002740 if (check_magic_endian(header->magic, header->size, true, ph) < 0) {
2741 pr_debug("endian/magic failed\n");
Tom Zanussi8dc58102010-04-01 23:59:15 -05002742 return -1;
Stephane Eranian114382a2012-02-09 23:21:08 +01002743 }
2744
2745 if (ph->needs_swap)
2746 header->size = bswap_64(header->size);
Tom Zanussi8dc58102010-04-01 23:59:15 -05002747
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07002748 if (repipe && do_write(&ff, header, sizeof(*header)) < 0)
Tom Zanussi454c4072010-05-01 01:41:20 -05002749 return -1;
2750
Tom Zanussi8dc58102010-04-01 23:59:15 -05002751 return 0;
2752}
2753
Jiri Olsad4339562013-07-17 19:49:41 +02002754static int perf_header__read_pipe(struct perf_session *session)
Tom Zanussi8dc58102010-04-01 23:59:15 -05002755{
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002756 struct perf_header *header = &session->header;
Tom Zanussi8dc58102010-04-01 23:59:15 -05002757 struct perf_pipe_file_header f_header;
2758
Jiri Olsacc9784bd2013-10-15 16:27:34 +02002759 if (perf_file_header__read_pipe(&f_header, header,
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01002760 perf_data__fd(session->data),
Tom Zanussi454c4072010-05-01 01:41:20 -05002761 session->repipe) < 0) {
Tom Zanussi8dc58102010-04-01 23:59:15 -05002762 pr_debug("incompatible file format\n");
2763 return -EINVAL;
2764 }
2765
Tom Zanussi8dc58102010-04-01 23:59:15 -05002766 return 0;
2767}
2768
Stephane Eranian69996df2012-02-09 23:21:06 +01002769static int read_attr(int fd, struct perf_header *ph,
2770 struct perf_file_attr *f_attr)
2771{
2772 struct perf_event_attr *attr = &f_attr->attr;
2773 size_t sz, left;
2774 size_t our_sz = sizeof(f_attr->attr);
Jiri Olsa727ebd52013-11-28 11:30:14 +01002775 ssize_t ret;
Stephane Eranian69996df2012-02-09 23:21:06 +01002776
2777 memset(f_attr, 0, sizeof(*f_attr));
2778
2779 /* read minimal guaranteed structure */
2780 ret = readn(fd, attr, PERF_ATTR_SIZE_VER0);
2781 if (ret <= 0) {
2782 pr_debug("cannot read %d bytes of header attr\n",
2783 PERF_ATTR_SIZE_VER0);
2784 return -1;
2785 }
2786
2787 /* on file perf_event_attr size */
2788 sz = attr->size;
Stephane Eranian114382a2012-02-09 23:21:08 +01002789
Stephane Eranian69996df2012-02-09 23:21:06 +01002790 if (ph->needs_swap)
2791 sz = bswap_32(sz);
2792
2793 if (sz == 0) {
2794 /* assume ABI0 */
2795 sz = PERF_ATTR_SIZE_VER0;
2796 } else if (sz > our_sz) {
2797 pr_debug("file uses a more recent and unsupported ABI"
2798 " (%zu bytes extra)\n", sz - our_sz);
2799 return -1;
2800 }
2801 /* what we have not yet read and that we know about */
2802 left = sz - PERF_ATTR_SIZE_VER0;
2803 if (left) {
2804 void *ptr = attr;
2805 ptr += PERF_ATTR_SIZE_VER0;
2806
2807 ret = readn(fd, ptr, left);
2808 }
2809 /* read perf_file_section, ids are read in caller */
2810 ret = readn(fd, &f_attr->ids, sizeof(f_attr->ids));
2811
2812 return ret <= 0 ? -1 : 0;
2813}
2814
Namhyung Kim831394b2012-09-06 11:10:46 +09002815static int perf_evsel__prepare_tracepoint_event(struct perf_evsel *evsel,
2816 struct pevent *pevent)
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002817{
Namhyung Kim831394b2012-09-06 11:10:46 +09002818 struct event_format *event;
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002819 char bf[128];
2820
Namhyung Kim831394b2012-09-06 11:10:46 +09002821 /* already prepared */
2822 if (evsel->tp_format)
2823 return 0;
2824
Namhyung Kim3dce2ce2013-03-21 16:18:48 +09002825 if (pevent == NULL) {
2826 pr_debug("broken or missing trace data\n");
2827 return -1;
2828 }
2829
Namhyung Kim831394b2012-09-06 11:10:46 +09002830 event = pevent_find_event(pevent, evsel->attr.config);
Namhyung Kima7619ae2013-04-18 21:24:16 +09002831 if (event == NULL) {
2832 pr_debug("cannot find event format for %d\n", (int)evsel->attr.config);
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002833 return -1;
Namhyung Kima7619ae2013-04-18 21:24:16 +09002834 }
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002835
Namhyung Kim831394b2012-09-06 11:10:46 +09002836 if (!evsel->name) {
2837 snprintf(bf, sizeof(bf), "%s:%s", event->system, event->name);
2838 evsel->name = strdup(bf);
2839 if (evsel->name == NULL)
2840 return -1;
2841 }
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002842
Arnaldo Carvalho de Melofcf65bf2012-08-07 09:58:03 -03002843 evsel->tp_format = event;
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002844 return 0;
2845}
2846
Namhyung Kim831394b2012-09-06 11:10:46 +09002847static int perf_evlist__prepare_tracepoint_events(struct perf_evlist *evlist,
2848 struct pevent *pevent)
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002849{
2850 struct perf_evsel *pos;
2851
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -03002852 evlist__for_each_entry(evlist, pos) {
Namhyung Kim831394b2012-09-06 11:10:46 +09002853 if (pos->attr.type == PERF_TYPE_TRACEPOINT &&
2854 perf_evsel__prepare_tracepoint_event(pos, pevent))
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002855 return -1;
2856 }
2857
2858 return 0;
2859}
2860
Jiri Olsad4339562013-07-17 19:49:41 +02002861int perf_session__read_header(struct perf_session *session)
Tom Zanussi8dc58102010-04-01 23:59:15 -05002862{
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01002863 struct perf_data *data = session->data;
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002864 struct perf_header *header = &session->header;
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002865 struct perf_file_header f_header;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002866 struct perf_file_attr f_attr;
2867 u64 f_id;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002868 int nr_attrs, nr_ids, i, j;
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01002869 int fd = perf_data__fd(data);
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002870
Namhyung Kim334fe7a2013-03-11 16:43:12 +09002871 session->evlist = perf_evlist__new();
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002872 if (session->evlist == NULL)
2873 return -ENOMEM;
2874
Kan Liang2c071442015-08-28 05:48:05 -04002875 session->evlist->env = &header->env;
Arnaldo Carvalho de Melo4cde9982015-09-09 12:25:00 -03002876 session->machines.host.env = &header->env;
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01002877 if (perf_data__is_pipe(data))
Jiri Olsad4339562013-07-17 19:49:41 +02002878 return perf_header__read_pipe(session);
Tom Zanussi8dc58102010-04-01 23:59:15 -05002879
Stephane Eranian69996df2012-02-09 23:21:06 +01002880 if (perf_file_header__read(&f_header, header, fd) < 0)
Arnaldo Carvalho de Melo4dc0a042009-11-19 14:55:55 -02002881 return -EINVAL;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002882
Namhyung Kimb314e5c2013-09-30 17:19:48 +09002883 /*
2884 * Sanity check that perf.data was written cleanly; data size is
2885 * initialized to 0 and updated only if the on_exit function is run.
2886 * If data size is still 0 then the file contains only partial
2887 * information. Just warn user and process it as much as it can.
2888 */
2889 if (f_header.data.size == 0) {
2890 pr_warning("WARNING: The %s file's data size field is 0 which is unexpected.\n"
2891 "Was the 'perf record' command properly terminated?\n",
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01002892 data->path);
Namhyung Kimb314e5c2013-09-30 17:19:48 +09002893 }
2894
Stephane Eranian69996df2012-02-09 23:21:06 +01002895 nr_attrs = f_header.attrs.size / f_header.attr_size;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002896 lseek(fd, f_header.attrs.offset, SEEK_SET);
2897
2898 for (i = 0; i < nr_attrs; i++) {
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002899 struct perf_evsel *evsel;
Peter Zijlstra1c222bc2009-08-06 20:57:41 +02002900 off_t tmp;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002901
Stephane Eranian69996df2012-02-09 23:21:06 +01002902 if (read_attr(fd, header, &f_attr) < 0)
Arnaldo Carvalho de Melo769885f2009-12-28 22:48:32 -02002903 goto out_errno;
Arnaldo Carvalho de Meloba215942010-01-14 12:23:10 -02002904
David Ahern1060ab82015-04-09 16:15:46 -04002905 if (header->needs_swap) {
2906 f_attr.ids.size = bswap_64(f_attr.ids.size);
2907 f_attr.ids.offset = bswap_64(f_attr.ids.offset);
David Aherneda39132011-07-15 12:34:09 -06002908 perf_event__attr_swap(&f_attr.attr);
David Ahern1060ab82015-04-09 16:15:46 -04002909 }
David Aherneda39132011-07-15 12:34:09 -06002910
Peter Zijlstra1c222bc2009-08-06 20:57:41 +02002911 tmp = lseek(fd, 0, SEEK_CUR);
Arnaldo Carvalho de Meloef503832013-11-07 16:41:19 -03002912 evsel = perf_evsel__new(&f_attr.attr);
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002913
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002914 if (evsel == NULL)
2915 goto out_delete_evlist;
Arnaldo Carvalho de Melo0807d2d2012-09-26 12:48:18 -03002916
2917 evsel->needs_swap = header->needs_swap;
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002918 /*
2919 * Do it before so that if perf_evsel__alloc_id fails, this
2920 * entry gets purged too at perf_evlist__delete().
2921 */
2922 perf_evlist__add(session->evlist, evsel);
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002923
2924 nr_ids = f_attr.ids.size / sizeof(u64);
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002925 /*
2926 * We don't have the cpu and thread maps on the header, so
2927 * for allocating the perf_sample_id table we fake 1 cpu and
2928 * hattr->ids threads.
2929 */
2930 if (perf_evsel__alloc_id(evsel, 1, nr_ids))
2931 goto out_delete_evlist;
2932
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002933 lseek(fd, f_attr.ids.offset, SEEK_SET);
2934
2935 for (j = 0; j < nr_ids; j++) {
Arnaldo Carvalho de Melo1c0b04d2011-03-09 08:13:19 -03002936 if (perf_header__getbuffer64(header, fd, &f_id, sizeof(f_id)))
Arnaldo Carvalho de Melo769885f2009-12-28 22:48:32 -02002937 goto out_errno;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002938
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002939 perf_evlist__id_add(session->evlist, evsel, 0, j, f_id);
Arnaldo Carvalho de Melo4dc0a042009-11-19 14:55:55 -02002940 }
Arnaldo Carvalho de Melo11deb1f2009-11-17 01:18:09 -02002941
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002942 lseek(fd, tmp, SEEK_SET);
2943 }
2944
Arnaldo Carvalho de Melod04b35f2011-11-11 22:17:32 -02002945 symbol_conf.nr_events = nr_attrs;
2946
Jiri Olsa29f5ffd2013-12-03 14:09:23 +01002947 perf_header__process_sections(header, fd, &session->tevent,
Stephane Eranianfbe96f22011-09-30 15:40:40 +02002948 perf_file_section__process);
Frederic Weisbecker4778d2e2009-11-11 04:51:05 +01002949
Namhyung Kim831394b2012-09-06 11:10:46 +09002950 if (perf_evlist__prepare_tracepoint_events(session->evlist,
Jiri Olsa29f5ffd2013-12-03 14:09:23 +01002951 session->tevent.pevent))
Arnaldo Carvalho de Melocb9dd492012-06-11 19:03:32 -03002952 goto out_delete_evlist;
2953
Arnaldo Carvalho de Melo4dc0a042009-11-19 14:55:55 -02002954 return 0;
Arnaldo Carvalho de Melo769885f2009-12-28 22:48:32 -02002955out_errno:
2956 return -errno;
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03002957
2958out_delete_evlist:
2959 perf_evlist__delete(session->evlist);
2960 session->evlist = NULL;
2961 return -ENOMEM;
Peter Zijlstra7c6a1c62009-06-25 17:05:54 +02002962}
Frederic Weisbecker0d3a5c82009-08-16 20:56:37 +02002963
Arnaldo Carvalho de Melo45694aa2011-11-28 08:30:20 -02002964int perf_event__synthesize_attr(struct perf_tool *tool,
Robert Richterf4d83432012-08-16 21:10:17 +02002965 struct perf_event_attr *attr, u32 ids, u64 *id,
Arnaldo Carvalho de Melo743eb862011-11-28 07:56:39 -02002966 perf_event__handler_t process)
Frederic Weisbecker0d3a5c82009-08-16 20:56:37 +02002967{
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02002968 union perf_event *ev;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05002969 size_t size;
2970 int err;
2971
2972 size = sizeof(struct perf_event_attr);
Irina Tirdea9ac3e482012-09-11 01:15:01 +03002973 size = PERF_ALIGN(size, sizeof(u64));
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05002974 size += sizeof(struct perf_event_header);
2975 size += ids * sizeof(u64);
2976
2977 ev = malloc(size);
2978
Chris Samuelce47dc52010-11-13 13:35:06 +11002979 if (ev == NULL)
2980 return -ENOMEM;
2981
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05002982 ev->attr.attr = *attr;
2983 memcpy(ev->attr.id, id, ids * sizeof(u64));
2984
2985 ev->attr.header.type = PERF_RECORD_HEADER_ATTR;
Robert Richterf4d83432012-08-16 21:10:17 +02002986 ev->attr.header.size = (u16)size;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05002987
Robert Richterf4d83432012-08-16 21:10:17 +02002988 if (ev->attr.header.size == size)
2989 err = process(tool, ev, NULL, NULL);
2990 else
2991 err = -E2BIG;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05002992
2993 free(ev);
2994
2995 return err;
2996}
2997
David Carrillo-Cisnerose9def1b2017-07-17 21:25:48 -07002998int perf_event__synthesize_features(struct perf_tool *tool,
2999 struct perf_session *session,
3000 struct perf_evlist *evlist,
3001 perf_event__handler_t process)
3002{
3003 struct perf_header *header = &session->header;
3004 struct feat_fd ff;
3005 struct feature_event *fe;
3006 size_t sz, sz_hdr;
3007 int feat, ret;
3008
3009 sz_hdr = sizeof(fe->header);
3010 sz = sizeof(union perf_event);
3011 /* get a nice alignment */
3012 sz = PERF_ALIGN(sz, page_size);
3013
3014 memset(&ff, 0, sizeof(ff));
3015
3016 ff.buf = malloc(sz);
3017 if (!ff.buf)
3018 return -ENOMEM;
3019
3020 ff.size = sz - sz_hdr;
3021
3022 for_each_set_bit(feat, header->adds_features, HEADER_FEAT_BITS) {
3023 if (!feat_ops[feat].synthesize) {
3024 pr_debug("No record header feature for header :%d\n", feat);
3025 continue;
3026 }
3027
3028 ff.offset = sizeof(*fe);
3029
3030 ret = feat_ops[feat].write(&ff, evlist);
3031 if (ret || ff.offset <= (ssize_t)sizeof(*fe)) {
3032 pr_debug("Error writing feature\n");
3033 continue;
3034 }
3035 /* ff.buf may have changed due to realloc in do_write() */
3036 fe = ff.buf;
3037 memset(fe, 0, sizeof(*fe));
3038
3039 fe->feat_id = feat;
3040 fe->header.type = PERF_RECORD_HEADER_FEATURE;
3041 fe->header.size = ff.offset;
3042
3043 ret = process(tool, ff.buf, NULL, NULL);
3044 if (ret) {
3045 free(ff.buf);
3046 return ret;
3047 }
3048 }
3049 free(ff.buf);
3050 return 0;
3051}
3052
3053int perf_event__process_feature(struct perf_tool *tool,
3054 union perf_event *event,
3055 struct perf_session *session __maybe_unused)
3056{
3057 struct feat_fd ff = { .fd = 0 };
3058 struct feature_event *fe = (struct feature_event *)event;
3059 int type = fe->header.type;
3060 u64 feat = fe->feat_id;
3061
3062 if (type < 0 || type >= PERF_RECORD_HEADER_MAX) {
3063 pr_warning("invalid record type %d in pipe-mode\n", type);
3064 return 0;
3065 }
3066 if (feat == HEADER_RESERVED || feat > HEADER_LAST_FEATURE) {
3067 pr_warning("invalid record type %d in pipe-mode\n", type);
3068 return -1;
3069 }
3070
3071 if (!feat_ops[feat].process)
3072 return 0;
3073
3074 ff.buf = (void *)fe->data;
3075 ff.size = event->header.size - sizeof(event->header);
3076 ff.ph = &session->header;
3077
3078 if (feat_ops[feat].process(&ff, NULL))
3079 return -1;
3080
3081 if (!feat_ops[feat].print || !tool->show_feat_hdr)
3082 return 0;
3083
3084 if (!feat_ops[feat].full_only ||
3085 tool->show_feat_hdr >= SHOW_FEAT_HEADER_FULL_INFO) {
3086 feat_ops[feat].print(&ff, stdout);
3087 } else {
3088 fprintf(stdout, "# %s info available, use -I to display\n",
3089 feat_ops[feat].name);
3090 }
3091
3092 return 0;
3093}
3094
Jiri Olsaa6e52812015-10-25 15:51:37 +01003095static struct event_update_event *
3096event_update_event__new(size_t size, u64 type, u64 id)
3097{
3098 struct event_update_event *ev;
3099
3100 size += sizeof(*ev);
3101 size = PERF_ALIGN(size, sizeof(u64));
3102
3103 ev = zalloc(size);
3104 if (ev) {
3105 ev->header.type = PERF_RECORD_EVENT_UPDATE;
3106 ev->header.size = (u16)size;
3107 ev->type = type;
3108 ev->id = id;
3109 }
3110 return ev;
3111}
3112
3113int
3114perf_event__synthesize_event_update_unit(struct perf_tool *tool,
3115 struct perf_evsel *evsel,
3116 perf_event__handler_t process)
3117{
3118 struct event_update_event *ev;
3119 size_t size = strlen(evsel->unit);
3120 int err;
3121
3122 ev = event_update_event__new(size + 1, PERF_EVENT_UPDATE__UNIT, evsel->id[0]);
3123 if (ev == NULL)
3124 return -ENOMEM;
3125
3126 strncpy(ev->data, evsel->unit, size);
3127 err = process(tool, (union perf_event *)ev, NULL, NULL);
3128 free(ev);
3129 return err;
3130}
3131
Jiri Olsadaeecbc2015-10-25 15:51:38 +01003132int
3133perf_event__synthesize_event_update_scale(struct perf_tool *tool,
3134 struct perf_evsel *evsel,
3135 perf_event__handler_t process)
3136{
3137 struct event_update_event *ev;
3138 struct event_update_event_scale *ev_data;
3139 int err;
3140
3141 ev = event_update_event__new(sizeof(*ev_data), PERF_EVENT_UPDATE__SCALE, evsel->id[0]);
3142 if (ev == NULL)
3143 return -ENOMEM;
3144
3145 ev_data = (struct event_update_event_scale *) ev->data;
3146 ev_data->scale = evsel->scale;
3147 err = process(tool, (union perf_event*) ev, NULL, NULL);
3148 free(ev);
3149 return err;
3150}
3151
Jiri Olsa802c9042015-10-25 15:51:39 +01003152int
3153perf_event__synthesize_event_update_name(struct perf_tool *tool,
3154 struct perf_evsel *evsel,
3155 perf_event__handler_t process)
3156{
3157 struct event_update_event *ev;
3158 size_t len = strlen(evsel->name);
3159 int err;
3160
3161 ev = event_update_event__new(len + 1, PERF_EVENT_UPDATE__NAME, evsel->id[0]);
3162 if (ev == NULL)
3163 return -ENOMEM;
3164
3165 strncpy(ev->data, evsel->name, len);
3166 err = process(tool, (union perf_event*) ev, NULL, NULL);
3167 free(ev);
3168 return err;
3169}
Jiri Olsadaeecbc2015-10-25 15:51:38 +01003170
Jiri Olsa86ebb092015-10-25 15:51:40 +01003171int
3172perf_event__synthesize_event_update_cpus(struct perf_tool *tool,
3173 struct perf_evsel *evsel,
3174 perf_event__handler_t process)
3175{
3176 size_t size = sizeof(struct event_update_event);
3177 struct event_update_event *ev;
3178 int max, err;
3179 u16 type;
3180
3181 if (!evsel->own_cpus)
3182 return 0;
3183
3184 ev = cpu_map_data__alloc(evsel->own_cpus, &size, &type, &max);
3185 if (!ev)
3186 return -ENOMEM;
3187
3188 ev->header.type = PERF_RECORD_EVENT_UPDATE;
3189 ev->header.size = (u16)size;
3190 ev->type = PERF_EVENT_UPDATE__CPUS;
3191 ev->id = evsel->id[0];
3192
3193 cpu_map_data__synthesize((struct cpu_map_data *) ev->data,
3194 evsel->own_cpus,
3195 type, max);
3196
3197 err = process(tool, (union perf_event*) ev, NULL, NULL);
3198 free(ev);
3199 return err;
3200}
3201
Jiri Olsac853f932015-10-25 15:51:41 +01003202size_t perf_event__fprintf_event_update(union perf_event *event, FILE *fp)
3203{
3204 struct event_update_event *ev = &event->event_update;
3205 struct event_update_event_scale *ev_scale;
3206 struct event_update_event_cpus *ev_cpus;
3207 struct cpu_map *map;
3208 size_t ret;
3209
3210 ret = fprintf(fp, "\n... id: %" PRIu64 "\n", ev->id);
3211
3212 switch (ev->type) {
3213 case PERF_EVENT_UPDATE__SCALE:
3214 ev_scale = (struct event_update_event_scale *) ev->data;
3215 ret += fprintf(fp, "... scale: %f\n", ev_scale->scale);
3216 break;
3217 case PERF_EVENT_UPDATE__UNIT:
3218 ret += fprintf(fp, "... unit: %s\n", ev->data);
3219 break;
3220 case PERF_EVENT_UPDATE__NAME:
3221 ret += fprintf(fp, "... name: %s\n", ev->data);
3222 break;
3223 case PERF_EVENT_UPDATE__CPUS:
3224 ev_cpus = (struct event_update_event_cpus *) ev->data;
3225 ret += fprintf(fp, "... ");
3226
3227 map = cpu_map__new_data(&ev_cpus->cpus);
3228 if (map)
3229 ret += cpu_map__fprintf(map, fp);
3230 else
3231 ret += fprintf(fp, "failed to get cpus\n");
3232 break;
3233 default:
3234 ret += fprintf(fp, "... unknown type\n");
3235 break;
3236 }
3237
3238 return ret;
3239}
Jiri Olsa86ebb092015-10-25 15:51:40 +01003240
Arnaldo Carvalho de Melo45694aa2011-11-28 08:30:20 -02003241int perf_event__synthesize_attrs(struct perf_tool *tool,
Arnaldo Carvalho de Melod20deb62011-11-25 08:19:45 -02003242 struct perf_session *session,
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03003243 perf_event__handler_t process)
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003244{
Robert Richter6606f872012-08-16 21:10:19 +02003245 struct perf_evsel *evsel;
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03003246 int err = 0;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003247
Arnaldo Carvalho de Meloe5cadb92016-06-23 11:26:15 -03003248 evlist__for_each_entry(session->evlist, evsel) {
Robert Richter6606f872012-08-16 21:10:19 +02003249 err = perf_event__synthesize_attr(tool, &evsel->attr, evsel->ids,
3250 evsel->id, process);
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003251 if (err) {
3252 pr_debug("failed to create perf header attribute\n");
3253 return err;
3254 }
3255 }
3256
3257 return err;
3258}
3259
Adrian Hunter47c3d102013-07-04 16:20:21 +03003260int perf_event__process_attr(struct perf_tool *tool __maybe_unused,
3261 union perf_event *event,
Arnaldo Carvalho de Melo10d0f082011-11-11 22:45:41 -02003262 struct perf_evlist **pevlist)
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003263{
Robert Richterf4d83432012-08-16 21:10:17 +02003264 u32 i, ids, n_ids;
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03003265 struct perf_evsel *evsel;
Arnaldo Carvalho de Melo10d0f082011-11-11 22:45:41 -02003266 struct perf_evlist *evlist = *pevlist;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003267
Arnaldo Carvalho de Melo10d0f082011-11-11 22:45:41 -02003268 if (evlist == NULL) {
Namhyung Kim334fe7a2013-03-11 16:43:12 +09003269 *pevlist = evlist = perf_evlist__new();
Arnaldo Carvalho de Melo10d0f082011-11-11 22:45:41 -02003270 if (evlist == NULL)
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003271 return -ENOMEM;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003272 }
3273
Arnaldo Carvalho de Meloef503832013-11-07 16:41:19 -03003274 evsel = perf_evsel__new(&event->attr.attr);
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03003275 if (evsel == NULL)
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003276 return -ENOMEM;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003277
Arnaldo Carvalho de Melo10d0f082011-11-11 22:45:41 -02003278 perf_evlist__add(evlist, evsel);
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03003279
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02003280 ids = event->header.size;
3281 ids -= (void *)&event->attr.id - (void *)event;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003282 n_ids = ids / sizeof(u64);
Arnaldo Carvalho de Meloa91e5432011-03-10 11:15:54 -03003283 /*
3284 * We don't have the cpu and thread maps on the header, so
3285 * for allocating the perf_sample_id table we fake 1 cpu and
3286 * hattr->ids threads.
3287 */
3288 if (perf_evsel__alloc_id(evsel, 1, n_ids))
3289 return -ENOMEM;
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003290
3291 for (i = 0; i < n_ids; i++) {
Arnaldo Carvalho de Melo10d0f082011-11-11 22:45:41 -02003292 perf_evlist__id_add(evlist, evsel, 0, i, event->attr.id[i]);
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003293 }
3294
Adrian Hunter7e0d6fc2013-07-04 16:20:29 +03003295 symbol_conf.nr_events = evlist->nr_entries;
3296
Tom Zanussi2c46dbb2010-04-01 23:59:19 -05003297 return 0;
3298}
Tom Zanussicd19a032010-04-01 23:59:20 -05003299
Jiri Olsaffe777252015-10-25 15:51:36 +01003300int perf_event__process_event_update(struct perf_tool *tool __maybe_unused,
3301 union perf_event *event,
3302 struct perf_evlist **pevlist)
3303{
3304 struct event_update_event *ev = &event->event_update;
Jiri Olsadaeecbc2015-10-25 15:51:38 +01003305 struct event_update_event_scale *ev_scale;
Jiri Olsa86ebb092015-10-25 15:51:40 +01003306 struct event_update_event_cpus *ev_cpus;
Jiri Olsaffe777252015-10-25 15:51:36 +01003307 struct perf_evlist *evlist;
3308 struct perf_evsel *evsel;
Jiri Olsa86ebb092015-10-25 15:51:40 +01003309 struct cpu_map *map;
Jiri Olsaffe777252015-10-25 15:51:36 +01003310
3311 if (!pevlist || *pevlist == NULL)
3312 return -EINVAL;
3313
3314 evlist = *pevlist;
3315
3316 evsel = perf_evlist__id2evsel(evlist, ev->id);
3317 if (evsel == NULL)
3318 return -EINVAL;
3319
Jiri Olsaa6e52812015-10-25 15:51:37 +01003320 switch (ev->type) {
3321 case PERF_EVENT_UPDATE__UNIT:
3322 evsel->unit = strdup(ev->data);
Jiri Olsadaeecbc2015-10-25 15:51:38 +01003323 break;
Jiri Olsa802c9042015-10-25 15:51:39 +01003324 case PERF_EVENT_UPDATE__NAME:
3325 evsel->name = strdup(ev->data);
3326 break;
Jiri Olsadaeecbc2015-10-25 15:51:38 +01003327 case PERF_EVENT_UPDATE__SCALE:
3328 ev_scale = (struct event_update_event_scale *) ev->data;
3329 evsel->scale = ev_scale->scale;
Arnaldo Carvalho de Melo8434a2e2017-02-08 21:57:22 -03003330 break;
Jiri Olsa86ebb092015-10-25 15:51:40 +01003331 case PERF_EVENT_UPDATE__CPUS:
3332 ev_cpus = (struct event_update_event_cpus *) ev->data;
3333
3334 map = cpu_map__new_data(&ev_cpus->cpus);
3335 if (map)
3336 evsel->own_cpus = map;
3337 else
3338 pr_err("failed to get event_update cpus\n");
Jiri Olsaa6e52812015-10-25 15:51:37 +01003339 default:
3340 break;
3341 }
3342
Jiri Olsaffe777252015-10-25 15:51:36 +01003343 return 0;
3344}
3345
Arnaldo Carvalho de Melo45694aa2011-11-28 08:30:20 -02003346int perf_event__synthesize_tracing_data(struct perf_tool *tool, int fd,
Arnaldo Carvalho de Melod20deb62011-11-25 08:19:45 -02003347 struct perf_evlist *evlist,
Arnaldo Carvalho de Melo743eb862011-11-28 07:56:39 -02003348 perf_event__handler_t process)
Tom Zanussi92155452010-04-01 23:59:21 -05003349{
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02003350 union perf_event ev;
Jiri Olsa29208e52011-10-20 15:59:43 +02003351 struct tracing_data *tdata;
Tom Zanussi92155452010-04-01 23:59:21 -05003352 ssize_t size = 0, aligned_size = 0, padding;
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07003353 struct feat_fd ff;
Irina Tirdea1d037ca2012-09-11 01:15:03 +03003354 int err __maybe_unused = 0;
Tom Zanussi92155452010-04-01 23:59:21 -05003355
Jiri Olsa29208e52011-10-20 15:59:43 +02003356 /*
3357 * We are going to store the size of the data followed
3358 * by the data contents. Since the fd descriptor is a pipe,
3359 * we cannot seek back to store the size of the data once
3360 * we know it. Instead we:
3361 *
3362 * - write the tracing data to the temp file
3363 * - get/write the data size to pipe
3364 * - write the tracing data from the temp file
3365 * to the pipe
3366 */
3367 tdata = tracing_data_get(&evlist->entries, fd, true);
3368 if (!tdata)
3369 return -1;
3370
Tom Zanussi92155452010-04-01 23:59:21 -05003371 memset(&ev, 0, sizeof(ev));
3372
3373 ev.tracing_data.header.type = PERF_RECORD_HEADER_TRACING_DATA;
Jiri Olsa29208e52011-10-20 15:59:43 +02003374 size = tdata->size;
Irina Tirdea9ac3e482012-09-11 01:15:01 +03003375 aligned_size = PERF_ALIGN(size, sizeof(u64));
Tom Zanussi92155452010-04-01 23:59:21 -05003376 padding = aligned_size - size;
3377 ev.tracing_data.header.size = sizeof(ev.tracing_data);
3378 ev.tracing_data.size = aligned_size;
3379
Arnaldo Carvalho de Melo45694aa2011-11-28 08:30:20 -02003380 process(tool, &ev, NULL, NULL);
Tom Zanussi92155452010-04-01 23:59:21 -05003381
Jiri Olsa29208e52011-10-20 15:59:43 +02003382 /*
3383 * The put function will copy all the tracing data
3384 * stored in temp file to the pipe.
3385 */
3386 tracing_data_put(tdata);
3387
David Carrillo-Cisnerosccebbeb2017-07-17 21:25:39 -07003388 ff = (struct feat_fd){ .fd = fd };
3389 if (write_padded(&ff, NULL, 0, padding))
David Carrillo-Cisneros2ff53652017-07-17 21:25:36 -07003390 return -1;
Tom Zanussi92155452010-04-01 23:59:21 -05003391
3392 return aligned_size;
3393}
3394
Adrian Hunter47c3d102013-07-04 16:20:21 +03003395int perf_event__process_tracing_data(struct perf_tool *tool __maybe_unused,
3396 union perf_event *event,
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02003397 struct perf_session *session)
Tom Zanussi92155452010-04-01 23:59:21 -05003398{
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02003399 ssize_t size_read, padding, size = event->tracing_data.size;
Jiri Olsa8ceb41d2017-01-23 22:07:59 +01003400 int fd = perf_data__fd(session->data);
Jiri Olsacc9784bd2013-10-15 16:27:34 +02003401 off_t offset = lseek(fd, 0, SEEK_CUR);
Tom Zanussi92155452010-04-01 23:59:21 -05003402 char buf[BUFSIZ];
3403
3404 /* setup for reading amidst mmap */
Jiri Olsacc9784bd2013-10-15 16:27:34 +02003405 lseek(fd, offset + sizeof(struct tracing_data_event),
Tom Zanussi92155452010-04-01 23:59:21 -05003406 SEEK_SET);
3407
Jiri Olsa29f5ffd2013-12-03 14:09:23 +01003408 size_read = trace_report(fd, &session->tevent,
Arnaldo Carvalho de Meloda378962012-06-27 13:08:42 -03003409 session->repipe);
Irina Tirdea9ac3e482012-09-11 01:15:01 +03003410 padding = PERF_ALIGN(size_read, sizeof(u64)) - size_read;
Tom Zanussi92155452010-04-01 23:59:21 -05003411
Jiri Olsacc9784bd2013-10-15 16:27:34 +02003412 if (readn(fd, buf, padding) < 0) {
Arnaldo Carvalho de Melo2caa48a2013-01-24 22:34:33 -03003413 pr_err("%s: reading input file", __func__);
3414 return -1;
3415 }
Tom Zanussi454c4072010-05-01 01:41:20 -05003416 if (session->repipe) {
3417 int retw = write(STDOUT_FILENO, buf, padding);
Arnaldo Carvalho de Melo2caa48a2013-01-24 22:34:33 -03003418 if (retw <= 0 || retw != padding) {
3419 pr_err("%s: repiping tracing data padding", __func__);
3420 return -1;
3421 }
Tom Zanussi454c4072010-05-01 01:41:20 -05003422 }
Tom Zanussi92155452010-04-01 23:59:21 -05003423
Arnaldo Carvalho de Melo2caa48a2013-01-24 22:34:33 -03003424 if (size_read + padding != size) {
3425 pr_err("%s: tracing data size mismatch", __func__);
3426 return -1;
3427 }
Tom Zanussi92155452010-04-01 23:59:21 -05003428
Namhyung Kim831394b2012-09-06 11:10:46 +09003429 perf_evlist__prepare_tracepoint_events(session->evlist,
Jiri Olsa29f5ffd2013-12-03 14:09:23 +01003430 session->tevent.pevent);
Arnaldo Carvalho de Melo8b6ee4c2012-08-07 23:36:16 -03003431
Tom Zanussi92155452010-04-01 23:59:21 -05003432 return size_read + padding;
3433}
Tom Zanussic7929e42010-04-01 23:59:22 -05003434
Arnaldo Carvalho de Melo45694aa2011-11-28 08:30:20 -02003435int perf_event__synthesize_build_id(struct perf_tool *tool,
Arnaldo Carvalho de Melod20deb62011-11-25 08:19:45 -02003436 struct dso *pos, u16 misc,
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02003437 perf_event__handler_t process,
Arnaldo Carvalho de Melo743eb862011-11-28 07:56:39 -02003438 struct machine *machine)
Tom Zanussic7929e42010-04-01 23:59:22 -05003439{
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02003440 union perf_event ev;
Tom Zanussic7929e42010-04-01 23:59:22 -05003441 size_t len;
3442 int err = 0;
3443
3444 if (!pos->hit)
3445 return err;
3446
3447 memset(&ev, 0, sizeof(ev));
3448
3449 len = pos->long_name_len + 1;
Irina Tirdea9ac3e482012-09-11 01:15:01 +03003450 len = PERF_ALIGN(len, NAME_ALIGN);
Tom Zanussic7929e42010-04-01 23:59:22 -05003451 memcpy(&ev.build_id.build_id, pos->build_id, sizeof(pos->build_id));
3452 ev.build_id.header.type = PERF_RECORD_HEADER_BUILD_ID;
3453 ev.build_id.header.misc = misc;
Arnaldo Carvalho de Melo23346f22010-04-27 21:17:50 -03003454 ev.build_id.pid = machine->pid;
Tom Zanussic7929e42010-04-01 23:59:22 -05003455 ev.build_id.header.size = sizeof(ev.build_id) + len;
3456 memcpy(&ev.build_id.filename, pos->long_name, pos->long_name_len);
3457
Arnaldo Carvalho de Melo45694aa2011-11-28 08:30:20 -02003458 err = process(tool, &ev, NULL, machine);
Tom Zanussic7929e42010-04-01 23:59:22 -05003459
3460 return err;
3461}
3462
Irina Tirdea1d037ca2012-09-11 01:15:03 +03003463int perf_event__process_build_id(struct perf_tool *tool __maybe_unused,
Arnaldo Carvalho de Melod20deb62011-11-25 08:19:45 -02003464 union perf_event *event,
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02003465 struct perf_session *session)
Tom Zanussic7929e42010-04-01 23:59:22 -05003466{
Arnaldo Carvalho de Melo8115d602011-01-29 14:01:45 -02003467 __event_process_build_id(&event->build_id,
3468 event->build_id.filename,
Zhang, Yanmina1645ce2010-04-19 13:32:50 +08003469 session);
Tom Zanussic7929e42010-04-01 23:59:22 -05003470 return 0;
3471}