summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorNathan Fontenot <nfont@linux.vnet.ibm.com>2018-10-29 14:43:36 -0400
committerMichael Ellerman <mpe@ellerman.id.au>2019-01-30 07:28:56 -0500
commit81b61324922c67f73813d8a9c175f3c153f6a1c6 (patch)
tree2cb9bcb1bbe6d9d2724e2bd8f5e5aab0cbb5210a
parent63da6caeb84cfad3d1e5774b7049dd1d2c9dec62 (diff)
powerpc/pseries: Perform full re-add of CPU for topology update post-migration
On pseries systems, performing a partition migration can result in altering the nodes a CPU is assigned to on the destination system. For exampl, pre-migration on the source system CPUs are in node 1 and 3, post-migration on the destination system CPUs are in nodes 2 and 3. Handling the node change for a CPU can cause corruption in the slab cache if we hit a timing where a CPUs node is changed while cache_reap() is invoked. The corruption occurs because the slab cache code appears to rely on the CPU and slab cache pages being on the same node. The current dynamic updating of a CPUs node done in arch/powerpc/mm/numa.c does not prevent us from hitting this scenario. Changing the device tree property update notification handler that recognizes an affinity change for a CPU to do a full DLPAR remove and add of the CPU instead of dynamically changing its node resolves this issue. Signed-off-by: Nathan Fontenot <nfont@linux.vnet.ibm.com> Signed-off-by: Michael W. Bringmann <mwb@linux.vnet.ibm.com> Tested-by: Michael W. Bringmann <mwb@linux.vnet.ibm.com> Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
-rw-r--r--arch/powerpc/include/asm/topology.h2
-rw-r--r--arch/powerpc/mm/numa.c9
-rw-r--r--arch/powerpc/platforms/pseries/hotplug-cpu.c19
3 files changed, 22 insertions, 8 deletions
diff --git a/arch/powerpc/include/asm/topology.h b/arch/powerpc/include/asm/topology.h
index a4a718dbfec6..f85e2b01c3df 100644
--- a/arch/powerpc/include/asm/topology.h
+++ b/arch/powerpc/include/asm/topology.h
@@ -132,6 +132,8 @@ static inline void shared_proc_topology_init(void) {}
132#define topology_sibling_cpumask(cpu) (per_cpu(cpu_sibling_map, cpu)) 132#define topology_sibling_cpumask(cpu) (per_cpu(cpu_sibling_map, cpu))
133#define topology_core_cpumask(cpu) (per_cpu(cpu_core_map, cpu)) 133#define topology_core_cpumask(cpu) (per_cpu(cpu_core_map, cpu))
134#define topology_core_id(cpu) (cpu_to_core_id(cpu)) 134#define topology_core_id(cpu) (cpu_to_core_id(cpu))
135
136int dlpar_cpu_readd(int cpu);
135#endif 137#endif
136#endif 138#endif
137 139
diff --git a/arch/powerpc/mm/numa.c b/arch/powerpc/mm/numa.c
index 87f0dd004295..b5d1c45c1475 100644
--- a/arch/powerpc/mm/numa.c
+++ b/arch/powerpc/mm/numa.c
@@ -1460,13 +1460,6 @@ static void reset_topology_timer(void)
1460 1460
1461#ifdef CONFIG_SMP 1461#ifdef CONFIG_SMP
1462 1462
1463static void stage_topology_update(int core_id)
1464{
1465 cpumask_or(&cpu_associativity_changes_mask,
1466 &cpu_associativity_changes_mask, cpu_sibling_mask(core_id));
1467 reset_topology_timer();
1468}
1469
1470static int dt_update_callback(struct notifier_block *nb, 1463static int dt_update_callback(struct notifier_block *nb,
1471 unsigned long action, void *data) 1464 unsigned long action, void *data)
1472{ 1465{
@@ -1479,7 +1472,7 @@ static int dt_update_callback(struct notifier_block *nb,
1479 !of_prop_cmp(update->prop->name, "ibm,associativity")) { 1472 !of_prop_cmp(update->prop->name, "ibm,associativity")) {
1480 u32 core_id; 1473 u32 core_id;
1481 of_property_read_u32(update->dn, "reg", &core_id); 1474 of_property_read_u32(update->dn, "reg", &core_id);
1482 stage_topology_update(core_id); 1475 rc = dlpar_cpu_readd(core_id);
1483 rc = NOTIFY_OK; 1476 rc = NOTIFY_OK;
1484 } 1477 }
1485 break; 1478 break;
diff --git a/arch/powerpc/platforms/pseries/hotplug-cpu.c b/arch/powerpc/platforms/pseries/hotplug-cpu.c
index 2f8e62163602..97feb6e79f1a 100644
--- a/arch/powerpc/platforms/pseries/hotplug-cpu.c
+++ b/arch/powerpc/platforms/pseries/hotplug-cpu.c
@@ -802,6 +802,25 @@ static int dlpar_cpu_add_by_count(u32 cpus_to_add)
802 return rc; 802 return rc;
803} 803}
804 804
805int dlpar_cpu_readd(int cpu)
806{
807 struct device_node *dn;
808 struct device *dev;
809 u32 drc_index;
810 int rc;
811
812 dev = get_cpu_device(cpu);
813 dn = dev->of_node;
814
815 rc = of_property_read_u32(dn, "ibm,my-drc-index", &drc_index);
816
817 rc = dlpar_cpu_remove_by_index(drc_index);
818 if (!rc)
819 rc = dlpar_cpu_add(drc_index);
820
821 return rc;
822}
823
805int dlpar_cpu(struct pseries_hp_errorlog *hp_elog) 824int dlpar_cpu(struct pseries_hp_errorlog *hp_elog)
806{ 825{
807 u32 count, drc_index; 826 u32 count, drc_index;