2005-04-17 02:20:36 +04:00
# ifndef __X86_64_MMU_CONTEXT_H
# define __X86_64_MMU_CONTEXT_H
# include <asm/desc.h>
# include <asm/atomic.h>
# include <asm/pgalloc.h>
# include <asm/pda.h>
# include <asm/pgtable.h>
# include <asm/tlbflush.h>
2008-01-30 15:33:19 +03:00
# ifndef CONFIG_PARAVIRT
2007-05-02 21:27:14 +04:00
# include <asm-generic/mm_hooks.h>
2008-01-30 15:33:19 +03:00
# endif
2005-04-17 02:20:36 +04:00
/*
* possibly do the LDT unload here ?
*/
int init_new_context ( struct task_struct * tsk , struct mm_struct * mm ) ;
void destroy_context ( struct mm_struct * mm ) ;
static inline void enter_lazy_tlb ( struct mm_struct * mm , struct task_struct * tsk )
{
2006-01-12 00:46:09 +03:00
# ifdef CONFIG_SMP
2008-03-23 11:02:43 +03:00
if ( read_pda ( mmu_state ) = = TLBSTATE_OK )
2005-04-17 02:20:36 +04:00
write_pda ( mmu_state , TLBSTATE_LAZY ) ;
# endif
2006-01-12 00:46:09 +03:00
}
2005-04-17 02:20:36 +04:00
2008-03-23 11:02:43 +03:00
static inline void switch_mm ( struct mm_struct * prev , struct mm_struct * next ,
2005-04-17 02:20:36 +04:00
struct task_struct * tsk )
{
unsigned cpu = smp_processor_id ( ) ;
if ( likely ( prev ! = next ) ) {
/* stop flush ipis for the previous mm */
2006-03-24 14:15:11 +03:00
cpu_clear ( cpu , prev - > cpu_vm_mask ) ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_SMP
write_pda ( mmu_state , TLBSTATE_OK ) ;
write_pda ( active_mm , next ) ;
# endif
2006-03-24 14:15:11 +03:00
cpu_set ( cpu , next - > cpu_vm_mask ) ;
2005-04-17 02:20:36 +04:00
load_cr3 ( next - > pgd ) ;
2008-03-23 11:02:43 +03:00
if ( unlikely ( next - > context . ldt ! = prev - > context . ldt ) )
2008-01-30 15:31:14 +03:00
load_LDT_nolock ( & next - > context ) ;
2005-04-17 02:20:36 +04:00
}
# ifdef CONFIG_SMP
else {
write_pda ( mmu_state , TLBSTATE_OK ) ;
if ( read_pda ( active_mm ) ! = next )
2008-01-30 15:30:28 +03:00
BUG ( ) ;
2006-03-24 14:15:11 +03:00
if ( ! cpu_test_and_set ( cpu , next - > cpu_vm_mask ) ) {
2008-03-23 11:02:43 +03:00
/* We were in lazy tlb mode and leave_mm disabled
2005-04-17 02:20:36 +04:00
* tlb flush IPI delivery . We must reload CR3
* to make sure to use no freed page tables .
*/
load_cr3 ( next - > pgd ) ;
2008-01-30 15:31:14 +03:00
load_LDT_nolock ( & next - > context ) ;
2005-04-17 02:20:36 +04:00
}
}
# endif
}
2008-03-23 11:02:43 +03:00
# define deactivate_mm(tsk, mm) \
do { \
load_gs_index ( 0 ) ; \
asm volatile ( " movl %0,%%fs " : : " r " ( 0 ) ) ; \
} while ( 0 )
2005-04-17 02:20:36 +04:00
2008-03-23 11:02:43 +03:00
# define activate_mm(prev, next) \
switch_mm ( ( prev ) , ( next ) , NULL )
2005-04-17 02:20:36 +04:00
# endif