perf tools: Merge trace.info content into perf.data
authorFrederic Weisbecker <fweisbec@gmail.com>
Tue, 6 Oct 2009 21:36:47 +0000 (23:36 +0200)
committerIngo Molnar <mingo@elte.hu>
Wed, 7 Oct 2009 06:36:10 +0000 (08:36 +0200)
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>
tools/perf/builtin-record.c
tools/perf/builtin-sched.c
tools/perf/builtin-trace.c
tools/perf/util/header.c
tools/perf/util/header.h
tools/perf/util/trace-event-info.c
tools/perf/util/trace-event-read.c
tools/perf/util/trace-event.h

index 494f8c7d7521e8e5dbb7e6f566f7aa4f94b487bc..59af03d80d0742bd3ac05685ba28c19776ef25da 100644 (file)
@@ -17,7 +17,6 @@
 #include "util/header.h"
 #include "util/event.h"
 #include "util/debug.h"
-#include "util/trace-event.h"
 
 #include <unistd.h>
 #include <sched.h>
@@ -566,17 +565,17 @@ static int __cmd_record(int argc, const char **argv)
        else
                header = perf_header__new();
 
-
        if (raw_samples) {
-               read_tracing_data(attrs, nr_counters);
+               perf_header__set_trace_info();
        } else {
                for (i = 0; i < nr_counters; i++) {
                        if (attrs[i].sample_type & PERF_SAMPLE_RAW) {
-                               read_tracing_data(attrs, nr_counters);
+                               perf_header__set_trace_info();
                                break;
                        }
                }
        }
+
        atexit(atexit_header);
 
        if (!system_wide) {
index 4470f2535706e76909e09031647406ac3f19f0ac..18871380b0155cd1562a376e1d7208e935695214 100644 (file)
@@ -1634,7 +1634,6 @@ static int read_events(void)
        uint32_t size;
        char *buf;
 
-       trace_report();
        register_idle_thread(&threads, &last_match);
 
        input = open(input_name, O_RDONLY);
index d573d4ea6c21f08641f7a314cf0dc829b8993575..d9abb4ae5f795fbe40db652643652265b959abae 100644 (file)
@@ -149,7 +149,6 @@ static int __cmd_trace(void)
        uint32_t size;
        char *buf;
 
-       trace_report();
        register_idle_thread(&threads, &last_match);
 
        input = open(input_name, O_RDONLY);
index e306857b2c2b98e3624bf30b96be74130edab831..212fade7ee74952f04ef4858937e3fa739b55ec1 100644 (file)
@@ -5,6 +5,8 @@
 
 #include "util.h"
 #include "header.h"
+#include "../perf.h"
+#include "trace-event.h"
 
 /*
  * Create new perf.data header attribute:
@@ -62,6 +64,8 @@ struct perf_header *perf_header__new(void)
 
        self->data_offset = 0;
        self->data_size = 0;
+       self->trace_info_offset = 0;
+       self->trace_info_size = 0;
 
        return self;
 }
@@ -145,8 +149,16 @@ struct perf_file_header {
        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) {
@@ -198,6 +210,23 @@ void perf_header__write(struct perf_header *self, int fd)
        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);
 
@@ -217,6 +246,10 @@ void perf_header__write(struct perf_header *self, int fd)
                        .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);
@@ -290,6 +323,15 @@ struct perf_header *perf_header__read(int fd)
                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;
 
index a2916b652a1b5960caad669bbc2d177c12210e0d..30aee5160dc0536b0b870b2bdd78e022c313615f 100644 (file)
@@ -21,6 +21,8 @@ struct perf_header {
        u64 data_size;
        u64 event_offset;
        u64 event_size;
+       u64 trace_info_offset;
+       u64 trace_info_size;
 };
 
 struct perf_header *perf_header__read(int fd);
@@ -40,7 +42,7 @@ void perf_header_attr__add_id(struct perf_header_attr *self, u64 id);
 u64 perf_header__sample_type(struct perf_header *header);
 struct perf_event_attr *
 perf_header__find_attr(u64 id, struct perf_header *header);
-
+void perf_header__set_trace_info(void);
 
 struct perf_header *perf_header__new(void);
 
index af4b0573b37fb3dfaeed2f57c9028bdbc62a1809..831052d4b4fbef86482af2e73eda14b458093ea5 100644 (file)
@@ -496,14 +496,12 @@ get_tracepoints_path(struct perf_event_attr *pattrs, int nb_events)
 
        return path.next;
 }
-void read_tracing_data(struct perf_event_attr *pattrs, int nb_events)
+void read_tracing_data(int fd, struct perf_event_attr *pattrs, int nb_events)
 {
        char buf[BUFSIZ];
        struct tracepoint_path *tps;
 
-       output_fd = open(output_file, O_WRONLY | O_CREAT | O_TRUNC | O_LARGEFILE, 0644);
-       if (output_fd < 0)
-               die("creating file '%s'", output_file);
+       output_fd = fd;
 
        buf[0] = 23;
        buf[1] = 8;
index 1b5c847d2c223f96a42f9f4a49a5a1afbb4aead4..44292e06cca41557c05b6e66281a6d4ed427f063 100644 (file)
@@ -458,9 +458,8 @@ struct record *trace_read_data(int cpu)
        return data;
 }
 
-void trace_report(void)
+void trace_report(int fd)
 {
-       const char *input_file = "trace.info";
        char buf[BUFSIZ];
        char test[] = { 23, 8, 68 };
        char *version;
@@ -468,9 +467,7 @@ void trace_report(void)
        int show_funcs = 0;
        int show_printk = 0;
 
-       input_fd = open(input_file, O_RDONLY);
-       if (input_fd < 0)
-               die("opening '%s'\n", input_file);
+       input_fd = fd;
 
        read_or_die(buf, 3);
        if (memcmp(buf, test, 3) != 0)
index 5f59a39fb88bd5f58458d28f4423ef5da0c9c11f..da77e073c867398670282de2303ea8f016d421d4 100644 (file)
@@ -158,7 +158,7 @@ struct record *trace_read_data(int cpu);
 
 void parse_set_info(int nr_cpus, int long_sz);
 
-void trace_report(void);
+void trace_report(int fd);
 
 void *malloc_or_die(unsigned int size);
 
@@ -244,6 +244,6 @@ unsigned long long
 raw_field_value(struct event *event, const char *name, void *data);
 void *raw_field_ptr(struct event *event, const char *name, void *data);
 
-void read_tracing_data(struct perf_event_attr *pattrs, int nb_events);
+void read_tracing_data(int fd, struct perf_event_attr *pattrs, int nb_events);
 
 #endif /* __PERF_TRACE_EVENTS_H */