2005-04-16 15:20:36 -07:00
/*
2008-08-02 10:55:55 +01:00
* arch / arm / include / asm / tlb . h
2005-04-16 15:20:36 -07:00
*
* Copyright ( C ) 2002 Russell King
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* Experimentation shows that on a StrongARM , it appears to be faster
* to use the " invalidate whole tlb " rather than " invalidate single
* tlb " for this.
*
* This appears true for both the process fork + exit case , as well as
* the munmap - large - area case .
*/
# ifndef __ASMARM_TLB_H
# define __ASMARM_TLB_H
# include <asm/cacheflush.h>
# include <asm/tlbflush.h>
2006-02-24 21:41:25 +00:00
# ifndef CONFIG_MMU
# include <linux/pagemap.h>
# include <asm-generic/tlb.h>
# else /* !CONFIG_MMU */
2005-04-16 15:20:36 -07:00
# include <asm/pgalloc.h>
/*
* TLB handling . This allows us to remove pages from the page
* tables , and efficiently handle the TLB issues .
*/
struct mmu_gather {
struct mm_struct * mm ;
unsigned int fullmm ;
2009-04-14 13:07:35 +01:00
unsigned long range_start ;
unsigned long range_end ;
2005-04-16 15:20:36 -07:00
} ;
DECLARE_PER_CPU ( struct mmu_gather , mmu_gathers ) ;
static inline struct mmu_gather *
tlb_gather_mmu ( struct mm_struct * mm , unsigned int full_mm_flush )
{
2005-10-29 18:16:01 -07:00
struct mmu_gather * tlb = & get_cpu_var ( mmu_gathers ) ;
2005-04-16 15:20:36 -07:00
tlb - > mm = mm ;
tlb - > fullmm = full_mm_flush ;
return tlb ;
}
static inline void
tlb_finish_mmu ( struct mmu_gather * tlb , unsigned long start , unsigned long end )
{
if ( tlb - > fullmm )
2005-10-29 18:16:03 -07:00
flush_tlb_mm ( tlb - > mm ) ;
2005-04-16 15:20:36 -07:00
/* keep the page table cache within bounds */
check_pgt_cache ( ) ;
2005-10-29 18:16:01 -07:00
put_cpu_var ( mmu_gathers ) ;
2005-04-16 15:20:36 -07:00
}
2009-04-14 13:07:35 +01:00
/*
* Memorize the range for the TLB flush .
*/
static inline void
tlb_remove_tlb_entry ( struct mmu_gather * tlb , pte_t * ptep , unsigned long addr )
{
if ( ! tlb - > fullmm ) {
if ( addr < tlb - > range_start )
tlb - > range_start = addr ;
if ( addr + PAGE_SIZE > tlb - > range_end )
tlb - > range_end = addr + PAGE_SIZE ;
}
}
2005-04-16 15:20:36 -07:00
/*
* In the case of tlb vma handling , we can optimise these away in the
* case where we ' re doing a full MM flush . When we ' re doing a munmap ,
* the vmas are adjusted to only cover the region to be torn down .
*/
static inline void
tlb_start_vma ( struct mmu_gather * tlb , struct vm_area_struct * vma )
{
2009-04-14 13:07:35 +01:00
if ( ! tlb - > fullmm ) {
2005-04-16 15:20:36 -07:00
flush_cache_range ( vma , vma - > vm_start , vma - > vm_end ) ;
2009-04-14 13:07:35 +01:00
tlb - > range_start = TASK_SIZE ;
tlb - > range_end = 0 ;
}
2005-04-16 15:20:36 -07:00
}
static inline void
tlb_end_vma ( struct mmu_gather * tlb , struct vm_area_struct * vma )
{
2009-04-14 13:07:35 +01:00
if ( ! tlb - > fullmm & & tlb - > range_end > 0 )
flush_tlb_range ( vma , tlb - > range_start , tlb - > range_end ) ;
2005-04-16 15:20:36 -07:00
}
# define tlb_remove_page(tlb,page) free_page_and_swap_cache(page)
2008-02-04 22:29:14 -08:00
# define pte_free_tlb(tlb, ptep) pte_free((tlb)->mm, ptep)
# define pmd_free_tlb(tlb, pmdp) pmd_free((tlb)->mm, pmdp)
2005-04-16 15:20:36 -07:00
# define tlb_migrate_finish(mm) do { } while (0)
2006-02-24 21:41:25 +00:00
# endif /* CONFIG_MMU */
2005-04-16 15:20:36 -07:00
# endif