mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-12-02 16:56:49 +07:00
03456a158d
This drops the trace.info file and move its contents into the common perf.data file. This is done by creating a new trace_info section into this file. A user of perf headers needs to call perf_header__set_trace_info() to save the trace meta informations into the perf.data file. A file created by perf after his patch is unsupported by previous version because the size of the headers have increased. That said, it's two new fields that have been added in the end of the headers, and those could be ignored by previous versions if they just handled the dynamic header size and then ignore the unknow part. The offsets guarantee the compatibility. We'll do a -stable fix for that. But current previous versions handle the header size using its static size, not dynamic, then it's not backward compatible with trace records. Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Arnaldo Carvalho de Melo <acme@redhat.com> Cc: Paul Mackerras <paulus@samba.org> Cc: Mike Galbraith <efault@gmx.de> Cc: Paul Mackerras <paulus@samba.org> LKML-Reference: <20091006213643.GA5343@nowhere> Signed-off-by: Ingo Molnar <mingo@elte.hu>
382 lines
7.6 KiB
C
382 lines
7.6 KiB
C
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "util.h"
|
|
#include "header.h"
|
|
#include "../perf.h"
|
|
#include "trace-event.h"
|
|
|
|
/*
|
|
* Create new perf.data header attribute:
|
|
*/
|
|
struct perf_header_attr *perf_header_attr__new(struct perf_event_attr *attr)
|
|
{
|
|
struct perf_header_attr *self = malloc(sizeof(*self));
|
|
|
|
if (!self)
|
|
die("nomem");
|
|
|
|
self->attr = *attr;
|
|
self->ids = 0;
|
|
self->size = 1;
|
|
self->id = malloc(sizeof(u64));
|
|
|
|
if (!self->id)
|
|
die("nomem");
|
|
|
|
return self;
|
|
}
|
|
|
|
void perf_header_attr__add_id(struct perf_header_attr *self, u64 id)
|
|
{
|
|
int pos = self->ids;
|
|
|
|
self->ids++;
|
|
if (self->ids > self->size) {
|
|
self->size *= 2;
|
|
self->id = realloc(self->id, self->size * sizeof(u64));
|
|
if (!self->id)
|
|
die("nomem");
|
|
}
|
|
self->id[pos] = id;
|
|
}
|
|
|
|
/*
|
|
* Create new perf.data header:
|
|
*/
|
|
struct perf_header *perf_header__new(void)
|
|
{
|
|
struct perf_header *self = malloc(sizeof(*self));
|
|
|
|
if (!self)
|
|
die("nomem");
|
|
|
|
self->frozen = 0;
|
|
|
|
self->attrs = 0;
|
|
self->size = 1;
|
|
self->attr = malloc(sizeof(void *));
|
|
|
|
if (!self->attr)
|
|
die("nomem");
|
|
|
|
self->data_offset = 0;
|
|
self->data_size = 0;
|
|
self->trace_info_offset = 0;
|
|
self->trace_info_size = 0;
|
|
|
|
return self;
|
|
}
|
|
|
|
void perf_header__add_attr(struct perf_header *self,
|
|
struct perf_header_attr *attr)
|
|
{
|
|
int pos = self->attrs;
|
|
|
|
if (self->frozen)
|
|
die("frozen");
|
|
|
|
self->attrs++;
|
|
if (self->attrs > self->size) {
|
|
self->size *= 2;
|
|
self->attr = realloc(self->attr, self->size * sizeof(void *));
|
|
if (!self->attr)
|
|
die("nomem");
|
|
}
|
|
self->attr[pos] = attr;
|
|
}
|
|
|
|
#define MAX_EVENT_NAME 64
|
|
|
|
struct perf_trace_event_type {
|
|
u64 event_id;
|
|
char name[MAX_EVENT_NAME];
|
|
};
|
|
|
|
static int event_count;
|
|
static struct perf_trace_event_type *events;
|
|
|
|
void perf_header__push_event(u64 id, const char *name)
|
|
{
|
|
if (strlen(name) > MAX_EVENT_NAME)
|
|
printf("Event %s will be truncated\n", name);
|
|
|
|
if (!events) {
|
|
events = malloc(sizeof(struct perf_trace_event_type));
|
|
if (!events)
|
|
die("nomem");
|
|
} else {
|
|
events = realloc(events, (event_count + 1) * sizeof(struct perf_trace_event_type));
|
|
if (!events)
|
|
die("nomem");
|
|
}
|
|
memset(&events[event_count], 0, sizeof(struct perf_trace_event_type));
|
|
events[event_count].event_id = id;
|
|
strncpy(events[event_count].name, name, MAX_EVENT_NAME - 1);
|
|
event_count++;
|
|
}
|
|
|
|
char *perf_header__find_event(u64 id)
|
|
{
|
|
int i;
|
|
for (i = 0 ; i < event_count; i++) {
|
|
if (events[i].event_id == id)
|
|
return events[i].name;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static const char *__perf_magic = "PERFFILE";
|
|
|
|
#define PERF_MAGIC (*(u64 *)__perf_magic)
|
|
|
|
struct perf_file_section {
|
|
u64 offset;
|
|
u64 size;
|
|
};
|
|
|
|
struct perf_file_attr {
|
|
struct perf_event_attr attr;
|
|
struct perf_file_section ids;
|
|
};
|
|
|
|
struct perf_file_header {
|
|
u64 magic;
|
|
u64 size;
|
|
u64 attr_size;
|
|
struct perf_file_section attrs;
|
|
struct perf_file_section data;
|
|
struct perf_file_section event_types;
|
|
struct perf_file_section trace_info;
|
|
};
|
|
|
|
static int trace_info;
|
|
|
|
void perf_header__set_trace_info(void)
|
|
{
|
|
trace_info = 1;
|
|
}
|
|
|
|
static void do_write(int fd, void *buf, size_t size)
|
|
{
|
|
while (size) {
|
|
int ret = write(fd, buf, size);
|
|
|
|
if (ret < 0)
|
|
die("failed to write");
|
|
|
|
size -= ret;
|
|
buf += ret;
|
|
}
|
|
}
|
|
|
|
void perf_header__write(struct perf_header *self, int fd)
|
|
{
|
|
struct perf_file_header f_header;
|
|
struct perf_file_attr f_attr;
|
|
struct perf_header_attr *attr;
|
|
int i;
|
|
|
|
lseek(fd, sizeof(f_header), SEEK_SET);
|
|
|
|
|
|
for (i = 0; i < self->attrs; i++) {
|
|
attr = self->attr[i];
|
|
|
|
attr->id_offset = lseek(fd, 0, SEEK_CUR);
|
|
do_write(fd, attr->id, attr->ids * sizeof(u64));
|
|
}
|
|
|
|
|
|
self->attr_offset = lseek(fd, 0, SEEK_CUR);
|
|
|
|
for (i = 0; i < self->attrs; i++) {
|
|
attr = self->attr[i];
|
|
|
|
f_attr = (struct perf_file_attr){
|
|
.attr = attr->attr,
|
|
.ids = {
|
|
.offset = attr->id_offset,
|
|
.size = attr->ids * sizeof(u64),
|
|
}
|
|
};
|
|
do_write(fd, &f_attr, sizeof(f_attr));
|
|
}
|
|
|
|
self->event_offset = lseek(fd, 0, SEEK_CUR);
|
|
self->event_size = event_count * sizeof(struct perf_trace_event_type);
|
|
if (events)
|
|
do_write(fd, events, self->event_size);
|
|
|
|
if (trace_info) {
|
|
static int trace_info_written;
|
|
|
|
/*
|
|
* Write it only once
|
|
*/
|
|
if (!trace_info_written) {
|
|
self->trace_info_offset = lseek(fd, 0, SEEK_CUR);
|
|
read_tracing_data(fd, attrs, nr_counters);
|
|
self->trace_info_size = lseek(fd, 0, SEEK_CUR) -
|
|
self->trace_info_offset;
|
|
trace_info_written = 1;
|
|
} else {
|
|
lseek(fd, self->trace_info_offset +
|
|
self->trace_info_size, SEEK_SET);
|
|
}
|
|
}
|
|
|
|
self->data_offset = lseek(fd, 0, SEEK_CUR);
|
|
|
|
f_header = (struct perf_file_header){
|
|
.magic = PERF_MAGIC,
|
|
.size = sizeof(f_header),
|
|
.attr_size = sizeof(f_attr),
|
|
.attrs = {
|
|
.offset = self->attr_offset,
|
|
.size = self->attrs * sizeof(f_attr),
|
|
},
|
|
.data = {
|
|
.offset = self->data_offset,
|
|
.size = self->data_size,
|
|
},
|
|
.event_types = {
|
|
.offset = self->event_offset,
|
|
.size = self->event_size,
|
|
},
|
|
.trace_info = {
|
|
.offset = self->trace_info_offset,
|
|
.size = self->trace_info_size,
|
|
},
|
|
};
|
|
|
|
lseek(fd, 0, SEEK_SET);
|
|
do_write(fd, &f_header, sizeof(f_header));
|
|
lseek(fd, self->data_offset + self->data_size, SEEK_SET);
|
|
|
|
self->frozen = 1;
|
|
}
|
|
|
|
static void do_read(int fd, void *buf, size_t size)
|
|
{
|
|
while (size) {
|
|
int ret = read(fd, buf, size);
|
|
|
|
if (ret < 0)
|
|
die("failed to read");
|
|
if (ret == 0)
|
|
die("failed to read: missing data");
|
|
|
|
size -= ret;
|
|
buf += ret;
|
|
}
|
|
}
|
|
|
|
struct perf_header *perf_header__read(int fd)
|
|
{
|
|
struct perf_header *self = perf_header__new();
|
|
struct perf_file_header f_header;
|
|
struct perf_file_attr f_attr;
|
|
u64 f_id;
|
|
|
|
int nr_attrs, nr_ids, i, j;
|
|
|
|
lseek(fd, 0, SEEK_SET);
|
|
do_read(fd, &f_header, sizeof(f_header));
|
|
|
|
if (f_header.magic != PERF_MAGIC ||
|
|
f_header.size != sizeof(f_header) ||
|
|
f_header.attr_size != sizeof(f_attr))
|
|
die("incompatible file format");
|
|
|
|
nr_attrs = f_header.attrs.size / sizeof(f_attr);
|
|
lseek(fd, f_header.attrs.offset, SEEK_SET);
|
|
|
|
for (i = 0; i < nr_attrs; i++) {
|
|
struct perf_header_attr *attr;
|
|
off_t tmp;
|
|
|
|
do_read(fd, &f_attr, sizeof(f_attr));
|
|
tmp = lseek(fd, 0, SEEK_CUR);
|
|
|
|
attr = perf_header_attr__new(&f_attr.attr);
|
|
|
|
nr_ids = f_attr.ids.size / sizeof(u64);
|
|
lseek(fd, f_attr.ids.offset, SEEK_SET);
|
|
|
|
for (j = 0; j < nr_ids; j++) {
|
|
do_read(fd, &f_id, sizeof(f_id));
|
|
|
|
perf_header_attr__add_id(attr, f_id);
|
|
}
|
|
perf_header__add_attr(self, attr);
|
|
lseek(fd, tmp, SEEK_SET);
|
|
}
|
|
|
|
if (f_header.event_types.size) {
|
|
lseek(fd, f_header.event_types.offset, SEEK_SET);
|
|
events = malloc(f_header.event_types.size);
|
|
if (!events)
|
|
die("nomem");
|
|
do_read(fd, events, f_header.event_types.size);
|
|
event_count = f_header.event_types.size / sizeof(struct perf_trace_event_type);
|
|
}
|
|
|
|
self->trace_info_offset = f_header.trace_info.offset;
|
|
self->trace_info_size = f_header.trace_info.size;
|
|
|
|
if (self->trace_info_size) {
|
|
lseek(fd, self->trace_info_offset, SEEK_SET);
|
|
trace_report(fd);
|
|
}
|
|
|
|
self->event_offset = f_header.event_types.offset;
|
|
self->event_size = f_header.event_types.size;
|
|
|
|
self->data_offset = f_header.data.offset;
|
|
self->data_size = f_header.data.size;
|
|
|
|
lseek(fd, self->data_offset, SEEK_SET);
|
|
|
|
self->frozen = 1;
|
|
|
|
return self;
|
|
}
|
|
|
|
u64 perf_header__sample_type(struct perf_header *header)
|
|
{
|
|
u64 type = 0;
|
|
int i;
|
|
|
|
for (i = 0; i < header->attrs; i++) {
|
|
struct perf_header_attr *attr = header->attr[i];
|
|
|
|
if (!type)
|
|
type = attr->attr.sample_type;
|
|
else if (type != attr->attr.sample_type)
|
|
die("non matching sample_type");
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
struct perf_event_attr *
|
|
perf_header__find_attr(u64 id, struct perf_header *header)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < header->attrs; i++) {
|
|
struct perf_header_attr *attr = header->attr[i];
|
|
int j;
|
|
|
|
for (j = 0; j < attr->ids; j++) {
|
|
if (attr->id[j] == id)
|
|
return &attr->attr;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|