2008-02-04 22:29:01 -08:00
# include <linux/mm.h>
# include <linux/highmem.h>
# include <linux/sched.h>
2009-12-14 17:59:59 -08:00
# include <linux/hugetlb.h>
2008-02-04 22:29:01 -08:00
static int walk_pte_range ( pmd_t * pmd , unsigned long addr , unsigned long end ,
2008-06-12 15:21:47 -07:00
struct mm_walk * walk )
2008-02-04 22:29:01 -08:00
{
pte_t * pte ;
int err = 0 ;
pte = pte_offset_map ( pmd , addr ) ;
2008-04-28 02:11:47 -07:00
for ( ; ; ) {
2008-06-12 15:21:47 -07:00
err = walk - > pte_entry ( pte , addr , addr + PAGE_SIZE , walk ) ;
2008-02-04 22:29:01 -08:00
if ( err )
break ;
2008-04-28 02:11:47 -07:00
addr + = PAGE_SIZE ;
if ( addr = = end )
break ;
pte + + ;
}
2008-02-04 22:29:01 -08:00
pte_unmap ( pte ) ;
return err ;
}
static int walk_pmd_range ( pud_t * pud , unsigned long addr , unsigned long end ,
2008-06-12 15:21:47 -07:00
struct mm_walk * walk )
2008-02-04 22:29:01 -08:00
{
pmd_t * pmd ;
unsigned long next ;
int err = 0 ;
pmd = pmd_offset ( pud , addr ) ;
do {
2011-03-22 16:32:56 -07:00
again :
2008-02-04 22:29:01 -08:00
next = pmd_addr_end ( addr , end ) ;
2015-02-11 15:28:06 -08:00
if ( pmd_none ( * pmd ) | | ! walk - > vma ) {
2008-02-04 22:29:01 -08:00
if ( walk - > pte_hole )
2008-06-12 15:21:47 -07:00
err = walk - > pte_hole ( addr , next , walk ) ;
2008-02-04 22:29:01 -08:00
if ( err )
break ;
continue ;
}
2011-03-22 16:32:56 -07:00
/*
* This implies that each - > pmd_entry ( ) handler
* needs to know about pmd_trans_huge ( ) pmds
*/
2008-02-04 22:29:01 -08:00
if ( walk - > pmd_entry )
2008-06-12 15:21:47 -07:00
err = walk - > pmd_entry ( pmd , addr , next , walk ) ;
2011-03-22 16:32:56 -07:00
if ( err )
break ;
/*
* Check this here so we only break down trans_huge
* pages when we _need_ to
*/
if ( ! walk - > pte_entry )
continue ;
2016-01-15 16:52:42 -08:00
split_huge_pmd ( walk - > vma , pmd , addr ) ;
2015-02-11 15:27:37 -08:00
if ( pmd_trans_unstable ( pmd ) )
2011-03-22 16:32:56 -07:00
goto again ;
err = walk_pte_range ( pmd , addr , next , walk ) ;
2008-02-04 22:29:01 -08:00
if ( err )
break ;
} while ( pmd + + , addr = next , addr ! = end ) ;
return err ;
}
static int walk_pud_range ( pgd_t * pgd , unsigned long addr , unsigned long end ,
2008-06-12 15:21:47 -07:00
struct mm_walk * walk )
2008-02-04 22:29:01 -08:00
{
pud_t * pud ;
unsigned long next ;
int err = 0 ;
pud = pud_offset ( pgd , addr ) ;
do {
next = pud_addr_end ( addr , end ) ;
if ( pud_none_or_clear_bad ( pud ) ) {
if ( walk - > pte_hole )
2008-06-12 15:21:47 -07:00
err = walk - > pte_hole ( addr , next , walk ) ;
2008-02-04 22:29:01 -08:00
if ( err )
break ;
continue ;
}
2015-02-11 15:27:34 -08:00
if ( walk - > pmd_entry | | walk - > pte_entry )
2008-06-12 15:21:47 -07:00
err = walk_pmd_range ( pud , addr , next , walk ) ;
2008-02-04 22:29:01 -08:00
if ( err )
break ;
} while ( pud + + , addr = next , addr ! = end ) ;
return err ;
}
2015-02-11 15:27:37 -08:00
static int walk_pgd_range ( unsigned long addr , unsigned long end ,
struct mm_walk * walk )
{
pgd_t * pgd ;
unsigned long next ;
int err = 0 ;
pgd = pgd_offset ( walk - > mm , addr ) ;
do {
next = pgd_addr_end ( addr , end ) ;
if ( pgd_none_or_clear_bad ( pgd ) ) {
if ( walk - > pte_hole )
err = walk - > pte_hole ( addr , next , walk ) ;
if ( err )
break ;
continue ;
}
if ( walk - > pmd_entry | | walk - > pte_entry )
err = walk_pud_range ( pgd , addr , next , walk ) ;
if ( err )
break ;
} while ( pgd + + , addr = next , addr ! = end ) ;
return err ;
}
2010-04-06 14:35:04 -07:00
# ifdef CONFIG_HUGETLB_PAGE
static unsigned long hugetlb_entry_end ( struct hstate * h , unsigned long addr ,
unsigned long end )
{
unsigned long boundary = ( addr & huge_page_mask ( h ) ) + huge_page_size ( h ) ;
return boundary < end ? boundary : end ;
}
2015-02-11 15:27:37 -08:00
static int walk_hugetlb_range ( unsigned long addr , unsigned long end ,
2010-04-06 14:35:04 -07:00
struct mm_walk * walk )
{
2015-02-11 15:27:37 -08:00
struct vm_area_struct * vma = walk - > vma ;
2010-04-06 14:35:04 -07:00
struct hstate * h = hstate_vma ( vma ) ;
unsigned long next ;
unsigned long hmask = huge_page_mask ( h ) ;
pte_t * pte ;
int err = 0 ;
do {
next = hugetlb_entry_end ( h , addr , end ) ;
pte = huge_pte_offset ( walk - > mm , addr & hmask ) ;
if ( pte & & walk - > hugetlb_entry )
err = walk - > hugetlb_entry ( pte , hmask , addr , next , walk ) ;
if ( err )
2015-02-11 15:27:37 -08:00
break ;
2010-04-06 14:35:04 -07:00
} while ( addr = next , addr ! = end ) ;
2015-02-11 15:27:37 -08:00
return err ;
2010-04-06 14:35:04 -07:00
}
2011-07-25 17:12:09 -07:00
# else /* CONFIG_HUGETLB_PAGE */
2015-02-11 15:27:37 -08:00
static int walk_hugetlb_range ( unsigned long addr , unsigned long end ,
2011-07-25 17:12:09 -07:00
struct mm_walk * walk )
{
return 0 ;
}
# endif /* CONFIG_HUGETLB_PAGE */
2015-02-11 15:27:37 -08:00
/*
* Decide whether we really walk over the current vma on [ @ start , @ end )
* or skip it via the returned value . Return 0 if we do walk over the
* current vma , and return 1 if we skip the vma . Negative values means
* error , where we abort the current walk .
*/
static int walk_page_test ( unsigned long start , unsigned long end ,
struct mm_walk * walk )
{
struct vm_area_struct * vma = walk - > vma ;
2011-07-25 17:12:09 -07:00
2015-02-11 15:27:37 -08:00
if ( walk - > test_walk )
return walk - > test_walk ( start , end , walk ) ;
/*
2015-02-11 15:28:06 -08:00
* vma ( VM_PFNMAP ) doesn ' t have any valid struct pages behind VM_PFNMAP
* range , so we don ' t walk over it as we do for normal vmas . However ,
* Some callers are interested in handling hole range and they don ' t
* want to just ignore any single address range . Such users certainly
* define their - > pte_hole ( ) callbacks , so let ' s delegate them to handle
* vma ( VM_PFNMAP ) .
2015-02-11 15:27:37 -08:00
*/
2015-02-11 15:28:06 -08:00
if ( vma - > vm_flags & VM_PFNMAP ) {
int err = 1 ;
if ( walk - > pte_hole )
err = walk - > pte_hole ( start , end , walk ) ;
return err ? err : 1 ;
}
2015-02-11 15:27:37 -08:00
return 0 ;
}
static int __walk_page_range ( unsigned long start , unsigned long end ,
struct mm_walk * walk )
{
int err = 0 ;
struct vm_area_struct * vma = walk - > vma ;
if ( vma & & is_vm_hugetlb_page ( vma ) ) {
if ( walk - > hugetlb_entry )
err = walk_hugetlb_range ( start , end , walk ) ;
} else
err = walk_pgd_range ( start , end , walk ) ;
return err ;
}
2010-04-06 14:35:04 -07:00
2008-02-04 22:29:01 -08:00
/**
2015-02-11 15:27:37 -08:00
* walk_page_range - walk page table with caller specific callbacks
2008-02-04 22:29:01 -08:00
*
2015-02-11 15:27:37 -08:00
* Recursively walk the page table tree of the process represented by @ walk - > mm
* within the virtual address range [ @ start , @ end ) . During walking , we can do
* some caller - specific works for each entry , by setting up pmd_entry ( ) ,
* pte_entry ( ) , and / or hugetlb_entry ( ) . If you don ' t set up for some of these
* callbacks , the associated entries / pages are just ignored .
* The return values of these callbacks are commonly defined like below :
* - 0 : succeeded to handle the current entry , and if you don ' t reach the
* end address yet , continue to walk .
* - > 0 : succeeded to handle the current entry , and return to the caller
* with caller specific value .
* - < 0 : failed to handle the current entry , and return to the caller
* with error code .
2008-02-04 22:29:01 -08:00
*
2015-02-11 15:27:37 -08:00
* Before starting to walk page table , some callers want to check whether
* they really want to walk over the current vma , typically by checking
* its vm_flags . walk_page_test ( ) and @ walk - > test_walk ( ) are used for this
* purpose .
2008-02-04 22:29:01 -08:00
*
2015-02-11 15:27:37 -08:00
* struct mm_walk keeps current values of some common data like vma and pmd ,
* which are useful for the access from callbacks . If you want to pass some
* caller - specific data to callbacks , @ walk - > private should be helpful .
2011-07-25 17:12:10 -07:00
*
2015-02-11 15:27:37 -08:00
* Locking :
* Callers of walk_page_range ( ) and walk_page_vma ( ) should hold
* @ walk - > mm - > mmap_sem , because these function traverse vma list and / or
* access to vma ' s data .
2008-02-04 22:29:01 -08:00
*/
2015-02-11 15:27:37 -08:00
int walk_page_range ( unsigned long start , unsigned long end ,
2008-06-12 15:21:47 -07:00
struct mm_walk * walk )
2008-02-04 22:29:01 -08:00
{
int err = 0 ;
2015-02-11 15:27:37 -08:00
unsigned long next ;
struct vm_area_struct * vma ;
2008-02-04 22:29:01 -08:00
2015-02-11 15:27:37 -08:00
if ( start > = end )
return - EINVAL ;
2008-02-04 22:29:01 -08:00
2008-06-12 15:21:47 -07:00
if ( ! walk - > mm )
return - EINVAL ;
2014-10-09 15:28:39 -07:00
VM_BUG_ON_MM ( ! rwsem_is_locked ( & walk - > mm - > mmap_sem ) , walk - > mm ) ;
2013-05-24 15:55:36 -07:00
2015-02-11 15:27:37 -08:00
vma = find_vma ( walk - > mm , start ) ;
2008-02-04 22:29:01 -08:00
do {
2015-02-11 15:27:37 -08:00
if ( ! vma ) { /* after the last vma */
walk - > vma = NULL ;
next = end ;
} else if ( start < vma - > vm_start ) { /* outside vma */
walk - > vma = NULL ;
next = min ( end , vma - > vm_start ) ;
} else { /* inside vma */
walk - > vma = vma ;
next = min ( end , vma - > vm_end ) ;
vma = vma - > vm_next ;
2010-11-24 12:57:10 -08:00
2015-02-11 15:27:37 -08:00
err = walk_page_test ( start , next , walk ) ;
2015-03-25 15:55:14 -07:00
if ( err > 0 ) {
/*
* positive return values are purely for
* controlling the pagewalk , so should never
* be passed to the callers .
*/
err = 0 ;
2013-05-24 15:55:36 -07:00
continue ;
2015-03-25 15:55:14 -07:00
}
2015-02-11 15:27:37 -08:00
if ( err < 0 )
2008-02-04 22:29:01 -08:00
break ;
}
2015-02-11 15:27:37 -08:00
if ( walk - > vma | | walk - > pte_hole )
err = __walk_page_range ( start , next , walk ) ;
2008-02-04 22:29:01 -08:00
if ( err )
break ;
2015-02-11 15:27:37 -08:00
} while ( start = next , start < end ) ;
2008-02-04 22:29:01 -08:00
return err ;
}
2015-02-11 15:27:40 -08:00
int walk_page_vma ( struct vm_area_struct * vma , struct mm_walk * walk )
{
int err ;
if ( ! walk - > mm )
return - EINVAL ;
VM_BUG_ON ( ! rwsem_is_locked ( & walk - > mm - > mmap_sem ) ) ;
VM_BUG_ON ( ! vma ) ;
walk - > vma = vma ;
err = walk_page_test ( vma - > vm_start , vma - > vm_end , walk ) ;
if ( err > 0 )
return 0 ;
if ( err < 0 )
return err ;
return __walk_page_range ( vma - > vm_start , vma - > vm_end , walk ) ;
}