Skip to content

Commit

Permalink
dump_stack: serialize the output from dump_stack()
Browse files Browse the repository at this point in the history
Add functionality to serialize the output from dump_stack() to avoid
mangling of the output when dump_stack is called simultaneously from
multiple cpus.

[[email protected]: fix comment indenting, avoid inclusion of <asm/> files - use <linux/> where possiblem fix uniprocessor build (__dump_stack undefined), remove unneeded ifdef around smp.h inclusion]
Signed-off-by: Alex Thorlton <[email protected]>
Reported-by: Russ Anderson <[email protected]>
Reviewed-by: Robin Holt <[email protected]>
Cc: Vineet Gupta <[email protected]>
Cc: David S. Miller <[email protected]>
Cc: Richard Kuo <[email protected]>
Cc: Jesper Nilsson <[email protected]>
Signed-off-by: Andrew Morton <[email protected]>
Signed-off-by: Linus Torvalds <[email protected]>
  • Loading branch information
Alex Thorlton authored and torvalds committed Jul 3, 2013
1 parent f170168 commit b58d977
Showing 1 changed file with 45 additions and 2 deletions.
47 changes: 45 additions & 2 deletions lib/dump_stack.c
Original file line number Diff line number Diff line change
Expand Up @@ -6,15 +6,58 @@
#include <linux/kernel.h>
#include <linux/export.h>
#include <linux/sched.h>
#include <linux/smp.h>
#include <linux/atomic.h>

static void __dump_stack(void)
{
dump_stack_print_info(KERN_DEFAULT);
show_stack(NULL, NULL);
}

/**
* dump_stack - dump the current task information and its stack trace
*
* Architectures can override this implementation by implementing its own.
*/
#ifdef CONFIG_SMP
static atomic_t dump_lock = ATOMIC_INIT(-1);

void dump_stack(void)
{
dump_stack_print_info(KERN_DEFAULT);
show_stack(NULL, NULL);
int was_locked;
int old;
int cpu;

/*
* Permit this cpu to perform nested stack dumps while serialising
* against other CPUs
*/
preempt_disable();

retry:
cpu = smp_processor_id();
old = atomic_cmpxchg(&dump_lock, -1, cpu);
if (old == -1) {
was_locked = 0;
} else if (old == cpu) {
was_locked = 1;
} else {
cpu_relax();
goto retry;
}

__dump_stack();

if (!was_locked)
atomic_set(&dump_lock, -1);

preempt_enable();
}
#else
void dump_stack(void)
{
__dump_stack();
}
#endif
EXPORT_SYMBOL(dump_stack);

0 comments on commit b58d977

Please sign in to comment.