s390/dumpstack: implement and use return_address()
authorHeiko Carstens <heiko.carstens@de.ibm.com>
Mon, 2 May 2016 12:38:29 +0000 (14:38 +0200)
committerMartin Schwidefsky <schwidefsky@de.ibm.com>
Wed, 4 May 2016 14:29:45 +0000 (16:29 +0200)
Implement return_address() and use it instead of __builtin_return_address(n).

__builtin_return_address(n) is not guaranteed to work for n > 0,
therefore implement a private return_address() function which walks
the stack frames and returns the proper return address.

This way we get also rid of a compile warning which gcc 6.1 emits and
look like all other architectures.

Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
arch/s390/include/asm/ftrace.h
arch/s390/kernel/dumpstack.c

index 836c56290499b84c0dad7785e9aa5979d68ed0bf..64053d9ac3f23b7cb1bf23aa42aa435b9f2ae0f8 100644 (file)
@@ -12,7 +12,9 @@
 
 #ifndef __ASSEMBLY__
 
-#define ftrace_return_address(n) __builtin_return_address(n)
+unsigned long return_address(int depth);
+
+#define ftrace_return_address(n) return_address(n)
 
 void _mcount(void);
 void ftrace_caller(void);
index 1b6081c0aff9238641f995c973560d2f8d33b686..69f9908ac44ccc18b04d5946a50364e40d75be91 100644 (file)
@@ -89,6 +89,30 @@ void dump_trace(dump_trace_func_t func, void *data, struct task_struct *task,
 }
 EXPORT_SYMBOL_GPL(dump_trace);
 
+struct return_address_data {
+       unsigned long address;
+       int depth;
+};
+
+static int __return_address(void *data, unsigned long address)
+{
+       struct return_address_data *rd = data;
+
+       if (rd->depth--)
+               return 0;
+       rd->address = address;
+       return 1;
+}
+
+unsigned long return_address(int depth)
+{
+       struct return_address_data rd = { .depth = depth + 2 };
+
+       dump_trace(__return_address, &rd, NULL, current_stack_pointer());
+       return rd.address;
+}
+EXPORT_SYMBOL_GPL(return_address);
+
 static int show_address(void *data, unsigned long address)
 {
        printk("([<%016lx>] %pSR)\n", address, (void *)address);