2011-01-13 15:46:40 -08:00
/*
* mm / pgtable - generic . c
*
* Generic pgtable methods declared in asm - generic / pgtable . h
*
* Copyright ( C ) 2010 Linus Torvalds
*/
2011-01-25 15:07:11 -08:00
# include <linux/pagemap.h>
2011-01-13 15:46:40 -08:00
# include <asm/tlb.h>
# include <asm-generic/pgtable.h>
2013-09-11 14:21:28 -07: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-13 15:46:40 -08:00
# ifndef __HAVE_ARCH_PTEP_SET_ACCESS_FLAGS
/*
2012-11-06 09:56:01 +00: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-13 15:46:40 -08: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 09:56:01 +00:00
flush_tlb_fix_spurious_fault ( vma , address ) ;
2011-01-13 15:46:40 -08:00
}
return changed ;
}
# endif
2015-07-09 17:19:30 +05:30
# 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_PTEP_CLEAR_FLUSH
pte_t ptep_clear_flush ( struct vm_area_struct * vma , unsigned long address ,
pte_t * ptep )
{
struct mm_struct * mm = ( vma ) - > vm_mm ;
pte_t pte ;
pte = ptep_get_and_clear ( mm , address , ptep ) ;
if ( pte_accessible ( mm , pte ) )
flush_tlb_page ( vma , address ) ;
return pte ;
}
# endif
2015-07-09 17:22:44 +05:30
# ifdef CONFIG_TRANSPARENT_HUGEPAGE
2011-01-13 15:46:40 -08:00
# 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 )
{
int changed = ! pmd_same ( * pmdp , entry ) ;
VM_BUG_ON ( address & ~ HPAGE_PMD_MASK ) ;
if ( changed ) {
set_pmd_at ( vma - > vm_mm , address , pmdp , entry ) ;
2015-02-20 10:36:28 +05:30
flush_pmd_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
2011-01-13 15:46:40 -08:00
}
return changed ;
}
# 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-21 16:34:27 -07:00
VM_BUG_ON ( address & ~ HPAGE_PMD_MASK ) ;
2011-01-13 15:46:40 -08:00
young = pmdp_test_and_clear_young ( vma , address , pmdp ) ;
if ( young )
2015-02-20 10:36:28 +05:30
flush_pmd_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
2011-01-13 15:46:40 -08:00
return young ;
}
# endif
2015-06-24 16:57:44 -07:00
# ifndef __HAVE_ARCH_PMDP_HUGE_CLEAR_FLUSH
pmd_t pmdp_huge_clear_flush ( struct vm_area_struct * vma , unsigned long address ,
pmd_t * pmdp )
2011-01-13 15:46:40 -08:00
{
pmd_t pmd ;
VM_BUG_ON ( address & ~ HPAGE_PMD_MASK ) ;
2016-01-15 16:56:52 -08:00
VM_BUG_ON ( ! pmd_trans_huge ( * pmdp ) & & ! pmd_devmap ( * pmdp ) ) ;
2015-06-24 16:57:44 -07:00
pmd = pmdp_huge_get_and_clear ( vma - > vm_mm , address , pmdp ) ;
2015-02-20 10:36:28 +05:30
flush_pmd_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
2011-01-13 15:46:40 -08:00
return pmd ;
}
# endif
2012-10-08 16:30:07 -07:00
# ifndef __HAVE_ARCH_PGTABLE_DEPOSIT
2013-06-05 17:14:02 -07:00
void pgtable_trans_huge_deposit ( struct mm_struct * mm , pmd_t * pmdp ,
pgtable_t pgtable )
2012-10-08 16:30:07 -07:00
{
2013-11-14 14:31:04 -08:00
assert_spin_locked ( pmd_lockptr ( mm , pmdp ) ) ;
2012-10-08 16:30:07 -07:00
/* FIFO */
2013-11-14 14:30:59 -08:00
if ( ! pmd_huge_pte ( mm , pmdp ) )
2012-10-08 16:30:07 -07:00
INIT_LIST_HEAD ( & pgtable - > lru ) ;
else
2013-11-14 14:30:59 -08:00
list_add ( & pgtable - > lru , & pmd_huge_pte ( mm , pmdp ) - > lru ) ;
pmd_huge_pte ( mm , pmdp ) = pgtable ;
2012-10-08 16:30:07 -07:00
}
# endif
# ifndef __HAVE_ARCH_PGTABLE_WITHDRAW
/* no "address" argument so destroys page coloring of some arch */
2013-06-05 17:14:02 -07:00
pgtable_t pgtable_trans_huge_withdraw ( struct mm_struct * mm , pmd_t * pmdp )
2012-10-08 16:30:07 -07:00
{
pgtable_t pgtable ;
2013-11-14 14:31:04 -08:00
assert_spin_locked ( pmd_lockptr ( mm , pmdp ) ) ;
2012-10-08 16:30:07 -07:00
/* FIFO */
2013-11-14 14:30:59 -08:00
pgtable = pmd_huge_pte ( mm , pmdp ) ;
2016-01-14 15:19:32 -08:00
pmd_huge_pte ( mm , pmdp ) = list_first_entry_or_null ( & pgtable - > lru ,
struct page , lru ) ;
if ( pmd_huge_pte ( mm , pmdp ) )
2012-10-08 16:30:07 -07:00
list_del ( & pgtable - > lru ) ;
return pgtable ;
}
# endif
2012-10-08 16:30:09 -07:00
# ifndef __HAVE_ARCH_PMDP_INVALIDATE
void pmdp_invalidate ( struct vm_area_struct * vma , unsigned long address ,
pmd_t * pmdp )
{
2013-12-18 17:08:34 -08:00
pmd_t entry = * pmdp ;
2014-08-29 15:18:33 -07:00
set_pmd_at ( vma - > vm_mm , address , pmdp , pmd_mknotpresent ( entry ) ) ;
2015-02-20 10:36:28 +05:30
flush_pmd_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
2012-10-08 16:30:09 -07:00
}
# endif
2015-06-24 16:57:42 -07:00
# ifndef pmdp_collapse_flush
pmd_t pmdp_collapse_flush ( struct vm_area_struct * vma , unsigned long address ,
pmd_t * pmdp )
{
2015-06-24 16:57:44 -07:00
/*
* pmd and hugepage pte format are same . So we could
* use the same function .
*/
2015-06-24 16:57:42 -07:00
pmd_t pmd ;
VM_BUG_ON ( address & ~ HPAGE_PMD_MASK ) ;
VM_BUG_ON ( pmd_trans_huge ( * pmdp ) ) ;
2015-06-24 16:57:44 -07:00
pmd = pmdp_huge_get_and_clear ( vma - > vm_mm , address , pmdp ) ;
2016-02-11 16:13:09 -08:00
/* collapse entails shooting down ptes not pmd */
flush_tlb_range ( vma , address , address + HPAGE_PMD_SIZE ) ;
2015-06-24 16:57:42 -07:00
return pmd ;
}
# endif
2015-07-09 17:22:44 +05:30
# endif /* CONFIG_TRANSPARENT_HUGEPAGE */