| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | /* | 
					
						
							|  |  |  |  *  linux/arch/arm/lib/backtrace.S | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Copyright (C) 1995, 1996 Russell King | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or modify
 | 
					
						
							|  |  |  |  * it under the terms of the GNU General Public License version 2 as | 
					
						
							|  |  |  |  * published by the Free Software Foundation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 27/03/03 Ian Molton Clean up CONFIG_CPU | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include <linux/linkage.h> | 
					
						
							|  |  |  | #include <asm/assembler.h> | 
					
						
							|  |  |  | 		.text | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @ fp is 0 or stack frame
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define frame	r4 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | #define sv_fp	r5 | 
					
						
							|  |  |  | #define sv_pc	r6 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #define mask	r7 | 
					
						
							|  |  |  | #define offset	r8 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ENTRY(c_backtrace) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-03-25 21:58:03 +00:00
										 |  |  | #if !defined(CONFIG_FRAME_POINTER) || !defined(CONFIG_PRINTK) | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		mov	pc, lr | 
					
						
							| 
									
										
										
										
											2008-08-28 11:22:32 +01:00
										 |  |  | ENDPROC(c_backtrace) | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #else | 
					
						
							|  |  |  | 		stmfd	sp!, {r4 - r8, lr}	@ Save an extra register so we have a location...
 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		movs	frame, r0		@ if frame pointer is zero
 | 
					
						
							|  |  |  | 		beq	no_frame		@ we have no stack frames
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		tst	r1, #0x10		@ 26 or 32-bit mode?
 | 
					
						
							| 
									
										
										
										
											2009-07-24 12:32:57 +01:00
										 |  |  |  ARM(		moveq	mask, #0xfc000003	) | 
					
						
							|  |  |  |  THUMB(		moveq	mask, #0xfc000000	) | 
					
						
							|  |  |  |  THUMB(		orreq	mask, #0x03		) | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		movne	mask, #0		@ mask for 32-bit
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 1:		stmfd	sp!, {pc}		@ calculate offset of PC stored
 | 
					
						
							|  |  |  | 		ldr	r0, [sp], #4		@ by stmfd for this CPU
 | 
					
						
							|  |  |  | 		adr	r1, 1b | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		sub	offset, r0, r1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | /* | 
					
						
							|  |  |  |  * Stack frame layout: | 
					
						
							|  |  |  |  *             optionally saved caller registers (r4 - r10) | 
					
						
							|  |  |  |  *             saved fp | 
					
						
							|  |  |  |  *             saved sp | 
					
						
							|  |  |  |  *             saved lr | 
					
						
							|  |  |  |  *    frame => saved pc | 
					
						
							|  |  |  |  *             optionally saved arguments (r0 - r3) | 
					
						
							|  |  |  |  * saved sp => <next word> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Functions start with the following code sequence: | 
					
						
							|  |  |  |  *                  mov   ip, sp | 
					
						
							|  |  |  |  *                  stmfd sp!, {r0 - r3} (optional) | 
					
						
							|  |  |  |  * corrected pc =>  stmfd sp!, {..., fp, ip, lr, pc} | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | for_each_frame:	tst	frame, mask		@ Check for address exceptions
 | 
					
						
							|  |  |  | 		bne	no_frame | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 1001:		ldr	sv_pc, [frame, #0]	@ get saved pc
 | 
					
						
							|  |  |  | 1002:		ldr	sv_fp, [frame, #-12]	@ get saved fp
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		sub	sv_pc, sv_pc, offset	@ Correct PC for prefetching
 | 
					
						
							|  |  |  | 		bic	sv_pc, sv_pc, mask	@ mask PC/LR for the mode
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 1003:		ldr	r2, [sv_pc, #-4]	@ if stmfd sp!, {args} exists,
 | 
					
						
							|  |  |  | 		ldr	r3, .Ldsi+4		@ adjust saved 'pc' back one
 | 
					
						
							|  |  |  | 		teq	r3, r2, lsr #10		@ instruction
 | 
					
						
							|  |  |  | 		subne	r0, sv_pc, #4		@ allow for mov
 | 
					
						
							|  |  |  | 		subeq	r0, sv_pc, #8		@ allow for mov + stmia
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ldr	r1, [frame, #-4]	@ get saved lr
 | 
					
						
							|  |  |  | 		mov	r2, frame | 
					
						
							|  |  |  | 		bic	r1, r1, mask		@ mask PC/LR for the mode
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		bl	dump_backtrace_entry | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		ldr	r1, [sv_pc, #-4]	@ if stmfd sp!, {args} exists,
 | 
					
						
							|  |  |  | 		ldr	r3, .Ldsi+4 | 
					
						
							| 
									
										
										
										
											2013-10-20 15:34:10 +01:00
										 |  |  | 		teq	r3, r1, lsr #11 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		ldreq	r0, [frame, #-8]	@ get sp
 | 
					
						
							|  |  |  | 		subeq	r0, r0, #4		@ point at the last arg
 | 
					
						
							|  |  |  | 		bleq	.Ldumpstm		@ dump saved registers
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 1004:		ldr	r1, [sv_pc, #0]		@ if stmfd sp!, {..., fp, ip, lr, pc}
 | 
					
						
							|  |  |  | 		ldr	r3, .Ldsi		@ instruction exists,
 | 
					
						
							| 
									
										
										
										
											2013-10-20 15:34:10 +01:00
										 |  |  | 		teq	r3, r1, lsr #11 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		subeq	r0, frame, #16 | 
					
						
							|  |  |  | 		bleq	.Ldumpstm		@ dump saved registers
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		teq	sv_fp, #0		@ zero saved fp means
 | 
					
						
							|  |  |  | 		beq	no_frame		@ no further frames
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		cmp	sv_fp, frame		@ next frame must be
 | 
					
						
							|  |  |  | 		mov	frame, sv_fp		@ above the current frame
 | 
					
						
							|  |  |  | 		bhi	for_each_frame | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 1006:		adr	r0, .Lbad | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		mov	r1, frame | 
					
						
							|  |  |  | 		bl	printk | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | no_frame:	ldmfd	sp!, {r4 - r8, pc} | 
					
						
							| 
									
										
										
										
											2008-08-28 11:22:32 +01:00
										 |  |  | ENDPROC(c_backtrace) | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2010-04-19 10:15:03 +01:00
										 |  |  | 		.pushsection __ex_table,"a" | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		.align	3
 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		.long	1001b, 1006b | 
					
						
							|  |  |  | 		.long	1002b, 1006b | 
					
						
							|  |  |  | 		.long	1003b, 1006b | 
					
						
							|  |  |  | 		.long	1004b, 1006b | 
					
						
							| 
									
										
										
										
											2010-04-19 10:15:03 +01:00
										 |  |  | 		.popsection | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define instr r4 | 
					
						
							|  |  |  | #define reg   r5 | 
					
						
							|  |  |  | #define stack r6 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | .Ldumpstm:	stmfd	sp!, {instr, reg, stack, r7, lr} | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		mov	stack, r0 | 
					
						
							|  |  |  | 		mov	instr, r1 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		mov	reg, #10 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		mov	r7, #0 | 
					
						
							|  |  |  | 1:		mov	r3, #1 | 
					
						
							| 
									
										
										
										
											2009-07-24 12:32:57 +01:00
										 |  |  |  ARM(		tst	instr, r3, lsl reg	) | 
					
						
							|  |  |  |  THUMB(		lsl	r3, reg			) | 
					
						
							|  |  |  |  THUMB(		tst	instr, r3		) | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		beq	2f | 
					
						
							|  |  |  | 		add	r7, r7, #1 | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		teq	r7, #6 | 
					
						
							| 
									
										
										
										
											2013-10-20 15:34:10 +01:00
										 |  |  | 		moveq	r7, #0 | 
					
						
							|  |  |  | 		adr	r3, .Lcr | 
					
						
							|  |  |  | 		addne	r3, r3, #1		@ skip newline
 | 
					
						
							|  |  |  | 		ldr	r2, [stack], #-4 | 
					
						
							|  |  |  | 		mov	r1, reg | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 		adr	r0, .Lfp | 
					
						
							|  |  |  | 		bl	printk | 
					
						
							|  |  |  | 2:		subs	reg, reg, #1 | 
					
						
							|  |  |  | 		bpl	1b | 
					
						
							|  |  |  | 		teq	r7, #0 | 
					
						
							|  |  |  | 		adrne	r0, .Lcr | 
					
						
							|  |  |  | 		blne	printk | 
					
						
							| 
									
										
										
										
											2007-03-02 15:01:36 +00:00
										 |  |  | 		ldmfd	sp!, {instr, reg, stack, r7, pc} | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-20 15:34:10 +01:00
										 |  |  | .Lfp:		.asciz	" r%d:%08x%s" | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | .Lcr:		.asciz	"\n" | 
					
						
							|  |  |  | .Lbad:		.asciz	"Backtrace aborted due to bad frame pointer <%p>\n" | 
					
						
							|  |  |  | 		.align | 
					
						
							| 
									
										
										
										
											2013-10-20 15:34:10 +01:00
										 |  |  | .Ldsi:		.word	0xe92dd800 >> 11	@ stmfd sp!, {... fp, ip, lr, pc}
 | 
					
						
							|  |  |  | 		.word	0xe92d0000 >> 11	@ stmfd sp!, {}
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif |