2 * Arm specific backtracing code for oprofile
4 * Copyright 2005 Openedhand Ltd.
6 * Author: Richard Purdie <rpurdie@openedhand.com>
8 * Based on i386 oprofile backtrace code by John Levon, David Smith
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License version 2 as
12 * published by the Free Software Foundation.
16 #include <linux/oprofile.h>
17 #include <linux/sched.h>
19 #include <linux/uaccess.h>
20 #include <asm/ptrace.h>
21 #include <asm/stacktrace.h>
23 static int report_trace(struct stackframe
*frame
, void *d
)
25 unsigned int *depth
= d
;
28 oprofile_add_trace(frame
->pc
);
36 * The registers we're interested in are at the end of the variable
37 * length saved register structure. The fp points at the end of this
38 * structure so the address of this struct is:
39 * (struct frame_tail *)(xxx->fp)-1
42 struct frame_tail
*fp
;
45 } __attribute__((packed
));
47 static struct frame_tail
* user_backtrace(struct frame_tail
*tail
)
49 struct frame_tail buftail
[2];
51 /* Also check accessibility of one struct frame_tail beyond */
52 if (!access_ok(VERIFY_READ
, tail
, sizeof(buftail
)))
54 if (__copy_from_user_inatomic(buftail
, tail
, sizeof(buftail
)))
57 oprofile_add_trace(buftail
[0].lr
);
59 /* frame pointers should strictly progress back up the stack
60 * (towards higher addresses) */
61 if (tail
>= buftail
[0].fp
)
64 return buftail
[0].fp
-1;
67 void arm_backtrace(struct pt_regs
* const regs
, unsigned int depth
)
69 struct frame_tail
*tail
= ((struct frame_tail
*) regs
->ARM_fp
) - 1;
71 if (!user_mode(regs
)) {
72 struct stackframe frame
;
73 frame
.fp
= regs
->ARM_fp
;
74 frame
.sp
= regs
->ARM_sp
;
75 frame
.lr
= regs
->ARM_lr
;
76 frame
.pc
= regs
->ARM_pc
;
77 walk_stackframe(&frame
, report_trace
, &depth
);
81 while (depth
-- && tail
&& !((unsigned long) tail
& 3))
82 tail
= user_backtrace(tail
);