2008-02-05 09:29:01 +03:00
# include <linux/mm.h>
# include <linux/highmem.h>
# include <linux/sched.h>
static int walk_pte_range ( pmd_t * pmd , unsigned long addr , unsigned long end ,
2008-06-13 02:21:47 +04:00
struct mm_walk * walk )
2008-02-05 09:29:01 +03:00
{
pte_t * pte ;
int err = 0 ;
pte = pte_offset_map ( pmd , addr ) ;
2008-04-28 13:11:47 +04:00
for ( ; ; ) {
2008-06-13 02:21:47 +04:00
err = walk - > pte_entry ( pte , addr , addr + PAGE_SIZE , walk ) ;
2008-02-05 09:29:01 +03:00
if ( err )
break ;
2008-04-28 13:11:47 +04:00
addr + = PAGE_SIZE ;
if ( addr = = end )
break ;
pte + + ;
}
2008-02-05 09:29:01 +03:00
pte_unmap ( pte ) ;
return err ;
}
static int walk_pmd_range ( pud_t * pud , unsigned long addr , unsigned long end ,
2008-06-13 02:21:47 +04:00
struct mm_walk * walk )
2008-02-05 09:29:01 +03:00
{
pmd_t * pmd ;
unsigned long next ;
int err = 0 ;
pmd = pmd_offset ( pud , addr ) ;
do {
next = pmd_addr_end ( addr , end ) ;
if ( pmd_none_or_clear_bad ( pmd ) ) {
if ( walk - > pte_hole )
2008-06-13 02:21:47 +04:00
err = walk - > pte_hole ( addr , next , walk ) ;
2008-02-05 09:29:01 +03:00
if ( err )
break ;
continue ;
}
if ( walk - > pmd_entry )
2008-06-13 02:21:47 +04:00
err = walk - > pmd_entry ( pmd , addr , next , walk ) ;
2008-02-05 09:29:01 +03:00
if ( ! err & & walk - > pte_entry )
2008-06-13 02:21:47 +04:00
err = walk_pte_range ( pmd , addr , next , walk ) ;
2008-02-05 09:29:01 +03: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-13 02:21:47 +04:00
struct mm_walk * walk )
2008-02-05 09:29:01 +03: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-13 02:21:47 +04:00
err = walk - > pte_hole ( addr , next , walk ) ;
2008-02-05 09:29:01 +03:00
if ( err )
break ;
continue ;
}
if ( walk - > pud_entry )
2008-06-13 02:21:47 +04:00
err = walk - > pud_entry ( pud , addr , next , walk ) ;
2008-02-05 09:29:01 +03:00
if ( ! err & & ( walk - > pmd_entry | | walk - > pte_entry ) )
2008-06-13 02:21:47 +04:00
err = walk_pmd_range ( pud , addr , next , walk ) ;
2008-02-05 09:29:01 +03:00
if ( err )
break ;
} while ( pud + + , addr = next , addr ! = end ) ;
return err ;
}
/**
* walk_page_range - walk a memory map ' s page tables with a callback
2008-03-20 03:00:40 +03:00
* @ mm : memory map to walk
* @ addr : starting address
* @ end : ending address
* @ walk : set of callbacks to invoke for each level of the tree
2008-02-05 09:29:01 +03:00
*
* Recursively walk the page table for the memory area in a VMA ,
* calling supplied callbacks . Callbacks are called in - order ( first
* PGD , first PUD , first PMD , first PTE , second PTE . . . second PMD ,
* etc . ) . If lower - level callbacks are omitted , walking depth is reduced .
*
2008-06-13 02:21:47 +04:00
* Each callback receives an entry pointer and the start and end of the
* associated range , and a copy of the original mm_walk for access to
* the - > private or - > mm fields .
2008-02-05 09:29:01 +03:00
*
* No locks are taken , but the bottom level iterator will map PTE
* directories from highmem if necessary .
*
* If any callback returns a non - zero value , the walk is aborted and
* the return value is propagated back to the caller . Otherwise 0 is returned .
*/
2008-06-13 02:21:47 +04:00
int walk_page_range ( unsigned long addr , unsigned long end ,
struct mm_walk * walk )
2008-02-05 09:29:01 +03:00
{
pgd_t * pgd ;
unsigned long next ;
int err = 0 ;
if ( addr > = end )
return err ;
2008-06-13 02:21:47 +04:00
if ( ! walk - > mm )
return - EINVAL ;
pgd = pgd_offset ( walk - > mm , addr ) ;
2008-02-05 09:29:01 +03:00
do {
next = pgd_addr_end ( addr , end ) ;
if ( pgd_none_or_clear_bad ( pgd ) ) {
if ( walk - > pte_hole )
2008-06-13 02:21:47 +04:00
err = walk - > pte_hole ( addr , next , walk ) ;
2008-02-05 09:29:01 +03:00
if ( err )
break ;
continue ;
}
if ( walk - > pgd_entry )
2008-06-13 02:21:47 +04:00
err = walk - > pgd_entry ( pgd , addr , next , walk ) ;
2008-02-05 09:29:01 +03:00
if ( ! err & &
( walk - > pud_entry | | walk - > pmd_entry | | walk - > pte_entry ) )
2008-06-13 02:21:47 +04:00
err = walk_pud_range ( pgd , addr , next , walk ) ;
2008-02-05 09:29:01 +03:00
if ( err )
break ;
} while ( pgd + + , addr = next , addr ! = end ) ;
return err ;
}