|
|
d11878 |
--- crash-trace-command-2.0/trace.c.orig
|
|
|
d11878 |
+++ crash-trace-command-2.0/trace.c
|
|
|
d11878 |
@@ -38,6 +38,10 @@ static int trace_buffer_available;
|
|
|
d11878 |
* max_buffer is supported
|
|
|
d11878 |
*/
|
|
|
d11878 |
static int max_buffer_available;
|
|
|
d11878 |
+/*
|
|
|
d11878 |
+ * multiple trace instances are supported
|
|
|
d11878 |
+ */
|
|
|
d11878 |
+static int multiple_instances_available;
|
|
|
d11878 |
|
|
|
d11878 |
#define koffset(struct, member) struct##_##member##_offset
|
|
|
d11878 |
|
|
|
d11878 |
@@ -78,6 +82,8 @@ static int koffset(ftrace_event_field, o
|
|
|
d11878 |
static int koffset(ftrace_event_field, size);
|
|
|
d11878 |
static int koffset(ftrace_event_field, is_signed);
|
|
|
d11878 |
|
|
|
d11878 |
+static int koffset(trace_array, name);
|
|
|
d11878 |
+
|
|
|
d11878 |
static int koffset(POINTER_SYM, POINTER) = 0;
|
|
|
d11878 |
|
|
|
d11878 |
struct ring_buffer_per_cpu {
|
|
|
d11878 |
@@ -101,16 +107,25 @@ struct ring_buffer_per_cpu {
|
|
|
d11878 |
};
|
|
|
d11878 |
|
|
|
d11878 |
static ulong global_trace;
|
|
|
d11878 |
-static ulong global_trace_buffer;
|
|
|
d11878 |
-static ulong global_max_buffer;
|
|
|
d11878 |
-static ulong global_ring_buffer;
|
|
|
d11878 |
-static unsigned global_pages;
|
|
|
d11878 |
-static struct ring_buffer_per_cpu *global_buffers;
|
|
|
d11878 |
-
|
|
|
d11878 |
static ulong max_tr_trace;
|
|
|
d11878 |
-static ulong max_tr_ring_buffer;
|
|
|
d11878 |
-static unsigned max_tr_pages;
|
|
|
d11878 |
-static struct ring_buffer_per_cpu *max_tr_buffers;
|
|
|
d11878 |
+
|
|
|
d11878 |
+struct trace_instance {
|
|
|
d11878 |
+ char name[NAME_MAX + 1];
|
|
|
d11878 |
+ ulong trace_buffer;
|
|
|
d11878 |
+ ulong max_buffer;
|
|
|
d11878 |
+ ulong ring_buffer;
|
|
|
d11878 |
+ unsigned pages;
|
|
|
d11878 |
+ struct ring_buffer_per_cpu *buffers;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ ulong max_tr_ring_buffer;
|
|
|
d11878 |
+ unsigned max_tr_pages;
|
|
|
d11878 |
+ struct ring_buffer_per_cpu *max_tr_buffers;
|
|
|
d11878 |
+};
|
|
|
d11878 |
+
|
|
|
d11878 |
+static ulong ftrace_trace_arrays;
|
|
|
d11878 |
+static struct trace_instance global_trace_instance;
|
|
|
d11878 |
+static struct trace_instance *trace_instances = NULL;
|
|
|
d11878 |
+static int instance_count;
|
|
|
d11878 |
|
|
|
d11878 |
static ulong ftrace_events;
|
|
|
d11878 |
static ulong current_trace;
|
|
|
d11878 |
@@ -229,6 +244,9 @@ static int init_offsets(void)
|
|
|
d11878 |
init_offset(ftrace_event_field, size);
|
|
|
d11878 |
init_offset(ftrace_event_field, is_signed);
|
|
|
d11878 |
|
|
|
d11878 |
+ if (MEMBER_EXISTS("trace_array", "name"))
|
|
|
d11878 |
+ init_offset(trace_array, name);
|
|
|
d11878 |
+
|
|
|
d11878 |
return 0;
|
|
|
d11878 |
#undef init_offset
|
|
|
d11878 |
}
|
|
|
d11878 |
@@ -435,61 +453,140 @@ out_fail:
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
-static int ftrace_int_global_trace(void)
|
|
|
d11878 |
+static int ftrace_init_trace(struct trace_instance *ti, ulong instance_addr)
|
|
|
d11878 |
{
|
|
|
d11878 |
if (trace_buffer_available) {
|
|
|
d11878 |
- global_trace_buffer = global_trace + koffset(trace_array, trace_buffer);
|
|
|
d11878 |
- read_value(global_ring_buffer, global_trace_buffer, trace_buffer, buffer);
|
|
|
d11878 |
+ ti->trace_buffer = instance_addr +
|
|
|
d11878 |
+ koffset(trace_array, trace_buffer);
|
|
|
d11878 |
+ read_value(ti->ring_buffer, ti->trace_buffer,
|
|
|
d11878 |
+ trace_buffer, buffer);
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (max_buffer_available) {
|
|
|
d11878 |
+ ti->max_buffer = instance_addr +
|
|
|
d11878 |
+ koffset(trace_array, max_buffer);
|
|
|
d11878 |
+ read_value(ti->max_tr_ring_buffer, ti->max_buffer,
|
|
|
d11878 |
+ trace_buffer, buffer);
|
|
|
d11878 |
+ }
|
|
|
d11878 |
} else {
|
|
|
d11878 |
- read_value(global_ring_buffer, global_trace, trace_array, buffer);
|
|
|
d11878 |
- read_value(global_pages, global_ring_buffer, ring_buffer, pages);
|
|
|
d11878 |
+ read_value(ti->ring_buffer, instance_addr, trace_array, buffer);
|
|
|
d11878 |
+ read_value(ti->pages, ti->ring_buffer, ring_buffer, pages);
|
|
|
d11878 |
+
|
|
|
d11878 |
+ read_value(ti->max_tr_ring_buffer, max_tr_trace, trace_array, buffer);
|
|
|
d11878 |
+ if (ti->max_tr_ring_buffer)
|
|
|
d11878 |
+ read_value(ti->max_tr_pages, ti->max_tr_ring_buffer, ring_buffer, pages);
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
- global_buffers = calloc(sizeof(*global_buffers), nr_cpu_ids);
|
|
|
d11878 |
- if (global_buffers == NULL)
|
|
|
d11878 |
+ ti->buffers = calloc(sizeof(*ti->buffers), nr_cpu_ids);
|
|
|
d11878 |
+ if (ti->buffers == NULL)
|
|
|
d11878 |
+ goto out_fail;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (ftrace_init_buffers(ti->buffers, ti->ring_buffer,
|
|
|
d11878 |
+ ti->pages) < 0)
|
|
|
d11878 |
+ goto out_fail;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (!ti->max_tr_ring_buffer)
|
|
|
d11878 |
+ return 0;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ ti->max_tr_buffers = calloc(sizeof(*ti->max_tr_buffers), nr_cpu_ids);
|
|
|
d11878 |
+ if (ti->max_tr_buffers == NULL)
|
|
|
d11878 |
goto out_fail;
|
|
|
d11878 |
|
|
|
d11878 |
- if (ftrace_init_buffers(global_buffers, global_ring_buffer,
|
|
|
d11878 |
- global_pages) < 0)
|
|
|
d11878 |
+ if (ftrace_init_buffers(ti->max_tr_buffers, ti->max_tr_ring_buffer,
|
|
|
d11878 |
+ ti->max_tr_pages) < 0)
|
|
|
d11878 |
goto out_fail;
|
|
|
d11878 |
|
|
|
d11878 |
return 0;
|
|
|
d11878 |
|
|
|
d11878 |
out_fail:
|
|
|
d11878 |
- free(global_buffers);
|
|
|
d11878 |
+ free(ti->max_tr_buffers);
|
|
|
d11878 |
+ free(ti->buffers);
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
-static int ftrace_int_max_tr_trace(void)
|
|
|
d11878 |
+static void ftrace_destroy_all_instance_buffers()
|
|
|
d11878 |
{
|
|
|
d11878 |
- if (trace_buffer_available) {
|
|
|
d11878 |
- if (!max_buffer_available)
|
|
|
d11878 |
- return 0;
|
|
|
d11878 |
+ int i;
|
|
|
d11878 |
|
|
|
d11878 |
- global_max_buffer = global_trace + koffset(trace_array, max_buffer);
|
|
|
d11878 |
- read_value(max_tr_ring_buffer, global_max_buffer, trace_buffer, buffer);
|
|
|
d11878 |
- } else {
|
|
|
d11878 |
- read_value(max_tr_ring_buffer, max_tr_trace, trace_array, buffer);
|
|
|
d11878 |
+ for (i = 0; i < instance_count; i++)
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ struct trace_instance *ti = &trace_instances[i];
|
|
|
d11878 |
|
|
|
d11878 |
- if (!max_tr_ring_buffer)
|
|
|
d11878 |
- return 0;
|
|
|
d11878 |
+ if (ti->max_tr_ring_buffer) {
|
|
|
d11878 |
+ ftrace_destroy_buffers(ti->max_tr_buffers);
|
|
|
d11878 |
+ free(ti->max_tr_buffers);
|
|
|
d11878 |
+ }
|
|
|
d11878 |
|
|
|
d11878 |
- read_value(max_tr_pages, max_tr_ring_buffer, ring_buffer, pages);
|
|
|
d11878 |
+ ftrace_destroy_buffers(ti->buffers);
|
|
|
d11878 |
+ free(ti->buffers);
|
|
|
d11878 |
}
|
|
|
d11878 |
+}
|
|
|
d11878 |
|
|
|
d11878 |
- max_tr_buffers = calloc(sizeof(*max_tr_buffers), nr_cpu_ids);
|
|
|
d11878 |
- if (max_tr_buffers == NULL)
|
|
|
d11878 |
- goto out_fail;
|
|
|
d11878 |
+static void ftrace_destroy_instances()
|
|
|
d11878 |
+{
|
|
|
d11878 |
+ ftrace_destroy_all_instance_buffers();
|
|
|
d11878 |
+ free(trace_instances);
|
|
|
d11878 |
+}
|
|
|
d11878 |
|
|
|
d11878 |
- if (ftrace_init_buffers(max_tr_buffers, max_tr_ring_buffer,
|
|
|
d11878 |
- max_tr_pages) < 0)
|
|
|
d11878 |
- goto out_fail;
|
|
|
d11878 |
+static int ftrace_init_instances()
|
|
|
d11878 |
+{
|
|
|
d11878 |
+ int i;
|
|
|
d11878 |
+ struct trace_instance *ti;
|
|
|
d11878 |
+ struct list_data list_data;
|
|
|
d11878 |
+ struct list_data *ld = &list_data;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (!multiple_instances_available)
|
|
|
d11878 |
+ return 0;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ BZERO(ld, sizeof(struct list_data));
|
|
|
d11878 |
+ ld->start = ftrace_trace_arrays;
|
|
|
d11878 |
+ ld->end = global_trace;
|
|
|
d11878 |
+ ld->flags = LIST_ALLOCATE;
|
|
|
d11878 |
+ instance_count = do_list(ld);
|
|
|
d11878 |
+
|
|
|
d11878 |
+ /* The do_list count includes the list_head, which is not a
|
|
|
d11878 |
+ * proper instance */
|
|
|
d11878 |
+ instance_count--;
|
|
|
d11878 |
+ if (instance_count <= 0)
|
|
|
d11878 |
+ return 0;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ trace_instances = calloc(sizeof(struct trace_instance), instance_count);
|
|
|
d11878 |
+
|
|
|
d11878 |
+ /* We start i at 1 to skip over the list_head and continue to the last
|
|
|
d11878 |
+ * instance, which lies at index instance_count */
|
|
|
d11878 |
+ for (i = 1; i <= instance_count; i++)
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ ulong instance_ptr;
|
|
|
d11878 |
+ ulong name_addr;
|
|
|
d11878 |
+ int ret;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ ti = &trace_instances[i-1];
|
|
|
d11878 |
+ instance_ptr = ld->list_ptr[i];
|
|
|
d11878 |
+ read_value(name_addr, instance_ptr, trace_array, name);
|
|
|
d11878 |
+ if (!name_addr)
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ console("Instance name is NULL\n");
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+ else if (!read_string(name_addr, ti->name, sizeof(ti->name)))
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ console("Failed to read instance name at address %p\n", (void*)name_addr);
|
|
|
d11878 |
+ goto out_fail;
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+
|
|
|
d11878 |
+ ret = ftrace_init_trace(ti, instance_ptr);
|
|
|
d11878 |
+ if (ret < 0)
|
|
|
d11878 |
+ goto out_fail;
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+ FREEBUF(ld->list_ptr);
|
|
|
d11878 |
|
|
|
d11878 |
return 0;
|
|
|
d11878 |
|
|
|
d11878 |
out_fail:
|
|
|
d11878 |
- free(max_tr_buffers);
|
|
|
d11878 |
- max_tr_ring_buffer = 0;
|
|
|
d11878 |
+ /* We've already freed the current instance's trace buffer info, so
|
|
|
d11878 |
+ * we'll clear that out to avoid double freeing in
|
|
|
d11878 |
+ * ftrace_destroy_instances() */
|
|
|
d11878 |
+ BZERO(ti, sizeof(struct trace_instance));
|
|
|
d11878 |
+ ftrace_destroy_instances();
|
|
|
d11878 |
+
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
@@ -504,7 +601,7 @@ static int ftrace_init_current_tracer(vo
|
|
|
d11878 |
} else {
|
|
|
d11878 |
read_value(addr, current_trace, POINTER_SYM, POINTER);
|
|
|
d11878 |
}
|
|
|
d11878 |
-
|
|
|
d11878 |
+
|
|
|
d11878 |
read_value(addr, addr, tracer, name);
|
|
|
d11878 |
read_string(addr, tmp, 128);
|
|
|
d11878 |
|
|
|
d11878 |
@@ -524,9 +621,11 @@ static int ftrace_init(void)
|
|
|
d11878 |
struct syment *sym_max_tr_trace;
|
|
|
d11878 |
struct syment *sym_ftrace_events;
|
|
|
d11878 |
struct syment *sym_current_trace;
|
|
|
d11878 |
+ struct syment *sym_ftrace_trace_arrays;
|
|
|
d11878 |
|
|
|
d11878 |
sym_global_trace = symbol_search("global_trace");
|
|
|
d11878 |
sym_ftrace_events = symbol_search("ftrace_events");
|
|
|
d11878 |
+ sym_ftrace_trace_arrays = symbol_search("ftrace_trace_arrays");
|
|
|
d11878 |
|
|
|
d11878 |
if (sym_global_trace == NULL || sym_ftrace_events == NULL)
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
@@ -534,6 +633,13 @@ static int ftrace_init(void)
|
|
|
d11878 |
global_trace = sym_global_trace->value;
|
|
|
d11878 |
ftrace_events = sym_ftrace_events->value;
|
|
|
d11878 |
|
|
|
d11878 |
+ if (sym_ftrace_trace_arrays)
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ multiple_instances_available = 1;
|
|
|
d11878 |
+ ftrace_trace_arrays = sym_ftrace_trace_arrays->value;
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+
|
|
|
d11878 |
+
|
|
|
d11878 |
if (MEMBER_EXISTS("trace_array", "current_trace")) {
|
|
|
d11878 |
encapsulated_current_trace = 1;
|
|
|
d11878 |
} else {
|
|
|
d11878 |
@@ -564,28 +670,31 @@ static int ftrace_init(void)
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
print_offsets();
|
|
|
d11878 |
|
|
|
d11878 |
- if (ftrace_int_global_trace() < 0)
|
|
|
d11878 |
+ if (ftrace_init_trace(&global_trace_instance, global_trace) < 0)
|
|
|
d11878 |
goto out_0;
|
|
|
d11878 |
|
|
|
d11878 |
- ftrace_int_max_tr_trace();
|
|
|
d11878 |
+ if (ftrace_init_instances() < 0)
|
|
|
d11878 |
+ goto out_1;
|
|
|
d11878 |
|
|
|
d11878 |
if (ftrace_init_event_types() < 0)
|
|
|
d11878 |
- goto out_1;
|
|
|
d11878 |
+ goto out_2;
|
|
|
d11878 |
|
|
|
d11878 |
if (ftrace_init_current_tracer() < 0)
|
|
|
d11878 |
- goto out_2;
|
|
|
d11878 |
+ goto out_3;
|
|
|
d11878 |
|
|
|
d11878 |
return 0;
|
|
|
d11878 |
|
|
|
d11878 |
-out_2:
|
|
|
d11878 |
+out_3:
|
|
|
d11878 |
ftrace_destroy_event_types();
|
|
|
d11878 |
+out_2:
|
|
|
d11878 |
+ ftrace_destroy_instances();
|
|
|
d11878 |
out_1:
|
|
|
d11878 |
- if (max_tr_ring_buffer) {
|
|
|
d11878 |
- ftrace_destroy_buffers(max_tr_buffers);
|
|
|
d11878 |
- free(max_tr_buffers);
|
|
|
d11878 |
+ if (global_trace_instance.max_tr_ring_buffer) {
|
|
|
d11878 |
+ ftrace_destroy_buffers(global_trace_instance.max_tr_buffers);
|
|
|
d11878 |
+ free(global_trace_instance.max_tr_buffers);
|
|
|
d11878 |
}
|
|
|
d11878 |
- ftrace_destroy_buffers(global_buffers);
|
|
|
d11878 |
- free(global_buffers);
|
|
|
d11878 |
+ ftrace_destroy_buffers(global_trace_instance.buffers);
|
|
|
d11878 |
+ free(global_trace_instance.buffers);
|
|
|
d11878 |
out_0:
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
}
|
|
|
d11878 |
@@ -595,13 +704,15 @@ static void ftrace_destroy(void)
|
|
|
d11878 |
free(current_tracer_name);
|
|
|
d11878 |
ftrace_destroy_event_types();
|
|
|
d11878 |
|
|
|
d11878 |
- if (max_tr_ring_buffer) {
|
|
|
d11878 |
- ftrace_destroy_buffers(max_tr_buffers);
|
|
|
d11878 |
- free(max_tr_buffers);
|
|
|
d11878 |
+ ftrace_destroy_instances();
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (global_trace_instance.max_tr_ring_buffer) {
|
|
|
d11878 |
+ ftrace_destroy_buffers(global_trace_instance.max_tr_buffers);
|
|
|
d11878 |
+ free(global_trace_instance.max_tr_buffers);
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
- ftrace_destroy_buffers(global_buffers);
|
|
|
d11878 |
- free(global_buffers);
|
|
|
d11878 |
+ ftrace_destroy_buffers(global_trace_instance.buffers);
|
|
|
d11878 |
+ free(global_trace_instance.buffers);
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
static int ftrace_dump_page(int fd, ulong page, void *page_tmp)
|
|
|
d11878 |
@@ -652,7 +763,8 @@ static int try_mkdir(const char *pathnam
|
|
|
d11878 |
return 0;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
-static int ftrace_dump_buffers(const char *per_cpu_path)
|
|
|
d11878 |
+static int ftrace_dump_buffers(const char *per_cpu_path,
|
|
|
d11878 |
+ struct trace_instance *ti)
|
|
|
d11878 |
{
|
|
|
d11878 |
int i;
|
|
|
d11878 |
void *page_tmp;
|
|
|
d11878 |
@@ -664,7 +776,7 @@ static int ftrace_dump_buffers(const cha
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
|
|
|
d11878 |
for (i = 0; i < nr_cpu_ids; i++) {
|
|
|
d11878 |
- struct ring_buffer_per_cpu *cpu_buffer = &global_buffers[i];
|
|
|
d11878 |
+ struct ring_buffer_per_cpu *cpu_buffer = &ti->buffers[i];
|
|
|
d11878 |
|
|
|
d11878 |
if (!cpu_buffer->kaddr)
|
|
|
d11878 |
continue;
|
|
|
d11878 |
@@ -679,7 +791,7 @@ static int ftrace_dump_buffers(const cha
|
|
|
d11878 |
if (fd < 0)
|
|
|
d11878 |
goto out_fail;
|
|
|
d11878 |
|
|
|
d11878 |
- ftrace_dump_buffer(fd, cpu_buffer, global_pages, page_tmp);
|
|
|
d11878 |
+ ftrace_dump_buffer(fd, cpu_buffer, ti->pages, page_tmp);
|
|
|
d11878 |
close(fd);
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
@@ -1015,8 +1127,6 @@ static void ftrace_destroy_event_types(v
|
|
|
d11878 |
free(ftrace_common_fields);
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
-#define TRACE_EVENT_FL_TRACEPOINT 0x40
|
|
|
d11878 |
-
|
|
|
d11878 |
static
|
|
|
d11878 |
int ftrace_get_event_type_name(ulong call, char *name, int len)
|
|
|
d11878 |
{
|
|
|
d11878 |
@@ -1024,34 +1134,35 @@ int ftrace_get_event_type_name(ulong cal
|
|
|
d11878 |
static int name_offset;
|
|
|
d11878 |
static int flags_offset;
|
|
|
d11878 |
static int tp_name_offset;
|
|
|
d11878 |
- uint flags;
|
|
|
d11878 |
+ static long tracepoint_flag;
|
|
|
d11878 |
|
|
|
d11878 |
+ uint flags;
|
|
|
d11878 |
ulong name_addr;
|
|
|
d11878 |
|
|
|
d11878 |
if (inited)
|
|
|
d11878 |
goto work;
|
|
|
d11878 |
|
|
|
d11878 |
- inited = 1;
|
|
|
d11878 |
- name_offset = MAX(MEMBER_OFFSET("ftrace_event_call", "name"),
|
|
|
d11878 |
- MEMBER_OFFSET("trace_event_call", "name"));
|
|
|
d11878 |
- if (name_offset >= 0)
|
|
|
d11878 |
- goto work;
|
|
|
d11878 |
-
|
|
|
d11878 |
- name_offset = MAX(ANON_MEMBER_OFFSET("ftrace_event_call", "name"),
|
|
|
d11878 |
- ANON_MEMBER_OFFSET("trace_event_call", "name"));
|
|
|
d11878 |
- if (name_offset < 0)
|
|
|
d11878 |
- return -1;
|
|
|
d11878 |
+ name_offset = MAX(MEMBER_OFFSET("ftrace_event_call", "tp"),
|
|
|
d11878 |
+ MEMBER_OFFSET("trace_event_call", "tp"));
|
|
|
d11878 |
+ if (name_offset >= 0) {
|
|
|
d11878 |
+ flags_offset = MAX(MEMBER_OFFSET("ftrace_event_call", "flags"),
|
|
|
d11878 |
+ MEMBER_OFFSET("trace_event_call", "flags"));
|
|
|
d11878 |
+ if (flags_offset < 0)
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
|
|
|
d11878 |
- flags_offset = MAX(MEMBER_OFFSET("ftrace_event_call", "flags"),
|
|
|
d11878 |
- MEMBER_OFFSET("trace_event_call", "flags"));
|
|
|
d11878 |
- if (flags_offset < 0)
|
|
|
d11878 |
- return -1;
|
|
|
d11878 |
+ tp_name_offset = MEMBER_OFFSET("tracepoint", "name");
|
|
|
d11878 |
+ if (tp_name_offset < 0)
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
|
|
|
d11878 |
- tp_name_offset = MEMBER_OFFSET("tracepoint", "name");
|
|
|
d11878 |
- if (tp_name_offset < 0)
|
|
|
d11878 |
- return -1;
|
|
|
d11878 |
+ if (!enumerator_value("TRACE_EVENT_FL_TRACEPOINT", &tracepoint_flag))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
|
|
|
d11878 |
- inited = 2;
|
|
|
d11878 |
+ inited = 2;
|
|
|
d11878 |
+ } else {
|
|
|
d11878 |
+ name_offset = MAX(MEMBER_OFFSET("ftrace_event_call", "name"),
|
|
|
d11878 |
+ MEMBER_OFFSET("trace_event_call", "name"));
|
|
|
d11878 |
+ inited = 1;
|
|
|
d11878 |
+ }
|
|
|
d11878 |
|
|
|
d11878 |
work:
|
|
|
d11878 |
if (name_offset < 0)
|
|
|
d11878 |
@@ -1067,7 +1178,7 @@ work:
|
|
|
d11878 |
RETURN_ON_ERROR))
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
|
|
|
d11878 |
- if (flags & TRACE_EVENT_FL_TRACEPOINT) {
|
|
|
d11878 |
+ if (flags & (uint)tracepoint_flag) {
|
|
|
d11878 |
if (!readmem(name_addr + tp_name_offset, KVADDR,
|
|
|
d11878 |
&name_addr, sizeof(name_addr),
|
|
|
d11878 |
"read tracepoint name", RETURN_ON_ERROR))
|
|
|
d11878 |
@@ -1476,26 +1587,72 @@ static int dump_kallsyms(const char *dum
|
|
|
d11878 |
|
|
|
d11878 |
static int trace_cmd_data_output(int fd);
|
|
|
d11878 |
|
|
|
d11878 |
+#define FTRACE_DUMP_SYMBOLS (1 << 0)
|
|
|
d11878 |
+#define FTRACE_DUMP_META_DATA (1 << 1)
|
|
|
d11878 |
+
|
|
|
d11878 |
+static int populate_ftrace_dir_tree(struct trace_instance *ti,
|
|
|
d11878 |
+ char *root, uint flags)
|
|
|
d11878 |
+{
|
|
|
d11878 |
+ char path[PATH_MAX];
|
|
|
d11878 |
+ int ret;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ ret = mkdir(root, 0755);
|
|
|
d11878 |
+ if (ret < 0) {
|
|
|
d11878 |
+ if (errno == EEXIST)
|
|
|
d11878 |
+ error(INFO, "mkdir: %s exists\n", root);
|
|
|
d11878 |
+ return FALSE;
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+
|
|
|
d11878 |
+ snprintf(path, sizeof(path), "%s/per_cpu", root);
|
|
|
d11878 |
+ if (try_mkdir(path, 0755) < 0)
|
|
|
d11878 |
+ return FALSE;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (ftrace_dump_buffers(path, ti) < 0)
|
|
|
d11878 |
+ return FALSE;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (flags & FTRACE_DUMP_META_DATA) {
|
|
|
d11878 |
+ /* Dump event types */
|
|
|
d11878 |
+ snprintf(path, sizeof(path), "%s/events", root);
|
|
|
d11878 |
+ if (try_mkdir(path, 0755) < 0)
|
|
|
d11878 |
+ return FALSE;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (ftrace_dump_event_types(path) < 0)
|
|
|
d11878 |
+ return FALSE;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ /* Dump pids with corresponding cmdlines */
|
|
|
d11878 |
+ if (dump_saved_cmdlines(root) < 0)
|
|
|
d11878 |
+ return FALSE;
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (flags & FTRACE_DUMP_SYMBOLS) {
|
|
|
d11878 |
+ /* Dump all symbols of the kernel */
|
|
|
d11878 |
+ dump_kallsyms(root);
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+
|
|
|
d11878 |
+ return TRUE;
|
|
|
d11878 |
+}
|
|
|
d11878 |
+
|
|
|
d11878 |
static void ftrace_dump(int argc, char *argv[])
|
|
|
d11878 |
{
|
|
|
d11878 |
int c;
|
|
|
d11878 |
- int dump_meta_data = 0;
|
|
|
d11878 |
- int dump_symbols = 0;
|
|
|
d11878 |
+ int i;
|
|
|
d11878 |
+ uint flags = 0;
|
|
|
d11878 |
char *dump_tracing_dir;
|
|
|
d11878 |
- char path[PATH_MAX];
|
|
|
d11878 |
- int ret;
|
|
|
d11878 |
+ char instance_path[PATH_MAX];
|
|
|
d11878 |
|
|
|
d11878 |
while ((c = getopt(argc, argv, "smt")) != EOF) {
|
|
|
d11878 |
switch(c)
|
|
|
d11878 |
{
|
|
|
d11878 |
case 's':
|
|
|
d11878 |
- dump_symbols = 1;
|
|
|
d11878 |
+ flags |= FTRACE_DUMP_SYMBOLS;
|
|
|
d11878 |
break;
|
|
|
d11878 |
case 'm':
|
|
|
d11878 |
- dump_meta_data = 1;
|
|
|
d11878 |
+ flags |= FTRACE_DUMP_META_DATA;
|
|
|
d11878 |
break;
|
|
|
d11878 |
case 't':
|
|
|
d11878 |
- if (dump_symbols || dump_meta_data || argc - optind > 1)
|
|
|
d11878 |
+ if (flags & FTRACE_DUMP_SYMBOLS ||
|
|
|
d11878 |
+ flags & FTRACE_DUMP_META_DATA ||
|
|
|
d11878 |
+ argc - optind > 1)
|
|
|
d11878 |
cmd_usage(pc->curcmd, SYNOPSIS);
|
|
|
d11878 |
else {
|
|
|
d11878 |
char *trace_dat = "trace.dat";
|
|
|
d11878 |
@@ -1526,38 +1683,34 @@ static void ftrace_dump(int argc, char *
|
|
|
d11878 |
return;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
- ret = mkdir(dump_tracing_dir, 0755);
|
|
|
d11878 |
- if (ret < 0) {
|
|
|
d11878 |
- if (errno == EEXIST)
|
|
|
d11878 |
- error(INFO, "mkdir: %s exists\n", dump_tracing_dir);
|
|
|
d11878 |
+ if (!populate_ftrace_dir_tree(&global_trace_instance, dump_tracing_dir, flags))
|
|
|
d11878 |
return;
|
|
|
d11878 |
- }
|
|
|
d11878 |
|
|
|
d11878 |
- snprintf(path, sizeof(path), "%s/per_cpu", dump_tracing_dir);
|
|
|
d11878 |
- if (try_mkdir(path, 0755) < 0)
|
|
|
d11878 |
+ if (!multiple_instances_available || instance_count == 0)
|
|
|
d11878 |
return;
|
|
|
d11878 |
|
|
|
d11878 |
- if (ftrace_dump_buffers(path) < 0)
|
|
|
d11878 |
+ /* Create an instances directory, and dump instance data in there */
|
|
|
d11878 |
+ snprintf(instance_path, sizeof(instance_path),
|
|
|
d11878 |
+ "%s/instances", dump_tracing_dir);
|
|
|
d11878 |
+ if (try_mkdir(instance_path, 0755) < 0)
|
|
|
d11878 |
return;
|
|
|
d11878 |
|
|
|
d11878 |
- if (dump_meta_data) {
|
|
|
d11878 |
- /* Dump event types */
|
|
|
d11878 |
- snprintf(path, sizeof(path), "%s/events", dump_tracing_dir);
|
|
|
d11878 |
- if (try_mkdir(path, 0755) < 0)
|
|
|
d11878 |
- return;
|
|
|
d11878 |
+ /* Don't care about the flags anymore */
|
|
|
d11878 |
+ flags = 0;
|
|
|
d11878 |
|
|
|
d11878 |
- if (ftrace_dump_event_types(path) < 0)
|
|
|
d11878 |
- return;
|
|
|
d11878 |
+ for (i = 0; i < instance_count; i++)
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ struct trace_instance *ti = &trace_instances[i];
|
|
|
d11878 |
+
|
|
|
d11878 |
+ snprintf(instance_path, sizeof(instance_path),
|
|
|
d11878 |
+ "%s/instances/%s", dump_tracing_dir,
|
|
|
d11878 |
+ ti->name);
|
|
|
d11878 |
|
|
|
d11878 |
- /* Dump pids with corresponding cmdlines */
|
|
|
d11878 |
- if (dump_saved_cmdlines(dump_tracing_dir) < 0)
|
|
|
d11878 |
- return;
|
|
|
d11878 |
+ if (populate_ftrace_dir_tree(ti, instance_path, flags) < 0)
|
|
|
d11878 |
+ break;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
- if (dump_symbols) {
|
|
|
d11878 |
- /* Dump all symbols of the kernel */
|
|
|
d11878 |
- dump_kallsyms(dump_tracing_dir);
|
|
|
d11878 |
- }
|
|
|
d11878 |
+ return;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
static void ftrace_show(int argc, char *argv[])
|
|
|
d11878 |
@@ -2161,26 +2314,69 @@ static int save_ftrace_cmdlines(int fd)
|
|
|
d11878 |
return tmp_file_flush(fd);
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
-static int save_res_data(int fd, int nr_cpu_buffers)
|
|
|
d11878 |
+/* From trace-cmd.h */
|
|
|
d11878 |
+enum {
|
|
|
d11878 |
+ TRACECMD_OPTION_DONE, /* 0 */
|
|
|
d11878 |
+ TRACECMD_OPTION_DATE, /* 1 */
|
|
|
d11878 |
+ TRACECMD_OPTION_CPUSTAT, /* 2 */
|
|
|
d11878 |
+ TRACECMD_OPTION_BUFFER, /* 3 */
|
|
|
d11878 |
+ TRACECMD_OPTION_TRACECLOCK, /* 4 */
|
|
|
d11878 |
+ TRACECMD_OPTION_UNAME, /* 5 */
|
|
|
d11878 |
+ TRACECMD_OPTION_HOOK, /* 6 */
|
|
|
d11878 |
+};
|
|
|
d11878 |
+
|
|
|
d11878 |
+static int write_options(int fd, unsigned long long *buffer_offsets)
|
|
|
d11878 |
{
|
|
|
d11878 |
- unsigned short option = 0;
|
|
|
d11878 |
+ int i;
|
|
|
d11878 |
+ unsigned short option;
|
|
|
d11878 |
|
|
|
d11878 |
- if (write_and_check(fd, &nr_cpu_buffers, 4))
|
|
|
d11878 |
- return -1;
|
|
|
d11878 |
+ if (!multiple_instances_available)
|
|
|
d11878 |
+ return 0;
|
|
|
d11878 |
|
|
|
d11878 |
if (write_and_check(fd, "options ", 10))
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
|
|
|
d11878 |
+ option = TRACECMD_OPTION_BUFFER;
|
|
|
d11878 |
+ for (i = 0; i < instance_count; i++)
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ char *name = trace_instances[i].name;
|
|
|
d11878 |
+ size_t name_size = strlen(name) + 1; /* Name length + '\0' */
|
|
|
d11878 |
+ unsigned long long option_size = 8 + name_size;
|
|
|
d11878 |
+ unsigned long long offset;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ offset = buffer_offsets ? buffer_offsets[i] : 0;
|
|
|
d11878 |
+ if (write_and_check(fd, &option, 2))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+ if (write_and_check(fd, &option_size, 4))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+ if (write_and_check(fd, &offset, 8))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+ if (write_and_check(fd, name, name_size))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+
|
|
|
d11878 |
+ option = TRACECMD_OPTION_DONE;
|
|
|
d11878 |
if (write_and_check(fd, &option, 2))
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
|
|
|
d11878 |
+ return 0;
|
|
|
d11878 |
+}
|
|
|
d11878 |
+
|
|
|
d11878 |
+static int save_res_data(int fd, int nr_cpu_buffers, unsigned long long *buffer_offsets)
|
|
|
d11878 |
+{
|
|
|
d11878 |
+ if (write_and_check(fd, &nr_cpu_buffers, 4))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (write_options(fd, buffer_offsets))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+
|
|
|
d11878 |
if (write_and_check(fd, "flyrecord", 10))
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
|
|
|
d11878 |
return 0;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
-static int save_record_data(int fd, int nr_cpu_buffers)
|
|
|
d11878 |
+static int save_record_data(int fd, int nr_cpu_buffers, struct trace_instance *ti)
|
|
|
d11878 |
{
|
|
|
d11878 |
int i, j;
|
|
|
d11878 |
unsigned long long offset, buffer_offset;
|
|
|
d11878 |
@@ -2192,7 +2388,7 @@ static int save_record_data(int fd, int
|
|
|
d11878 |
buffer_offset = offset;
|
|
|
d11878 |
|
|
|
d11878 |
for (i = 0; i < nr_cpu_ids; i++) {
|
|
|
d11878 |
- struct ring_buffer_per_cpu *cpu_buffer = &global_buffers[i];
|
|
|
d11878 |
+ struct ring_buffer_per_cpu *cpu_buffer = &ti->buffers[i];
|
|
|
d11878 |
unsigned long long buffer_size;
|
|
|
d11878 |
|
|
|
d11878 |
if (!cpu_buffer->kaddr)
|
|
|
d11878 |
@@ -2212,7 +2408,7 @@ static int save_record_data(int fd, int
|
|
|
d11878 |
|
|
|
d11878 |
lseek(fd, offset, SEEK_SET);
|
|
|
d11878 |
for (i = 0; i < nr_cpu_ids; i++) {
|
|
|
d11878 |
- struct ring_buffer_per_cpu *cpu_buffer = &global_buffers[i];
|
|
|
d11878 |
+ struct ring_buffer_per_cpu *cpu_buffer = &ti->buffers[i];
|
|
|
d11878 |
|
|
|
d11878 |
if (!cpu_buffer->kaddr)
|
|
|
d11878 |
continue;
|
|
|
d11878 |
@@ -2231,13 +2427,13 @@ static int save_record_data(int fd, int
|
|
|
d11878 |
return 0;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
-static int __trace_cmd_data_output(int fd)
|
|
|
d11878 |
+static int get_nr_cpu_buffers(struct trace_instance *ti)
|
|
|
d11878 |
{
|
|
|
d11878 |
int i;
|
|
|
d11878 |
int nr_cpu_buffers = 0;
|
|
|
d11878 |
|
|
|
d11878 |
for (i = 0; i < nr_cpu_ids; i++) {
|
|
|
d11878 |
- struct ring_buffer_per_cpu *cpu_buffer = &global_buffers[i];
|
|
|
d11878 |
+ struct ring_buffer_per_cpu *cpu_buffer = &ti->buffers[i];
|
|
|
d11878 |
|
|
|
d11878 |
if (!cpu_buffer->kaddr)
|
|
|
d11878 |
continue;
|
|
|
d11878 |
@@ -2245,6 +2441,19 @@ static int __trace_cmd_data_output(int f
|
|
|
d11878 |
nr_cpu_buffers++;
|
|
|
d11878 |
}
|
|
|
d11878 |
|
|
|
d11878 |
+ return nr_cpu_buffers;
|
|
|
d11878 |
+}
|
|
|
d11878 |
+
|
|
|
d11878 |
+static int __trace_cmd_data_output(int fd)
|
|
|
d11878 |
+{
|
|
|
d11878 |
+ int nr_cpu_buffers;
|
|
|
d11878 |
+ unsigned long long global_res_data_offset;
|
|
|
d11878 |
+ unsigned long long *instance_offsets;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ instance_offsets = calloc(sizeof(unsigned long long), instance_count);
|
|
|
d11878 |
+
|
|
|
d11878 |
+ nr_cpu_buffers = get_nr_cpu_buffers(&global_trace_instance);
|
|
|
d11878 |
+
|
|
|
d11878 |
if (save_initial_data(fd))
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
if (save_header_files(fd))
|
|
|
d11878 |
@@ -2257,9 +2466,38 @@ static int __trace_cmd_data_output(int f
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
if (save_ftrace_cmdlines(fd))
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
- if (save_res_data(fd, nr_cpu_buffers))
|
|
|
d11878 |
+
|
|
|
d11878 |
+ /* We don't have the instance buffer offsets yet, so we'll write in 0s
|
|
|
d11878 |
+ * for now, and fix it up after we have that information available */
|
|
|
d11878 |
+ global_res_data_offset = lseek(fd, 0, SEEK_CUR);
|
|
|
d11878 |
+ if (save_res_data(fd, nr_cpu_buffers, NULL))
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
- if (save_record_data(fd, nr_cpu_buffers))
|
|
|
d11878 |
+ if (save_record_data(fd, nr_cpu_buffers, &global_trace_instance))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (multiple_instances_available)
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ int i;
|
|
|
d11878 |
+
|
|
|
d11878 |
+ for (i = 0; i < instance_count; i++)
|
|
|
d11878 |
+ {
|
|
|
d11878 |
+ struct trace_instance *ti = &trace_instances[i];
|
|
|
d11878 |
+ nr_cpu_buffers = get_nr_cpu_buffers(ti);
|
|
|
d11878 |
+
|
|
|
d11878 |
+ /* Save off the instance offset for fixup later */
|
|
|
d11878 |
+ instance_offsets[i] = lseek(fd, 0, SEEK_CUR);
|
|
|
d11878 |
+
|
|
|
d11878 |
+ if (write_and_check(fd, "flyrecord", 10))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+ if (save_record_data(fd, nr_cpu_buffers, ti))
|
|
|
d11878 |
+ return -1;
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+ }
|
|
|
d11878 |
+
|
|
|
d11878 |
+ /* Fix up the global trace's options header with the instance offsets */
|
|
|
d11878 |
+ lseek(fd, global_res_data_offset, SEEK_SET);
|
|
|
d11878 |
+ nr_cpu_buffers = get_nr_cpu_buffers(&global_trace_instance);
|
|
|
d11878 |
+ if (save_res_data(fd, nr_cpu_buffers, instance_offsets))
|
|
|
d11878 |
return -1;
|
|
|
d11878 |
|
|
|
d11878 |
return 0;
|