[PATCH v3 3/3] tools lib traceevent: Add support for __print_array()

2015-01-19 Thread Javi Merino
Trace can now generate traces with variable element size arrays.  Add support 
to parse them.

Cc: Namhyung Kim 
Cc: Arnaldo Carvalho de Melo 
Cc: Steven Rostedt 
Cc: Jiri Olsa 
Signed-off-by: Javi Merino 
---

Changes since v2[0]:
  - Avoid repeating the alloc and process of fields in prcoess_int_array()
  - print a warning if the size of the array is not valid

[0] http://thread.gmane.org/gmane.linux.kernel/1867165/focus=1867166

 tools/lib/traceevent/event-parse.c | 91 ++
 tools/lib/traceevent/event-parse.h |  8 
 2 files changed, 99 insertions(+)

diff --git a/tools/lib/traceevent/event-parse.c 
b/tools/lib/traceevent/event-parse.c
index 9d063b829907..8626c57cd769 100644
--- a/tools/lib/traceevent/event-parse.c
+++ b/tools/lib/traceevent/event-parse.c
@@ -757,6 +757,11 @@ static void free_arg(struct print_arg *arg)
free_arg(arg->hex.field);
free_arg(arg->hex.size);
break;
+   case PRINT_INT_ARRAY:
+   free_arg(arg->int_array.field);
+   free_arg(arg->int_array.size);
+   free_arg(arg->int_array.el_size);
+   break;
case PRINT_TYPE:
free(arg->typecast.type);
free_arg(arg->typecast.item);
@@ -2536,6 +2541,32 @@ out:
 }
 
 static enum event_type
+process_int_array(struct event_format *event, struct print_arg *arg, char 
**tok)
+{
+   memset(arg, 0, sizeof(*arg));
+   arg->type = PRINT_INT_ARRAY;
+
+   if (alloc_and_process_arg(event, ",", >int_array.field))
+   goto out;
+
+   if (alloc_and_process_arg(event, ",", >int_array.size))
+   goto free_field;
+
+   if (alloc_and_process_arg(event, ")", >int_array.el_size))
+   goto free_size;
+
+   return read_token_item(tok);
+
+free_size:
+free_arg(arg->int_array.size);
+free_field:
+   free_arg(arg->int_array.field);
+out:
+   *tok = NULL;
+   return EVENT_ERROR;
+}
+
+static enum event_type
 process_dynamic_array(struct event_format *event, struct print_arg *arg, char 
**tok)
 {
struct format_field *field;
@@ -2830,6 +2861,10 @@ process_function(struct event_format *event, struct 
print_arg *arg,
free_token(token);
return process_hex(event, arg, tok);
}
+   if (strcmp(token, "__print_array") == 0) {
+   free_token(token);
+   return process_int_array(event, arg, tok);
+   }
if (strcmp(token, "__get_str") == 0) {
free_token(token);
return process_str(event, arg, tok);
@@ -3358,6 +3393,7 @@ eval_num_arg(void *data, int size, struct event_format 
*event, struct print_arg
break;
case PRINT_FLAGS:
case PRINT_SYMBOL:
+   case PRINT_INT_ARRAY:
case PRINT_HEX:
break;
case PRINT_TYPE:
@@ -3768,6 +3804,52 @@ static void print_str_arg(struct trace_seq *s, void 
*data, int size,
}
break;
 
+   case PRINT_INT_ARRAY: {
+   void *num;
+   int el_size;
+
+   if (arg->int_array.field->type == PRINT_DYNAMIC_ARRAY) {
+   unsigned long offset;
+   offset = pevent_read_number(pevent,
+   data + 
arg->int_array.field->dynarray.field->offset,
+   arg->int_array.field->dynarray.field->size);
+   num = data + (offset & 0x);
+   } else {
+   field = arg->int_array.field->field.field;
+   if (!field) {
+   str = arg->int_array.field->field.name;
+   field = pevent_find_any_field(event, str);
+   if (!field)
+   goto out_warning_field;
+   arg->int_array.field->field.field = field;
+   }
+   num = data + field->offset;
+   }
+   len = eval_num_arg(data, size, event, arg->int_array.size);
+   el_size = eval_num_arg(data, size, event, 
arg->int_array.el_size);
+   el_size /= 8;
+   for (i = 0; i < len; i++) {
+   if (i)
+   trace_seq_putc(s, ' ');
+
+   if (el_size == 1) {
+   trace_seq_printf(s, "%u", *(uint8_t *)num);
+   } else if (el_size == 2) {
+   trace_seq_printf(s, "%u", *(uint16_t *)num);
+   } else if (el_size == 4) {
+   trace_seq_printf(s, "%u", *(uint32_t *)num);
+   } else if (el_size == 8) {
+   trace_seq_printf(s, "%llu", *(uint64_t *)num);
+   } else {
+   

[PATCH v3 3/3] tools lib traceevent: Add support for __print_array()

2015-01-19 Thread Javi Merino
Trace can now generate traces with variable element size arrays.  Add support 
to parse them.

Cc: Namhyung Kim namhy...@kernel.org
Cc: Arnaldo Carvalho de Melo a...@redhat.com
Cc: Steven Rostedt srost...@redhat.com
Cc: Jiri Olsa jo...@redhat.com
Signed-off-by: Javi Merino javi.mer...@arm.com
---

Changes since v2[0]:
  - Avoid repeating the alloc and process of fields in prcoess_int_array()
  - print a warning if the size of the array is not valid

[0] http://thread.gmane.org/gmane.linux.kernel/1867165/focus=1867166

 tools/lib/traceevent/event-parse.c | 91 ++
 tools/lib/traceevent/event-parse.h |  8 
 2 files changed, 99 insertions(+)

diff --git a/tools/lib/traceevent/event-parse.c 
b/tools/lib/traceevent/event-parse.c
index 9d063b829907..8626c57cd769 100644
--- a/tools/lib/traceevent/event-parse.c
+++ b/tools/lib/traceevent/event-parse.c
@@ -757,6 +757,11 @@ static void free_arg(struct print_arg *arg)
free_arg(arg-hex.field);
free_arg(arg-hex.size);
break;
+   case PRINT_INT_ARRAY:
+   free_arg(arg-int_array.field);
+   free_arg(arg-int_array.size);
+   free_arg(arg-int_array.el_size);
+   break;
case PRINT_TYPE:
free(arg-typecast.type);
free_arg(arg-typecast.item);
@@ -2536,6 +2541,32 @@ out:
 }
 
 static enum event_type
+process_int_array(struct event_format *event, struct print_arg *arg, char 
**tok)
+{
+   memset(arg, 0, sizeof(*arg));
+   arg-type = PRINT_INT_ARRAY;
+
+   if (alloc_and_process_arg(event, ,, arg-int_array.field))
+   goto out;
+
+   if (alloc_and_process_arg(event, ,, arg-int_array.size))
+   goto free_field;
+
+   if (alloc_and_process_arg(event, ), arg-int_array.el_size))
+   goto free_size;
+
+   return read_token_item(tok);
+
+free_size:
+free_arg(arg-int_array.size);
+free_field:
+   free_arg(arg-int_array.field);
+out:
+   *tok = NULL;
+   return EVENT_ERROR;
+}
+
+static enum event_type
 process_dynamic_array(struct event_format *event, struct print_arg *arg, char 
**tok)
 {
struct format_field *field;
@@ -2830,6 +2861,10 @@ process_function(struct event_format *event, struct 
print_arg *arg,
free_token(token);
return process_hex(event, arg, tok);
}
+   if (strcmp(token, __print_array) == 0) {
+   free_token(token);
+   return process_int_array(event, arg, tok);
+   }
if (strcmp(token, __get_str) == 0) {
free_token(token);
return process_str(event, arg, tok);
@@ -3358,6 +3393,7 @@ eval_num_arg(void *data, int size, struct event_format 
*event, struct print_arg
break;
case PRINT_FLAGS:
case PRINT_SYMBOL:
+   case PRINT_INT_ARRAY:
case PRINT_HEX:
break;
case PRINT_TYPE:
@@ -3768,6 +3804,52 @@ static void print_str_arg(struct trace_seq *s, void 
*data, int size,
}
break;
 
+   case PRINT_INT_ARRAY: {
+   void *num;
+   int el_size;
+
+   if (arg-int_array.field-type == PRINT_DYNAMIC_ARRAY) {
+   unsigned long offset;
+   offset = pevent_read_number(pevent,
+   data + 
arg-int_array.field-dynarray.field-offset,
+   arg-int_array.field-dynarray.field-size);
+   num = data + (offset  0x);
+   } else {
+   field = arg-int_array.field-field.field;
+   if (!field) {
+   str = arg-int_array.field-field.name;
+   field = pevent_find_any_field(event, str);
+   if (!field)
+   goto out_warning_field;
+   arg-int_array.field-field.field = field;
+   }
+   num = data + field-offset;
+   }
+   len = eval_num_arg(data, size, event, arg-int_array.size);
+   el_size = eval_num_arg(data, size, event, 
arg-int_array.el_size);
+   el_size /= 8;
+   for (i = 0; i  len; i++) {
+   if (i)
+   trace_seq_putc(s, ' ');
+
+   if (el_size == 1) {
+   trace_seq_printf(s, %u, *(uint8_t *)num);
+   } else if (el_size == 2) {
+   trace_seq_printf(s, %u, *(uint16_t *)num);
+   } else if (el_size == 4) {
+   trace_seq_printf(s, %u, *(uint32_t *)num);
+   } else if (el_size == 8) {
+   trace_seq_printf(s, %llu, *(uint64_t *)num);
+