Commit 773438d6 authored by Xiao Guangrong's avatar Xiao Guangrong Committed by James Toy

There is a race between generic_smp_call_function_*() and hotplug_cfd() in

many cases, see below examples:

1: hotplug_cfd() can free cfd->cpumask, the system will crash if the
   cpu's cfd still in the call_function list:


      CPU A:                         CPU B

 smp_call_function_many()	    ......
   cpu_down()                      ......
  hotplug_cfd() ->                 ......
 free_cpumask_var(cfd->cpumask)  (receive function IPI interrupte)
                                /* read cfd->cpumask */
                          generic_smp_call_function_interrupt() ->
                         cpumask_test_and_clear_cpu(cpu, data->cpumask)

                         	CRASH!!!

2: It's not handle call_function list when cpu down, It's will lead to
   dead-wait if other path is waiting this cpu to execute function

    CPU A:                           CPU B

 smp_call_function_many(wait=0)
        ......			    CPU B down
   smp_call_function_many() -->  (cpu down before recevie function
    csd_lock(&data->csd);         IPI interrupte)

    DEAD-WAIT!!!!

  So, CPU A will dead-wait in csd_lock(), the same as
  smp_call_function_single()
Signed-off-by: default avatarXiao Guangrong <xiaoguangrong@cn.fujitsu.com>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Jens Axboe <jens.axboe@oracle.com>
Cc: Nick Piggin <nickpiggin@yahoo.com.au>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Rusty Russell <rusty@rustcorp.com.au>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
parent 822713a2
...@@ -113,14 +113,10 @@ void generic_exec_single(int cpu, struct call_single_data *data, int wait) ...@@ -113,14 +113,10 @@ void generic_exec_single(int cpu, struct call_single_data *data, int wait)
csd_lock_wait(data); csd_lock_wait(data);
} }
/* static void
* Invoked by arch to handle an IPI for call function. Must be called with __generic_smp_call_function_interrupt(int cpu, int run_callbacks)
* interrupts disabled.
*/
void generic_smp_call_function_interrupt(void)
{ {
struct call_function_data *data; struct call_function_data *data;
int cpu = smp_processor_id();
/* /*
* Ensure entry is visible on call_function_queue after we have * Ensure entry is visible on call_function_queue after we have
...@@ -159,12 +155,18 @@ void generic_smp_call_function_interrupt(void) ...@@ -159,12 +155,18 @@ void generic_smp_call_function_interrupt(void)
} }
/* /*
* Invoked by arch to handle an IPI for call function single. Must be * Invoked by arch to handle an IPI for call function. Must be called with
* called from the arch with interrupts disabled. * interrupts disabled.
*/ */
void generic_smp_call_function_single_interrupt(void) void generic_smp_call_function_interrupt(void)
{
__generic_smp_call_function_interrupt(smp_processor_id(), 1);
}
static void
__generic_smp_call_function_single_interrupt(int cpu, int run_callbacks)
{ {
struct call_single_queue *q = &__get_cpu_var(call_single_queue); struct call_single_queue *q = &per_cpu(call_single_queue, cpu);
unsigned int data_flags; unsigned int data_flags;
LIST_HEAD(list); LIST_HEAD(list);
...@@ -195,6 +197,15 @@ void generic_smp_call_function_single_interrupt(void) ...@@ -195,6 +197,15 @@ void generic_smp_call_function_single_interrupt(void)
} }
} }
/*
* Invoked by arch to handle an IPI for call function single. Must be
* called from the arch with interrupts disabled.
*/
void generic_smp_call_function_single_interrupt(void)
{
__generic_smp_call_function_single_interrupt(smp_processor_id(), 1);
}
static DEFINE_PER_CPU(struct call_single_data, csd_data); static DEFINE_PER_CPU(struct call_single_data, csd_data);
/* /*
...@@ -443,6 +454,7 @@ static int ...@@ -443,6 +454,7 @@ static int
hotplug_cfd(struct notifier_block *nfb, unsigned long action, void *hcpu) hotplug_cfd(struct notifier_block *nfb, unsigned long action, void *hcpu)
{ {
long cpu = (long)hcpu; long cpu = (long)hcpu;
unsigned long flags;
struct call_function_data *cfd = &per_cpu(cfd_data, cpu); struct call_function_data *cfd = &per_cpu(cfd_data, cpu);
switch (action) { switch (action) {
...@@ -459,6 +471,12 @@ hotplug_cfd(struct notifier_block *nfb, unsigned long action, void *hcpu) ...@@ -459,6 +471,12 @@ hotplug_cfd(struct notifier_block *nfb, unsigned long action, void *hcpu)
case CPU_DEAD: case CPU_DEAD:
case CPU_DEAD_FROZEN: case CPU_DEAD_FROZEN:
local_irq_save(flags);
__generic_smp_call_function_interrupt(cpu, 0);
__generic_smp_call_function_single_interrupt(cpu, 0);
local_irq_restore(flags);
csd_lock_wait(&cfd->csd);
free_cpumask_var(cfd->cpumask); free_cpumask_var(cfd->cpumask);
break; break;
#endif #endif
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment