Re: [PATCH] i386: Execute stack overflow warning on interrupt stack

From: Eric Sandeen
Date: Mon May 05 2008 - 09:43:18 EST


Andi Kleen wrote:
> i386: Execute stack overflow warning on interrupt stack
>
> [Repost. This was posted deep in the 4K flame war some time ago. Probably
> very few people read it completely, so the patch was missed.]
>
> Previously the reporting printk would run on the process stack, which risks
> overflow an already low stack. Instead execute it on the interrupt stack.
> This makes it more likely for the printk to make it actually out.
>
> It adds one not taken test/branch more to the interrupt path when
> stack overflow checking is enabled. We could avoid that by duplicating
> more code, but that seemed not worth it.
>
> Based on an observation by Eric Sandeen.
>
> Signed-off-by: Andi Kleen <andi@xxxxxxxxxxxxxx>
>
> Index: linux/arch/x86/kernel/irq_32.c
> ===================================================================
> --- linux.orig/arch/x86/kernel/irq_32.c
> +++ linux/arch/x86/kernel/irq_32.c
> @@ -61,6 +61,26 @@ static union irq_ctx *hardirq_ctx[NR_CPU
> static union irq_ctx *softirq_ctx[NR_CPUS] __read_mostly;
> #endif
>
> +static void stack_overflow(void)
> +{
> + printk("low stack detected by irq handler\n");

We've lost the information about how close we are... that'd be nice to
keep if possible....

Can we keep the old printk string and pass the remaining stack in as an arg?

-Eric

> + dump_stack();
> +}
> +
> +static inline void call_on_stack2(void *func, unsigned long stack,
> + unsigned long arg1, unsigned long arg2)
> +{
> + unsigned long bx;
> + asm volatile(
> + " xchgl %%ebx,%%esp \n"
> + " call *%%edi \n"
> + " movl %%ebx,%%esp \n"
> + : "=a" (arg1), "=d" (arg2), "=b" (bx)
> + : "0" (arg1), "1" (arg2), "2" (stack),
> + "D" (func)
> + : "memory", "cc", "ecx');
> +}
> +
> /*
> * do_IRQ handles all normal device IRQ's (the special
> * SMP cross-CPU interrupts have their own specific
> @@ -76,6 +96,7 @@ unsigned int do_IRQ(struct pt_regs *regs
> union irq_ctx *curctx, *irqctx;
> u32 *isp;
> #endif
> + int overflow = 0;
>
> if (unlikely((unsigned)irq >= NR_IRQS)) {
> printk(KERN_EMERG "%s: cannot handle IRQ %d\n",
> @@ -92,11 +113,8 @@ unsigned int do_IRQ(struct pt_regs *regs
>
> __asm__ __volatile__("andl %%esp,%0" :
> "=r" (sp) : "0" (THREAD_SIZE - 1));
> - if (unlikely(sp < (sizeof(struct thread_info) + STACK_WARN))) {
> - printk("do_IRQ: stack overflow: %ld\n",
> - sp - sizeof(struct thread_info));
> - dump_stack();
> - }
> + if (unlikely(sp < (sizeof(struct thread_info) + STACK_WARN)))
> + overflow = 1;
> }
> #endif
>
> @@ -112,8 +130,6 @@ unsigned int do_IRQ(struct pt_regs *regs
> * current stack (which is the irq stack already after all)
> */
> if (curctx != irqctx) {
> - int arg1, arg2, bx;
> -
> /* build the stack frame on the IRQ stack */
> isp = (u32*) ((char*)irqctx + sizeof(*irqctx));
> irqctx->tinfo.task = curctx->tinfo.task;
> @@ -127,18 +143,19 @@ unsigned int do_IRQ(struct pt_regs *regs
> (irqctx->tinfo.preempt_count & ~SOFTIRQ_MASK) |
> (curctx->tinfo.preempt_count & SOFTIRQ_MASK);
>
> - asm volatile(
> - " xchgl %%ebx,%%esp \n"
> - " call *%%edi \n"
> - " movl %%ebx,%%esp \n"
> - : "=a" (arg1), "=d" (arg2), "=b" (bx)
> - : "0" (irq), "1" (desc), "2" (isp),
> - "D" (desc->handle_irq)
> - : "memory", "cc", "ecx"
> - );
> + /* Execute warning on interrupt stack */
> + if (unlikely(overflow))
> + call_on_stack2(stack_overflow, isp, 0, 0);
> +
> + call_on_stack2(desc->handle_irq, isp, irq, desc);
> } else
> #endif
> - desc->handle_irq(irq, desc);
> + {
> + /* AK: Slightly bogus here. Just return? */
> + if (overflow)
> + stack_overflow();
> + desc->handle_irq(irq, desc);
> + }
>
> irq_exit();
> set_irq_regs(old_regs);
>

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