summaryrefslogtreecommitdiff
path: root/patches/0024-mm-slub-separate-detaching-of-partial-list-in-unfree.patch
diff options
context:
space:
mode:
Diffstat (limited to 'patches/0024-mm-slub-separate-detaching-of-partial-list-in-unfree.patch')
-rw-r--r--patches/0024-mm-slub-separate-detaching-of-partial-list-in-unfree.patch156
1 files changed, 156 insertions, 0 deletions
diff --git a/patches/0024-mm-slub-separate-detaching-of-partial-list-in-unfree.patch b/patches/0024-mm-slub-separate-detaching-of-partial-list-in-unfree.patch
new file mode 100644
index 000000000000..7a01091b8456
--- /dev/null
+++ b/patches/0024-mm-slub-separate-detaching-of-partial-list-in-unfree.patch
@@ -0,0 +1,156 @@
+From: Vlastimil Babka <vbabka@suse.cz>
+Date: Thu, 20 May 2021 16:39:51 +0200
+Subject: [PATCH 24/33] mm, slub: separate detaching of partial list in
+ unfreeze_partials() from unfreezing
+
+Unfreezing partial list can be split to two phases - detaching the list from
+struct kmem_cache_cpu, and processing the list. The whole operation does not
+need to be protected by disabled irqs. Restructure the code to separate the
+detaching (with disabled irqs) and unfreezing (with irq disabling to be reduced
+in the next patch).
+
+Also, unfreeze_partials() can be called from another cpu on behalf of a cpu
+that is being offlined, where disabling irqs on the local cpu has no sense, so
+restructure the code as follows:
+
+- __unfreeze_partials() is the bulk of unfreeze_partials() that processes the
+ detached percpu partial list
+- unfreeze_partials() detaches list from current cpu with irqs disabled and
+ calls __unfreeze_partials()
+- unfreeze_partials_cpu() is to be called for the offlined cpu so it needs no
+ irq disabling, and is called from __flush_cpu_slab()
+- flush_cpu_slab() is for the local cpu thus it needs to call
+ unfreeze_partials(). So it can't simply call
+ __flush_cpu_slab(smp_processor_id()) anymore and we have to open-code the
+ proper calls.
+
+Signed-off-by: Vlastimil Babka <vbabka@suse.cz>
+---
+ mm/slub.c | 73 +++++++++++++++++++++++++++++++++++++++++++-------------------
+ 1 file changed, 51 insertions(+), 22 deletions(-)
+
+--- a/mm/slub.c
++++ b/mm/slub.c
+@@ -2347,25 +2347,15 @@ static void deactivate_slab(struct kmem_
+ }
+ }
+
+-/*
+- * Unfreeze all the cpu partial slabs.
+- *
+- * This function must be called with preemption or migration
+- * disabled with c local to the cpu.
+- */
+-static void unfreeze_partials(struct kmem_cache *s,
+- struct kmem_cache_cpu *c)
+-{
+ #ifdef CONFIG_SLUB_CPU_PARTIAL
++static void __unfreeze_partials(struct kmem_cache *s, struct page *partial_page)
++{
+ struct kmem_cache_node *n = NULL, *n2 = NULL;
+- struct page *page, *partial_page, *discard_page = NULL;
++ struct page *page, *discard_page = NULL;
+ unsigned long flags;
+
+ local_irq_save(flags);
+
+- partial_page = slub_percpu_partial(c);
+- c->partial = NULL;
+-
+ while (partial_page) {
+ struct page new;
+ struct page old;
+@@ -2420,10 +2410,45 @@ static void unfreeze_partials(struct kme
+ discard_slab(s, page);
+ stat(s, FREE_SLAB);
+ }
++}
+
+-#endif /* CONFIG_SLUB_CPU_PARTIAL */
++/*
++ * Unfreeze all the cpu partial slabs.
++ */
++static void unfreeze_partials(struct kmem_cache *s)
++{
++ struct page *partial_page;
++ unsigned long flags;
++
++ local_irq_save(flags);
++ partial_page = this_cpu_read(s->cpu_slab->partial);
++ this_cpu_write(s->cpu_slab->partial, NULL);
++ local_irq_restore(flags);
++
++ if (partial_page)
++ __unfreeze_partials(s, partial_page);
++}
++
++static void unfreeze_partials_cpu(struct kmem_cache *s,
++ struct kmem_cache_cpu *c)
++{
++ struct page *partial_page;
++
++ partial_page = slub_percpu_partial(c);
++ c->partial = NULL;
++
++ if (partial_page)
++ __unfreeze_partials(s, partial_page);
+ }
+
++#else /* CONFIG_SLUB_CPU_PARTIAL */
++
++static inline void unfreeze_partials(struct kmem_cache *s) { }
++static inline void unfreeze_partials_cpu(struct kmem_cache *s,
++ struct kmem_cache_cpu *c) { }
++
++#endif /* CONFIG_SLUB_CPU_PARTIAL */
++
+ /*
+ * Put a page that was just frozen (in __slab_free|get_partial_node) into a
+ * partial page slot if available.
+@@ -2452,7 +2477,7 @@ static void put_cpu_partial(struct kmem_
+ * partial array is full. Move the existing
+ * set to the per node partial list.
+ */
+- unfreeze_partials(s, this_cpu_ptr(s->cpu_slab));
++ unfreeze_partials(s);
+ oldpage = NULL;
+ pobjects = 0;
+ pages = 0;
+@@ -2487,11 +2512,6 @@ static inline void flush_slab(struct kme
+ stat(s, CPUSLAB_FLUSH);
+ }
+
+-/*
+- * Flush cpu slab.
+- *
+- * Called from IPI handler with interrupts disabled.
+- */
+ static inline void __flush_cpu_slab(struct kmem_cache *s, int cpu)
+ {
+ struct kmem_cache_cpu *c = per_cpu_ptr(s->cpu_slab, cpu);
+@@ -2499,14 +2519,23 @@ static inline void __flush_cpu_slab(stru
+ if (c->page)
+ flush_slab(s, c);
+
+- unfreeze_partials(s, c);
++ unfreeze_partials_cpu(s, c);
+ }
+
++/*
++ * Flush cpu slab.
++ *
++ * Called from IPI handler with interrupts disabled.
++ */
+ static void flush_cpu_slab(void *d)
+ {
+ struct kmem_cache *s = d;
++ struct kmem_cache_cpu *c = this_cpu_ptr(s->cpu_slab);
++
++ if (c->page)
++ flush_slab(s, c);
+
+- __flush_cpu_slab(s, smp_processor_id());
++ unfreeze_partials(s);
+ }
+
+ static bool has_cpu_slab(int cpu, void *info)