Add config compat support for detailed x86_32 syscalls
authorMathieu Desnoyers <mathieu.desnoyers@efficios.com>
Tue, 27 Sep 2011 16:24:09 +0000 (12:24 -0400)
committerMathieu Desnoyers <mathieu.desnoyers@efficios.com>
Tue, 27 Sep 2011 16:24:09 +0000 (12:24 -0400)
Signed-off-by: Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
instrumentation/syscalls/headers/compat_syscalls_integers.h [new file with mode: 0644]
instrumentation/syscalls/headers/compat_syscalls_pointers.h [new file with mode: 0644]
ltt-events.h
lttng-syscalls.c

diff --git a/instrumentation/syscalls/headers/compat_syscalls_integers.h b/instrumentation/syscalls/headers/compat_syscalls_integers.h
new file mode 100644 (file)
index 0000000..dabc4bf
--- /dev/null
@@ -0,0 +1,3 @@
+#ifdef CONFIG_X86_64
+#include "x86-32-syscalls-3.1.0-rc6_integers.h"
+#endif
diff --git a/instrumentation/syscalls/headers/compat_syscalls_pointers.h b/instrumentation/syscalls/headers/compat_syscalls_pointers.h
new file mode 100644 (file)
index 0000000..a84423c
--- /dev/null
@@ -0,0 +1,3 @@
+#ifdef CONFIG_X86_64
+#include "x86-32-syscalls-3.1.0-rc6_pointers.h"
+#endif
index ccb9df829094f60ad9c7b4a20adf7cae421b96b5..3ffdcf46045840037d81f1fd42aa7188c213a4e9 100644 (file)
@@ -246,6 +246,7 @@ struct ltt_channel {
        struct ltt_channel_ops *ops;
        struct ltt_transport *transport;
        struct ltt_event **sc_table;    /* for syscall tracing */
+       struct ltt_event **compat_sc_table;
        struct ltt_event *sc_unknown;   /* for unknown syscalls */
        struct ltt_event *sc_compat_unknown;
        struct ltt_event *sc_exit;      /* for syscall exit */
index b712502b362d6f68942d57cfff5ce9928e344df1..5135dfe2f88501f47c4436df3b42f878cbb6b800 100644 (file)
@@ -59,8 +59,6 @@ struct trace_syscall_entry {
        unsigned int nrargs;
 };
 
-#define CREATE_SYSCALL_TABLE
-
 #undef TRACE_SYSCALL_TABLE
 #define TRACE_SYSCALL_TABLE(_template, _name, _nr, _nrargs)    \
        [ _nr ] = {                                             \
@@ -70,11 +68,19 @@ struct trace_syscall_entry {
                .desc = &__event_desc___##_name,                \
        },
 
-static struct trace_syscall_entry sc_table[] = {
+#define CREATE_SYSCALL_TABLE
+
+static const struct trace_syscall_entry sc_table[] = {
 #include "instrumentation/syscalls/headers/syscalls_integers.h"
 #include "instrumentation/syscalls/headers/syscalls_pointers.h"
 };
 
+/* Create compatibility syscall table */
+static const struct trace_syscall_entry compat_sc_table[] = {
+#include "instrumentation/syscalls/headers/compat_syscalls_integers.h"
+#include "instrumentation/syscalls/headers/compat_syscalls_pointers.h"
+};
+
 #undef CREATE_SYSCALL_TABLE
 
 static void syscall_entry_unknown(struct ltt_event *event,
@@ -93,24 +99,33 @@ static void syscall_entry_unknown(struct ltt_event *event,
  */
 static void syscall_entry_probe(void *__data, struct pt_regs *regs, long id)
 {
-       struct trace_syscall_entry *entry;
        struct ltt_channel *chan = __data;
-       struct ltt_event *event;
+       struct ltt_event *event, *unknown_event;
+       const struct trace_syscall_entry *table, *entry;
+       size_t table_len;
 
        if (unlikely(is_compat_task())) {
-               syscall_entry_unknown(chan->sc_compat_unknown, regs, id);
-               return;
+               table = compat_sc_table;
+               table_len = ARRAY_SIZE(compat_sc_table);
+               unknown_event = chan->sc_compat_unknown;
+       } else {
+               table = sc_table;
+               table_len = ARRAY_SIZE(sc_table);
+               unknown_event = chan->sc_unknown;
        }
-       if (unlikely(id >= ARRAY_SIZE(sc_table))) {
-               syscall_entry_unknown(chan->sc_unknown, regs, id);
+       if (unlikely(id >= table_len)) {
+               syscall_entry_unknown(unknown_event, regs, id);
                return;
        }
-       event = chan->sc_table[id];
+       if (unlikely(is_compat_task()))
+               event = chan->compat_sc_table[id];
+       else
+               event = chan->sc_table[id];
        if (unlikely(!event)) {
-               syscall_entry_unknown(chan->sc_unknown, regs, id);
+               syscall_entry_unknown(unknown_event, regs, id);
                return;
        }
-       entry = &sc_table[id];
+       entry = &table[id];
        WARN_ON_ONCE(!entry);
 
        switch (entry->nrargs) {
@@ -201,9 +216,48 @@ static void syscall_entry_probe(void *__data, struct pt_regs *regs, long id)
        }
 }
 
-int lttng_syscalls_register(struct ltt_channel *chan, void *filter)
+static
+int fill_table(const struct trace_syscall_entry *table, size_t table_len,
+       struct ltt_event **chan_table, struct ltt_channel *chan, void *filter)
 {
        unsigned int i;
+
+       /* Allocate events for each syscall, insert into table */
+       for (i = 0; i < table_len; i++) {
+               struct lttng_kernel_event ev;
+               const struct lttng_event_desc *desc = table[i].desc;
+
+               if (!desc) {
+                       /* Unknown syscall */
+                       continue;
+               }
+               /*
+                * Skip those already populated by previous failed
+                * register for this channel.
+                */
+               if (chan_table[i])
+                       continue;
+               memset(&ev, 0, sizeof(ev));
+               strncpy(ev.name, desc->name, LTTNG_SYM_NAME_LEN);
+               ev.name[LTTNG_SYM_NAME_LEN - 1] = '\0';
+               ev.instrumentation = LTTNG_KERNEL_NOOP;
+               chan_table[i] = ltt_event_create(chan, &ev, filter,
+                                               desc);
+               if (!chan_table[i]) {
+                       /*
+                        * If something goes wrong in event registration
+                        * after the first one, we have no choice but to
+                        * leave the previous events in there, until
+                        * deleted by session teardown.
+                        */
+                       return -EINVAL;
+               }
+       }
+       return 0;
+}
+
+int lttng_syscalls_register(struct ltt_channel *chan, void *filter)
+{
        int ret;
 
        wrapper_vmalloc_sync_all();
@@ -216,6 +270,15 @@ int lttng_syscalls_register(struct ltt_channel *chan, void *filter)
                        return -ENOMEM;
        }
 
+#ifdef CONFIG_COMPAT
+       if (!chan->compat_sc_table) {
+               /* create syscall table mapping compat syscall to events */
+               chan->compat_sc_table = kzalloc(sizeof(struct ltt_event *)
+                                       * ARRAY_SIZE(compat_sc_table), GFP_KERNEL);
+               if (!chan->compat_sc_table)
+                       return -ENOMEM;
+       }
+#endif
        if (!chan->sc_unknown) {
                struct lttng_kernel_event ev;
                const struct lttng_event_desc *desc =
@@ -264,37 +327,16 @@ int lttng_syscalls_register(struct ltt_channel *chan, void *filter)
                }
        }
 
-       /* Allocate events for each syscall, insert into table */
-       for (i = 0; i < ARRAY_SIZE(sc_table); i++) {
-               struct lttng_kernel_event ev;
-               const struct lttng_event_desc *desc = sc_table[i].desc;
-
-               if (!desc) {
-                       /* Unknown syscall */
-                       continue;
-               }
-               /*
-                * Skip those already populated by previous failed
-                * register for this channel.
-                */
-               if (chan->sc_table[i])
-                       continue;
-               memset(&ev, 0, sizeof(ev));
-               strncpy(ev.name, desc->name, LTTNG_SYM_NAME_LEN);
-               ev.name[LTTNG_SYM_NAME_LEN - 1] = '\0';
-               ev.instrumentation = LTTNG_KERNEL_NOOP;
-               chan->sc_table[i] = ltt_event_create(chan, &ev, filter,
-                                                    desc);
-               if (!chan->sc_table[i]) {
-                       /*
-                        * If something goes wrong in event registration
-                        * after the first one, we have no choice but to
-                        * leave the previous events in there, until
-                        * deleted by session teardown.
-                        */
-                       return -EINVAL;
-               }
-       }
+       ret = fill_table(sc_table, ARRAY_SIZE(sc_table),
+                       chan->sc_table, chan, filter);
+       if (ret)
+               return ret;
+#ifdef CONFIG_COMPAT
+       ret = fill_table(compat_sc_table, ARRAY_SIZE(compat_sc_table),
+                       chan->compat_sc_table, chan, filter);
+       if (ret)
+               return ret;
+#endif
        ret = tracepoint_probe_register("sys_enter",
                        (void *) syscall_entry_probe, chan);
        if (ret)
@@ -333,5 +375,8 @@ int lttng_syscalls_unregister(struct ltt_channel *chan)
                return ret;
        /* ltt_event destroy will be performed by ltt_session_destroy() */
        kfree(chan->sc_table);
+#ifdef CONFIG_COMPAT
+       kfree(chan->compat_sc_table);
+#endif
        return 0;
 }
This page took 0.034562 seconds and 4 git commands to generate.