aboutsummaryrefslogtreecommitdiffstats
path: root/arch/s390/mm
diff options
context:
space:
mode:
Diffstat (limited to 'arch/s390/mm')
-rw-r--r--arch/s390/mm/gup.c11
-rw-r--r--arch/s390/mm/pgtable.c18
2 files changed, 28 insertions, 1 deletions
diff --git a/arch/s390/mm/gup.c b/arch/s390/mm/gup.c
index eeaf8023851f..60acb93a4680 100644
--- a/arch/s390/mm/gup.c
+++ b/arch/s390/mm/gup.c
@@ -115,7 +115,16 @@ static inline int gup_pmd_range(pud_t *pudp, pud_t pud, unsigned long addr,
115 pmd = *pmdp; 115 pmd = *pmdp;
116 barrier(); 116 barrier();
117 next = pmd_addr_end(addr, end); 117 next = pmd_addr_end(addr, end);
118 if (pmd_none(pmd)) 118 /*
119 * The pmd_trans_splitting() check below explains why
120 * pmdp_splitting_flush() has to serialize with
121 * smp_call_function() against our disabled IRQs, to stop
122 * this gup-fast code from running while we set the
123 * splitting bit in the pmd. Returning zero will take
124 * the slow path that will call wait_split_huge_page()
125 * if the pmd is still in splitting state.
126 */
127 if (pmd_none(pmd) || pmd_trans_splitting(pmd))
119 return 0; 128 return 0;
120 if (unlikely(pmd_huge(pmd))) { 129 if (unlikely(pmd_huge(pmd))) {
121 if (!gup_huge_pmd(pmdp, pmd, addr, next, 130 if (!gup_huge_pmd(pmdp, pmd, addr, next,
diff --git a/arch/s390/mm/pgtable.c b/arch/s390/mm/pgtable.c
index b402991e43d7..a6131d1fe6c0 100644
--- a/arch/s390/mm/pgtable.c
+++ b/arch/s390/mm/pgtable.c
@@ -866,3 +866,21 @@ bool kernel_page_present(struct page *page)
866 return cc == 0; 866 return cc == 0;
867} 867}
868#endif /* CONFIG_HIBERNATION && CONFIG_DEBUG_PAGEALLOC */ 868#endif /* CONFIG_HIBERNATION && CONFIG_DEBUG_PAGEALLOC */
869
870#ifdef CONFIG_TRANSPARENT_HUGEPAGE
871static void pmdp_splitting_flush_sync(void *arg)
872{
873 /* Simply deliver the interrupt */
874}
875
876void pmdp_splitting_flush(struct vm_area_struct *vma, unsigned long address,
877 pmd_t *pmdp)
878{
879 VM_BUG_ON(address & ~HPAGE_PMD_MASK);
880 if (!test_and_set_bit(_SEGMENT_ENTRY_SPLIT_BIT,
881 (unsigned long *) pmdp)) {
882 /* need to serialize against gup-fast (IRQ disabled) */
883 smp_call_function(pmdp_splitting_flush_sync, NULL, 1);
884 }
885}
886#endif /* CONFIG_TRANSPARENT_HUGEPAGE */