Commit 84dbf66c authored by Srikar Dronamraju's avatar Srikar Dronamraju Committed by Michael Ellerman

powerpc/smp: Use GFP_ATOMIC while allocating tmp mask

Qian Cai reported a regression where CPU Hotplug fails with the latest
powerpc/next

BUG: sleeping function called from invalid context at mm/slab.h:494
in_atomic(): 1, irqs_disabled(): 1, non_block: 0, pid: 0, name: swapper/88
no locks held by swapper/88/0.
irq event stamp: 18074448
hardirqs last  enabled at (18074447): [<c0000000001a2a7c>] tick_nohz_idle_enter+0x9c/0x110
hardirqs last disabled at (18074448): [<c000000000106798>] do_idle+0x138/0x3b0
do_idle at kernel/sched/idle.c:253 (discriminator 1)
softirqs last  enabled at (18074440): [<c0000000000bbec4>] irq_enter_rcu+0x94/0xa0
softirqs last disabled at (18074439): [<c0000000000bbea0>] irq_enter_rcu+0x70/0xa0
CPU: 88 PID: 0 Comm: swapper/88 Tainted: G        W         5.9.0-rc8-next-20201007 #1
Call Trace:
[c00020000a4bfcf0] [c000000000649e98] dump_stack+0xec/0x144 (unreliable)
[c00020000a4bfd30] [c0000000000f6c34] ___might_sleep+0x2f4/0x310
[c00020000a4bfdb0] [c000000000354f94] slab_pre_alloc_hook.constprop.82+0x124/0x190
[c00020000a4bfe00] [c00000000035e9e8] __kmalloc_node+0x88/0x3a0
slab_alloc_node at mm/slub.c:2817
(inlined by) __kmalloc_node at mm/slub.c:4013
[c00020000a4bfe80] [c0000000006494d8] alloc_cpumask_var_node+0x38/0x80
kmalloc_node at include/linux/slab.h:577
(inlined by) alloc_cpumask_var_node at lib/cpumask.c:116
[c00020000a4bfef0] [c00000000003eedc] start_secondary+0x27c/0x800
update_mask_by_l2 at arch/powerpc/kernel/smp.c:1267
(inlined by) add_cpu_to_masks at arch/powerpc/kernel/smp.c:1387
(inlined by) start_secondary at arch/powerpc/kernel/smp.c:1420
[c00020000a4bff90] [c00000000000c468] start_secondary_resume+0x10/0x14

Allocating a temporary mask while performing a CPU Hotplug operation
with CONFIG_CPUMASK_OFFSTACK enabled, leads to calling a sleepable
function from a atomic context. Fix this by allocating the temporary
mask with GFP_ATOMIC flag. Also instead of having to allocate twice,
allocate the mask in the caller so that we only have to allocate once.
If the allocation fails, assume the mask to be same as sibling mask, which
will make the scheduler to drop this domain for this CPU.

Fixes: 70a94089 ("powerpc/smp: Optimize update_coregroup_mask")
Fixes: 3ab33d6d ("powerpc/smp: Optimize update_mask_by_l2")
Reported-by: default avatarQian Cai <cai@redhat.com>
Signed-off-by: default avatarSrikar Dronamraju <srikar@linux.vnet.ibm.com>
Signed-off-by: default avatarMichael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20201019042716.106234-3-srikar@linux.vnet.ibm.com
parent 966730a6
...@@ -1240,38 +1240,33 @@ static struct device_node *cpu_to_l2cache(int cpu) ...@@ -1240,38 +1240,33 @@ static struct device_node *cpu_to_l2cache(int cpu)
return cache; return cache;
} }
static bool update_mask_by_l2(int cpu) static bool update_mask_by_l2(int cpu, cpumask_var_t *mask)
{ {
struct cpumask *(*submask_fn)(int) = cpu_sibling_mask; struct cpumask *(*submask_fn)(int) = cpu_sibling_mask;
struct device_node *l2_cache, *np; struct device_node *l2_cache, *np;
cpumask_var_t mask;
int i; int i;
if (has_big_cores) if (has_big_cores)
submask_fn = cpu_smallcore_mask; submask_fn = cpu_smallcore_mask;
l2_cache = cpu_to_l2cache(cpu); l2_cache = cpu_to_l2cache(cpu);
if (!l2_cache) { if (!l2_cache || !*mask) {
/* /* Assume only core siblings share cache with this CPU */
* If no l2cache for this CPU, assume all siblings to share
* cache with this CPU.
*/
for_each_cpu(i, submask_fn(cpu)) for_each_cpu(i, submask_fn(cpu))
set_cpus_related(cpu, i, cpu_l2_cache_mask); set_cpus_related(cpu, i, cpu_l2_cache_mask);
return false; return false;
} }
alloc_cpumask_var_node(&mask, GFP_KERNEL, cpu_to_node(cpu)); cpumask_and(*mask, cpu_online_mask, cpu_cpu_mask(cpu));
cpumask_and(mask, cpu_online_mask, cpu_cpu_mask(cpu));
/* Update l2-cache mask with all the CPUs that are part of submask */ /* Update l2-cache mask with all the CPUs that are part of submask */
or_cpumasks_related(cpu, cpu, submask_fn, cpu_l2_cache_mask); or_cpumasks_related(cpu, cpu, submask_fn, cpu_l2_cache_mask);
/* Skip all CPUs already part of current CPU l2-cache mask */ /* Skip all CPUs already part of current CPU l2-cache mask */
cpumask_andnot(mask, mask, cpu_l2_cache_mask(cpu)); cpumask_andnot(*mask, *mask, cpu_l2_cache_mask(cpu));
for_each_cpu(i, mask) { for_each_cpu(i, *mask) {
/* /*
* when updating the marks the current CPU has not been marked * when updating the marks the current CPU has not been marked
* online, but we need to update the cache masks * online, but we need to update the cache masks
...@@ -1281,15 +1276,14 @@ static bool update_mask_by_l2(int cpu) ...@@ -1281,15 +1276,14 @@ static bool update_mask_by_l2(int cpu)
/* Skip all CPUs already part of current CPU l2-cache */ /* Skip all CPUs already part of current CPU l2-cache */
if (np == l2_cache) { if (np == l2_cache) {
or_cpumasks_related(cpu, i, submask_fn, cpu_l2_cache_mask); or_cpumasks_related(cpu, i, submask_fn, cpu_l2_cache_mask);
cpumask_andnot(mask, mask, submask_fn(i)); cpumask_andnot(*mask, *mask, submask_fn(i));
} else { } else {
cpumask_andnot(mask, mask, cpu_l2_cache_mask(i)); cpumask_andnot(*mask, *mask, cpu_l2_cache_mask(i));
} }
of_node_put(np); of_node_put(np);
} }
of_node_put(l2_cache); of_node_put(l2_cache);
free_cpumask_var(mask);
return true; return true;
} }
...@@ -1332,40 +1326,46 @@ static inline void add_cpu_to_smallcore_masks(int cpu) ...@@ -1332,40 +1326,46 @@ static inline void add_cpu_to_smallcore_masks(int cpu)
} }
} }
static void update_coregroup_mask(int cpu) static void update_coregroup_mask(int cpu, cpumask_var_t *mask)
{ {
struct cpumask *(*submask_fn)(int) = cpu_sibling_mask; struct cpumask *(*submask_fn)(int) = cpu_sibling_mask;
cpumask_var_t mask;
int coregroup_id = cpu_to_coregroup_id(cpu); int coregroup_id = cpu_to_coregroup_id(cpu);
int i; int i;
alloc_cpumask_var_node(&mask, GFP_KERNEL, cpu_to_node(cpu));
cpumask_and(mask, cpu_online_mask, cpu_cpu_mask(cpu));
if (shared_caches) if (shared_caches)
submask_fn = cpu_l2_cache_mask; submask_fn = cpu_l2_cache_mask;
if (!*mask) {
/* Assume only siblings are part of this CPU's coregroup */
for_each_cpu(i, submask_fn(cpu))
set_cpus_related(cpu, i, cpu_coregroup_mask);
return;
}
cpumask_and(*mask, cpu_online_mask, cpu_cpu_mask(cpu));
/* Update coregroup mask with all the CPUs that are part of submask */ /* Update coregroup mask with all the CPUs that are part of submask */
or_cpumasks_related(cpu, cpu, submask_fn, cpu_coregroup_mask); or_cpumasks_related(cpu, cpu, submask_fn, cpu_coregroup_mask);
/* Skip all CPUs already part of coregroup mask */ /* Skip all CPUs already part of coregroup mask */
cpumask_andnot(mask, mask, cpu_coregroup_mask(cpu)); cpumask_andnot(*mask, *mask, cpu_coregroup_mask(cpu));
for_each_cpu(i, mask) { for_each_cpu(i, *mask) {
/* Skip all CPUs not part of this coregroup */ /* Skip all CPUs not part of this coregroup */
if (coregroup_id == cpu_to_coregroup_id(i)) { if (coregroup_id == cpu_to_coregroup_id(i)) {
or_cpumasks_related(cpu, i, submask_fn, cpu_coregroup_mask); or_cpumasks_related(cpu, i, submask_fn, cpu_coregroup_mask);
cpumask_andnot(mask, mask, submask_fn(i)); cpumask_andnot(*mask, *mask, submask_fn(i));
} else { } else {
cpumask_andnot(mask, mask, cpu_coregroup_mask(i)); cpumask_andnot(*mask, *mask, cpu_coregroup_mask(i));
} }
} }
free_cpumask_var(mask);
} }
static void add_cpu_to_masks(int cpu) static void add_cpu_to_masks(int cpu)
{ {
int first_thread = cpu_first_thread_sibling(cpu); int first_thread = cpu_first_thread_sibling(cpu);
cpumask_var_t mask;
int i; int i;
/* /*
...@@ -1379,10 +1379,15 @@ static void add_cpu_to_masks(int cpu) ...@@ -1379,10 +1379,15 @@ static void add_cpu_to_masks(int cpu)
set_cpus_related(i, cpu, cpu_sibling_mask); set_cpus_related(i, cpu, cpu_sibling_mask);
add_cpu_to_smallcore_masks(cpu); add_cpu_to_smallcore_masks(cpu);
update_mask_by_l2(cpu);
/* In CPU-hotplug path, hence use GFP_ATOMIC */
alloc_cpumask_var_node(&mask, GFP_ATOMIC, cpu_to_node(cpu));
update_mask_by_l2(cpu, &mask);
if (has_coregroup_support()) if (has_coregroup_support())
update_coregroup_mask(cpu); update_coregroup_mask(cpu, &mask);
free_cpumask_var(mask);
} }
/* Activate a secondary processor. */ /* Activate a secondary processor. */
......
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