Re: backtracer stuck in apic_timer_interrupt

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



>Do you know why the backtracer gets stuck in apic_timer_interrupt?
>Somehow it didn't manage to go over to the process stack, but it
should
>have.

No, I don't, I just verified that the annotations are correct (in that
unwinds can pass this function). If you see this again, I'd need the
raw
stack dump to compare. However, while doing this I also saw a case
where it wouldn't properly unwind (but one level up from
apic_timer_interrupt), however in this case it was expectedly getting
stuck because the interrupt had hit a procedure that already entered
it
epilogue, which gcc fails to annotate.

Further, while doing this, I found myself ending up with an unwound
stack trace followed by a legacy one, but without any separator
indicating so. Therefore I looked at the fallback logic you recently
changed and found that it was still not quite right. Hence I created
below patch (for both i386 and x86-64).

Jan

The unwinder fallback logic still had potential for falling through to
the legacy stack trace code without printing an indication (at once
serving as a separator) of this.

Further, the stack pointer retrieval for the fallback should be as
restrictive as possible (in order to avoid having the legacy stack
tracer try to access invalid memory). The patch tightens that, but
this could certainly be further improved.

Also making the call_trace command line option now conditional upon
CONFIG_STACK_UNWIND (as it's meaningless otherwise).

Signed-off-by: Jan Beulich <[email protected]>

--- linux-2.6.18-rc4/arch/i386/kernel/traps.c	2006-08-15
11:29:35.000000000 +0200
+++
2.6.18-rc4-unwind-x86-fallback/arch/i386/kernel/traps.c	2006-08-15
10:52:43.000000000 +0200
@@ -92,7 +92,11 @@ asmlinkage void spurious_interrupt_bug(v
 asmlinkage void machine_check(void);
 
 static int kstack_depth_to_print = 24;
+#ifdef CONFIG_STACK_UNWIND
 static int call_trace = 1;
+#else
+#define call_trace (-1)
+#endif
 ATOMIC_NOTIFIER_HEAD(i386die_chain);
 
 int register_die_notifier(struct notifier_block *nb)
@@ -187,22 +191,21 @@ static void show_trace_log_lvl(struct ta
 			if (unwind_init_blocked(&info, task) == 0)
 				unw_ret = show_trace_unwind(&info,
log_lvl);
 		}
-		if (unw_ret > 0 && !arch_unw_user_mode(&info)) {
-#ifdef CONFIG_STACK_UNWIND
-			print_symbol("DWARF2 unwinder stuck at %s\n",
-				     UNW_PC(&info));
-			if (call_trace == 1) {
-				printk("Leftover inexact
backtrace:\n");
-				if (UNW_SP(&info))
+		if (unw_ret > 0) {
+			if (call_trace == 1 &&
!arch_unw_user_mode(&info)) {
+				print_symbol("DWARF2 unwinder stuck at
%s\n",
+					     UNW_PC(&info));
+				if (UNW_SP(&info) >= PAGE_OFFSET) {
+					printk("Leftover inexact
backtrace:\n");
 					stack = (void *)UNW_SP(&info);
-			} else if (call_trace > 1)
+				} else
+					printk("Full inexact backtrace
again:\n");
+			} else if (call_trace >= 1)
 				return;
 			else
 				printk("Full inexact backtrace
again:\n");
-#else
+		} else
 			printk("Inexact backtrace:\n");
-#endif
-		}
 	}
 
 	if (task == current) {
@@ -1241,6 +1244,7 @@ static int __init kstack_setup(char *s)
 }
 __setup("kstack=", kstack_setup);
 
+#ifdef CONFIG_STACK_UNWIND
 static int __init call_trace_setup(char *s)
 {
 	if (strcmp(s, "old") == 0)
@@ -1254,3 +1258,4 @@ static int __init call_trace_setup(char 
 	return 1;
 }
 __setup("call_trace=", call_trace_setup);
+#endif
--- linux-2.6.18-rc4/arch/x86_64/kernel/traps.c	2006-08-15
11:29:41.000000000 +0200
+++
2.6.18-rc4-unwind-x86-fallback/arch/x86_64/kernel/traps.c	2006-08-15
10:49:34.000000000 +0200
@@ -107,7 +107,11 @@ static inline void preempt_conditional_c
 }
 
 static int kstack_depth_to_print = 12;
+#ifdef CONFIG_STACK_UNWIND
 static int call_trace = 1;
+#else
+#define call_trace (-1)
+#endif
 
 #ifdef CONFIG_KALLSYMS
 # include <linux/kallsyms.h>
@@ -274,21 +278,21 @@ void show_trace(struct task_struct *tsk,
 			if (unwind_init_blocked(&info, tsk) == 0)
 				unw_ret = show_trace_unwind(&info,
NULL);
 		}
-		if (unw_ret > 0 && !arch_unw_user_mode(&info)) {
-#ifdef CONFIG_STACK_UNWIND
-			unsigned long rip = info.regs.rip;
-			print_symbol("DWARF2 unwinder stuck at %s\n",
rip);
-			if (call_trace == 1) {
-				printk("Leftover inexact
backtrace:\n");
-				stack = (unsigned long *)info.regs.rsp;
-			} else if (call_trace > 1)
+		if (unw_ret > 0) {
+			if (call_trace == 1 &&
!arch_unw_user_mode(&info)) {
+				print_symbol("DWARF2 unwinder stuck at
%s\n",
+					     UNW_PC(&info));
+				if ((long)UNW_SP(&info) < 0) {
+					printk("Leftover inexact
backtrace:\n");
+					stack = (unsigned long
*)UNW_SP(&info);
+				} else
+					printk("Full inexact backtrace
again:\n");
+			} else if (call_trace >= 1)
 				return;
 			else
 				printk("Full inexact backtrace
again:\n");
-#else
+		} else
 			printk("Inexact backtrace:\n");
-#endif
-		}
 	}
 
 	/*
@@ -1120,6 +1124,7 @@ static int __init kstack_setup(char *s)
 }
 __setup("kstack=", kstack_setup);
 
+#ifdef CONFIG_STACK_UNWIND
 static int __init call_trace_setup(char *s)
 {
 	if (strcmp(s, "old") == 0)
@@ -1133,3 +1138,4 @@ static int __init call_trace_setup(char 
 	return 1;
 }
 __setup("call_trace=", call_trace_setup);
+#endif
--- linux-2.6.18-rc4/include/asm-i386/unwind.h	2006-08-15
11:30:00.000000000 +0200
+++
2.6.18-rc4-unwind-x86-fallback/include/asm-i386/unwind.h	2006-08-15
09:36:19.000000000 +0200
@@ -87,6 +87,7 @@ static inline int arch_unw_user_mode(con
 #else
 
 #define UNW_PC(frame) ((void)(frame), 0)
+#define UNW_SP(frame) ((void)(frame), 0)
 
 static inline int arch_unw_user_mode(const void *info)
 {
--- linux-2.6.18-rc4/include/asm-x86_64/unwind.h	2006-08-15
11:30:02.000000000 +0200
+++
2.6.18-rc4-unwind-x86-fallback/include/asm-x86_64/unwind.h	2006-08-15
09:36:19.000000000 +0200
@@ -95,6 +95,7 @@ static inline int arch_unw_user_mode(con
 #else
 
 #define UNW_PC(frame) ((void)(frame), 0)
+#define UNW_SP(frame) ((void)(frame), 0)
 
 static inline int arch_unw_user_mode(const void *info)
 {

-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [email protected]
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

[Index of Archives]     [Kernel Newbies]     [Netfilter]     [Bugtraq]     [Photo]     [Stuff]     [Gimp]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Video 4 Linux]     [Linux for the blind]     [Linux Resources]
  Powered by Linux