tools lib traceevent: Introduced new traceevent API, for adding new plugins directories.
authorTzvetomir Stoyanov (VMware) <tz.stoyanov@gmail.com>
Thu, 2 Jul 2020 18:53:49 +0000 (14:53 -0400)
committerArnaldo Carvalho de Melo <acme@redhat.com>
Mon, 6 Jul 2020 11:35:23 +0000 (08:35 -0300)
Implement new traceevent plugin API, which can be used to add new plugins
directories:

enum tep_plugin_load_priority {
TEP_PLUGIN_FIRST,
TEP_PLUGIN_LAST,
};

int tep_add_plugin_path(struct tep_handle *tep, char *path,
enum tep_plugin_load_priority prio);

It adds the "path" as new plugin directory, in the context of the
handler "tep". The tep_load_plugins() API searches for plugins in this
new location. Depending of the priority "prio", the plugins from this
directory are loaded before (TEP_PLUGIN_FIRST) or after
(TEP_PLUGIN_LAST) the ordinary libtraceevent plugin locations.

Link: http://lore.kernel.org/linux-trace-devel/20191007114947.17104-2-tz.stoyanov@gmail.com
Link: http://lore.kernel.org/linux-trace-devel/20200625100516.365338-6-tz.stoyanov@gmail.com
Signed-off-by: Tzvetomir Stoyanov (VMware) <tz.stoyanov@gmail.com>
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: linux-trace-devel@vger.kernel.org
Link: http://lore.kernel.org/lkml/20200702185704.248123446@goodmis.org
Signed-off-by: Steven Rostedt (VMware) <rostedt@goodmis.org>
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
tools/lib/traceevent/event-parse-local.h
tools/lib/traceevent/event-parse.c
tools/lib/traceevent/event-parse.h
tools/lib/traceevent/event-plugin.c

index cee4698..96a0b0c 100644 (file)
@@ -13,6 +13,7 @@ struct func_map;
 struct func_list;
 struct event_handler;
 struct func_resolver;
+struct tep_plugins_dir;
 
 struct tep_handle {
        int ref_count;
@@ -47,7 +48,6 @@ struct tep_handle {
        struct printk_list *printklist;
        unsigned int printk_count;
 
-
        struct tep_event **events;
        int nr_events;
        struct tep_event **sort_events;
@@ -81,10 +81,13 @@ struct tep_handle {
 
        /* cache */
        struct tep_event *last_event;
+
+       struct tep_plugins_dir *plugins_dir;
 };
 
 void tep_free_event(struct tep_event *event);
 void tep_free_format_field(struct tep_format_field *field);
+void tep_free_plugin_paths(struct tep_handle *tep);
 
 unsigned short tep_data2host2(struct tep_handle *tep, unsigned short data);
 unsigned int tep_data2host4(struct tep_handle *tep, unsigned int data);
index 5b36c58..629437d 100644 (file)
@@ -7119,6 +7119,7 @@ void tep_free(struct tep_handle *tep)
        free(tep->events);
        free(tep->sort_events);
        free(tep->func_resolver);
+       tep_free_plugin_paths(tep);
 
        free(tep);
 }
index 02c0438..91f462f 100644 (file)
@@ -393,6 +393,13 @@ struct tep_plugin_list;
 
 #define INVALID_PLUGIN_LIST_OPTION     ((char **)((unsigned long)-1))
 
+enum tep_plugin_load_priority {
+       TEP_PLUGIN_FIRST,
+       TEP_PLUGIN_LAST,
+};
+
+int tep_add_plugin_path(struct tep_handle *tep, char *path,
+                       enum tep_plugin_load_priority prio);
 struct tep_plugin_list *tep_load_plugins(struct tep_handle *tep);
 void tep_unload_plugins(struct tep_plugin_list *plugin_list,
                        struct tep_handle *tep);
index e8f4329..1d4f180 100644 (file)
@@ -39,6 +39,12 @@ struct tep_plugin_list {
        void                    *handle;
 };
 
+struct tep_plugins_dir {
+       struct tep_plugins_dir          *next;
+       char                            *path;
+       enum tep_plugin_load_priority   prio;
+};
+
 static void lower_case(char *str)
 {
        if (!str)
@@ -544,6 +550,7 @@ void tep_load_plugins_hook(struct tep_handle *tep, const char *suffix,
                                               void *data),
                           void *data)
 {
+       struct tep_plugins_dir *dir = NULL;
        char *home;
        char *path;
        char *envdir;
@@ -552,6 +559,15 @@ void tep_load_plugins_hook(struct tep_handle *tep, const char *suffix,
        if (tep && tep->flags & TEP_DISABLE_PLUGINS)
                return;
 
+       if (tep)
+               dir = tep->plugins_dir;
+       while (dir) {
+               if (dir->prio == TEP_PLUGIN_FIRST)
+                       load_plugins_dir(tep, suffix, dir->path,
+                                        load_plugin, data);
+               dir = dir->next;
+       }
+
        /*
         * If a system plugin directory was defined,
         * check that first.
@@ -586,6 +602,15 @@ void tep_load_plugins_hook(struct tep_handle *tep, const char *suffix,
 
        load_plugins_dir(tep, suffix, path, load_plugin, data);
 
+       if (tep)
+               dir = tep->plugins_dir;
+       while (dir) {
+               if (dir->prio == TEP_PLUGIN_LAST)
+                       load_plugins_dir(tep, suffix, dir->path,
+                                        load_plugin, data);
+               dir = dir->next;
+       }
+
        free(path);
 }
 
@@ -598,6 +623,51 @@ tep_load_plugins(struct tep_handle *tep)
        return list;
 }
 
+/**
+ * tep_add_plugin_path - Add a new plugin directory.
+ * @tep: Trace event handler.
+ * @path: Path to a directory. All files with extension .so in that
+ *       directory will be loaded as plugins.
+ *@prio: Load priority of the plugins in that directory.
+ *
+ * Returns -1 in case of an error, 0 otherwise.
+ */
+int tep_add_plugin_path(struct tep_handle *tep, char *path,
+                       enum tep_plugin_load_priority prio)
+{
+       struct tep_plugins_dir *dir;
+
+       if (!tep || !path)
+               return -1;
+
+       dir = calloc(1, sizeof(*dir));
+       if (!dir)
+               return -1;
+
+       dir->path = strdup(path);
+       dir->prio = prio;
+       dir->next = tep->plugins_dir;
+       tep->plugins_dir = dir;
+
+       return 0;
+}
+
+void tep_free_plugin_paths(struct tep_handle *tep)
+{
+       struct tep_plugins_dir *dir;
+
+       if (!tep)
+               return;
+
+       dir = tep->plugins_dir;
+       while (dir) {
+               tep->plugins_dir = tep->plugins_dir->next;
+               free(dir->path);
+               free(dir);
+               dir = tep->plugins_dir;
+       }
+}
+
 void
 tep_unload_plugins(struct tep_plugin_list *plugin_list, struct tep_handle *tep)
 {