Skip to content

Commit adb11e7

Browse files
Sebastian Andrzej Siewiortorvalds
authored andcommitted
mm/munlock: protect the per-CPU pagevec by a local_lock_t
The access to mlock_pvec is protected by disabling preemption via get_cpu_var() or implicit by having preemption disabled by the caller (in mlock_page_drain() case). This breaks on PREEMPT_RT since folio_lruvec_lock_irq() acquires a sleeping lock in this section. Create struct mlock_pvec which consits of the local_lock_t and the pagevec. Acquire the local_lock() before accessing the per-CPU pagevec. Replace mlock_page_drain() with a _local() version which is invoked on the local CPU and acquires the local_lock_t and a _remote() version which uses the pagevec from a remote CPU which offline. Link: https://lkml.kernel.org/r/YjizWi9IY0mpvIfb@linutronix.de Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de> Acked-by: Hugh Dickins <hughd@google.com> Cc: Vlastimil Babka <vbabka@suse.cz> Cc: Matthew Wilcox <willy@infradead.org> Cc: Thomas Gleixner <tglx@linutronix.de> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
1 parent 577e984 commit adb11e7

6 files changed

Lines changed: 47 additions & 16 deletions

File tree

mm/internal.h

Lines changed: 4 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -456,7 +456,8 @@ static inline void munlock_vma_page(struct page *page,
456456
}
457457
void mlock_new_page(struct page *page);
458458
bool need_mlock_page_drain(int cpu);
459-
void mlock_page_drain(int cpu);
459+
void mlock_page_drain_local(void);
460+
void mlock_page_drain_remote(int cpu);
460461

461462
extern pmd_t maybe_pmd_mkwrite(pmd_t pmd, struct vm_area_struct *vma);
462463

@@ -539,7 +540,8 @@ static inline void munlock_vma_page(struct page *page,
539540
struct vm_area_struct *vma, bool compound) { }
540541
static inline void mlock_new_page(struct page *page) { }
541542
static inline bool need_mlock_page_drain(int cpu) { return false; }
542-
static inline void mlock_page_drain(int cpu) { }
543+
static inline void mlock_page_drain_local(void) { }
544+
static inline void mlock_page_drain_remote(int cpu) { }
543545
static inline void vunmap_range_noflush(unsigned long start, unsigned long end)
544546
{
545547
}

mm/migrate.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -246,7 +246,7 @@ static bool remove_migration_pte(struct folio *folio,
246246
set_pte_at(vma->vm_mm, pvmw.address, pvmw.pte, pte);
247247
}
248248
if (vma->vm_flags & VM_LOCKED)
249-
mlock_page_drain(smp_processor_id());
249+
mlock_page_drain_local();
250250

251251
trace_remove_migration_pte(pvmw.address, pte_val(pte),
252252
compound_order(new));

mm/mlock.c

Lines changed: 36 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,14 @@
2828

2929
#include "internal.h"
3030

31-
static DEFINE_PER_CPU(struct pagevec, mlock_pvec);
31+
struct mlock_pvec {
32+
local_lock_t lock;
33+
struct pagevec vec;
34+
};
35+
36+
static DEFINE_PER_CPU(struct mlock_pvec, mlock_pvec) = {
37+
.lock = INIT_LOCAL_LOCK(lock),
38+
};
3239

3340
bool can_do_mlock(void)
3441
{
@@ -203,18 +210,30 @@ static void mlock_pagevec(struct pagevec *pvec)
203210
pagevec_reinit(pvec);
204211
}
205212

206-
void mlock_page_drain(int cpu)
213+
void mlock_page_drain_local(void)
214+
{
215+
struct pagevec *pvec;
216+
217+
local_lock(&mlock_pvec.lock);
218+
pvec = this_cpu_ptr(&mlock_pvec.vec);
219+
if (pagevec_count(pvec))
220+
mlock_pagevec(pvec);
221+
local_unlock(&mlock_pvec.lock);
222+
}
223+
224+
void mlock_page_drain_remote(int cpu)
207225
{
208226
struct pagevec *pvec;
209227

210-
pvec = &per_cpu(mlock_pvec, cpu);
228+
WARN_ON_ONCE(cpu_online(cpu));
229+
pvec = &per_cpu(mlock_pvec.vec, cpu);
211230
if (pagevec_count(pvec))
212231
mlock_pagevec(pvec);
213232
}
214233

215234
bool need_mlock_page_drain(int cpu)
216235
{
217-
return pagevec_count(&per_cpu(mlock_pvec, cpu));
236+
return pagevec_count(&per_cpu(mlock_pvec.vec, cpu));
218237
}
219238

220239
/**
@@ -223,7 +242,10 @@ bool need_mlock_page_drain(int cpu)
223242
*/
224243
void mlock_folio(struct folio *folio)
225244
{
226-
struct pagevec *pvec = &get_cpu_var(mlock_pvec);
245+
struct pagevec *pvec;
246+
247+
local_lock(&mlock_pvec.lock);
248+
pvec = this_cpu_ptr(&mlock_pvec.vec);
227249

228250
if (!folio_test_set_mlocked(folio)) {
229251
int nr_pages = folio_nr_pages(folio);
@@ -236,7 +258,7 @@ void mlock_folio(struct folio *folio)
236258
if (!pagevec_add(pvec, mlock_lru(&folio->page)) ||
237259
folio_test_large(folio) || lru_cache_disabled())
238260
mlock_pagevec(pvec);
239-
put_cpu_var(mlock_pvec);
261+
local_unlock(&mlock_pvec.lock);
240262
}
241263

242264
/**
@@ -245,9 +267,11 @@ void mlock_folio(struct folio *folio)
245267
*/
246268
void mlock_new_page(struct page *page)
247269
{
248-
struct pagevec *pvec = &get_cpu_var(mlock_pvec);
270+
struct pagevec *pvec;
249271
int nr_pages = thp_nr_pages(page);
250272

273+
local_lock(&mlock_pvec.lock);
274+
pvec = this_cpu_ptr(&mlock_pvec.vec);
251275
SetPageMlocked(page);
252276
mod_zone_page_state(page_zone(page), NR_MLOCK, nr_pages);
253277
__count_vm_events(UNEVICTABLE_PGMLOCKED, nr_pages);
@@ -256,7 +280,7 @@ void mlock_new_page(struct page *page)
256280
if (!pagevec_add(pvec, mlock_new(page)) ||
257281
PageHead(page) || lru_cache_disabled())
258282
mlock_pagevec(pvec);
259-
put_cpu_var(mlock_pvec);
283+
local_unlock(&mlock_pvec.lock);
260284
}
261285

262286
/**
@@ -265,8 +289,10 @@ void mlock_new_page(struct page *page)
265289
*/
266290
void munlock_page(struct page *page)
267291
{
268-
struct pagevec *pvec = &get_cpu_var(mlock_pvec);
292+
struct pagevec *pvec;
269293

294+
local_lock(&mlock_pvec.lock);
295+
pvec = this_cpu_ptr(&mlock_pvec.vec);
270296
/*
271297
* TestClearPageMlocked(page) must be left to __munlock_page(),
272298
* which will check whether the page is multiply mlocked.
@@ -276,7 +302,7 @@ void munlock_page(struct page *page)
276302
if (!pagevec_add(pvec, page) ||
277303
PageHead(page) || lru_cache_disabled())
278304
mlock_pagevec(pvec);
279-
put_cpu_var(mlock_pvec);
305+
local_unlock(&mlock_pvec.lock);
280306
}
281307

282308
static int mlock_pte_range(pmd_t *pmd, unsigned long addr,

mm/page_alloc.c

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8367,6 +8367,7 @@ static int page_alloc_cpu_dead(unsigned int cpu)
83678367
struct zone *zone;
83688368

83698369
lru_add_drain_cpu(cpu);
8370+
mlock_page_drain_remote(cpu);
83708371
drain_pages(cpu);
83718372

83728373
/*

mm/rmap.c

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1683,7 +1683,7 @@ static bool try_to_unmap_one(struct folio *folio, struct vm_area_struct *vma,
16831683
*/
16841684
page_remove_rmap(subpage, vma, folio_test_hugetlb(folio));
16851685
if (vma->vm_flags & VM_LOCKED)
1686-
mlock_page_drain(smp_processor_id());
1686+
mlock_page_drain_local();
16871687
folio_put(folio);
16881688
}
16891689

@@ -1961,7 +1961,7 @@ static bool try_to_migrate_one(struct folio *folio, struct vm_area_struct *vma,
19611961
*/
19621962
page_remove_rmap(subpage, vma, folio_test_hugetlb(folio));
19631963
if (vma->vm_flags & VM_LOCKED)
1964-
mlock_page_drain(smp_processor_id());
1964+
mlock_page_drain_local();
19651965
folio_put(folio);
19661966
}
19671967

mm/swap.c

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -624,7 +624,6 @@ void lru_add_drain_cpu(int cpu)
624624
pagevec_lru_move_fn(pvec, lru_lazyfree_fn);
625625

626626
activate_page_drain(cpu);
627-
mlock_page_drain(cpu);
628627
}
629628

630629
/**
@@ -706,6 +705,7 @@ void lru_add_drain(void)
706705
local_lock(&lru_pvecs.lock);
707706
lru_add_drain_cpu(smp_processor_id());
708707
local_unlock(&lru_pvecs.lock);
708+
mlock_page_drain_local();
709709
}
710710

711711
/*
@@ -720,6 +720,7 @@ static void lru_add_and_bh_lrus_drain(void)
720720
lru_add_drain_cpu(smp_processor_id());
721721
local_unlock(&lru_pvecs.lock);
722722
invalidate_bh_lrus_cpu();
723+
mlock_page_drain_local();
723724
}
724725

725726
void lru_add_drain_cpu_zone(struct zone *zone)
@@ -728,6 +729,7 @@ void lru_add_drain_cpu_zone(struct zone *zone)
728729
lru_add_drain_cpu(smp_processor_id());
729730
drain_local_pages(zone);
730731
local_unlock(&lru_pvecs.lock);
732+
mlock_page_drain_local();
731733
}
732734

733735
#ifdef CONFIG_SMP

0 commit comments

Comments
 (0)