2011-01-14 02:46:40 +03:00
/*
* mm / pgtable - generic . c
*
* Generic pgtable methods declared in asm - generic / pgtable . h
*
* Copyright ( C ) 2010 Linus Torvalds
*/
2011-01-26 02:07:11 +03:00
# include <linux/pagemap.h>
2011-01-14 02:46:40 +03:00
# include <asm/tlb.h>
# include <asm-generic/pgtable.h>
2013-09-12 01:21:28 +04:00
/*
* If a p ? d_bad entry is found while walking page tables , report
* the error , before resetting entry to p ? d_none . Usually ( but
* very seldom ) called out from the p ? d_none_or_clear_bad macros .
*/
void pgd_clear_bad ( pgd_t * pgd )
{
pgd_ERROR ( * pgd ) ;
pgd_clear ( pgd ) ;
}
void pud_clear_bad ( pud_t * pud )
{
pud_ERROR ( * pud ) ;
pud_clear ( pud ) ;
}
void pmd_clear_bad ( pmd_t * pmd )
{
pmd_ERROR ( * pmd ) ;
pmd_clear ( pmd ) ;
}
2011-01-14 02:46:40 +03:00
# ifndef __HAVE_ARCH_PTEP_SET_ACCESS_FLAGS
/*
2012-11-06 13:56:01 +04:00
* Only sets the access flags ( dirty , accessed ) , as well as write
* permission . Furthermore , we know it always gets set to a " more
2011-01-14 02:46:40 +03:00
* permissive " setting, which allows most architectures to optimize
* this . We return whether the PTE actually changed , which in turn
* instructs the caller to do things like update__mmu_cache . This
* used to be done in the caller , but sparc needs minor faults to
* force that call on sun4c so we changed this macro slightly
*/
int ptep_set_access_flags ( struct vm_area_struct * vma ,
unsigned long address , pte_t * ptep ,
pte_t entry , int dirty )
{
int changed = ! pte_same ( * ptep , entry ) ;
if ( changed ) {
set_pte_at ( vma - > vm_mm , address , ptep , entry ) ;
2012-11-06 13:56:01 +04:00
flush_tlb_fix_spurious_fault ( vma , address ) ;
2011-01-14 02:46:40 +03:00
}
return changed ;
}
# endif
# ifndef __HAVE_ARCH_PMDP_SET_ACCESS_FLAGS
int pmdp_set_access_flags ( struct vm_area_struct * vma ,
unsigned long address , pmd_t * pmdp ,
pmd_t entry , int dirty )
{
# ifdef CONFIG_TRANSPARENT_HUGEPAGE
int changed = ! pmd_same ( * pmdp , entry ) ;
VM_BUG_ON ( address & ~ HPAGE_PMD_MASK ) ;
if ( changed ) {
set_pmd_at ( vma - > vm_mm , address , pmdp , entry ) ;
flush_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
}
return changed ;
# else /* CONFIG_TRANSPARENT_HUGEPAGE */
BUG ( ) ;
return 0 ;
# endif /* CONFIG_TRANSPARENT_HUGEPAGE */
}
# endif
# ifndef __HAVE_ARCH_PTEP_CLEAR_YOUNG_FLUSH
int ptep_clear_flush_young ( struct vm_area_struct * vma ,
unsigned long address , pte_t * ptep )
{
int young ;
young = ptep_test_and_clear_young ( vma , address , ptep ) ;
if ( young )
flush_tlb_page ( vma , address ) ;
return young ;
}
# endif
# ifndef __HAVE_ARCH_PMDP_CLEAR_YOUNG_FLUSH
int pmdp_clear_flush_young ( struct vm_area_struct * vma ,
unsigned long address , pmd_t * pmdp )
{
int young ;
2012-03-22 03:34:27 +04:00
# ifdef CONFIG_TRANSPARENT_HUGEPAGE
VM_BUG_ON ( address & ~ HPAGE_PMD_MASK ) ;
# else
2011-01-14 02:46:40 +03:00
BUG ( ) ;
# endif /* CONFIG_TRANSPARENT_HUGEPAGE */
young = pmdp_test_and_clear_young ( vma , address , pmdp ) ;
if ( young )
flush_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
return young ;
}
# endif
# ifndef __HAVE_ARCH_PTEP_CLEAR_FLUSH
pte_t ptep_clear_flush ( struct vm_area_struct * vma , unsigned long address ,
pte_t * ptep )
{
pte_t pte ;
pte = ptep_get_and_clear ( ( vma ) - > vm_mm , address , ptep ) ;
2012-10-09 17:31:59 +04:00
if ( pte_accessible ( pte ) )
flush_tlb_page ( vma , address ) ;
2011-01-14 02:46:40 +03:00
return pte ;
}
# endif
# ifndef __HAVE_ARCH_PMDP_CLEAR_FLUSH
2011-01-17 00:10:39 +03:00
# ifdef CONFIG_TRANSPARENT_HUGEPAGE
2011-01-14 02:46:40 +03:00
pmd_t pmdp_clear_flush ( struct vm_area_struct * vma , unsigned long address ,
pmd_t * pmdp )
{
pmd_t pmd ;
VM_BUG_ON ( address & ~ HPAGE_PMD_MASK ) ;
pmd = pmdp_get_and_clear ( vma - > vm_mm , address , pmdp ) ;
flush_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
return pmd ;
}
2011-01-17 00:10:39 +03:00
# endif /* CONFIG_TRANSPARENT_HUGEPAGE */
2011-01-14 02:46:40 +03:00
# endif
# ifndef __HAVE_ARCH_PMDP_SPLITTING_FLUSH
2011-01-17 00:10:39 +03:00
# ifdef CONFIG_TRANSPARENT_HUGEPAGE
2012-03-28 21:59:18 +04:00
void pmdp_splitting_flush ( struct vm_area_struct * vma , unsigned long address ,
pmd_t * pmdp )
2011-01-14 02:46:40 +03:00
{
pmd_t pmd = pmd_mksplitting ( * pmdp ) ;
VM_BUG_ON ( address & ~ HPAGE_PMD_MASK ) ;
set_pmd_at ( vma - > vm_mm , address , pmdp , pmd ) ;
/* tlb flush only to serialize against gup-fast */
flush_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
}
2011-01-17 00:10:39 +03:00
# endif /* CONFIG_TRANSPARENT_HUGEPAGE */
2011-01-14 02:46:40 +03:00
# endif
2012-10-09 03:30:07 +04:00
# ifndef __HAVE_ARCH_PGTABLE_DEPOSIT
# ifdef CONFIG_TRANSPARENT_HUGEPAGE
2013-06-06 04:14:02 +04:00
void pgtable_trans_huge_deposit ( struct mm_struct * mm , pmd_t * pmdp ,
pgtable_t pgtable )
2012-10-09 03:30:07 +04:00
{
2013-11-15 02:31:04 +04:00
assert_spin_locked ( pmd_lockptr ( mm , pmdp ) ) ;
2012-10-09 03:30:07 +04:00
/* FIFO */
2013-11-15 02:30:59 +04:00
if ( ! pmd_huge_pte ( mm , pmdp ) )
2012-10-09 03:30:07 +04:00
INIT_LIST_HEAD ( & pgtable - > lru ) ;
else
2013-11-15 02:30:59 +04:00
list_add ( & pgtable - > lru , & pmd_huge_pte ( mm , pmdp ) - > lru ) ;
pmd_huge_pte ( mm , pmdp ) = pgtable ;
2012-10-09 03:30:07 +04:00
}
# endif /* CONFIG_TRANSPARENT_HUGEPAGE */
# endif
# ifndef __HAVE_ARCH_PGTABLE_WITHDRAW
# ifdef CONFIG_TRANSPARENT_HUGEPAGE
/* no "address" argument so destroys page coloring of some arch */
2013-06-06 04:14:02 +04:00
pgtable_t pgtable_trans_huge_withdraw ( struct mm_struct * mm , pmd_t * pmdp )
2012-10-09 03:30:07 +04:00
{
pgtable_t pgtable ;
2013-11-15 02:31:04 +04:00
assert_spin_locked ( pmd_lockptr ( mm , pmdp ) ) ;
2012-10-09 03:30:07 +04:00
/* FIFO */
2013-11-15 02:30:59 +04:00
pgtable = pmd_huge_pte ( mm , pmdp ) ;
2012-10-09 03:30:07 +04:00
if ( list_empty ( & pgtable - > lru ) )
2013-11-15 02:30:59 +04:00
pmd_huge_pte ( mm , pmdp ) = NULL ;
2012-10-09 03:30:07 +04:00
else {
2013-11-15 02:30:59 +04:00
pmd_huge_pte ( mm , pmdp ) = list_entry ( pgtable - > lru . next ,
2012-10-09 03:30:07 +04:00
struct page , lru ) ;
list_del ( & pgtable - > lru ) ;
}
return pgtable ;
}
# endif /* CONFIG_TRANSPARENT_HUGEPAGE */
# endif
2012-10-09 03:30:09 +04:00
# ifndef __HAVE_ARCH_PMDP_INVALIDATE
# ifdef CONFIG_TRANSPARENT_HUGEPAGE
void pmdp_invalidate ( struct vm_area_struct * vma , unsigned long address ,
pmd_t * pmdp )
{
2013-12-19 05:08:34 +04:00
pmd_t entry = * pmdp ;
if ( pmd_numa ( entry ) )
entry = pmd_mknonnuma ( entry ) ;
2012-10-09 03:30:09 +04:00
set_pmd_at ( vma - > vm_mm , address , pmdp , pmd_mknotpresent ( * pmdp ) ) ;
flush_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
}
# endif /* CONFIG_TRANSPARENT_HUGEPAGE */
# endif