2005-09-26 16:04:21 +10:00
/*
* MMU context allocation for 64 - bit kernels .
*
* Copyright ( C ) 2004 Anton Blanchard , IBM Corp . < anton @ samba . org >
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*
*/
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/string.h>
# include <linux/types.h>
# include <linux/mm.h>
# include <linux/spinlock.h>
# include <linux/idr.h>
# include <asm/mmu_context.h>
static DEFINE_SPINLOCK ( mmu_context_lock ) ;
static DEFINE_IDR ( mmu_context_idr ) ;
2008-12-18 19:13:24 +00:00
/*
* The proto - VSID space has 2 ^ 35 - 1 segments available for user mappings .
* Each segment contains 2 ^ 28 bytes . Each context maps 2 ^ 44 bytes ,
* so we can support 2 ^ 19 - 1 contexts ( 19 = = 35 + 28 - 44 ) .
*/
# define NO_CONTEXT 0
# define MAX_CONTEXT ((1UL << 19) - 1)
2005-09-26 16:04:21 +10:00
int init_new_context ( struct task_struct * tsk , struct mm_struct * mm )
{
int index ;
int err ;
again :
if ( ! idr_pre_get ( & mmu_context_idr , GFP_KERNEL ) )
return - ENOMEM ;
spin_lock ( & mmu_context_lock ) ;
err = idr_get_new_above ( & mmu_context_idr , NULL , 1 , & index ) ;
spin_unlock ( & mmu_context_lock ) ;
if ( err = = - EAGAIN )
goto again ;
else if ( err )
return err ;
if ( index > MAX_CONTEXT ) {
2006-06-27 08:46:09 -04:00
spin_lock ( & mmu_context_lock ) ;
2005-09-26 16:04:21 +10:00
idr_remove ( & mmu_context_idr , index ) ;
2006-06-27 08:46:09 -04:00
spin_unlock ( & mmu_context_lock ) ;
2005-09-26 16:04:21 +10:00
return - ENOMEM ;
}
2007-05-08 16:27:27 +10:00
/* The old code would re-promote on fork, we don't do that
* when using slices as it could cause problem promoting slices
* that have been forced down to 4 K
*/
2007-08-15 16:51:18 +10:00
if ( slice_mm_new_context ( mm ) )
2007-05-08 16:27:27 +10:00
slice_set_user_psize ( mm , mmu_virtual_psize ) ;
2007-08-15 16:33:55 +10:00
mm - > context . id = index ;
2005-09-26 16:04:21 +10:00
return 0 ;
}
void destroy_context ( struct mm_struct * mm )
{
spin_lock ( & mmu_context_lock ) ;
idr_remove ( & mmu_context_idr , mm - > context . id ) ;
spin_unlock ( & mmu_context_lock ) ;
mm - > context . id = NO_CONTEXT ;
}