diff options
author | Nathan Fontenot <nfont@linux.vnet.ibm.com> | 2018-10-29 14:43:36 -0400 |
---|---|---|
committer | Michael Ellerman <mpe@ellerman.id.au> | 2019-01-30 07:28:56 -0500 |
commit | 81b61324922c67f73813d8a9c175f3c153f6a1c6 (patch) | |
tree | 2cb9bcb1bbe6d9d2724e2bd8f5e5aab0cbb5210a | |
parent | 63da6caeb84cfad3d1e5774b7049dd1d2c9dec62 (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.h | 2 | ||||
-rw-r--r-- | arch/powerpc/mm/numa.c | 9 | ||||
-rw-r--r-- | arch/powerpc/platforms/pseries/hotplug-cpu.c | 19 |
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 | |||
136 | int 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 | ||
1463 | static 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 | |||
1470 | static int dt_update_callback(struct notifier_block *nb, | 1463 | static 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 | ||
805 | int 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 | |||
805 | int dlpar_cpu(struct pseries_hp_errorlog *hp_elog) | 824 | int dlpar_cpu(struct pseries_hp_errorlog *hp_elog) |
806 | { | 825 | { |
807 | u32 count, drc_index; | 826 | u32 count, drc_index; |