[ltt-dev] [PATCH V1] ARM: Set bit 0 for thumb mode in kallsyms_lookup_name returned address
Mathieu Desnoyers
compudj at krystal.dyndns.org
Mon Sep 19 10:57:19 EDT 2011
* Avik Sil (avik.sil at linaro.org) wrote:
> This patch fixes the undefined instruction oops due to execution
> of thumb-2 code in ARM mode. The zero bit in the symbol address
> returned by kallsyms_lookup_name is not set, leading to switching
> to ARM mode that generates oops while executing thumb-2 code. For
> detailed discussion, see [1].
> [1] http://lists.casi.polymtl.ca/pipermail/ltt-dev/2011-September/005176.html
>
> v1:
> - include wrapper function kallsyms_lookup_funcptr as suggested
> by Dave Martin
>
> Signed-off-by: Avik Sil <avik.sil at linaro.org>
Merged, thanks!
Mathieu
> ---
> lttng-context-prio.c | 3 ++-
> wrapper/ftrace.h | 5 +++--
> wrapper/kallsyms.h | 28 ++++++++++++++++++++++++++++
> wrapper/splice.c | 3 ++-
> wrapper/vmalloc.h | 3 ++-
> 5 files changed, 37 insertions(+), 5 deletions(-)
> create mode 100644 wrapper/kallsyms.h
>
> diff --git a/lttng-context-prio.c b/lttng-context-prio.c
> index ad1c42f..1ee3a54 100644
> --- a/lttng-context-prio.c
> +++ b/lttng-context-prio.c
> @@ -13,6 +13,7 @@
> #include "ltt-events.h"
> #include "wrapper/ringbuffer/frontend_types.h"
> #include "wrapper/vmalloc.h"
> +#include "wrapper/kallsyms.h"
> #include "ltt-tracer.h"
>
> static
> @@ -20,7 +21,7 @@ int (*wrapper_task_prio_sym)(struct task_struct *t);
>
> int wrapper_task_prio_init(void)
> {
> - wrapper_task_prio_sym = (void *) kallsyms_lookup_name("task_prio");
> + wrapper_task_prio_sym = (void *) kallsyms_lookup_funcptr("task_prio");
> if (!wrapper_task_prio_sym) {
> printk(KERN_WARNING "LTTng: task_prio symbol lookup failed.\n");
> return -EINVAL;
> diff --git a/wrapper/ftrace.h b/wrapper/ftrace.h
> index 9c18cc5..ace33c5 100644
> --- a/wrapper/ftrace.h
> +++ b/wrapper/ftrace.h
> @@ -16,6 +16,7 @@
> #ifdef CONFIG_KALLSYMS
>
> #include <linux/kallsyms.h>
> +#include "kallsyms.h"
>
> static inline
> int wrapper_register_ftrace_function_probe(char *glob,
> @@ -24,7 +25,7 @@ int wrapper_register_ftrace_function_probe(char *glob,
> int (*register_ftrace_function_probe_sym)(char *glob,
> struct ftrace_probe_ops *ops, void *data);
>
> - register_ftrace_function_probe_sym = (void *) kallsyms_lookup_name("register_ftrace_function_probe");
> + register_ftrace_function_probe_sym = (void *) kallsyms_lookup_funcptr("register_ftrace_function_probe");
> if (register_ftrace_function_probe_sym) {
> return register_ftrace_function_probe_sym(glob, ops, data);
> } else {
> @@ -40,7 +41,7 @@ void wrapper_unregister_ftrace_function_probe(char *glob,
> void (*unregister_ftrace_function_probe_sym)(char *glob,
> struct ftrace_probe_ops *ops, void *data);
>
> - unregister_ftrace_function_probe_sym = (void *) kallsyms_lookup_name("unregister_ftrace_function_probe");
> + unregister_ftrace_function_probe_sym = (void *) kallsyms_lookup_funcptr("unregister_ftrace_function_probe");
> if (unregister_ftrace_function_probe_sym) {
> unregister_ftrace_function_probe_sym(glob, ops, data);
> } else {
> diff --git a/wrapper/kallsyms.h b/wrapper/kallsyms.h
> new file mode 100644
> index 0000000..bb45f38
> --- /dev/null
> +++ b/wrapper/kallsyms.h
> @@ -0,0 +1,28 @@
> +#ifndef _LTT_WRAPPER_KALLSYMS_H
> +#define _LTT_WRAPPER_KALLSYMS_H
> +
> +/*
> + * Copyright (C) 2011 Avik Sil (avik.sil at linaro.org)
> + *
> + * wrapper around kallsyms_lookup_name. Implements arch-dependent code for
> + * arches where the address of the start of the function body is different
> + * from the pointer which can be used to call the function, e.g. ARM THUMB2.
> + *
> + * Dual LGPL v2.1/GPL v2 license.
> + */
> +
> +static inline
> +unsigned long kallsyms_lookup_funcptr(const char *name)
> +{
> + unsigned long addr;
> +
> + addr = kallsyms_lookup_name(name);
> +#ifdef CONFIG_ARM
> +#ifdef CONFIG_THUMB2_KERNEL
> + if (addr)
> + addr |= 1; /* set bit 0 in address for thumb mode */
> +#endif
> +#endif
> + return addr;
> +}
> +#endif /* _LTT_WRAPPER_KALLSYMS_H */
> diff --git a/wrapper/splice.c b/wrapper/splice.c
> index edc499c..ba224ee 100644
> --- a/wrapper/splice.c
> +++ b/wrapper/splice.c
> @@ -13,6 +13,7 @@
> #include <linux/kallsyms.h>
> #include <linux/fs.h>
> #include <linux/splice.h>
> +#include "kallsyms.h"
>
> static
> ssize_t (*splice_to_pipe_sym)(struct pipe_inode_info *pipe,
> @@ -22,7 +23,7 @@ ssize_t wrapper_splice_to_pipe(struct pipe_inode_info *pipe,
> struct splice_pipe_desc *spd)
> {
> if (!splice_to_pipe_sym)
> - splice_to_pipe_sym = (void *) kallsyms_lookup_name("splice_to_pipe");
> + splice_to_pipe_sym = (void *) kallsyms_lookup_funcptr("splice_to_pipe");
> if (splice_to_pipe_sym) {
> return splice_to_pipe_sym(pipe, spd);
> } else {
> diff --git a/wrapper/vmalloc.h b/wrapper/vmalloc.h
> index 7d87855..765f2ad 100644
> --- a/wrapper/vmalloc.h
> +++ b/wrapper/vmalloc.h
> @@ -14,13 +14,14 @@
> #ifdef CONFIG_KALLSYMS
>
> #include <linux/kallsyms.h>
> +#include "kallsyms.h"
>
> static inline
> void wrapper_vmalloc_sync_all(void)
> {
> void (*vmalloc_sync_all_sym)(void);
>
> - vmalloc_sync_all_sym = (void *) kallsyms_lookup_name("vmalloc_sync_all");
> + vmalloc_sync_all_sym = (void *) kallsyms_lookup_funcptr("vmalloc_sync_all");
> if (vmalloc_sync_all_sym) {
> vmalloc_sync_all_sym();
> } else {
> --
> 1.7.0.4
>
>
> _______________________________________________
> ltt-dev mailing list
> ltt-dev at lists.casi.polymtl.ca
> http://lists.casi.polymtl.ca/cgi-bin/mailman/listinfo/ltt-dev
>
--
Mathieu Desnoyers
Operating System Efficiency R&D Consultant
EfficiOS Inc.
http://www.efficios.com
More information about the lttng-dev
mailing list