1 #ifndef _ASM_X86_TLBFLUSH_H
 
   2 #define _ASM_X86_TLBFLUSH_H
 
   5 #include <linux/sched.h>
 
   7 #include <asm/processor.h>
 
   8 #include <asm/system.h>
 
  10 #ifdef CONFIG_PARAVIRT
 
  11 #include <asm/paravirt.h>
 
  13 #define __flush_tlb() __native_flush_tlb()
 
  14 #define __flush_tlb_global() __native_flush_tlb_global()
 
  15 #define __flush_tlb_single(addr) __native_flush_tlb_single(addr)
 
  18 static inline void __native_flush_tlb(void)
 
  20         write_cr3(read_cr3());
 
  23 static inline void __native_flush_tlb_global(void)
 
  29          * Read-modify-write to CR4 - protect it from preemption and
 
  30          * from interrupts. (Use the raw variant because this code can
 
  31          * be called from deep inside debugging code.)
 
  33         raw_local_irq_save(flags);
 
  37         write_cr4(cr4 & ~X86_CR4_PGE);
 
  38         /* write old PGE again and flush TLBs */
 
  41         raw_local_irq_restore(flags);
 
  44 static inline void __native_flush_tlb_single(unsigned long addr)
 
  46         asm volatile("invlpg (%0)" ::"r" (addr) : "memory");
 
  49 static inline void __flush_tlb_all(void)
 
  57 static inline void __flush_tlb_one(unsigned long addr)
 
  60                 __flush_tlb_single(addr);
 
  66 # define TLB_FLUSH_ALL  0xffffffff
 
  68 # define TLB_FLUSH_ALL  -1ULL
 
  74  *  - flush_tlb() flushes the current mm struct TLBs
 
  75  *  - flush_tlb_all() flushes all processes TLBs
 
  76  *  - flush_tlb_mm(mm) flushes the specified mm context TLB's
 
  77  *  - flush_tlb_page(vma, vmaddr) flushes one page
 
  78  *  - flush_tlb_range(vma, start, end) flushes a range of pages
 
  79  *  - flush_tlb_kernel_range(start, end) flushes a range of kernel pages
 
  80  *  - flush_tlb_others(cpumask, mm, va) flushes TLBs on other cpus
 
  82  * ..but the i386 has somewhat limited tlb flushing capabilities,
 
  83  * and page-granular flushes are available only on i486 and up.
 
  85  * x86-64 can only flush individual pages or full VMs. For a range flush
 
  86  * we always do the full VM. Might be worth trying if for a small
 
  87  * range a few INVLPGs in a row are a win.
 
  92 #define flush_tlb() __flush_tlb()
 
  93 #define flush_tlb_all() __flush_tlb_all()
 
  94 #define local_flush_tlb() __flush_tlb()
 
  96 static inline void flush_tlb_mm(struct mm_struct *mm)
 
  98         if (mm == current->active_mm)
 
 102 static inline void flush_tlb_page(struct vm_area_struct *vma,
 
 105         if (vma->vm_mm == current->active_mm)
 
 106                 __flush_tlb_one(addr);
 
 109 static inline void flush_tlb_range(struct vm_area_struct *vma,
 
 110                                    unsigned long start, unsigned long end)
 
 112         if (vma->vm_mm == current->active_mm)
 
 116 static inline void native_flush_tlb_others(const struct cpumask *cpumask,
 
 117                                            struct mm_struct *mm,
 
 122 static inline void reset_lazy_tlbstate(void)
 
 130 #define local_flush_tlb() __flush_tlb()
 
 132 extern void flush_tlb_all(void);
 
 133 extern void flush_tlb_current_task(void);
 
 134 extern void flush_tlb_mm(struct mm_struct *);
 
 135 extern void flush_tlb_page(struct vm_area_struct *, unsigned long);
 
 137 #define flush_tlb()     flush_tlb_current_task()
 
 139 static inline void flush_tlb_range(struct vm_area_struct *vma,
 
 140                                    unsigned long start, unsigned long end)
 
 142         flush_tlb_mm(vma->vm_mm);
 
 145 void native_flush_tlb_others(const struct cpumask *cpumask,
 
 146                              struct mm_struct *mm, unsigned long va);
 
 148 #define TLBSTATE_OK     1
 
 149 #define TLBSTATE_LAZY   2
 
 152         struct mm_struct *active_mm;
 
 155 DECLARE_PER_CPU(struct tlb_state, cpu_tlbstate);
 
 157 static inline void reset_lazy_tlbstate(void)
 
 159         percpu_write(cpu_tlbstate.state, 0);
 
 160         percpu_write(cpu_tlbstate.active_mm, &init_mm);
 
 165 #ifndef CONFIG_PARAVIRT
 
 166 #define flush_tlb_others(mask, mm, va)  native_flush_tlb_others(mask, mm, va)
 
 169 static inline void flush_tlb_kernel_range(unsigned long start,
 
 175 extern void zap_low_mappings(void);
 
 177 #endif /* _ASM_X86_TLBFLUSH_H */