aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/mm.h32
-rw-r--r--include/linux/mm_types.h7
-rw-r--r--kernel/fork.c4
-rw-r--r--mm/Kconfig3
4 files changed, 43 insertions, 3 deletions
diff --git a/include/linux/mm.h b/include/linux/mm.h
index 861cad53b744..255750d9b1be 100644
--- a/include/linux/mm.h
+++ b/include/linux/mm.h
@@ -1378,13 +1378,45 @@ static inline void pgtable_page_dtor(struct page *page)
1378 ((unlikely(pmd_none(*(pmd))) && __pte_alloc_kernel(pmd, address))? \ 1378 ((unlikely(pmd_none(*(pmd))) && __pte_alloc_kernel(pmd, address))? \
1379 NULL: pte_offset_kernel(pmd, address)) 1379 NULL: pte_offset_kernel(pmd, address))
1380 1380
1381#if USE_SPLIT_PMD_PTLOCKS
1382
1383static inline spinlock_t *pmd_lockptr(struct mm_struct *mm, pmd_t *pmd)
1384{
1385 return &virt_to_page(pmd)->ptl;
1386}
1387
1388static inline bool pgtable_pmd_page_ctor(struct page *page)
1389{
1390 spin_lock_init(&page->ptl);
1391#ifdef CONFIG_TRANSPARENT_HUGEPAGE
1392 page->pmd_huge_pte = NULL;
1393#endif
1394 return true;
1395}
1396
1397static inline void pgtable_pmd_page_dtor(struct page *page)
1398{
1399#ifdef CONFIG_TRANSPARENT_HUGEPAGE
1400 VM_BUG_ON(page->pmd_huge_pte);
1401#endif
1402}
1403
1404#define pmd_huge_pte(mm, pmd) (virt_to_page(pmd)->pmd_huge_pte)
1405
1406#else
1407
1381static inline spinlock_t *pmd_lockptr(struct mm_struct *mm, pmd_t *pmd) 1408static inline spinlock_t *pmd_lockptr(struct mm_struct *mm, pmd_t *pmd)
1382{ 1409{
1383 return &mm->page_table_lock; 1410 return &mm->page_table_lock;
1384} 1411}
1385 1412
1413static inline bool pgtable_pmd_page_ctor(struct page *page) { return true; }
1414static inline void pgtable_pmd_page_dtor(struct page *page) {}
1415
1386#define pmd_huge_pte(mm, pmd) ((mm)->pmd_huge_pte) 1416#define pmd_huge_pte(mm, pmd) ((mm)->pmd_huge_pte)
1387 1417
1418#endif
1419
1388static inline spinlock_t *pmd_lock(struct mm_struct *mm, pmd_t *pmd) 1420static inline spinlock_t *pmd_lock(struct mm_struct *mm, pmd_t *pmd)
1389{ 1421{
1390 spinlock_t *ptl = pmd_lockptr(mm, pmd); 1422 spinlock_t *ptl = pmd_lockptr(mm, pmd);
diff --git a/include/linux/mm_types.h b/include/linux/mm_types.h
index 566df579c51f..934261099975 100644
--- a/include/linux/mm_types.h
+++ b/include/linux/mm_types.h
@@ -24,6 +24,8 @@
24struct address_space; 24struct address_space;
25 25
26#define USE_SPLIT_PTE_PTLOCKS (NR_CPUS >= CONFIG_SPLIT_PTLOCK_CPUS) 26#define USE_SPLIT_PTE_PTLOCKS (NR_CPUS >= CONFIG_SPLIT_PTLOCK_CPUS)
27#define USE_SPLIT_PMD_PTLOCKS (USE_SPLIT_PTE_PTLOCKS && \
28 IS_ENABLED(CONFIG_ARCH_ENABLE_SPLIT_PMD_PTLOCK))
27 29
28/* 30/*
29 * Each physical page in the system has a struct page associated with 31 * Each physical page in the system has a struct page associated with
@@ -63,6 +65,9 @@ struct page {
63 * this page is only used to 65 * this page is only used to
64 * free other pages. 66 * free other pages.
65 */ 67 */
68#if defined(CONFIG_TRANSPARENT_HUGEPAGE) && USE_SPLIT_PMD_PTLOCKS
69 pgtable_t pmd_huge_pte; /* protected by page->ptl */
70#endif
66 }; 71 };
67 72
68 union { 73 union {
@@ -406,7 +411,7 @@ struct mm_struct {
406#ifdef CONFIG_MMU_NOTIFIER 411#ifdef CONFIG_MMU_NOTIFIER
407 struct mmu_notifier_mm *mmu_notifier_mm; 412 struct mmu_notifier_mm *mmu_notifier_mm;
408#endif 413#endif
409#ifdef CONFIG_TRANSPARENT_HUGEPAGE 414#if defined(CONFIG_TRANSPARENT_HUGEPAGE) && !USE_SPLIT_PMD_PTLOCKS
410 pgtable_t pmd_huge_pte; /* protected by page_table_lock */ 415 pgtable_t pmd_huge_pte; /* protected by page_table_lock */
411#endif 416#endif
412#ifdef CONFIG_CPUMASK_OFFSTACK 417#ifdef CONFIG_CPUMASK_OFFSTACK
diff --git a/kernel/fork.c b/kernel/fork.c
index e2520756e005..728d5be9548c 100644
--- a/kernel/fork.c
+++ b/kernel/fork.c
@@ -560,7 +560,7 @@ static void check_mm(struct mm_struct *mm)
560 "mm:%p idx:%d val:%ld\n", mm, i, x); 560 "mm:%p idx:%d val:%ld\n", mm, i, x);
561 } 561 }
562 562
563#ifdef CONFIG_TRANSPARENT_HUGEPAGE 563#if defined(CONFIG_TRANSPARENT_HUGEPAGE) && !USE_SPLIT_PMD_PTLOCKS
564 VM_BUG_ON(mm->pmd_huge_pte); 564 VM_BUG_ON(mm->pmd_huge_pte);
565#endif 565#endif
566} 566}
@@ -814,7 +814,7 @@ struct mm_struct *dup_mm(struct task_struct *tsk)
814 memcpy(mm, oldmm, sizeof(*mm)); 814 memcpy(mm, oldmm, sizeof(*mm));
815 mm_init_cpumask(mm); 815 mm_init_cpumask(mm);
816 816
817#ifdef CONFIG_TRANSPARENT_HUGEPAGE 817#if defined(CONFIG_TRANSPARENT_HUGEPAGE) && !USE_SPLIT_PMD_PTLOCKS
818 mm->pmd_huge_pte = NULL; 818 mm->pmd_huge_pte = NULL;
819#endif 819#endif
820 if (!mm_init(mm, tsk)) 820 if (!mm_init(mm, tsk))
diff --git a/mm/Kconfig b/mm/Kconfig
index c28d247e524a..7aa02def4666 100644
--- a/mm/Kconfig
+++ b/mm/Kconfig
@@ -222,6 +222,9 @@ config SPLIT_PTLOCK_CPUS
222 default "999999" if !64BIT && GENERIC_LOCKBREAK 222 default "999999" if !64BIT && GENERIC_LOCKBREAK
223 default "4" 223 default "4"
224 224
225config ARCH_ENABLE_SPLIT_PMD_PTLOCK
226 boolean
227
225# 228#
226# support for memory balloon compaction 229# support for memory balloon compaction
227config BALLOON_COMPACTION 230config BALLOON_COMPACTION