[PATCH] powerpc/pseries: Verify CPU doesn't exist before adding
Nathan Fontenot
nfont at linux.vnet.ibm.com
Tue Oct 27 06:53:09 AEDT 2015
On 10/25/2015 11:30 AM, Denis Kirjanov wrote:
> On 10/23/15, Nathan Fontenot <nfont at linux.vnet.ibm.com> wrote:
>> When DLPAR adding a CPU we should verify that the CPU does not already
>> exist. Failure to do so can generate a kernel oops;
>>
>> [ 9.465585] kernel BUG at arch/powerpc/platforms/pseries/dlpar.c:382!
>> [ 9.465796] Oops: Exception in kernel mode, sig: 5 [#1]
>>
>> This oops can be generated by causing a probe to be performed on a cpu
>> by writing to the sysfs cpu probe file (/sys/devices/system/cpu/probe).
>> This patch adds a check for the existence of cpu prior to probing the cpu
>> so userspace doing the wrong thing won't trigger a BUG_ON().
>
> Hi Nathan,
>
> Can you please tell how to trigger the oops manually since I've tried to write
> a core number to the probe file, but with no luck. Always get -EINVAL.
Triggering the oops manually may be a bit trickier than just writing to the sysfs file.
First, make sure you are writing the drc index of a cpu that is already present.
You can see a list of present cpus with 'lsslot -c cpu'.
You may be able to trigger the oops by just writing the drc index to the sysfs file
but will likely get -EINVAL since the probe code tries to acquire the cpu from
firmware and call configure-connector, both of which will likely fail before we
try to online the cpu and see the BUG_ON.
For a kvm guest, you should be able to generate the oops by killing the rtas_errd
daemon, do a cpu hotplug add from the host, then reboot the guest. If the
rtas_errd daemon is set to autostart you should see the oops at boot, otherwise
you can manually start rtas_errd and see the oops.
For a Power LPAR, the process would be much trickier. You would have to do some
hacking to the drmgr command (which write the drc index to the probe file) to
trick it into trying to add the same cpu twice when invoking cpu hotplug from
the HMC.
Hope that helps.
-Nathan
>
> Thanks!
>>
>> Signed-off-by: Nathan Fontenot <nfont at linux.vnet.ibm.com>
>> ---
>> arch/powerpc/platforms/pseries/dlpar.c | 43
>> +++++++++++++++++++++++++++++---
>> 1 file changed, 39 insertions(+), 4 deletions(-)
>>
>> diff --git a/arch/powerpc/platforms/pseries/dlpar.c
>> b/arch/powerpc/platforms/pseries/dlpar.c
>> index f244dcb..fe6320d 100644
>> --- a/arch/powerpc/platforms/pseries/dlpar.c
>> +++ b/arch/powerpc/platforms/pseries/dlpar.c
>> @@ -381,6 +381,32 @@ out:
>>
>> }
>>
>> +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;
>> +}
>> +
>> static ssize_t dlpar_cpu_probe(const char *buf, size_t count)
>> {
>> struct device_node *dn, *parent;
>> @@ -391,14 +417,23 @@ static ssize_t dlpar_cpu_probe(const char *buf, size_t
>> count)
>> if (rc)
>> return -EINVAL;
>>
>> - rc = dlpar_acquire_drc(drc_index);
>> - if (rc)
>> - return -EINVAL;
>> -
>> parent = of_find_node_by_path("/cpus");
>> if (!parent)
>> return -ENODEV;
>>
>> + if (dlpar_cpu_exists(parent, drc_index)) {
>> + of_node_put(parent);
>> + printk(KERN_WARNING "CPU with drc index %x already exists\n",
>> + drc_index);
>> + return -EINVAL;
>> + }
>> +
>> + rc = dlpar_acquire_drc(drc_index);
>> + if (rc) {
>> + of_node_put(parent);
>> + return -EINVAL;
>> + }
>> +
>> dn = dlpar_configure_connector(cpu_to_be32(drc_index), parent);
>> of_node_put(parent);
>> if (!dn) {
>>
>> _______________________________________________
>> Linuxppc-dev mailing list
>> Linuxppc-dev at lists.ozlabs.org
>> https://lists.ozlabs.org/listinfo/linuxppc-dev
>
More information about the Linuxppc-dev
mailing list