The PPC mobility code may receive DLPAR CPU add/remove requests
to perform CPU changes at any time, including during LPAR migration
or RTAS requests or SMT changes.  When the operations are received
concurrently, there is an opportunity for DLPAR CPU remove requests
and other requests to overlap, and for one of the requests to be
interrupted after some shared state has been modified.  This patch
changes the duration for which cpu maps updates are suppressed
during DLPAR CPU operations made by 'drmgr' are suppressed to
reduce the period in which changes to shared state may occur.

Signed-off-by: Michael Bringmann <m...@linux.vnet.ibm.com>
---
 arch/powerpc/platforms/pseries/hotplug-cpu.c |   10 ++++++----
 1 file changed, 6 insertions(+), 4 deletions(-)

diff --git a/arch/powerpc/platforms/pseries/hotplug-cpu.c 
b/arch/powerpc/platforms/pseries/hotplug-cpu.c
index 2f8e621..fce46c56 100644
--- a/arch/powerpc/platforms/pseries/hotplug-cpu.c
+++ b/arch/powerpc/platforms/pseries/hotplug-cpu.c
@@ -356,7 +356,6 @@ static int dlpar_online_cpu(struct device_node *dn)
 
        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) {
@@ -378,7 +377,6 @@ static int dlpar_online_cpu(struct device_node *dn)
                        printk(KERN_WARNING "Could not find cpu to online "
                               "with physical id 0x%x\n", thread);
        }
-       cpu_maps_update_done();
 
 out:
        return rc;
@@ -523,7 +521,6 @@ static int dlpar_offline_cpu(struct device_node *dn)
 
        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) {
@@ -559,7 +556,6 @@ static int dlpar_offline_cpu(struct device_node *dn)
                if (cpu == num_possible_cpus())
                        printk(KERN_WARNING "Could not find cpu to offline with 
physical id 0x%x\n", thread);
        }
-       cpu_maps_update_done();
 
 out:
        return rc;
@@ -811,6 +807,7 @@ int dlpar_cpu(struct pseries_hp_errorlog *hp_elog)
        drc_index = hp_elog->_drc_u.drc_index;
 
        lock_device_hotplug();
+       cpu_maps_update_begin();
 
        switch (hp_elog->action) {
        case PSERIES_HP_ELOG_ACTION_REMOVE:
@@ -835,6 +832,7 @@ int dlpar_cpu(struct pseries_hp_errorlog *hp_elog)
                break;
        }
 
+       cpu_maps_update_done();
        unlock_device_hotplug();
        return rc;
 }
@@ -850,7 +848,9 @@ static ssize_t dlpar_cpu_probe(const char *buf, size_t 
count)
        if (rc)
                return -EINVAL;
 
+       cpu_maps_update_begin();
        rc = dlpar_cpu_add(drc_index);
+       cpu_maps_update_done();
 
        return rc ? rc : count;
 }
@@ -871,7 +871,9 @@ static ssize_t dlpar_cpu_release(const char *buf, size_t 
count)
                return -EINVAL;
        }
 
+       cpu_maps_update_begin();
        rc = dlpar_cpu_remove(dn, drc_index);
+       cpu_maps_update_done();
        of_node_put(dn);
 
        return rc ? rc : count;

Reply via email to