diff options
author | Wang Nan <wangnan0@huawei.com> | 2016-06-24 07:22:10 -0400 |
---|---|---|
committer | Arnaldo Carvalho de Melo <acme@redhat.com> | 2016-06-28 09:54:57 -0400 |
commit | f5a08ceda55bee91f879d2ac19edeb4a8916d04f (patch) | |
tree | 705f7815e5b26a3a6ecd049061942e20afc7c93e | |
parent | 8ee4c46c5ec2481dd18098c5604f791ff911d427 (diff) |
perf data ctf: Generate comm event to CTF output
If 'all' is selected, convert comm event to output CTF stream.
setup_non_sample_events() is called if non_sample is selected. It
creates a comm_class for comm event.
Use macros to generate and process_comm_event and add_comm_event. These
macros can be reused for other non-sample events.
Signed-off-by: Wang Nan <wangnan0@huawei.com>
Acked-by: Jiri Olsa <jolsa@kernel.org>
Cc: Zefan Li <lizefan@huawei.com>
Cc: pi3orama@163.com
Link: http://lkml.kernel.org/r/1466767332-114472-6-git-send-email-wangnan0@huawei.com
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
-rw-r--r-- | tools/perf/util/data-convert-bt.c | 110 |
1 files changed, 110 insertions, 0 deletions
diff --git a/tools/perf/util/data-convert-bt.c b/tools/perf/util/data-convert-bt.c index 3b3ac7c143e1..5dd62ba07438 100644 --- a/tools/perf/util/data-convert-bt.c +++ b/tools/perf/util/data-convert-bt.c | |||
@@ -69,6 +69,7 @@ struct ctf_writer { | |||
69 | }; | 69 | }; |
70 | struct bt_ctf_field_type *array[6]; | 70 | struct bt_ctf_field_type *array[6]; |
71 | } data; | 71 | } data; |
72 | struct bt_ctf_event_class *comm_class; | ||
72 | }; | 73 | }; |
73 | 74 | ||
74 | struct convert { | 75 | struct convert { |
@@ -763,6 +764,57 @@ static int process_sample_event(struct perf_tool *tool, | |||
763 | return cs ? 0 : -1; | 764 | return cs ? 0 : -1; |
764 | } | 765 | } |
765 | 766 | ||
767 | #define __NON_SAMPLE_SET_FIELD(_name, _type, _field) \ | ||
768 | do { \ | ||
769 | ret = value_set_##_type(cw, event, #_field, _event->_name._field);\ | ||
770 | if (ret) \ | ||
771 | return -1; \ | ||
772 | } while(0) | ||
773 | |||
774 | #define __FUNC_PROCESS_NON_SAMPLE(_name, body) \ | ||
775 | static int process_##_name##_event(struct perf_tool *tool, \ | ||
776 | union perf_event *_event, \ | ||
777 | struct perf_sample *sample, \ | ||
778 | struct machine *machine) \ | ||
779 | { \ | ||
780 | struct convert *c = container_of(tool, struct convert, tool);\ | ||
781 | struct ctf_writer *cw = &c->writer; \ | ||
782 | struct bt_ctf_event_class *event_class = cw->_name##_class;\ | ||
783 | struct bt_ctf_event *event; \ | ||
784 | struct ctf_stream *cs; \ | ||
785 | int ret; \ | ||
786 | \ | ||
787 | c->non_sample_count++; \ | ||
788 | c->events_size += _event->header.size; \ | ||
789 | event = bt_ctf_event_create(event_class); \ | ||
790 | if (!event) { \ | ||
791 | pr_err("Failed to create an CTF event\n"); \ | ||
792 | return -1; \ | ||
793 | } \ | ||
794 | \ | ||
795 | bt_ctf_clock_set_time(cw->clock, sample->time); \ | ||
796 | body \ | ||
797 | cs = ctf_stream(cw, 0); \ | ||
798 | if (cs) { \ | ||
799 | if (is_flush_needed(cs)) \ | ||
800 | ctf_stream__flush(cs); \ | ||
801 | \ | ||
802 | cs->count++; \ | ||
803 | bt_ctf_stream_append_event(cs->stream, event); \ | ||
804 | } \ | ||
805 | bt_ctf_event_put(event); \ | ||
806 | \ | ||
807 | return perf_event__process_##_name(tool, _event, sample, machine);\ | ||
808 | } | ||
809 | |||
810 | __FUNC_PROCESS_NON_SAMPLE(comm, | ||
811 | __NON_SAMPLE_SET_FIELD(comm, u32, pid); | ||
812 | __NON_SAMPLE_SET_FIELD(comm, u32, tid); | ||
813 | __NON_SAMPLE_SET_FIELD(comm, string, comm); | ||
814 | ) | ||
815 | #undef __NON_SAMPLE_SET_FIELD | ||
816 | #undef __FUNC_PROCESS_NON_SAMPLE | ||
817 | |||
766 | /* If dup < 0, add a prefix. Else, add _dupl_X suffix. */ | 818 | /* If dup < 0, add a prefix. Else, add _dupl_X suffix. */ |
767 | static char *change_name(char *name, char *orig_name, int dup) | 819 | static char *change_name(char *name, char *orig_name, int dup) |
768 | { | 820 | { |
@@ -1037,6 +1089,58 @@ static int setup_events(struct ctf_writer *cw, struct perf_session *session) | |||
1037 | return 0; | 1089 | return 0; |
1038 | } | 1090 | } |
1039 | 1091 | ||
1092 | #define __NON_SAMPLE_ADD_FIELD(t, n) \ | ||
1093 | do { \ | ||
1094 | pr2(" field '%s'\n", #n); \ | ||
1095 | if (bt_ctf_event_class_add_field(event_class, cw->data.t, #n)) {\ | ||
1096 | pr_err("Failed to add field '%s';\n", #n);\ | ||
1097 | return -1; \ | ||
1098 | } \ | ||
1099 | } while(0) | ||
1100 | |||
1101 | #define __FUNC_ADD_NON_SAMPLE_EVENT_CLASS(_name, body) \ | ||
1102 | static int add_##_name##_event(struct ctf_writer *cw) \ | ||
1103 | { \ | ||
1104 | struct bt_ctf_event_class *event_class; \ | ||
1105 | int ret; \ | ||
1106 | \ | ||
1107 | pr("Adding "#_name" event\n"); \ | ||
1108 | event_class = bt_ctf_event_class_create("perf_" #_name);\ | ||
1109 | if (!event_class) \ | ||
1110 | return -1; \ | ||
1111 | body \ | ||
1112 | \ | ||
1113 | ret = bt_ctf_stream_class_add_event_class(cw->stream_class, event_class);\ | ||
1114 | if (ret) { \ | ||
1115 | pr("Failed to add event class '"#_name"' into stream.\n");\ | ||
1116 | return ret; \ | ||
1117 | } \ | ||
1118 | \ | ||
1119 | cw->_name##_class = event_class; \ | ||
1120 | bt_ctf_event_class_put(event_class); \ | ||
1121 | return 0; \ | ||
1122 | } | ||
1123 | |||
1124 | __FUNC_ADD_NON_SAMPLE_EVENT_CLASS(comm, | ||
1125 | __NON_SAMPLE_ADD_FIELD(u32, pid); | ||
1126 | __NON_SAMPLE_ADD_FIELD(u32, tid); | ||
1127 | __NON_SAMPLE_ADD_FIELD(string, comm); | ||
1128 | ) | ||
1129 | |||
1130 | #undef __NON_SAMPLE_ADD_FIELD | ||
1131 | #undef __FUNC_ADD_NON_SAMPLE_EVENT_CLASS | ||
1132 | |||
1133 | static int setup_non_sample_events(struct ctf_writer *cw, | ||
1134 | struct perf_session *session __maybe_unused) | ||
1135 | { | ||
1136 | int ret; | ||
1137 | |||
1138 | ret = add_comm_event(cw); | ||
1139 | if (ret) | ||
1140 | return ret; | ||
1141 | return 0; | ||
1142 | } | ||
1143 | |||
1040 | static void cleanup_events(struct perf_session *session) | 1144 | static void cleanup_events(struct perf_session *session) |
1041 | { | 1145 | { |
1042 | struct perf_evlist *evlist = session->evlist; | 1146 | struct perf_evlist *evlist = session->evlist; |
@@ -1332,6 +1436,9 @@ int bt_convert__perf2ctf(const char *input, const char *path, | |||
1332 | struct ctf_writer *cw = &c.writer; | 1436 | struct ctf_writer *cw = &c.writer; |
1333 | int err = -1; | 1437 | int err = -1; |
1334 | 1438 | ||
1439 | if (opts->all) | ||
1440 | c.tool.comm = process_comm_event; | ||
1441 | |||
1335 | perf_config(convert__config, &c); | 1442 | perf_config(convert__config, &c); |
1336 | 1443 | ||
1337 | /* CTF writer */ | 1444 | /* CTF writer */ |
@@ -1356,6 +1463,9 @@ int bt_convert__perf2ctf(const char *input, const char *path, | |||
1356 | if (setup_events(cw, session)) | 1463 | if (setup_events(cw, session)) |
1357 | goto free_session; | 1464 | goto free_session; |
1358 | 1465 | ||
1466 | if (opts->all && setup_non_sample_events(cw, session)) | ||
1467 | goto free_session; | ||
1468 | |||
1359 | if (setup_streams(cw, session)) | 1469 | if (setup_streams(cw, session)) |
1360 | goto free_session; | 1470 | goto free_session; |
1361 | 1471 | ||