2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2006-12-05 17:52:36 +11:00
/*
* pseries CPU Hotplug infrastructure .
*
2006-12-05 17:52:38 +11:00
* Split out from arch / powerpc / platforms / pseries / setup . c
* arch / powerpc / kernel / rtas . c , and arch / powerpc / platforms / pseries / smp . c
2006-12-05 17:52:36 +11:00
*
* Peter Bergner , IBM March 2001.
* Copyright ( C ) 2001 IBM .
2006-12-05 17:52:38 +11:00
* Dave Engebretsen , Peter Bergner , and
* Mike Corrigan { engebret | bergner | mikec } @ us . ibm . com
* Plus various changes from other IBM teams . . .
2006-12-05 17:52:36 +11:00
*
* Copyright ( C ) 2006 Michael Ellerman , IBM Corporation
*/
2015-12-16 14:52:39 -06:00
# define pr_fmt(fmt) "pseries-hotplug-cpu: " fmt
2006-12-05 17:52:36 +11:00
# include <linux/kernel.h>
2011-04-04 13:46:58 +10:00
# include <linux/interrupt.h>
2006-12-05 17:52:36 +11:00
# include <linux/delay.h>
2011-05-27 14:25:11 -04:00
# include <linux/sched.h> /* for idle_task_exit */
2017-02-08 18:51:36 +01:00
# include <linux/sched/hotplug.h>
2006-12-05 17:52:36 +11:00
# include <linux/cpu.h>
2012-10-02 16:57:57 +00:00
# include <linux/of.h>
2015-12-16 14:54:05 -06:00
# include <linux/slab.h>
2006-12-05 17:52:36 +11:00
# include <asm/prom.h>
# include <asm/rtas.h>
# include <asm/firmware.h>
# include <asm/machdep.h>
# include <asm/vdso_datapage.h>
2011-04-04 13:46:58 +10:00
# include <asm/xics.h>
powerpc/xive: guest exploitation of the XIVE interrupt controller
This is the framework for using XIVE in a PowerVM guest. The support
is very similar to the native one in a much simpler form.
Each source is associated with an Event State Buffer (ESB). This is a
two bit state machine which is used to trigger events. The bits are
named "P" (pending) and "Q" (queued) and can be controlled by MMIO.
The Guest OS registers event (or notifications) queues on which the HW
will post event data for a target to notify.
Instead of OPAL calls, a set of Hypervisors call are used to configure
the interrupt sources and the event/notification queues of the guest:
- H_INT_GET_SOURCE_INFO
used to obtain the address of the MMIO page of the Event State
Buffer (PQ bits) entry associated with the source.
- H_INT_SET_SOURCE_CONFIG
assigns a source to a "target".
- H_INT_GET_SOURCE_CONFIG
determines to which "target" and "priority" is assigned to a source
- H_INT_GET_QUEUE_INFO
returns the address of the notification management page associated
with the specified "target" and "priority".
- H_INT_SET_QUEUE_CONFIG
sets or resets the event queue for a given "target" and "priority".
It is also used to set the notification config associated with the
queue, only unconditional notification for the moment. Reset is
performed with a queue size of 0 and queueing is disabled in that
case.
- H_INT_GET_QUEUE_CONFIG
returns the queue settings for a given "target" and "priority".
- H_INT_RESET
resets all of the partition's interrupt exploitation structures to
their initial state, losing all configuration set via the hcalls
H_INT_SET_SOURCE_CONFIG and H_INT_SET_QUEUE_CONFIG.
- H_INT_SYNC
issue a synchronisation on a source to make sure sure all
notifications have reached their queue.
As for XICS, the XIVE interface for the guest is described in the
device tree under the "interrupt-controller" node. A couple of new
properties are specific to XIVE :
- "reg"
contains the base address and size of the thread interrupt
managnement areas (TIMA), also called rings, for the User level and
for the Guest OS level. Only the Guest OS level is taken into
account today.
- "ibm,xive-eq-sizes"
the size of the event queues. One cell per size supported, contains
log2 of size, in ascending order.
- "ibm,xive-lisn-ranges"
the interrupt numbers ranges assigned to the guest. These are
allocated using a simple bitmap.
and also :
- "/ibm,plat-res-int-priorities"
contains a list of priorities that the hypervisor has reserved for
its own use.
Tested with a QEMU XIVE model for pseries and with the Power hypervisor.
Signed-off-by: Cédric Le Goater <clg@kaod.org>
Acked-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2017-08-30 21:46:11 +02:00
# include <asm/xive.h>
2013-08-22 15:23:52 +05:30
# include <asm/plpar_wrappers.h>
2018-01-26 13:41:59 -06:00
# include <asm/topology.h>
2013-08-22 15:23:52 +05:30
2015-12-16 14:50:21 -06:00
# include "pseries.h"
2006-12-05 17:52:36 +11:00
/* This version can't take the spinlock, because it never returns */
2014-02-20 21:13:52 +11:00
static int rtas_stop_self_token = RTAS_UNKNOWN_SERVICE ;
2006-12-05 17:52:36 +11:00
2006-12-05 17:52:37 +11:00
static void rtas_stop_self ( void )
2006-12-05 17:52:36 +11:00
{
2015-11-24 22:26:10 +11:00
static struct rtas_args args ;
2014-04-28 08:29:51 +08:00
2006-12-05 17:52:36 +11:00
local_irq_disable ( ) ;
2014-02-20 21:13:52 +11:00
BUG_ON ( rtas_stop_self_token = = RTAS_UNKNOWN_SERVICE ) ;
2006-12-05 17:52:36 +11:00
printk ( " cpu %u (hwid %u) Ready to die... \n " ,
smp_processor_id ( ) , hard_smp_processor_id ( ) ) ;
2015-11-24 22:26:10 +11:00
rtas_call_unlocked ( & args , rtas_stop_self_token , 0 , 1 , NULL ) ;
2006-12-05 17:52:36 +11:00
panic ( " Alas, I survived. \n " ) ;
}
2020-08-19 11:56:34 +10:00
static void pseries_cpu_offline_self ( void )
2006-12-05 17:52:37 +11:00
{
2009-10-29 19:22:53 +00:00
unsigned int hwcpu = hard_smp_processor_id ( ) ;
2006-12-05 17:52:37 +11:00
local_irq_disable ( ) ;
idle_task_exit ( ) ;
powerpc/xive: guest exploitation of the XIVE interrupt controller
This is the framework for using XIVE in a PowerVM guest. The support
is very similar to the native one in a much simpler form.
Each source is associated with an Event State Buffer (ESB). This is a
two bit state machine which is used to trigger events. The bits are
named "P" (pending) and "Q" (queued) and can be controlled by MMIO.
The Guest OS registers event (or notifications) queues on which the HW
will post event data for a target to notify.
Instead of OPAL calls, a set of Hypervisors call are used to configure
the interrupt sources and the event/notification queues of the guest:
- H_INT_GET_SOURCE_INFO
used to obtain the address of the MMIO page of the Event State
Buffer (PQ bits) entry associated with the source.
- H_INT_SET_SOURCE_CONFIG
assigns a source to a "target".
- H_INT_GET_SOURCE_CONFIG
determines to which "target" and "priority" is assigned to a source
- H_INT_GET_QUEUE_INFO
returns the address of the notification management page associated
with the specified "target" and "priority".
- H_INT_SET_QUEUE_CONFIG
sets or resets the event queue for a given "target" and "priority".
It is also used to set the notification config associated with the
queue, only unconditional notification for the moment. Reset is
performed with a queue size of 0 and queueing is disabled in that
case.
- H_INT_GET_QUEUE_CONFIG
returns the queue settings for a given "target" and "priority".
- H_INT_RESET
resets all of the partition's interrupt exploitation structures to
their initial state, losing all configuration set via the hcalls
H_INT_SET_SOURCE_CONFIG and H_INT_SET_QUEUE_CONFIG.
- H_INT_SYNC
issue a synchronisation on a source to make sure sure all
notifications have reached their queue.
As for XICS, the XIVE interface for the guest is described in the
device tree under the "interrupt-controller" node. A couple of new
properties are specific to XIVE :
- "reg"
contains the base address and size of the thread interrupt
managnement areas (TIMA), also called rings, for the User level and
for the Guest OS level. Only the Guest OS level is taken into
account today.
- "ibm,xive-eq-sizes"
the size of the event queues. One cell per size supported, contains
log2 of size, in ascending order.
- "ibm,xive-lisn-ranges"
the interrupt numbers ranges assigned to the guest. These are
allocated using a simple bitmap.
and also :
- "/ibm,plat-res-int-priorities"
contains a list of priorities that the hypervisor has reserved for
its own use.
Tested with a QEMU XIVE model for pseries and with the Power hypervisor.
Signed-off-by: Cédric Le Goater <clg@kaod.org>
Acked-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2017-08-30 21:46:11 +02:00
if ( xive_enabled ( ) )
xive_teardown_cpu ( ) ;
else
xics_teardown_cpu ( ) ;
2009-10-29 19:22:53 +00:00
2011-07-25 01:46:34 +00:00
unregister_slb_shadow ( hwcpu ) ;
2010-03-01 02:58:16 +00:00
rtas_stop_self ( ) ;
2009-10-29 19:22:53 +00:00
2006-12-05 17:52:37 +11:00
/* Should never get here... */
BUG ( ) ;
for ( ; ; ) ;
}
2006-12-05 17:52:39 +11:00
static int pseries_cpu_disable ( void )
2006-12-05 17:52:38 +11:00
{
int cpu = smp_processor_id ( ) ;
2009-09-24 09:34:48 -06:00
set_cpu_online ( cpu , false ) ;
2006-12-05 17:52:38 +11:00
vdso_data - > processorCount - - ;
/*fix boot_cpuid here*/
if ( cpu = = boot_cpuid )
2010-04-26 15:32:42 +00:00
boot_cpuid = cpumask_any ( cpu_online_mask ) ;
2006-12-05 17:52:38 +11:00
/* FIXME: abstract this to not be platform specific later on */
powerpc/xive: guest exploitation of the XIVE interrupt controller
This is the framework for using XIVE in a PowerVM guest. The support
is very similar to the native one in a much simpler form.
Each source is associated with an Event State Buffer (ESB). This is a
two bit state machine which is used to trigger events. The bits are
named "P" (pending) and "Q" (queued) and can be controlled by MMIO.
The Guest OS registers event (or notifications) queues on which the HW
will post event data for a target to notify.
Instead of OPAL calls, a set of Hypervisors call are used to configure
the interrupt sources and the event/notification queues of the guest:
- H_INT_GET_SOURCE_INFO
used to obtain the address of the MMIO page of the Event State
Buffer (PQ bits) entry associated with the source.
- H_INT_SET_SOURCE_CONFIG
assigns a source to a "target".
- H_INT_GET_SOURCE_CONFIG
determines to which "target" and "priority" is assigned to a source
- H_INT_GET_QUEUE_INFO
returns the address of the notification management page associated
with the specified "target" and "priority".
- H_INT_SET_QUEUE_CONFIG
sets or resets the event queue for a given "target" and "priority".
It is also used to set the notification config associated with the
queue, only unconditional notification for the moment. Reset is
performed with a queue size of 0 and queueing is disabled in that
case.
- H_INT_GET_QUEUE_CONFIG
returns the queue settings for a given "target" and "priority".
- H_INT_RESET
resets all of the partition's interrupt exploitation structures to
their initial state, losing all configuration set via the hcalls
H_INT_SET_SOURCE_CONFIG and H_INT_SET_QUEUE_CONFIG.
- H_INT_SYNC
issue a synchronisation on a source to make sure sure all
notifications have reached their queue.
As for XICS, the XIVE interface for the guest is described in the
device tree under the "interrupt-controller" node. A couple of new
properties are specific to XIVE :
- "reg"
contains the base address and size of the thread interrupt
managnement areas (TIMA), also called rings, for the User level and
for the Guest OS level. Only the Guest OS level is taken into
account today.
- "ibm,xive-eq-sizes"
the size of the event queues. One cell per size supported, contains
log2 of size, in ascending order.
- "ibm,xive-lisn-ranges"
the interrupt numbers ranges assigned to the guest. These are
allocated using a simple bitmap.
and also :
- "/ibm,plat-res-int-priorities"
contains a list of priorities that the hypervisor has reserved for
its own use.
Tested with a QEMU XIVE model for pseries and with the Power hypervisor.
Signed-off-by: Cédric Le Goater <clg@kaod.org>
Acked-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
2017-08-30 21:46:11 +02:00
if ( xive_enabled ( ) )
xive_smp_disable_cpu ( ) ;
else
xics_migrate_irqs_away ( ) ;
powerpc/64s: Trim offlined CPUs from mm_cpumasks
When offlining a CPU, powerpc/64s does not flush TLBs, rather it just
leaves the CPU set in mm_cpumasks, so it continues to receive TLBIEs
to manage its TLBs.
However the exit_flush_lazy_tlbs() function expects that after
returning, all CPUs (except self) have flushed TLBs for that mm, in
which case TLBIEL can be used for this flush. This breaks for offline
CPUs because they don't get the IPI to flush their TLB. This can lead
to stale translations.
Fix this by clearing the CPU from mm_cpumasks, then flushing all TLBs
before going offline.
These offlined CPU bits stuck in the cpumask also prevents the cpumask
from being trimmed back to local mode, which means continual broadcast
IPIs or TLBIEs are needed for TLB flushing. This patch prevents that
situation too.
A cast of many were involved in working this out, but in particular
Milton, Aneesh, Paul made key discoveries.
Fixes: 0cef77c7798a7 ("powerpc/64s/radix: flush remote CPUs out of single-threaded mm_cpumask")
Signed-off-by: Nicholas Piggin <npiggin@gmail.com>
Reviewed-by: Aneesh Kumar K.V <aneesh.kumar@linux.ibm.com>
Debugged-by: Milton Miller <miltonm@us.ibm.com>
Debugged-by: Aneesh Kumar K.V <aneesh.kumar@linux.ibm.com>
Debugged-by: Paul Mackerras <paulus@samba.org>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20201126102530.691335-5-npiggin@gmail.com
2020-11-26 20:25:30 +10:00
cleanup_cpu_mmu_context ( ) ;
2006-12-05 17:52:38 +11:00
return 0 ;
}
2009-10-29 19:22:53 +00:00
/*
* pseries_cpu_die : Wait for the cpu to die .
* @ cpu : logical processor id of the CPU whose death we ' re awaiting .
*
* This function is called from the context of the thread which is performing
* the cpu - offline . Here we wait for long enough to allow the cpu in question
* to self - destroy so that the cpu - offline thread can send the CPU_DEAD
* notifications .
*
2020-08-19 11:56:34 +10:00
* OTOH , pseries_cpu_offline_self ( ) is called by the @ cpu when it wants to
2009-10-29 19:22:53 +00:00
* self - destruct .
*/
2006-12-05 17:52:39 +11:00
static void pseries_cpu_die ( unsigned int cpu )
2006-12-05 17:52:38 +11:00
{
2009-10-29 19:22:53 +00:00
int cpu_status = 1 ;
2006-12-05 17:52:38 +11:00
unsigned int pcpu = get_hard_smp_processor_id ( cpu ) ;
powerpc/pseries/hotplug-cpu: wait indefinitely for vCPU death
For a power9 KVM guest with XIVE enabled, running a test loop
where we hotplug 384 vcpus and then unplug them, the following traces
can be seen (generally within a few loops) either from the unplugged
vcpu:
cpu 65 (hwid 65) Ready to die...
Querying DEAD? cpu 66 (66) shows 2
list_del corruption. next->prev should be c00a000002470208, but was c00a000002470048
------------[ cut here ]------------
kernel BUG at lib/list_debug.c:56!
Oops: Exception in kernel mode, sig: 5 [#1]
LE SMP NR_CPUS=2048 NUMA pSeries
Modules linked in: fuse nft_fib_inet nft_fib_ipv4 nft_fib_ipv6 ...
CPU: 66 PID: 0 Comm: swapper/66 Kdump: loaded Not tainted 4.18.0-221.el8.ppc64le #1
NIP: c0000000007ab50c LR: c0000000007ab508 CTR: 00000000000003ac
REGS: c0000009e5a17840 TRAP: 0700 Not tainted (4.18.0-221.el8.ppc64le)
MSR: 800000000282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE> CR: 28000842 XER: 20040000
...
NIP __list_del_entry_valid+0xac/0x100
LR __list_del_entry_valid+0xa8/0x100
Call Trace:
__list_del_entry_valid+0xa8/0x100 (unreliable)
free_pcppages_bulk+0x1f8/0x940
free_unref_page+0xd0/0x100
xive_spapr_cleanup_queue+0x148/0x1b0
xive_teardown_cpu+0x1bc/0x240
pseries_mach_cpu_die+0x78/0x2f0
cpu_die+0x48/0x70
arch_cpu_idle_dead+0x20/0x40
do_idle+0x2f4/0x4c0
cpu_startup_entry+0x38/0x40
start_secondary+0x7bc/0x8f0
start_secondary_prolog+0x10/0x14
or on the worker thread handling the unplug:
pseries-hotplug-cpu: Attempting to remove CPU <NULL>, drc index: 1000013a
Querying DEAD? cpu 314 (314) shows 2
BUG: Bad page state in process kworker/u768:3 pfn:95de1
cpu 314 (hwid 314) Ready to die...
page:c00a000002577840 refcount:0 mapcount:-128 mapping:0000000000000000 index:0x0
flags: 0x5ffffc00000000()
raw: 005ffffc00000000 5deadbeef0000100 5deadbeef0000200 0000000000000000
raw: 0000000000000000 0000000000000000 00000000ffffff7f 0000000000000000
page dumped because: nonzero mapcount
Modules linked in: kvm xt_CHECKSUM ipt_MASQUERADE xt_conntrack ...
CPU: 0 PID: 548 Comm: kworker/u768:3 Kdump: loaded Not tainted 4.18.0-224.el8.bz1856588.ppc64le #1
Workqueue: pseries hotplug workque pseries_hp_work_fn
Call Trace:
dump_stack+0xb0/0xf4 (unreliable)
bad_page+0x12c/0x1b0
free_pcppages_bulk+0x5bc/0x940
page_alloc_cpu_dead+0x118/0x120
cpuhp_invoke_callback.constprop.5+0xb8/0x760
_cpu_down+0x188/0x340
cpu_down+0x5c/0xa0
cpu_subsys_offline+0x24/0x40
device_offline+0xf0/0x130
dlpar_offline_cpu+0x1c4/0x2a0
dlpar_cpu_remove+0xb8/0x190
dlpar_cpu_remove_by_index+0x12c/0x150
dlpar_cpu+0x94/0x800
pseries_hp_work_fn+0x128/0x1e0
process_one_work+0x304/0x5d0
worker_thread+0xcc/0x7a0
kthread+0x1ac/0x1c0
ret_from_kernel_thread+0x5c/0x80
The latter trace is due to the following sequence:
page_alloc_cpu_dead
drain_pages
drain_pages_zone
free_pcppages_bulk
where drain_pages() in this case is called under the assumption that
the unplugged cpu is no longer executing. To ensure that is the case,
and early call is made to __cpu_die()->pseries_cpu_die(), which runs a
loop that waits for the cpu to reach a halted state by polling its
status via query-cpu-stopped-state RTAS calls. It only polls for 25
iterations before giving up, however, and in the trace above this
results in the following being printed only .1 seconds after the
hotplug worker thread begins processing the unplug request:
pseries-hotplug-cpu: Attempting to remove CPU <NULL>, drc index: 1000013a
Querying DEAD? cpu 314 (314) shows 2
At that point the worker thread assumes the unplugged CPU is in some
unknown/dead state and procedes with the cleanup, causing the race
with the XIVE cleanup code executed by the unplugged CPU.
Fix this by waiting indefinitely, but also making an effort to avoid
spurious lockup messages by allowing for rescheduling after polling
the CPU status and printing a warning if we wait for longer than 120s.
Fixes: eac1e731b59ee ("powerpc/xive: guest exploitation of the XIVE interrupt controller")
Suggested-by: Michael Ellerman <mpe@ellerman.id.au>
Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com>
Tested-by: Greg Kurz <groug@kaod.org>
Reviewed-by: Thiago Jung Bauermann <bauerman@linux.ibm.com>
Reviewed-by: Greg Kurz <groug@kaod.org>
[mpe: Trim oopses in change log slightly for readability]
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20200811161544.10513-1-mdroth@linux.vnet.ibm.com
2020-08-11 11:15:44 -05:00
unsigned long timeout = jiffies + msecs_to_jiffies ( 120000 ) ;
2006-12-05 17:52:38 +11:00
powerpc/pseries/hotplug-cpu: wait indefinitely for vCPU death
For a power9 KVM guest with XIVE enabled, running a test loop
where we hotplug 384 vcpus and then unplug them, the following traces
can be seen (generally within a few loops) either from the unplugged
vcpu:
cpu 65 (hwid 65) Ready to die...
Querying DEAD? cpu 66 (66) shows 2
list_del corruption. next->prev should be c00a000002470208, but was c00a000002470048
------------[ cut here ]------------
kernel BUG at lib/list_debug.c:56!
Oops: Exception in kernel mode, sig: 5 [#1]
LE SMP NR_CPUS=2048 NUMA pSeries
Modules linked in: fuse nft_fib_inet nft_fib_ipv4 nft_fib_ipv6 ...
CPU: 66 PID: 0 Comm: swapper/66 Kdump: loaded Not tainted 4.18.0-221.el8.ppc64le #1
NIP: c0000000007ab50c LR: c0000000007ab508 CTR: 00000000000003ac
REGS: c0000009e5a17840 TRAP: 0700 Not tainted (4.18.0-221.el8.ppc64le)
MSR: 800000000282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE> CR: 28000842 XER: 20040000
...
NIP __list_del_entry_valid+0xac/0x100
LR __list_del_entry_valid+0xa8/0x100
Call Trace:
__list_del_entry_valid+0xa8/0x100 (unreliable)
free_pcppages_bulk+0x1f8/0x940
free_unref_page+0xd0/0x100
xive_spapr_cleanup_queue+0x148/0x1b0
xive_teardown_cpu+0x1bc/0x240
pseries_mach_cpu_die+0x78/0x2f0
cpu_die+0x48/0x70
arch_cpu_idle_dead+0x20/0x40
do_idle+0x2f4/0x4c0
cpu_startup_entry+0x38/0x40
start_secondary+0x7bc/0x8f0
start_secondary_prolog+0x10/0x14
or on the worker thread handling the unplug:
pseries-hotplug-cpu: Attempting to remove CPU <NULL>, drc index: 1000013a
Querying DEAD? cpu 314 (314) shows 2
BUG: Bad page state in process kworker/u768:3 pfn:95de1
cpu 314 (hwid 314) Ready to die...
page:c00a000002577840 refcount:0 mapcount:-128 mapping:0000000000000000 index:0x0
flags: 0x5ffffc00000000()
raw: 005ffffc00000000 5deadbeef0000100 5deadbeef0000200 0000000000000000
raw: 0000000000000000 0000000000000000 00000000ffffff7f 0000000000000000
page dumped because: nonzero mapcount
Modules linked in: kvm xt_CHECKSUM ipt_MASQUERADE xt_conntrack ...
CPU: 0 PID: 548 Comm: kworker/u768:3 Kdump: loaded Not tainted 4.18.0-224.el8.bz1856588.ppc64le #1
Workqueue: pseries hotplug workque pseries_hp_work_fn
Call Trace:
dump_stack+0xb0/0xf4 (unreliable)
bad_page+0x12c/0x1b0
free_pcppages_bulk+0x5bc/0x940
page_alloc_cpu_dead+0x118/0x120
cpuhp_invoke_callback.constprop.5+0xb8/0x760
_cpu_down+0x188/0x340
cpu_down+0x5c/0xa0
cpu_subsys_offline+0x24/0x40
device_offline+0xf0/0x130
dlpar_offline_cpu+0x1c4/0x2a0
dlpar_cpu_remove+0xb8/0x190
dlpar_cpu_remove_by_index+0x12c/0x150
dlpar_cpu+0x94/0x800
pseries_hp_work_fn+0x128/0x1e0
process_one_work+0x304/0x5d0
worker_thread+0xcc/0x7a0
kthread+0x1ac/0x1c0
ret_from_kernel_thread+0x5c/0x80
The latter trace is due to the following sequence:
page_alloc_cpu_dead
drain_pages
drain_pages_zone
free_pcppages_bulk
where drain_pages() in this case is called under the assumption that
the unplugged cpu is no longer executing. To ensure that is the case,
and early call is made to __cpu_die()->pseries_cpu_die(), which runs a
loop that waits for the cpu to reach a halted state by polling its
status via query-cpu-stopped-state RTAS calls. It only polls for 25
iterations before giving up, however, and in the trace above this
results in the following being printed only .1 seconds after the
hotplug worker thread begins processing the unplug request:
pseries-hotplug-cpu: Attempting to remove CPU <NULL>, drc index: 1000013a
Querying DEAD? cpu 314 (314) shows 2
At that point the worker thread assumes the unplugged CPU is in some
unknown/dead state and procedes with the cleanup, causing the race
with the XIVE cleanup code executed by the unplugged CPU.
Fix this by waiting indefinitely, but also making an effort to avoid
spurious lockup messages by allowing for rescheduling after polling
the CPU status and printing a warning if we wait for longer than 120s.
Fixes: eac1e731b59ee ("powerpc/xive: guest exploitation of the XIVE interrupt controller")
Suggested-by: Michael Ellerman <mpe@ellerman.id.au>
Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com>
Tested-by: Greg Kurz <groug@kaod.org>
Reviewed-by: Thiago Jung Bauermann <bauerman@linux.ibm.com>
Reviewed-by: Greg Kurz <groug@kaod.org>
[mpe: Trim oopses in change log slightly for readability]
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20200811161544.10513-1-mdroth@linux.vnet.ibm.com
2020-08-11 11:15:44 -05:00
while ( true ) {
2020-06-12 00:12:21 -05:00
cpu_status = smp_query_cpu_stopped ( pcpu ) ;
if ( cpu_status = = QCSS_STOPPED | |
cpu_status = = QCSS_HARDWARE_ERROR )
break ;
2009-10-29 19:22:53 +00:00
powerpc/pseries/hotplug-cpu: wait indefinitely for vCPU death
For a power9 KVM guest with XIVE enabled, running a test loop
where we hotplug 384 vcpus and then unplug them, the following traces
can be seen (generally within a few loops) either from the unplugged
vcpu:
cpu 65 (hwid 65) Ready to die...
Querying DEAD? cpu 66 (66) shows 2
list_del corruption. next->prev should be c00a000002470208, but was c00a000002470048
------------[ cut here ]------------
kernel BUG at lib/list_debug.c:56!
Oops: Exception in kernel mode, sig: 5 [#1]
LE SMP NR_CPUS=2048 NUMA pSeries
Modules linked in: fuse nft_fib_inet nft_fib_ipv4 nft_fib_ipv6 ...
CPU: 66 PID: 0 Comm: swapper/66 Kdump: loaded Not tainted 4.18.0-221.el8.ppc64le #1
NIP: c0000000007ab50c LR: c0000000007ab508 CTR: 00000000000003ac
REGS: c0000009e5a17840 TRAP: 0700 Not tainted (4.18.0-221.el8.ppc64le)
MSR: 800000000282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE> CR: 28000842 XER: 20040000
...
NIP __list_del_entry_valid+0xac/0x100
LR __list_del_entry_valid+0xa8/0x100
Call Trace:
__list_del_entry_valid+0xa8/0x100 (unreliable)
free_pcppages_bulk+0x1f8/0x940
free_unref_page+0xd0/0x100
xive_spapr_cleanup_queue+0x148/0x1b0
xive_teardown_cpu+0x1bc/0x240
pseries_mach_cpu_die+0x78/0x2f0
cpu_die+0x48/0x70
arch_cpu_idle_dead+0x20/0x40
do_idle+0x2f4/0x4c0
cpu_startup_entry+0x38/0x40
start_secondary+0x7bc/0x8f0
start_secondary_prolog+0x10/0x14
or on the worker thread handling the unplug:
pseries-hotplug-cpu: Attempting to remove CPU <NULL>, drc index: 1000013a
Querying DEAD? cpu 314 (314) shows 2
BUG: Bad page state in process kworker/u768:3 pfn:95de1
cpu 314 (hwid 314) Ready to die...
page:c00a000002577840 refcount:0 mapcount:-128 mapping:0000000000000000 index:0x0
flags: 0x5ffffc00000000()
raw: 005ffffc00000000 5deadbeef0000100 5deadbeef0000200 0000000000000000
raw: 0000000000000000 0000000000000000 00000000ffffff7f 0000000000000000
page dumped because: nonzero mapcount
Modules linked in: kvm xt_CHECKSUM ipt_MASQUERADE xt_conntrack ...
CPU: 0 PID: 548 Comm: kworker/u768:3 Kdump: loaded Not tainted 4.18.0-224.el8.bz1856588.ppc64le #1
Workqueue: pseries hotplug workque pseries_hp_work_fn
Call Trace:
dump_stack+0xb0/0xf4 (unreliable)
bad_page+0x12c/0x1b0
free_pcppages_bulk+0x5bc/0x940
page_alloc_cpu_dead+0x118/0x120
cpuhp_invoke_callback.constprop.5+0xb8/0x760
_cpu_down+0x188/0x340
cpu_down+0x5c/0xa0
cpu_subsys_offline+0x24/0x40
device_offline+0xf0/0x130
dlpar_offline_cpu+0x1c4/0x2a0
dlpar_cpu_remove+0xb8/0x190
dlpar_cpu_remove_by_index+0x12c/0x150
dlpar_cpu+0x94/0x800
pseries_hp_work_fn+0x128/0x1e0
process_one_work+0x304/0x5d0
worker_thread+0xcc/0x7a0
kthread+0x1ac/0x1c0
ret_from_kernel_thread+0x5c/0x80
The latter trace is due to the following sequence:
page_alloc_cpu_dead
drain_pages
drain_pages_zone
free_pcppages_bulk
where drain_pages() in this case is called under the assumption that
the unplugged cpu is no longer executing. To ensure that is the case,
and early call is made to __cpu_die()->pseries_cpu_die(), which runs a
loop that waits for the cpu to reach a halted state by polling its
status via query-cpu-stopped-state RTAS calls. It only polls for 25
iterations before giving up, however, and in the trace above this
results in the following being printed only .1 seconds after the
hotplug worker thread begins processing the unplug request:
pseries-hotplug-cpu: Attempting to remove CPU <NULL>, drc index: 1000013a
Querying DEAD? cpu 314 (314) shows 2
At that point the worker thread assumes the unplugged CPU is in some
unknown/dead state and procedes with the cleanup, causing the race
with the XIVE cleanup code executed by the unplugged CPU.
Fix this by waiting indefinitely, but also making an effort to avoid
spurious lockup messages by allowing for rescheduling after polling
the CPU status and printing a warning if we wait for longer than 120s.
Fixes: eac1e731b59ee ("powerpc/xive: guest exploitation of the XIVE interrupt controller")
Suggested-by: Michael Ellerman <mpe@ellerman.id.au>
Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com>
Tested-by: Greg Kurz <groug@kaod.org>
Reviewed-by: Thiago Jung Bauermann <bauerman@linux.ibm.com>
Reviewed-by: Greg Kurz <groug@kaod.org>
[mpe: Trim oopses in change log slightly for readability]
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20200811161544.10513-1-mdroth@linux.vnet.ibm.com
2020-08-11 11:15:44 -05:00
if ( time_after ( jiffies , timeout ) ) {
pr_warn ( " CPU %i (hwid %i) didn't die after 120 seconds \n " ,
cpu , pcpu ) ;
timeout = jiffies + msecs_to_jiffies ( 120000 ) ;
}
cond_resched ( ) ;
2006-12-05 17:52:38 +11:00
}
2009-10-29 19:22:53 +00:00
powerpc/pseries/hotplug-cpu: wait indefinitely for vCPU death
For a power9 KVM guest with XIVE enabled, running a test loop
where we hotplug 384 vcpus and then unplug them, the following traces
can be seen (generally within a few loops) either from the unplugged
vcpu:
cpu 65 (hwid 65) Ready to die...
Querying DEAD? cpu 66 (66) shows 2
list_del corruption. next->prev should be c00a000002470208, but was c00a000002470048
------------[ cut here ]------------
kernel BUG at lib/list_debug.c:56!
Oops: Exception in kernel mode, sig: 5 [#1]
LE SMP NR_CPUS=2048 NUMA pSeries
Modules linked in: fuse nft_fib_inet nft_fib_ipv4 nft_fib_ipv6 ...
CPU: 66 PID: 0 Comm: swapper/66 Kdump: loaded Not tainted 4.18.0-221.el8.ppc64le #1
NIP: c0000000007ab50c LR: c0000000007ab508 CTR: 00000000000003ac
REGS: c0000009e5a17840 TRAP: 0700 Not tainted (4.18.0-221.el8.ppc64le)
MSR: 800000000282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE> CR: 28000842 XER: 20040000
...
NIP __list_del_entry_valid+0xac/0x100
LR __list_del_entry_valid+0xa8/0x100
Call Trace:
__list_del_entry_valid+0xa8/0x100 (unreliable)
free_pcppages_bulk+0x1f8/0x940
free_unref_page+0xd0/0x100
xive_spapr_cleanup_queue+0x148/0x1b0
xive_teardown_cpu+0x1bc/0x240
pseries_mach_cpu_die+0x78/0x2f0
cpu_die+0x48/0x70
arch_cpu_idle_dead+0x20/0x40
do_idle+0x2f4/0x4c0
cpu_startup_entry+0x38/0x40
start_secondary+0x7bc/0x8f0
start_secondary_prolog+0x10/0x14
or on the worker thread handling the unplug:
pseries-hotplug-cpu: Attempting to remove CPU <NULL>, drc index: 1000013a
Querying DEAD? cpu 314 (314) shows 2
BUG: Bad page state in process kworker/u768:3 pfn:95de1
cpu 314 (hwid 314) Ready to die...
page:c00a000002577840 refcount:0 mapcount:-128 mapping:0000000000000000 index:0x0
flags: 0x5ffffc00000000()
raw: 005ffffc00000000 5deadbeef0000100 5deadbeef0000200 0000000000000000
raw: 0000000000000000 0000000000000000 00000000ffffff7f 0000000000000000
page dumped because: nonzero mapcount
Modules linked in: kvm xt_CHECKSUM ipt_MASQUERADE xt_conntrack ...
CPU: 0 PID: 548 Comm: kworker/u768:3 Kdump: loaded Not tainted 4.18.0-224.el8.bz1856588.ppc64le #1
Workqueue: pseries hotplug workque pseries_hp_work_fn
Call Trace:
dump_stack+0xb0/0xf4 (unreliable)
bad_page+0x12c/0x1b0
free_pcppages_bulk+0x5bc/0x940
page_alloc_cpu_dead+0x118/0x120
cpuhp_invoke_callback.constprop.5+0xb8/0x760
_cpu_down+0x188/0x340
cpu_down+0x5c/0xa0
cpu_subsys_offline+0x24/0x40
device_offline+0xf0/0x130
dlpar_offline_cpu+0x1c4/0x2a0
dlpar_cpu_remove+0xb8/0x190
dlpar_cpu_remove_by_index+0x12c/0x150
dlpar_cpu+0x94/0x800
pseries_hp_work_fn+0x128/0x1e0
process_one_work+0x304/0x5d0
worker_thread+0xcc/0x7a0
kthread+0x1ac/0x1c0
ret_from_kernel_thread+0x5c/0x80
The latter trace is due to the following sequence:
page_alloc_cpu_dead
drain_pages
drain_pages_zone
free_pcppages_bulk
where drain_pages() in this case is called under the assumption that
the unplugged cpu is no longer executing. To ensure that is the case,
and early call is made to __cpu_die()->pseries_cpu_die(), which runs a
loop that waits for the cpu to reach a halted state by polling its
status via query-cpu-stopped-state RTAS calls. It only polls for 25
iterations before giving up, however, and in the trace above this
results in the following being printed only .1 seconds after the
hotplug worker thread begins processing the unplug request:
pseries-hotplug-cpu: Attempting to remove CPU <NULL>, drc index: 1000013a
Querying DEAD? cpu 314 (314) shows 2
At that point the worker thread assumes the unplugged CPU is in some
unknown/dead state and procedes with the cleanup, causing the race
with the XIVE cleanup code executed by the unplugged CPU.
Fix this by waiting indefinitely, but also making an effort to avoid
spurious lockup messages by allowing for rescheduling after polling
the CPU status and printing a warning if we wait for longer than 120s.
Fixes: eac1e731b59ee ("powerpc/xive: guest exploitation of the XIVE interrupt controller")
Suggested-by: Michael Ellerman <mpe@ellerman.id.au>
Signed-off-by: Michael Roth <mdroth@linux.vnet.ibm.com>
Tested-by: Greg Kurz <groug@kaod.org>
Reviewed-by: Thiago Jung Bauermann <bauerman@linux.ibm.com>
Reviewed-by: Greg Kurz <groug@kaod.org>
[mpe: Trim oopses in change log slightly for readability]
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20200811161544.10513-1-mdroth@linux.vnet.ibm.com
2020-08-11 11:15:44 -05:00
if ( cpu_status = = QCSS_HARDWARE_ERROR ) {
pr_warn ( " CPU %i (hwid %i) reported error while dying \n " ,
cpu , pcpu ) ;
2006-12-05 17:52:38 +11:00
}
2011-03-30 22:57:33 -03:00
/* Isolation and deallocation are definitely done by
2006-12-05 17:52:38 +11:00
* drslot_chrp_cpu . If they were not they would be
* done here . Change isolate state to Isolate and
* change allocation - state to Unusable .
*/
2018-02-14 01:08:12 +10:00
paca_ptrs [ cpu ] - > cpu_start = 0 ;
2006-12-05 17:52:38 +11:00
}
/*
2010-04-26 15:32:44 +00:00
* Update cpu_present_mask and paca ( s ) for a new cpu node . The wrinkle
2006-12-05 17:52:38 +11:00
* here is that a cpu device node may represent up to two logical cpus
* in the SMT case . We must honor the assumption in other code that
* the logical ids for sibling SMT threads x and y are adjacent , such
* that x ^ 1 = = y and y ^ 1 = = x .
*/
2006-12-05 17:52:39 +11:00
static int pseries_add_processor ( struct device_node * np )
2006-12-05 17:52:38 +11:00
{
unsigned int cpu ;
2010-04-26 15:32:42 +00:00
cpumask_var_t candidate_mask , tmp ;
2006-12-05 17:52:38 +11:00
int err = - ENOSPC , len , nthreads , i ;
2014-09-16 15:15:45 -05:00
const __be32 * intserv ;
2006-12-05 17:52:38 +11:00
2007-04-03 22:26:41 +10:00
intserv = of_get_property ( np , " ibm,ppc-interrupt-server#s " , & len ) ;
2006-12-05 17:52:38 +11:00
if ( ! intserv )
return 0 ;
2010-04-26 15:32:42 +00:00
zalloc_cpumask_var ( & candidate_mask , GFP_KERNEL ) ;
zalloc_cpumask_var ( & tmp , GFP_KERNEL ) ;
2006-12-05 17:52:38 +11:00
nthreads = len / sizeof ( u32 ) ;
for ( i = 0 ; i < nthreads ; i + + )
2010-04-26 15:32:42 +00:00
cpumask_set_cpu ( i , tmp ) ;
2006-12-05 17:52:38 +11:00
2008-01-25 21:08:02 +01:00
cpu_maps_update_begin ( ) ;
2006-12-05 17:52:38 +11:00
2010-04-26 15:32:42 +00:00
BUG_ON ( ! cpumask_subset ( cpu_present_mask , cpu_possible_mask ) ) ;
2006-12-05 17:52:38 +11:00
/* Get a bitmap of unoccupied slots. */
2010-04-26 15:32:42 +00:00
cpumask_xor ( candidate_mask , cpu_possible_mask , cpu_present_mask ) ;
if ( cpumask_empty ( candidate_mask ) ) {
2006-12-05 17:52:38 +11:00
/* If we get here, it most likely means that NR_CPUS is
* less than the partition ' s max processors setting .
*/
2017-08-21 10:16:47 -05:00
printk ( KERN_ERR " Cannot add cpu %pOF; this system configuration "
" supports %d logical cpus. \n " , np ,
2015-01-21 16:21:14 -06:00
num_possible_cpus ( ) ) ;
2006-12-05 17:52:38 +11:00
goto out_unlock ;
}
2010-04-26 15:32:42 +00:00
while ( ! cpumask_empty ( tmp ) )
if ( cpumask_subset ( tmp , candidate_mask ) )
2006-12-05 17:52:38 +11:00
/* Found a range where we can insert the new cpu(s) */
break ;
else
2010-04-26 15:32:42 +00:00
cpumask_shift_left ( tmp , tmp , nthreads ) ;
2006-12-05 17:52:38 +11:00
2010-04-26 15:32:42 +00:00
if ( cpumask_empty ( tmp ) ) {
2010-04-26 15:32:44 +00:00
printk ( KERN_ERR " Unable to find space in cpu_present_mask for "
2018-08-27 20:52:07 -05:00
" processor %pOFn with %d thread(s) \n " , np ,
2006-12-05 17:52:38 +11:00
nthreads ) ;
goto out_unlock ;
}
2010-04-26 15:32:42 +00:00
for_each_cpu ( cpu , tmp ) {
2011-04-28 05:07:23 +00:00
BUG_ON ( cpu_present ( cpu ) ) ;
2009-09-24 09:34:48 -06:00
set_cpu_present ( cpu , true ) ;
2014-09-16 15:15:45 -05:00
set_hard_smp_processor_id ( cpu , be32_to_cpu ( * intserv + + ) ) ;
2006-12-05 17:52:38 +11:00
}
err = 0 ;
out_unlock :
2008-01-25 21:08:02 +01:00
cpu_maps_update_done ( ) ;
2010-04-26 15:32:42 +00:00
free_cpumask_var ( candidate_mask ) ;
free_cpumask_var ( tmp ) ;
2006-12-05 17:52:38 +11:00
return err ;
}
/*
* Update the present map for a cpu node which is going away , and set
* the hard id in the paca ( s ) to - 1 to be consistent with boot time
* convention for non - present cpus .
*/
2006-12-05 17:52:39 +11:00
static void pseries_remove_processor ( struct device_node * np )
2006-12-05 17:52:38 +11:00
{
unsigned int cpu ;
int len , nthreads , i ;
2014-09-12 14:11:42 -05:00
const __be32 * intserv ;
u32 thread ;
2006-12-05 17:52:38 +11:00
2007-04-03 22:26:41 +10:00
intserv = of_get_property ( np , " ibm,ppc-interrupt-server#s " , & len ) ;
2006-12-05 17:52:38 +11:00
if ( ! intserv )
return ;
nthreads = len / sizeof ( u32 ) ;
2008-01-25 21:08:02 +01:00
cpu_maps_update_begin ( ) ;
2006-12-05 17:52:38 +11:00
for ( i = 0 ; i < nthreads ; i + + ) {
2014-09-12 14:11:42 -05:00
thread = be32_to_cpu ( intserv [ i ] ) ;
2006-12-05 17:52:38 +11:00
for_each_present_cpu ( cpu ) {
2014-09-12 14:11:42 -05:00
if ( get_hard_smp_processor_id ( cpu ) ! = thread )
2006-12-05 17:52:38 +11:00
continue ;
BUG_ON ( cpu_online ( cpu ) ) ;
2009-09-24 09:34:48 -06:00
set_cpu_present ( cpu , false ) ;
2006-12-05 17:52:38 +11:00
set_hard_smp_processor_id ( cpu , - 1 ) ;
2018-01-26 13:41:59 -06:00
update_numa_cpu_lookup_table ( cpu , - 1 ) ;
2006-12-05 17:52:38 +11:00
break ;
}
2010-04-26 15:32:42 +00:00
if ( cpu > = nr_cpu_ids )
2006-12-05 17:52:38 +11:00
printk ( KERN_WARNING " Could not find cpu to remove "
2014-09-12 14:11:42 -05:00
" with physical id 0x%x \n " , thread ) ;
2006-12-05 17:52:38 +11:00
}
2008-01-25 21:08:02 +01:00
cpu_maps_update_done ( ) ;
2006-12-05 17:52:38 +11:00
}
powerpc/pseries: safely roll back failed DLPAR cpu add
dlpar_online_cpu() attempts to online all threads of a core that has
been added to an LPAR. If onlining a non-primary thread
fails (e.g. due to an allocation failure), the core is left with at
least one thread online. dlpar_cpu_add() attempts to roll back the
whole operation, releasing the core back to the platform. However,
since some threads of the core being removed are still online, the
BUG_ON(cpu_online(cpu)) in pseries_remove_processor() strikes:
LE PAGE_SIZE=64K MMU=Hash SMP NR_CPUS=2048 NUMA pSeries
Modules linked in:
CPU: 3 PID: 8587 Comm: drmgr Not tainted 5.3.0-rc2-00190-g9b123d1ea237-dirty #46
NIP: c0000000000eeb2c LR: c0000000000eeac4 CTR: c0000000000ee9e0
REGS: c0000001f745b6c0 TRAP: 0700 Not tainted (5.3.0-rc2-00190-g9b123d1ea237-dirty)
MSR: 800000010282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE,TM[E]> CR: 44002448 XER: 00000000
CFAR: c00000000195d718 IRQMASK: 0
GPR00: c0000000000eeac4 c0000001f745b950 c0000000032f6200 0000000000000008
GPR04: 0000000000000008 c000000003349c78 0000000000000040 00000000000001ff
GPR08: 0000000000000008 0000000000000000 0000000000000001 0007ffffffffffff
GPR12: 0000000084002844 c00000001ecacb80 0000000000000000 0000000000000000
GPR16: 0000000000000000 0000000000000000 0000000000000000 0000000000000000
GPR20: 0000000000000000 0000000000000000 0000000000000000 0000000000000008
GPR24: c000000003349ee0 c00000000334a2e4 c0000000fca4d7a8 c000000001d20048
GPR28: 0000000000000001 ffffffffffffffff ffffffffffffffff c0000000fca4d7c4
NIP [c0000000000eeb2c] pseries_smp_notifier+0x14c/0x2e0
LR [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0
Call Trace:
[c0000001f745b950] [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0 (unreliable)
[c0000001f745ba10] [c0000000001ac774] notifier_call_chain+0xb4/0x190
[c0000001f745bab0] [c0000000001ad62c] blocking_notifier_call_chain+0x7c/0xb0
[c0000001f745baf0] [c00000000167bda0] of_detach_node+0xc0/0x110
[c0000001f745bb50] [c0000000000e7ae4] dlpar_detach_node+0x64/0xa0
[c0000001f745bb80] [c0000000000edefc] dlpar_cpu_add+0x31c/0x360
[c0000001f745bc10] [c0000000000ee980] dlpar_cpu_probe+0x50/0xb0
[c0000001f745bc50] [c00000000002cf70] arch_cpu_probe+0x40/0x70
[c0000001f745bc70] [c000000000ccd808] cpu_probe_store+0x48/0x80
[c0000001f745bcb0] [c000000000cbcef8] dev_attr_store+0x38/0x60
[c0000001f745bcd0] [c00000000059c980] sysfs_kf_write+0x70/0xb0
[c0000001f745bd10] [c00000000059afb8] kernfs_fop_write+0xf8/0x280
[c0000001f745bd60] [c0000000004b437c] __vfs_write+0x3c/0x70
[c0000001f745bd80] [c0000000004b8710] vfs_write+0xd0/0x220
[c0000001f745bdd0] [c0000000004b8acc] ksys_write+0x7c/0x140
[c0000001f745be20] [c00000000000bbd8] system_call+0x5c/0x68
Move dlpar_offline_cpu() up in the file so that dlpar_online_cpu() can
use it to re-offline any threads that have been onlined when an error
is encountered.
Signed-off-by: Nathan Lynch <nathanl@linux.ibm.com>
Fixes: e666ae0b10aa ("powerpc/pseries: Update CPU hotplug error recovery")
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20191016183611.10867-3-nathanl@linux.ibm.com
2019-10-16 13:36:11 -05:00
static int dlpar_offline_cpu ( struct device_node * dn )
{
int rc = 0 ;
unsigned int cpu ;
int len , nthreads , i ;
const __be32 * intserv ;
u32 thread ;
intserv = of_get_property ( dn , " ibm,ppc-interrupt-server#s " , & len ) ;
if ( ! intserv )
return - EINVAL ;
nthreads = len / sizeof ( u32 ) ;
cpu_maps_update_begin ( ) ;
for ( i = 0 ; i < nthreads ; i + + ) {
thread = be32_to_cpu ( intserv [ i ] ) ;
for_each_present_cpu ( cpu ) {
if ( get_hard_smp_processor_id ( cpu ) ! = thread )
continue ;
2020-06-12 00:12:21 -05:00
if ( ! cpu_online ( cpu ) )
powerpc/pseries: safely roll back failed DLPAR cpu add
dlpar_online_cpu() attempts to online all threads of a core that has
been added to an LPAR. If onlining a non-primary thread
fails (e.g. due to an allocation failure), the core is left with at
least one thread online. dlpar_cpu_add() attempts to roll back the
whole operation, releasing the core back to the platform. However,
since some threads of the core being removed are still online, the
BUG_ON(cpu_online(cpu)) in pseries_remove_processor() strikes:
LE PAGE_SIZE=64K MMU=Hash SMP NR_CPUS=2048 NUMA pSeries
Modules linked in:
CPU: 3 PID: 8587 Comm: drmgr Not tainted 5.3.0-rc2-00190-g9b123d1ea237-dirty #46
NIP: c0000000000eeb2c LR: c0000000000eeac4 CTR: c0000000000ee9e0
REGS: c0000001f745b6c0 TRAP: 0700 Not tainted (5.3.0-rc2-00190-g9b123d1ea237-dirty)
MSR: 800000010282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE,TM[E]> CR: 44002448 XER: 00000000
CFAR: c00000000195d718 IRQMASK: 0
GPR00: c0000000000eeac4 c0000001f745b950 c0000000032f6200 0000000000000008
GPR04: 0000000000000008 c000000003349c78 0000000000000040 00000000000001ff
GPR08: 0000000000000008 0000000000000000 0000000000000001 0007ffffffffffff
GPR12: 0000000084002844 c00000001ecacb80 0000000000000000 0000000000000000
GPR16: 0000000000000000 0000000000000000 0000000000000000 0000000000000000
GPR20: 0000000000000000 0000000000000000 0000000000000000 0000000000000008
GPR24: c000000003349ee0 c00000000334a2e4 c0000000fca4d7a8 c000000001d20048
GPR28: 0000000000000001 ffffffffffffffff ffffffffffffffff c0000000fca4d7c4
NIP [c0000000000eeb2c] pseries_smp_notifier+0x14c/0x2e0
LR [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0
Call Trace:
[c0000001f745b950] [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0 (unreliable)
[c0000001f745ba10] [c0000000001ac774] notifier_call_chain+0xb4/0x190
[c0000001f745bab0] [c0000000001ad62c] blocking_notifier_call_chain+0x7c/0xb0
[c0000001f745baf0] [c00000000167bda0] of_detach_node+0xc0/0x110
[c0000001f745bb50] [c0000000000e7ae4] dlpar_detach_node+0x64/0xa0
[c0000001f745bb80] [c0000000000edefc] dlpar_cpu_add+0x31c/0x360
[c0000001f745bc10] [c0000000000ee980] dlpar_cpu_probe+0x50/0xb0
[c0000001f745bc50] [c00000000002cf70] arch_cpu_probe+0x40/0x70
[c0000001f745bc70] [c000000000ccd808] cpu_probe_store+0x48/0x80
[c0000001f745bcb0] [c000000000cbcef8] dev_attr_store+0x38/0x60
[c0000001f745bcd0] [c00000000059c980] sysfs_kf_write+0x70/0xb0
[c0000001f745bd10] [c00000000059afb8] kernfs_fop_write+0xf8/0x280
[c0000001f745bd60] [c0000000004b437c] __vfs_write+0x3c/0x70
[c0000001f745bd80] [c0000000004b8710] vfs_write+0xd0/0x220
[c0000001f745bdd0] [c0000000004b8acc] ksys_write+0x7c/0x140
[c0000001f745be20] [c00000000000bbd8] system_call+0x5c/0x68
Move dlpar_offline_cpu() up in the file so that dlpar_online_cpu() can
use it to re-offline any threads that have been onlined when an error
is encountered.
Signed-off-by: Nathan Lynch <nathanl@linux.ibm.com>
Fixes: e666ae0b10aa ("powerpc/pseries: Update CPU hotplug error recovery")
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20191016183611.10867-3-nathanl@linux.ibm.com
2019-10-16 13:36:11 -05:00
break ;
powerpc/pseries/hotplug-cpu: Show 'last online CPU' error in dlpar_cpu_offline()
One of the reasons that dlpar_cpu_offline can fail is when attempting to
offline the last online CPU of the kernel. This can be observed in a
pseries QEMU guest that has hotplugged CPUs. If the user offlines all
other CPUs of the guest, and a hotplugged CPU is now the last online
CPU, trying to reclaim it will fail.
The current error message in this situation returns rc with -EBUSY and a
generic explanation, e.g.:
pseries-hotplug-cpu: Failed to offline CPU PowerPC,POWER9, rc: -16
EBUSY can be caused by other conditions, such as cpu_hotplug_disable
being true. Throwing a more specific error message for this case,
instead of just "Failed to offline CPU", makes it clearer that the error
is in fact a known error situation instead of other generic/unknown
cause.
This patch adds a 'last online' check in dlpar_cpu_offline() to catch
the 'last online CPU' offline error, eturning a more informative error
message:
pseries-hotplug-cpu: Unable to remove last online CPU PowerPC,POWER9
Signed-off-by: Daniel Henrique Barboza <danielhb413@gmail.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20210323205056.52768-2-danielhb413@gmail.com
2021-03-23 17:50:56 -03:00
/*
* device_offline ( ) will return - EBUSY ( via cpu_down ( ) ) if there
* is only one CPU left . Check it here to fail earlier and with a
* more informative error message , while also retaining the
* cpu_add_remove_lock to be sure that no CPUs are being
* online / offlined during this check .
*/
if ( num_online_cpus ( ) = = 1 ) {
pr_warn ( " Unable to remove last online CPU %pOFn \n " , dn ) ;
rc = - EBUSY ;
goto out_unlock ;
}
2020-06-12 00:12:21 -05:00
cpu_maps_update_done ( ) ;
rc = device_offline ( get_cpu_device ( cpu ) ) ;
if ( rc )
goto out ;
cpu_maps_update_begin ( ) ;
powerpc/pseries: safely roll back failed DLPAR cpu add
dlpar_online_cpu() attempts to online all threads of a core that has
been added to an LPAR. If onlining a non-primary thread
fails (e.g. due to an allocation failure), the core is left with at
least one thread online. dlpar_cpu_add() attempts to roll back the
whole operation, releasing the core back to the platform. However,
since some threads of the core being removed are still online, the
BUG_ON(cpu_online(cpu)) in pseries_remove_processor() strikes:
LE PAGE_SIZE=64K MMU=Hash SMP NR_CPUS=2048 NUMA pSeries
Modules linked in:
CPU: 3 PID: 8587 Comm: drmgr Not tainted 5.3.0-rc2-00190-g9b123d1ea237-dirty #46
NIP: c0000000000eeb2c LR: c0000000000eeac4 CTR: c0000000000ee9e0
REGS: c0000001f745b6c0 TRAP: 0700 Not tainted (5.3.0-rc2-00190-g9b123d1ea237-dirty)
MSR: 800000010282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE,TM[E]> CR: 44002448 XER: 00000000
CFAR: c00000000195d718 IRQMASK: 0
GPR00: c0000000000eeac4 c0000001f745b950 c0000000032f6200 0000000000000008
GPR04: 0000000000000008 c000000003349c78 0000000000000040 00000000000001ff
GPR08: 0000000000000008 0000000000000000 0000000000000001 0007ffffffffffff
GPR12: 0000000084002844 c00000001ecacb80 0000000000000000 0000000000000000
GPR16: 0000000000000000 0000000000000000 0000000000000000 0000000000000000
GPR20: 0000000000000000 0000000000000000 0000000000000000 0000000000000008
GPR24: c000000003349ee0 c00000000334a2e4 c0000000fca4d7a8 c000000001d20048
GPR28: 0000000000000001 ffffffffffffffff ffffffffffffffff c0000000fca4d7c4
NIP [c0000000000eeb2c] pseries_smp_notifier+0x14c/0x2e0
LR [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0
Call Trace:
[c0000001f745b950] [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0 (unreliable)
[c0000001f745ba10] [c0000000001ac774] notifier_call_chain+0xb4/0x190
[c0000001f745bab0] [c0000000001ad62c] blocking_notifier_call_chain+0x7c/0xb0
[c0000001f745baf0] [c00000000167bda0] of_detach_node+0xc0/0x110
[c0000001f745bb50] [c0000000000e7ae4] dlpar_detach_node+0x64/0xa0
[c0000001f745bb80] [c0000000000edefc] dlpar_cpu_add+0x31c/0x360
[c0000001f745bc10] [c0000000000ee980] dlpar_cpu_probe+0x50/0xb0
[c0000001f745bc50] [c00000000002cf70] arch_cpu_probe+0x40/0x70
[c0000001f745bc70] [c000000000ccd808] cpu_probe_store+0x48/0x80
[c0000001f745bcb0] [c000000000cbcef8] dev_attr_store+0x38/0x60
[c0000001f745bcd0] [c00000000059c980] sysfs_kf_write+0x70/0xb0
[c0000001f745bd10] [c00000000059afb8] kernfs_fop_write+0xf8/0x280
[c0000001f745bd60] [c0000000004b437c] __vfs_write+0x3c/0x70
[c0000001f745bd80] [c0000000004b8710] vfs_write+0xd0/0x220
[c0000001f745bdd0] [c0000000004b8acc] ksys_write+0x7c/0x140
[c0000001f745be20] [c00000000000bbd8] system_call+0x5c/0x68
Move dlpar_offline_cpu() up in the file so that dlpar_online_cpu() can
use it to re-offline any threads that have been onlined when an error
is encountered.
Signed-off-by: Nathan Lynch <nathanl@linux.ibm.com>
Fixes: e666ae0b10aa ("powerpc/pseries: Update CPU hotplug error recovery")
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20191016183611.10867-3-nathanl@linux.ibm.com
2019-10-16 13:36:11 -05:00
break ;
}
if ( cpu = = num_possible_cpus ( ) ) {
pr_warn ( " Could not find cpu to offline with physical id 0x%x \n " ,
thread ) ;
}
}
powerpc/pseries/hotplug-cpu: Show 'last online CPU' error in dlpar_cpu_offline()
One of the reasons that dlpar_cpu_offline can fail is when attempting to
offline the last online CPU of the kernel. This can be observed in a
pseries QEMU guest that has hotplugged CPUs. If the user offlines all
other CPUs of the guest, and a hotplugged CPU is now the last online
CPU, trying to reclaim it will fail.
The current error message in this situation returns rc with -EBUSY and a
generic explanation, e.g.:
pseries-hotplug-cpu: Failed to offline CPU PowerPC,POWER9, rc: -16
EBUSY can be caused by other conditions, such as cpu_hotplug_disable
being true. Throwing a more specific error message for this case,
instead of just "Failed to offline CPU", makes it clearer that the error
is in fact a known error situation instead of other generic/unknown
cause.
This patch adds a 'last online' check in dlpar_cpu_offline() to catch
the 'last online CPU' offline error, eturning a more informative error
message:
pseries-hotplug-cpu: Unable to remove last online CPU PowerPC,POWER9
Signed-off-by: Daniel Henrique Barboza <danielhb413@gmail.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20210323205056.52768-2-danielhb413@gmail.com
2021-03-23 17:50:56 -03:00
out_unlock :
powerpc/pseries: safely roll back failed DLPAR cpu add
dlpar_online_cpu() attempts to online all threads of a core that has
been added to an LPAR. If onlining a non-primary thread
fails (e.g. due to an allocation failure), the core is left with at
least one thread online. dlpar_cpu_add() attempts to roll back the
whole operation, releasing the core back to the platform. However,
since some threads of the core being removed are still online, the
BUG_ON(cpu_online(cpu)) in pseries_remove_processor() strikes:
LE PAGE_SIZE=64K MMU=Hash SMP NR_CPUS=2048 NUMA pSeries
Modules linked in:
CPU: 3 PID: 8587 Comm: drmgr Not tainted 5.3.0-rc2-00190-g9b123d1ea237-dirty #46
NIP: c0000000000eeb2c LR: c0000000000eeac4 CTR: c0000000000ee9e0
REGS: c0000001f745b6c0 TRAP: 0700 Not tainted (5.3.0-rc2-00190-g9b123d1ea237-dirty)
MSR: 800000010282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE,TM[E]> CR: 44002448 XER: 00000000
CFAR: c00000000195d718 IRQMASK: 0
GPR00: c0000000000eeac4 c0000001f745b950 c0000000032f6200 0000000000000008
GPR04: 0000000000000008 c000000003349c78 0000000000000040 00000000000001ff
GPR08: 0000000000000008 0000000000000000 0000000000000001 0007ffffffffffff
GPR12: 0000000084002844 c00000001ecacb80 0000000000000000 0000000000000000
GPR16: 0000000000000000 0000000000000000 0000000000000000 0000000000000000
GPR20: 0000000000000000 0000000000000000 0000000000000000 0000000000000008
GPR24: c000000003349ee0 c00000000334a2e4 c0000000fca4d7a8 c000000001d20048
GPR28: 0000000000000001 ffffffffffffffff ffffffffffffffff c0000000fca4d7c4
NIP [c0000000000eeb2c] pseries_smp_notifier+0x14c/0x2e0
LR [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0
Call Trace:
[c0000001f745b950] [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0 (unreliable)
[c0000001f745ba10] [c0000000001ac774] notifier_call_chain+0xb4/0x190
[c0000001f745bab0] [c0000000001ad62c] blocking_notifier_call_chain+0x7c/0xb0
[c0000001f745baf0] [c00000000167bda0] of_detach_node+0xc0/0x110
[c0000001f745bb50] [c0000000000e7ae4] dlpar_detach_node+0x64/0xa0
[c0000001f745bb80] [c0000000000edefc] dlpar_cpu_add+0x31c/0x360
[c0000001f745bc10] [c0000000000ee980] dlpar_cpu_probe+0x50/0xb0
[c0000001f745bc50] [c00000000002cf70] arch_cpu_probe+0x40/0x70
[c0000001f745bc70] [c000000000ccd808] cpu_probe_store+0x48/0x80
[c0000001f745bcb0] [c000000000cbcef8] dev_attr_store+0x38/0x60
[c0000001f745bcd0] [c00000000059c980] sysfs_kf_write+0x70/0xb0
[c0000001f745bd10] [c00000000059afb8] kernfs_fop_write+0xf8/0x280
[c0000001f745bd60] [c0000000004b437c] __vfs_write+0x3c/0x70
[c0000001f745bd80] [c0000000004b8710] vfs_write+0xd0/0x220
[c0000001f745bdd0] [c0000000004b8acc] ksys_write+0x7c/0x140
[c0000001f745be20] [c00000000000bbd8] system_call+0x5c/0x68
Move dlpar_offline_cpu() up in the file so that dlpar_online_cpu() can
use it to re-offline any threads that have been onlined when an error
is encountered.
Signed-off-by: Nathan Lynch <nathanl@linux.ibm.com>
Fixes: e666ae0b10aa ("powerpc/pseries: Update CPU hotplug error recovery")
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20191016183611.10867-3-nathanl@linux.ibm.com
2019-10-16 13:36:11 -05:00
cpu_maps_update_done ( ) ;
out :
return rc ;
}
2015-12-16 14:50:21 -06:00
static int dlpar_online_cpu ( struct device_node * dn )
{
int rc = 0 ;
unsigned int cpu ;
int len , nthreads , i ;
const __be32 * intserv ;
u32 thread ;
intserv = of_get_property ( dn , " ibm,ppc-interrupt-server#s " , & len ) ;
if ( ! intserv )
return - EINVAL ;
nthreads = len / sizeof ( u32 ) ;
cpu_maps_update_begin ( ) ;
for ( i = 0 ; i < nthreads ; i + + ) {
thread = be32_to_cpu ( intserv [ i ] ) ;
for_each_present_cpu ( cpu ) {
if ( get_hard_smp_processor_id ( cpu ) ! = thread )
continue ;
cpu_maps_update_done ( ) ;
2017-11-28 16:58:43 -06:00
find_and_online_cpu_nid ( cpu ) ;
2015-12-16 14:50:21 -06:00
rc = device_online ( get_cpu_device ( cpu ) ) ;
powerpc/pseries: safely roll back failed DLPAR cpu add
dlpar_online_cpu() attempts to online all threads of a core that has
been added to an LPAR. If onlining a non-primary thread
fails (e.g. due to an allocation failure), the core is left with at
least one thread online. dlpar_cpu_add() attempts to roll back the
whole operation, releasing the core back to the platform. However,
since some threads of the core being removed are still online, the
BUG_ON(cpu_online(cpu)) in pseries_remove_processor() strikes:
LE PAGE_SIZE=64K MMU=Hash SMP NR_CPUS=2048 NUMA pSeries
Modules linked in:
CPU: 3 PID: 8587 Comm: drmgr Not tainted 5.3.0-rc2-00190-g9b123d1ea237-dirty #46
NIP: c0000000000eeb2c LR: c0000000000eeac4 CTR: c0000000000ee9e0
REGS: c0000001f745b6c0 TRAP: 0700 Not tainted (5.3.0-rc2-00190-g9b123d1ea237-dirty)
MSR: 800000010282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE,TM[E]> CR: 44002448 XER: 00000000
CFAR: c00000000195d718 IRQMASK: 0
GPR00: c0000000000eeac4 c0000001f745b950 c0000000032f6200 0000000000000008
GPR04: 0000000000000008 c000000003349c78 0000000000000040 00000000000001ff
GPR08: 0000000000000008 0000000000000000 0000000000000001 0007ffffffffffff
GPR12: 0000000084002844 c00000001ecacb80 0000000000000000 0000000000000000
GPR16: 0000000000000000 0000000000000000 0000000000000000 0000000000000000
GPR20: 0000000000000000 0000000000000000 0000000000000000 0000000000000008
GPR24: c000000003349ee0 c00000000334a2e4 c0000000fca4d7a8 c000000001d20048
GPR28: 0000000000000001 ffffffffffffffff ffffffffffffffff c0000000fca4d7c4
NIP [c0000000000eeb2c] pseries_smp_notifier+0x14c/0x2e0
LR [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0
Call Trace:
[c0000001f745b950] [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0 (unreliable)
[c0000001f745ba10] [c0000000001ac774] notifier_call_chain+0xb4/0x190
[c0000001f745bab0] [c0000000001ad62c] blocking_notifier_call_chain+0x7c/0xb0
[c0000001f745baf0] [c00000000167bda0] of_detach_node+0xc0/0x110
[c0000001f745bb50] [c0000000000e7ae4] dlpar_detach_node+0x64/0xa0
[c0000001f745bb80] [c0000000000edefc] dlpar_cpu_add+0x31c/0x360
[c0000001f745bc10] [c0000000000ee980] dlpar_cpu_probe+0x50/0xb0
[c0000001f745bc50] [c00000000002cf70] arch_cpu_probe+0x40/0x70
[c0000001f745bc70] [c000000000ccd808] cpu_probe_store+0x48/0x80
[c0000001f745bcb0] [c000000000cbcef8] dev_attr_store+0x38/0x60
[c0000001f745bcd0] [c00000000059c980] sysfs_kf_write+0x70/0xb0
[c0000001f745bd10] [c00000000059afb8] kernfs_fop_write+0xf8/0x280
[c0000001f745bd60] [c0000000004b437c] __vfs_write+0x3c/0x70
[c0000001f745bd80] [c0000000004b8710] vfs_write+0xd0/0x220
[c0000001f745bdd0] [c0000000004b8acc] ksys_write+0x7c/0x140
[c0000001f745be20] [c00000000000bbd8] system_call+0x5c/0x68
Move dlpar_offline_cpu() up in the file so that dlpar_online_cpu() can
use it to re-offline any threads that have been onlined when an error
is encountered.
Signed-off-by: Nathan Lynch <nathanl@linux.ibm.com>
Fixes: e666ae0b10aa ("powerpc/pseries: Update CPU hotplug error recovery")
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20191016183611.10867-3-nathanl@linux.ibm.com
2019-10-16 13:36:11 -05:00
if ( rc ) {
dlpar_offline_cpu ( dn ) ;
2015-12-16 14:50:21 -06:00
goto out ;
powerpc/pseries: safely roll back failed DLPAR cpu add
dlpar_online_cpu() attempts to online all threads of a core that has
been added to an LPAR. If onlining a non-primary thread
fails (e.g. due to an allocation failure), the core is left with at
least one thread online. dlpar_cpu_add() attempts to roll back the
whole operation, releasing the core back to the platform. However,
since some threads of the core being removed are still online, the
BUG_ON(cpu_online(cpu)) in pseries_remove_processor() strikes:
LE PAGE_SIZE=64K MMU=Hash SMP NR_CPUS=2048 NUMA pSeries
Modules linked in:
CPU: 3 PID: 8587 Comm: drmgr Not tainted 5.3.0-rc2-00190-g9b123d1ea237-dirty #46
NIP: c0000000000eeb2c LR: c0000000000eeac4 CTR: c0000000000ee9e0
REGS: c0000001f745b6c0 TRAP: 0700 Not tainted (5.3.0-rc2-00190-g9b123d1ea237-dirty)
MSR: 800000010282b033 <SF,VEC,VSX,EE,FP,ME,IR,DR,RI,LE,TM[E]> CR: 44002448 XER: 00000000
CFAR: c00000000195d718 IRQMASK: 0
GPR00: c0000000000eeac4 c0000001f745b950 c0000000032f6200 0000000000000008
GPR04: 0000000000000008 c000000003349c78 0000000000000040 00000000000001ff
GPR08: 0000000000000008 0000000000000000 0000000000000001 0007ffffffffffff
GPR12: 0000000084002844 c00000001ecacb80 0000000000000000 0000000000000000
GPR16: 0000000000000000 0000000000000000 0000000000000000 0000000000000000
GPR20: 0000000000000000 0000000000000000 0000000000000000 0000000000000008
GPR24: c000000003349ee0 c00000000334a2e4 c0000000fca4d7a8 c000000001d20048
GPR28: 0000000000000001 ffffffffffffffff ffffffffffffffff c0000000fca4d7c4
NIP [c0000000000eeb2c] pseries_smp_notifier+0x14c/0x2e0
LR [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0
Call Trace:
[c0000001f745b950] [c0000000000eeac4] pseries_smp_notifier+0xe4/0x2e0 (unreliable)
[c0000001f745ba10] [c0000000001ac774] notifier_call_chain+0xb4/0x190
[c0000001f745bab0] [c0000000001ad62c] blocking_notifier_call_chain+0x7c/0xb0
[c0000001f745baf0] [c00000000167bda0] of_detach_node+0xc0/0x110
[c0000001f745bb50] [c0000000000e7ae4] dlpar_detach_node+0x64/0xa0
[c0000001f745bb80] [c0000000000edefc] dlpar_cpu_add+0x31c/0x360
[c0000001f745bc10] [c0000000000ee980] dlpar_cpu_probe+0x50/0xb0
[c0000001f745bc50] [c00000000002cf70] arch_cpu_probe+0x40/0x70
[c0000001f745bc70] [c000000000ccd808] cpu_probe_store+0x48/0x80
[c0000001f745bcb0] [c000000000cbcef8] dev_attr_store+0x38/0x60
[c0000001f745bcd0] [c00000000059c980] sysfs_kf_write+0x70/0xb0
[c0000001f745bd10] [c00000000059afb8] kernfs_fop_write+0xf8/0x280
[c0000001f745bd60] [c0000000004b437c] __vfs_write+0x3c/0x70
[c0000001f745bd80] [c0000000004b8710] vfs_write+0xd0/0x220
[c0000001f745bdd0] [c0000000004b8acc] ksys_write+0x7c/0x140
[c0000001f745be20] [c00000000000bbd8] system_call+0x5c/0x68
Move dlpar_offline_cpu() up in the file so that dlpar_online_cpu() can
use it to re-offline any threads that have been onlined when an error
is encountered.
Signed-off-by: Nathan Lynch <nathanl@linux.ibm.com>
Fixes: e666ae0b10aa ("powerpc/pseries: Update CPU hotplug error recovery")
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/20191016183611.10867-3-nathanl@linux.ibm.com
2019-10-16 13:36:11 -05:00
}
2015-12-16 14:50:21 -06:00
cpu_maps_update_begin ( ) ;
break ;
}
if ( cpu = = num_possible_cpus ( ) )
printk ( KERN_WARNING " Could not find cpu to online "
" with physical id 0x%x \n " , thread ) ;
}
cpu_maps_update_done ( ) ;
out :
return rc ;
}
static bool dlpar_cpu_exists ( struct device_node * parent , u32 drc_index )
{
struct device_node * child = NULL ;
u32 my_drc_index ;
bool found ;
int rc ;
/* Assume cpu doesn't exist */
found = false ;
for_each_child_of_node ( parent , child ) {
rc = of_property_read_u32 ( child , " ibm,my-drc-index " ,
& my_drc_index ) ;
if ( rc )
continue ;
if ( my_drc_index = = drc_index ) {
of_node_put ( child ) ;
found = true ;
break ;
}
}
return found ;
}
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
static bool drc_info_valid_index ( struct device_node * parent , u32 drc_index )
{
struct property * info ;
struct of_drc_info drc ;
const __be32 * value ;
u32 index ;
int count , i , j ;
info = of_find_property ( parent , " ibm,drc-info " , NULL ) ;
if ( ! info )
return false ;
value = of_prop_next_u32 ( info , NULL , & count ) ;
/* First value of ibm,drc-info is number of drc-info records */
if ( value )
value + + ;
else
return false ;
for ( i = 0 ; i < count ; i + + ) {
if ( of_read_drc_info_cell ( & info , & value , & drc ) )
return false ;
if ( strncmp ( drc . drc_type , " CPU " , 3 ) )
break ;
if ( drc_index > drc . last_drc_index )
continue ;
index = drc . drc_index_start ;
for ( j = 0 ; j < drc . num_sequential_elems ; j + + ) {
if ( drc_index = = index )
return true ;
index + = drc . sequential_inc ;
}
}
return false ;
}
2015-12-16 14:55:07 -06:00
static bool valid_cpu_drc_index ( struct device_node * parent , u32 drc_index )
{
bool found = false ;
int rc , index ;
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
if ( of_find_property ( parent , " ibm,drc-info " , NULL ) )
return drc_info_valid_index ( parent , drc_index ) ;
/* Note that the format of the ibm,drc-indexes array is
* the number of entries in the array followed by the array
* of drc values so we start looking at index = 1.
*/
index = 1 ;
2015-12-16 14:55:07 -06:00
while ( ! found ) {
u32 drc ;
rc = of_property_read_u32_index ( parent , " ibm,drc-indexes " ,
index + + , & drc ) ;
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
2015-12-16 14:55:07 -06:00
if ( rc )
break ;
if ( drc = = drc_index )
found = true ;
}
return found ;
}
2015-12-16 14:51:26 -06:00
static ssize_t dlpar_cpu_add ( u32 drc_index )
2015-12-16 14:50:21 -06:00
{
struct device_node * dn , * parent ;
2015-12-16 14:52:39 -06:00
int rc , saved_rc ;
pr_debug ( " Attempting to add CPU, drc index: %x \n " , drc_index ) ;
2015-12-16 14:50:21 -06:00
parent = of_find_node_by_path ( " /cpus " ) ;
2015-12-16 14:52:39 -06:00
if ( ! parent ) {
pr_warn ( " Failed to find CPU root node \" /cpus \" \n " ) ;
2015-12-16 14:50:21 -06:00
return - ENODEV ;
2015-12-16 14:52:39 -06:00
}
2015-12-16 14:50:21 -06:00
if ( dlpar_cpu_exists ( parent , drc_index ) ) {
of_node_put ( parent ) ;
2015-12-16 14:52:39 -06:00
pr_warn ( " CPU with drc index %x already exists \n " , drc_index ) ;
2015-12-16 14:50:21 -06:00
return - EINVAL ;
}
2015-12-16 14:55:07 -06:00
if ( ! valid_cpu_drc_index ( parent , drc_index ) ) {
of_node_put ( parent ) ;
pr_warn ( " Cannot find CPU (drc index %x) to add. \n " , drc_index ) ;
return - EINVAL ;
}
2015-12-16 14:50:21 -06:00
rc = dlpar_acquire_drc ( drc_index ) ;
if ( rc ) {
2015-12-16 14:52:39 -06:00
pr_warn ( " Failed to acquire DRC, rc: %d, drc index: %x \n " ,
rc , drc_index ) ;
2015-12-16 14:50:21 -06:00
of_node_put ( parent ) ;
return - EINVAL ;
}
dn = dlpar_configure_connector ( cpu_to_be32 ( drc_index ) , parent ) ;
2015-12-16 14:52:39 -06:00
if ( ! dn ) {
pr_warn ( " Failed call to configure-connector, drc index: %x \n " ,
drc_index ) ;
dlpar_release_drc ( drc_index ) ;
2017-09-20 17:02:51 -04:00
of_node_put ( parent ) ;
2015-12-16 14:50:21 -06:00
return - EINVAL ;
2015-12-16 14:52:39 -06:00
}
2015-12-16 14:50:21 -06:00
2017-08-21 10:16:49 -05:00
rc = dlpar_attach_node ( dn , parent ) ;
2017-09-20 17:02:51 -04:00
/* Regardless we are done with parent now */
of_node_put ( parent ) ;
2015-12-16 14:50:21 -06:00
if ( rc ) {
2015-12-16 14:52:39 -06:00
saved_rc = rc ;
2018-08-27 20:52:07 -05:00
pr_warn ( " Failed to attach node %pOFn, rc: %d, drc index: %x \n " ,
dn , rc , drc_index ) ;
2015-12-16 14:52:39 -06:00
rc = dlpar_release_drc ( drc_index ) ;
if ( ! rc )
dlpar_free_cc_nodes ( dn ) ;
return saved_rc ;
2015-12-16 14:50:21 -06:00
}
rc = dlpar_online_cpu ( dn ) ;
2015-12-16 14:52:39 -06:00
if ( rc ) {
saved_rc = rc ;
2018-08-27 20:52:07 -05:00
pr_warn ( " Failed to online cpu %pOFn, rc: %d, drc index: %x \n " ,
dn , rc , drc_index ) ;
2015-12-16 14:52:39 -06:00
rc = dlpar_detach_node ( dn ) ;
if ( ! rc )
dlpar_release_drc ( drc_index ) ;
return saved_rc ;
}
2018-08-27 20:52:07 -05:00
pr_debug ( " Successfully added CPU %pOFn, drc index: %x \n " , dn ,
2015-12-16 14:52:39 -06:00
drc_index ) ;
2015-12-16 14:51:26 -06:00
return rc ;
2015-12-16 14:50:21 -06:00
}
2015-12-16 14:51:26 -06:00
static ssize_t dlpar_cpu_remove ( struct device_node * dn , u32 drc_index )
{
int rc ;
2018-08-27 20:52:07 -05:00
pr_debug ( " Attempting to remove CPU %pOFn, drc index: %x \n " ,
dn , drc_index ) ;
2015-12-16 14:52:39 -06:00
2015-12-16 14:51:26 -06:00
rc = dlpar_offline_cpu ( dn ) ;
2015-12-16 14:52:39 -06:00
if ( rc ) {
2018-08-27 20:52:07 -05:00
pr_warn ( " Failed to offline CPU %pOFn, rc: %d \n " , dn , rc ) ;
2015-12-16 14:51:26 -06:00
return - EINVAL ;
2015-12-16 14:52:39 -06:00
}
2015-12-16 14:51:26 -06:00
rc = dlpar_release_drc ( drc_index ) ;
2015-12-16 14:52:39 -06:00
if ( rc ) {
2018-08-27 20:52:07 -05:00
pr_warn ( " Failed to release drc (%x) for CPU %pOFn, rc: %d \n " ,
drc_index , dn , rc ) ;
2015-12-16 14:52:39 -06:00
dlpar_online_cpu ( dn ) ;
2015-12-16 14:51:26 -06:00
return rc ;
2015-12-16 14:52:39 -06:00
}
2015-12-16 14:51:26 -06:00
rc = dlpar_detach_node ( dn ) ;
2015-12-16 14:52:39 -06:00
if ( rc ) {
int saved_rc = rc ;
2015-12-16 14:51:26 -06:00
2018-08-27 20:52:07 -05:00
pr_warn ( " Failed to detach CPU %pOFn, rc: %d " , dn , rc ) ;
2015-12-16 14:52:39 -06:00
rc = dlpar_acquire_drc ( drc_index ) ;
if ( ! rc )
dlpar_online_cpu ( dn ) ;
return saved_rc ;
}
pr_debug ( " Successfully removed CPU, drc index: %x \n " , drc_index ) ;
return 0 ;
2015-12-16 14:51:26 -06:00
}
2015-12-16 14:54:05 -06:00
static struct device_node * cpu_drc_index_to_dn ( u32 drc_index )
{
struct device_node * dn ;
u32 my_index ;
int rc ;
for_each_node_by_type ( dn , " cpu " ) {
rc = of_property_read_u32 ( dn , " ibm,my-drc-index " , & my_index ) ;
if ( rc )
continue ;
if ( my_index = = drc_index )
break ;
}
return dn ;
}
static int dlpar_cpu_remove_by_index ( u32 drc_index )
{
struct device_node * dn ;
int rc ;
dn = cpu_drc_index_to_dn ( drc_index ) ;
if ( ! dn ) {
pr_warn ( " Cannot find CPU (drc index %x) to remove \n " ,
drc_index ) ;
return - ENODEV ;
}
rc = dlpar_cpu_remove ( dn , drc_index ) ;
of_node_put ( dn ) ;
return rc ;
}
static int find_dlpar_cpus_to_remove ( u32 * cpu_drcs , int cpus_to_remove )
{
struct device_node * dn ;
int cpus_found = 0 ;
int rc ;
/* We want to find cpus_to_remove + 1 CPUs to ensure we do not
* remove the last CPU .
*/
for_each_node_by_type ( dn , " cpu " ) {
cpus_found + + ;
if ( cpus_found > cpus_to_remove ) {
of_node_put ( dn ) ;
break ;
}
/* Note that cpus_found is always 1 ahead of the index
* into the cpu_drcs array , so we use cpus_found - 1
*/
rc = of_property_read_u32 ( dn , " ibm,my-drc-index " ,
& cpu_drcs [ cpus_found - 1 ] ) ;
if ( rc ) {
2018-08-27 20:52:07 -05:00
pr_warn ( " Error occurred getting drc-index for %pOFn \n " ,
dn ) ;
2015-12-16 14:54:05 -06:00
of_node_put ( dn ) ;
return - 1 ;
}
}
if ( cpus_found < cpus_to_remove ) {
pr_warn ( " Failed to find enough CPUs (%d of %d) to remove \n " ,
cpus_found , cpus_to_remove ) ;
} else if ( cpus_found = = cpus_to_remove ) {
pr_warn ( " Cannot remove all CPUs \n " ) ;
}
return cpus_found ;
}
static int dlpar_cpu_remove_by_count ( u32 cpus_to_remove )
{
u32 * cpu_drcs ;
int cpus_found ;
int cpus_removed = 0 ;
int i , rc ;
pr_debug ( " Attempting to hot-remove %d CPUs \n " , cpus_to_remove ) ;
cpu_drcs = kcalloc ( cpus_to_remove , sizeof ( * cpu_drcs ) , GFP_KERNEL ) ;
if ( ! cpu_drcs )
return - EINVAL ;
cpus_found = find_dlpar_cpus_to_remove ( cpu_drcs , cpus_to_remove ) ;
if ( cpus_found < = cpus_to_remove ) {
kfree ( cpu_drcs ) ;
return - EINVAL ;
}
for ( i = 0 ; i < cpus_to_remove ; i + + ) {
rc = dlpar_cpu_remove_by_index ( cpu_drcs [ i ] ) ;
if ( rc )
break ;
cpus_removed + + ;
}
if ( cpus_removed ! = cpus_to_remove ) {
pr_warn ( " CPU hot-remove failed, adding back removed CPUs \n " ) ;
for ( i = 0 ; i < cpus_removed ; i + + )
dlpar_cpu_add ( cpu_drcs [ i ] ) ;
rc = - EINVAL ;
} else {
rc = 0 ;
}
kfree ( cpu_drcs ) ;
return rc ;
}
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
static int find_drc_info_cpus_to_add ( struct device_node * cpus ,
struct property * info ,
u32 * cpu_drcs , u32 cpus_to_add )
2015-12-16 14:55:07 -06:00
{
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
struct of_drc_info drc ;
const __be32 * value ;
u32 count , drc_index ;
2015-12-16 14:55:07 -06:00
int cpus_found = 0 ;
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
int i , j ;
2015-12-16 14:55:07 -06:00
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
if ( ! info )
2015-12-16 14:55:07 -06:00
return - 1 ;
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
value = of_prop_next_u32 ( info , NULL , & count ) ;
if ( value )
value + + ;
for ( i = 0 ; i < count ; i + + ) {
of_read_drc_info_cell ( & info , & value , & drc ) ;
if ( strncmp ( drc . drc_type , " CPU " , 3 ) )
break ;
drc_index = drc . drc_index_start ;
for ( j = 0 ; j < drc . num_sequential_elems ; j + + ) {
if ( dlpar_cpu_exists ( cpus , drc_index ) )
continue ;
cpu_drcs [ cpus_found + + ] = drc_index ;
if ( cpus_found = = cpus_to_add )
return cpus_found ;
drc_index + = drc . sequential_inc ;
}
2015-12-16 14:55:07 -06:00
}
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
return cpus_found ;
}
static int find_drc_index_cpus_to_add ( struct device_node * cpus ,
u32 * cpu_drcs , u32 cpus_to_add )
{
int cpus_found = 0 ;
int index , rc ;
u32 drc_index ;
2015-12-16 14:55:07 -06:00
/* Search the ibm,drc-indexes array for possible CPU drcs to
* add . Note that the format of the ibm , drc - indexes array is
* the number of entries in the array followed by the array
* of drc values so we start looking at index = 1.
*/
index = 1 ;
while ( cpus_found < cpus_to_add ) {
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
rc = of_property_read_u32_index ( cpus , " ibm,drc-indexes " ,
index + + , & drc_index ) ;
2015-12-16 14:55:07 -06:00
if ( rc )
break ;
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
if ( dlpar_cpu_exists ( cpus , drc_index ) )
2015-12-16 14:55:07 -06:00
continue ;
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
cpu_drcs [ cpus_found + + ] = drc_index ;
2015-12-16 14:55:07 -06:00
}
return cpus_found ;
}
static int dlpar_cpu_add_by_count ( u32 cpus_to_add )
{
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
struct device_node * parent ;
struct property * info ;
2015-12-16 14:55:07 -06:00
u32 * cpu_drcs ;
int cpus_added = 0 ;
int cpus_found ;
int i , rc ;
pr_debug ( " Attempting to hot-add %d CPUs \n " , cpus_to_add ) ;
cpu_drcs = kcalloc ( cpus_to_add , sizeof ( * cpu_drcs ) , GFP_KERNEL ) ;
if ( ! cpu_drcs )
return - EINVAL ;
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
parent = of_find_node_by_path ( " /cpus " ) ;
if ( ! parent ) {
pr_warn ( " Could not find CPU root node in device tree \n " ) ;
2020-11-10 21:07:52 -05:00
kfree ( cpu_drcs ) ;
powerpc/pseries: Add cpu DLPAR support for drc-info property
Older firmwares provided information about Dynamic Reconfig
Connectors (DRC) through several device tree properties, namely
ibm,drc-types, ibm,drc-indexes, ibm,drc-names, and
ibm,drc-power-domains. New firmwares have the ability to present this
same information in a much condensed format through a device tree
property called ibm,drc-info.
The existing cpu DLPAR hotplug code only understands the older DRC
property format when validating the drc-index of a cpu during a
hotplug add. This updates those code paths to use the ibm,drc-info
property, when present, instead for validation.
Signed-off-by: Tyrel Datwyler <tyreld@linux.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
Link: https://lore.kernel.org/r/1573449697-5448-4-git-send-email-tyreld@linux.ibm.com
2019-11-10 23:21:30 -06:00
return - 1 ;
}
info = of_find_property ( parent , " ibm,drc-info " , NULL ) ;
if ( info )
cpus_found = find_drc_info_cpus_to_add ( parent , info , cpu_drcs , cpus_to_add ) ;
else
cpus_found = find_drc_index_cpus_to_add ( parent , cpu_drcs , cpus_to_add ) ;
of_node_put ( parent ) ;
2015-12-16 14:55:07 -06:00
if ( cpus_found < cpus_to_add ) {
pr_warn ( " Failed to find enough CPUs (%d of %d) to add \n " ,
cpus_found , cpus_to_add ) ;
kfree ( cpu_drcs ) ;
return - EINVAL ;
}
for ( i = 0 ; i < cpus_to_add ; i + + ) {
rc = dlpar_cpu_add ( cpu_drcs [ i ] ) ;
if ( rc )
break ;
cpus_added + + ;
}
if ( cpus_added < cpus_to_add ) {
pr_warn ( " CPU hot-add failed, removing any added CPUs \n " ) ;
for ( i = 0 ; i < cpus_added ; i + + )
dlpar_cpu_remove_by_index ( cpu_drcs [ i ] ) ;
rc = - EINVAL ;
} else {
rc = 0 ;
}
kfree ( cpu_drcs ) ;
return rc ;
}
2015-12-16 14:54:05 -06:00
int dlpar_cpu ( struct pseries_hp_errorlog * hp_elog )
{
u32 count , drc_index ;
int rc ;
count = hp_elog - > _drc_u . drc_count ;
drc_index = hp_elog - > _drc_u . drc_index ;
lock_device_hotplug ( ) ;
switch ( hp_elog - > action ) {
case PSERIES_HP_ELOG_ACTION_REMOVE :
if ( hp_elog - > id_type = = PSERIES_HP_ELOG_ID_DRC_COUNT )
rc = dlpar_cpu_remove_by_count ( count ) ;
else if ( hp_elog - > id_type = = PSERIES_HP_ELOG_ID_DRC_INDEX )
rc = dlpar_cpu_remove_by_index ( drc_index ) ;
else
rc = - EINVAL ;
break ;
2015-12-16 14:55:07 -06:00
case PSERIES_HP_ELOG_ACTION_ADD :
if ( hp_elog - > id_type = = PSERIES_HP_ELOG_ID_DRC_COUNT )
rc = dlpar_cpu_add_by_count ( count ) ;
else if ( hp_elog - > id_type = = PSERIES_HP_ELOG_ID_DRC_INDEX )
rc = dlpar_cpu_add ( drc_index ) ;
else
rc = - EINVAL ;
break ;
2015-12-16 14:54:05 -06:00
default :
pr_err ( " Invalid action (%d) specified \n " , hp_elog - > action ) ;
rc = - EINVAL ;
break ;
}
unlock_device_hotplug ( ) ;
return rc ;
}
2015-12-16 14:51:26 -06:00
# ifdef CONFIG_ARCH_CPU_PROBE_RELEASE
static ssize_t dlpar_cpu_probe ( const char * buf , size_t count )
{
u32 drc_index ;
int rc ;
rc = kstrtou32 ( buf , 0 , & drc_index ) ;
if ( rc )
return - EINVAL ;
rc = dlpar_cpu_add ( drc_index ) ;
return rc ? rc : count ;
}
2015-12-16 14:50:21 -06:00
static ssize_t dlpar_cpu_release ( const char * buf , size_t count )
{
struct device_node * dn ;
u32 drc_index ;
int rc ;
dn = of_find_node_by_path ( buf ) ;
if ( ! dn )
return - EINVAL ;
rc = of_property_read_u32 ( dn , " ibm,my-drc-index " , & drc_index ) ;
if ( rc ) {
of_node_put ( dn ) ;
return - EINVAL ;
}
2015-12-16 14:51:26 -06:00
rc = dlpar_cpu_remove ( dn , drc_index ) ;
2015-12-16 14:50:21 -06:00
of_node_put ( dn ) ;
2015-12-16 14:51:26 -06:00
return rc ? rc : count ;
2015-12-16 14:50:21 -06:00
}
# endif /* CONFIG_ARCH_CPU_PROBE_RELEASE */
2006-12-05 17:52:39 +11:00
static int pseries_smp_notifier ( struct notifier_block * nb ,
2014-11-24 17:58:01 +00:00
unsigned long action , void * data )
2006-12-05 17:52:38 +11:00
{
2014-11-24 17:58:01 +00:00
struct of_reconfig_data * rd = data ;
2011-06-21 03:35:56 +00:00
int err = 0 ;
2006-12-05 17:52:38 +11:00
switch ( action ) {
2012-10-02 16:57:57 +00:00
case OF_RECONFIG_ATTACH_NODE :
2014-11-24 17:58:01 +00:00
err = pseries_add_processor ( rd - > dn ) ;
2006-12-05 17:52:38 +11:00
break ;
2012-10-02 16:57:57 +00:00
case OF_RECONFIG_DETACH_NODE :
2014-11-24 17:58:01 +00:00
pseries_remove_processor ( rd - > dn ) ;
2006-12-05 17:52:38 +11:00
break ;
}
2011-06-21 03:35:56 +00:00
return notifier_from_errno ( err ) ;
2006-12-05 17:52:38 +11:00
}
2006-12-05 17:52:39 +11:00
static struct notifier_block pseries_smp_nb = {
. notifier_call = pseries_smp_notifier ,
2006-12-05 17:52:38 +11:00
} ;
2006-12-05 17:52:36 +11:00
static int __init pseries_cpu_hotplug_init ( void )
{
2010-04-28 13:39:41 +00:00
int qcss_tok ;
2007-10-10 10:38:24 +10:00
2015-12-16 14:50:21 -06:00
# ifdef CONFIG_ARCH_CPU_PROBE_RELEASE
ppc_md . cpu_probe = dlpar_cpu_probe ;
ppc_md . cpu_release = dlpar_cpu_release ;
# endif /* CONFIG_ARCH_CPU_PROBE_RELEASE */
2014-02-20 21:13:52 +11:00
rtas_stop_self_token = rtas_token ( " stop-self " ) ;
2006-12-05 17:52:38 +11:00
qcss_tok = rtas_token ( " query-cpu-stopped-state " ) ;
2006-12-05 17:52:36 +11:00
2014-02-20 21:13:52 +11:00
if ( rtas_stop_self_token = = RTAS_UNKNOWN_SERVICE | |
2006-12-05 17:52:38 +11:00
qcss_tok = = RTAS_UNKNOWN_SERVICE ) {
printk ( KERN_INFO " CPU Hotplug not supported by firmware "
" - disabling. \n " ) ;
return 0 ;
}
2006-12-05 17:52:37 +11:00
2020-08-19 11:56:34 +10:00
smp_ops - > cpu_offline_self = pseries_cpu_offline_self ;
2006-12-05 17:52:39 +11:00
smp_ops - > cpu_disable = pseries_cpu_disable ;
smp_ops - > cpu_die = pseries_cpu_die ;
2006-12-05 17:52:38 +11:00
/* Processors can be added/removed only on LPAR */
2020-06-12 00:12:21 -05:00
if ( firmware_has_feature ( FW_FEATURE_LPAR ) )
2012-10-02 16:57:57 +00:00
of_reconfig_notifier_register ( & pseries_smp_nb ) ;
2006-12-05 17:52:38 +11:00
2006-12-05 17:52:36 +11:00
return 0 ;
}
2013-12-10 11:31:02 +11:00
machine_arch_initcall ( pseries , pseries_cpu_hotplug_init ) ;