Backporting options support for traceevent lib.

It's now possible to use following interface
to load options for 'struct pevent' object:

void traceevent_add_options(const char *name, struct plugin_option *options);
  - adds a set of options by a plugin

void traceevent_remove_options(struct plugin_option *options);
  - removes plugin options that were registered

void traceevent_add_option(const char *name, const char *val);
  - modifies plugin option

Signed-off-by: Jiri Olsa <jo...@redhat.com>
Cc: Corey Ashford <cjash...@linux.vnet.ibm.com>
Cc: Frederic Weisbecker <fweis...@gmail.com>
Cc: Ingo Molnar <mi...@elte.hu>
Cc: Namhyung Kim <namhy...@kernel.org>
Cc: Paul Mackerras <pau...@samba.org>
Cc: Peter Zijlstra <a.p.zijls...@chello.nl>
Cc: Arnaldo Carvalho de Melo <a...@redhat.com>
Cc: Steven Rostedt <rost...@goodmis.org>
Cc: David Ahern <dsah...@gmail.com>
---
 tools/lib/traceevent/Makefile       |   2 +-
 tools/lib/traceevent/event-option.c | 278 ++++++++++++++++++++++++++++++++++++
 tools/lib/traceevent/event-parse.h  |   4 +
 3 files changed, 283 insertions(+), 1 deletion(-)
 create mode 100644 tools/lib/traceevent/event-option.c

diff --git a/tools/lib/traceevent/Makefile b/tools/lib/traceevent/Makefile
index 539f3e5..99da843 100644
--- a/tools/lib/traceevent/Makefile
+++ b/tools/lib/traceevent/Makefile
@@ -180,7 +180,7 @@ $(obj)/%.o: $(src)/%.c
 %.o: $(src)/%.c
        $(Q)$(call do_compile)
 
-PEVENT_LIB_OBJS = event-parse.o event-plugin.o trace-seq.o parse-filter.o 
parse-utils.o
+PEVENT_LIB_OBJS = event-parse.o event-plugin.o trace-seq.o parse-filter.o 
parse-utils.o event-option.o
 PEVENT_LIB_OBJS += kbuffer-parse.o
 
 ALL_OBJS = $(PEVENT_LIB_OBJS)
diff --git a/tools/lib/traceevent/event-option.c 
b/tools/lib/traceevent/event-option.c
new file mode 100644
index 0000000..61181be
--- /dev/null
+++ b/tools/lib/traceevent/event-option.c
@@ -0,0 +1,278 @@
+
+#include <stdlib.h>
+#include <string.h>
+#include "event-parse.h"
+#include "event-utils.h"
+
+static struct trace_plugin_options {
+       struct trace_plugin_options     *next;
+       char                            *plugin;
+       char                            *option;
+       char                            *value;
+} *trace_plugin_options;
+
+static struct registered_plugin_options {
+       struct registered_plugin_options        *next;
+       struct plugin_option                    *options;
+} *registered_options;
+
+static void update_option(const char *file, struct plugin_option *option)
+{
+       struct trace_plugin_options *op;
+       char *plugin;
+
+       if (option->plugin_alias) {
+               plugin = strdup(option->plugin_alias);
+               if (!plugin)
+                       die("malloc");
+       } else {
+               char *p;
+               plugin = strdup(file);
+               if (!plugin)
+                       die("malloc");
+               p = strstr(plugin, ".");
+               if (p)
+                       *p = '\0';
+       }
+
+       /* first look for named options */
+       for (op = trace_plugin_options; op; op = op->next) {
+               if (!op->plugin)
+                       continue;
+               if (strcmp(op->plugin, plugin) != 0)
+                       continue;
+               if (strcmp(op->option, option->name) != 0)
+                       continue;
+
+               option->value = op->value;
+               option->set ^= 1;
+               goto out;
+       }
+
+       /* first look for unnamed options */
+       for (op = trace_plugin_options; op; op = op->next) {
+               if (op->plugin)
+                       continue;
+               if (strcmp(op->option, option->name) != 0)
+                       continue;
+
+               option->value = op->value;
+               option->set ^= 1;
+               break;
+       }
+
+ out:
+       free(plugin);
+}
+
+/**
+ * trace_util_add_options - Add a set of options by a plugin
+ * @name: The name of the plugin adding the options
+ * @options: The set of options being loaded
+ *
+ * Sets the options with the values that have been added by user.
+ */
+void traceevent_add_options(const char *name, struct plugin_option *options)
+{
+       struct registered_plugin_options *reg;
+
+       reg = malloc_or_die(sizeof(*reg));
+       reg->next = registered_options;
+       reg->options = options;
+       registered_options = reg;
+
+       while (options->name) {
+               update_option("ftrace", options);
+               options++;
+       }
+}
+
+/**
+ * trace_util_remove_options - remove plugin options that were registered
+ * @options: Options to removed that were registered with 
trace_util_add_options
+ */
+void traceevent_remove_options(struct plugin_option *options)
+{
+       struct registered_plugin_options **last;
+       struct registered_plugin_options *reg;
+
+       for (last = &registered_options; *last; last = &(*last)->next) {
+               if ((*last)->options == options) {
+                       reg = *last;
+                       *last = reg->next;
+                       free(reg);
+                       return;
+               }
+       }
+}
+
+static struct plugin_option *
+find_registered_option(const char *plugin, const char *option)
+{
+       struct registered_plugin_options *reg;
+       struct plugin_option *op;
+       const char *op_plugin;
+
+       for (reg = registered_options; reg; reg = reg->next) {
+               for (op = reg->options; op->name; op++) {
+                       if (op->plugin_alias)
+                               op_plugin = op->plugin_alias;
+                       else
+                               op_plugin = op->file;
+
+                       if (plugin && strcmp(plugin, op_plugin) != 0)
+                               continue;
+                       if (strcmp(option, op->name) != 0)
+                               continue;
+
+                       return op;
+               }
+       }
+
+       return NULL;
+}
+
+static void lower_case(char *str)
+{
+       if (!str)
+               return;
+       for (; *str; str++)
+               *str = tolower(*str);
+}
+
+static int update_option_value(struct plugin_option *op, const char *val)
+{
+       char *op_val;
+       int ret = 1;
+
+       if (!val) {
+               /* toggle, only if option is boolean */
+               if (op->value)
+                       /* Warn? */
+                       return 0;
+               op->set ^= 1;
+               return 1;
+       }
+
+       /*
+        * If the option has a value then it takes a string
+        * otherwise the option is a boolean.
+        */
+       if (op->value) {
+               op->value = (char *) val;
+               return 1;
+       }
+
+       /* Option is boolean, must be either "1", "0", "true" or "false" */
+
+       op_val = strdup(val);
+       if (!op_val)
+               die("malloc");
+       lower_case(op_val);
+
+       if (strcmp(val, "1") == 0 || strcmp(val, "true") == 0)
+               op->set = 1;
+       else if (strcmp(val, "0") == 0 || strcmp(val, "false") == 0)
+               op->set = 0;
+       else
+               /* Warn on else? */
+               ret = 0;
+       free(op_val);
+
+       return ret;
+}
+
+static int process_option(const char *plugin, const char *option, const char 
*val)
+{
+       struct plugin_option *op;
+
+       op = find_registered_option(plugin, option);
+       if (!op)
+               return 0;
+
+       return update_option_value(op, val);
+}
+
+static void parse_option_name(char **option, char **plugin)
+{
+       char *p;
+
+       *plugin = NULL;
+
+       if ((p = strstr(*option, ":"))) {
+               *plugin = *option;
+               *p = '\0';
+               *option = strdup(p + 1);
+               if (!*option)
+                       die("malloc");
+       }
+}
+
+/**
+ * trace_util_add_option - add an option/val pair to set plugin options
+ * @name: The name of the option (format: <plugin>:<option> or just <option>)
+ * @val: (optiona) the value for the option
+ *
+ * Modify a plugin option. If @val is given than the value of the option
+ * is set (note, some options just take a boolean, so @val must be either
+ * "1" or "0" or "true" or "false").
+ */
+void traceevent_add_option(const char *name, const char *val)
+{
+       struct trace_plugin_options *op;
+       char *option_str;
+       char *plugin;
+
+       option_str = strdup(name);
+       if (!option_str)
+               die("malloc");
+
+       parse_option_name(&option_str, &plugin);
+
+       /* If the option exists, update the val */
+       for (op = trace_plugin_options; op; op = op->next) {
+               /* Both must be NULL or not NULL */
+               if ((!plugin || !op->plugin) && plugin != op->plugin)
+                       continue;
+               if (plugin && strcmp(plugin, op->plugin) != 0)
+                       continue;
+               if (strcmp(op->option, option_str) != 0)
+                       continue;
+
+               /* update option */
+               free(op->value);
+               if (val) {
+                       op->value = strdup(val);
+                       if (!op->value)
+                               die("malloc");
+               } else
+                       op->value = NULL;
+
+               /* plugin and option_str don't get freed at the end */
+               free(plugin);
+               free(option_str);
+
+               plugin = op->plugin;
+               option_str = op->option;
+               break;
+       }
+
+       /* If not found, create */
+       if (!op) {
+               op = malloc_or_die(sizeof(*op));
+               memset(op, 0, sizeof(*op));
+               op->next = trace_plugin_options;
+               trace_plugin_options = op;
+
+               op->plugin = plugin;
+               op->option = option_str;
+
+               if (val) {
+                       op->value = strdup(val);
+                       if (!op->value)
+                               die("malloc");
+               }
+       }
+
+       process_option(plugin, option_str, val);
+}
diff --git a/tools/lib/traceevent/event-parse.h 
b/tools/lib/traceevent/event-parse.h
index 8273c6e..3aaf668 100644
--- a/tools/lib/traceevent/event-parse.h
+++ b/tools/lib/traceevent/event-parse.h
@@ -110,6 +110,10 @@ struct plugin_option {
        int                             set;
 };
 
+void traceevent_add_options(const char *name, struct plugin_option *options);
+void traceevent_remove_options(struct plugin_option *options);
+void traceevent_add_option(const char *name, const char *val);
+
 /*
  * Plugin hooks that can be called:
  *
-- 
1.7.11.7

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to