2005-04-16 15:20:36 -07:00
/* rwsem-spinlock.c: R/W semaphores: contention handling functions for
* generic spinlock implementation
*
* Copyright ( c ) 2001 David Howells ( dhowells @ redhat . com ) .
* - Derived partially from idea by Andrea Arcangeli < andrea @ suse . de >
* - Derived also from comments by Linus
*/
# include <linux/rwsem.h>
# include <linux/sched.h>
2011-11-16 21:29:17 -05:00
# include <linux/export.h>
2005-04-16 15:20:36 -07:00
struct rwsem_waiter {
struct list_head list ;
struct task_struct * task ;
unsigned int flags ;
# define RWSEM_WAITING_FOR_READ 0x00000001
# define RWSEM_WAITING_FOR_WRITE 0x00000002
} ;
2009-12-14 18:00:21 -08:00
int rwsem_is_locked ( struct rw_semaphore * sem )
{
int ret = 1 ;
unsigned long flags ;
2010-02-24 09:54:54 +01:00
if ( raw_spin_trylock_irqsave ( & sem - > wait_lock , flags ) ) {
2009-12-14 18:00:21 -08:00
ret = ( sem - > activity ! = 0 ) ;
2010-02-24 09:54:54 +01:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2009-12-14 18:00:21 -08:00
}
return ret ;
}
EXPORT_SYMBOL ( rwsem_is_locked ) ;
2005-04-16 15:20:36 -07:00
/*
* initialise the semaphore
*/
2006-07-03 00:24:53 -07:00
void __init_rwsem ( struct rw_semaphore * sem , const char * name ,
struct lock_class_key * key )
2005-04-16 15:20:36 -07:00
{
2006-07-03 00:24:53 -07:00
# ifdef CONFIG_DEBUG_LOCK_ALLOC
/*
* Make sure we are not reinitializing a held semaphore :
*/
debug_check_no_locks_freed ( ( void * ) sem , sizeof ( * sem ) ) ;
2006-10-11 01:45:14 -04:00
lockdep_init_map ( & sem - > dep_map , name , key , 0 ) ;
2006-07-03 00:24:53 -07:00
# endif
2005-04-16 15:20:36 -07:00
sem - > activity = 0 ;
2010-02-24 09:54:54 +01:00
raw_spin_lock_init ( & sem - > wait_lock ) ;
2005-04-16 15:20:36 -07:00
INIT_LIST_HEAD ( & sem - > wait_list ) ;
}
2009-12-14 18:00:20 -08:00
EXPORT_SYMBOL ( __init_rwsem ) ;
2005-04-16 15:20:36 -07:00
/*
* handle the lock release when processes blocked on it that can now run
* - if we come here , then :
* - the ' active count ' _reached_ zero
* - the ' waiting count ' is non - zero
* - the spinlock must be held by the caller
* - woken process blocks are discarded from the list after having task zeroed
* - writers are only woken if wakewrite is non - zero
*/
static inline struct rw_semaphore *
__rwsem_do_wake ( struct rw_semaphore * sem , int wakewrite )
{
struct rwsem_waiter * waiter ;
struct task_struct * tsk ;
int woken ;
waiter = list_entry ( sem - > wait_list . next , struct rwsem_waiter , list ) ;
if ( ! wakewrite ) {
if ( waiter - > flags & RWSEM_WAITING_FOR_WRITE )
goto out ;
goto dont_wake_writers ;
}
2013-02-01 18:59:16 +08:00
/*
* as we support write lock stealing , we can ' t set sem - > activity
* to - 1 here to indicate we get the lock . Instead , we wake it up
* to let it go get it again .
2005-04-16 15:20:36 -07:00
*/
if ( waiter - > flags & RWSEM_WAITING_FOR_WRITE ) {
2013-02-01 18:59:16 +08:00
wake_up_process ( waiter - > task ) ;
2005-04-16 15:20:36 -07:00
goto out ;
}
/* grant an infinite number of read locks to the front of the queue */
dont_wake_writers :
woken = 0 ;
while ( waiter - > flags & RWSEM_WAITING_FOR_READ ) {
struct list_head * next = waiter - > list . next ;
list_del ( & waiter - > list ) ;
tsk = waiter - > task ;
2005-05-01 08:58:47 -07:00
smp_mb ( ) ;
2005-04-16 15:20:36 -07:00
waiter - > task = NULL ;
wake_up_process ( tsk ) ;
put_task_struct ( tsk ) ;
woken + + ;
if ( list_empty ( & sem - > wait_list ) )
break ;
waiter = list_entry ( next , struct rwsem_waiter , list ) ;
}
sem - > activity + = woken ;
out :
return sem ;
}
/*
* wake a single writer
*/
static inline struct rw_semaphore *
__rwsem_wake_one_writer ( struct rw_semaphore * sem )
{
struct rwsem_waiter * waiter ;
waiter = list_entry ( sem - > wait_list . next , struct rwsem_waiter , list ) ;
2013-02-01 18:59:16 +08:00
wake_up_process ( waiter - > task ) ;
2005-04-16 15:20:36 -07:00
return sem ;
}
/*
* get a read lock on the semaphore
*/
2008-02-08 04:19:55 -08:00
void __sched __down_read ( struct rw_semaphore * sem )
2005-04-16 15:20:36 -07:00
{
struct rwsem_waiter waiter ;
struct task_struct * tsk ;
2010-04-07 11:52:46 -07:00
unsigned long flags ;
2005-04-16 15:20:36 -07:00
2010-02-24 09:54:54 +01:00
raw_spin_lock_irqsave ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
if ( sem - > activity > = 0 & & list_empty ( & sem - > wait_list ) ) {
/* granted */
sem - > activity + + ;
2010-02-24 09:54:54 +01:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
goto out ;
}
tsk = current ;
set_task_state ( tsk , TASK_UNINTERRUPTIBLE ) ;
/* set up my own style of waitqueue */
waiter . task = tsk ;
waiter . flags = RWSEM_WAITING_FOR_READ ;
get_task_struct ( tsk ) ;
list_add_tail ( & waiter . list , & sem - > wait_list ) ;
/* we don't need to touch the semaphore struct anymore */
2010-02-24 09:54:54 +01:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
/* wait to be given the lock */
for ( ; ; ) {
if ( ! waiter . task )
break ;
schedule ( ) ;
set_task_state ( tsk , TASK_UNINTERRUPTIBLE ) ;
}
tsk - > state = TASK_RUNNING ;
out :
2006-07-03 00:24:29 -07:00
;
2005-04-16 15:20:36 -07:00
}
/*
* trylock for reading - - returns 1 if successful , 0 if contention
*/
2008-02-08 04:19:55 -08:00
int __down_read_trylock ( struct rw_semaphore * sem )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
int ret = 0 ;
2010-02-24 09:54:54 +01:00
raw_spin_lock_irqsave ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
if ( sem - > activity > = 0 & & list_empty ( & sem - > wait_list ) ) {
/* granted */
sem - > activity + + ;
ret = 1 ;
}
2010-02-24 09:54:54 +01:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
return ret ;
}
/*
* get a write lock on the semaphore
*/
2008-02-08 04:19:55 -08:00
void __sched __down_write_nested ( struct rw_semaphore * sem , int subclass )
2005-04-16 15:20:36 -07:00
{
struct rwsem_waiter waiter ;
struct task_struct * tsk ;
2010-04-07 11:52:46 -07:00
unsigned long flags ;
2005-04-16 15:20:36 -07:00
2010-02-24 09:54:54 +01:00
raw_spin_lock_irqsave ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
/* set up my own style of waitqueue */
2013-02-01 18:59:16 +08:00
tsk = current ;
2005-04-16 15:20:36 -07:00
waiter . task = tsk ;
waiter . flags = RWSEM_WAITING_FOR_WRITE ;
list_add_tail ( & waiter . list , & sem - > wait_list ) ;
2013-02-01 18:59:16 +08:00
/* wait for someone to release the lock */
2005-04-16 15:20:36 -07:00
for ( ; ; ) {
2013-02-01 18:59:16 +08:00
/*
* That is the key to support write lock stealing : allows the
* task already on CPU to get the lock soon rather than put
* itself into sleep and waiting for system woke it or someone
* else in the head of the wait list up .
*/
if ( sem - > activity = = 0 )
2005-04-16 15:20:36 -07:00
break ;
set_task_state ( tsk , TASK_UNINTERRUPTIBLE ) ;
2013-02-01 18:59:16 +08:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
schedule ( ) ;
raw_spin_lock_irqsave ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
}
2013-02-01 18:59:16 +08:00
/* got the lock */
sem - > activity = - 1 ;
list_del ( & waiter . list ) ;
2005-04-16 15:20:36 -07:00
2013-02-01 18:59:16 +08:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
}
2008-02-08 04:19:55 -08:00
void __sched __down_write ( struct rw_semaphore * sem )
2006-07-03 00:24:53 -07:00
{
__down_write_nested ( sem , 0 ) ;
}
2005-04-16 15:20:36 -07:00
/*
* trylock for writing - - returns 1 if successful , 0 if contention
*/
2008-02-08 04:19:55 -08:00
int __down_write_trylock ( struct rw_semaphore * sem )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
int ret = 0 ;
2010-02-24 09:54:54 +01:00
raw_spin_lock_irqsave ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
2013-02-01 18:59:16 +08:00
if ( sem - > activity = = 0 ) {
/* got the lock */
2005-04-16 15:20:36 -07:00
sem - > activity = - 1 ;
ret = 1 ;
}
2010-02-24 09:54:54 +01:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
return ret ;
}
/*
* release a read lock on the semaphore
*/
2008-02-08 04:19:55 -08:00
void __up_read ( struct rw_semaphore * sem )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2010-02-24 09:54:54 +01:00
raw_spin_lock_irqsave ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
if ( - - sem - > activity = = 0 & & ! list_empty ( & sem - > wait_list ) )
sem = __rwsem_wake_one_writer ( sem ) ;
2010-02-24 09:54:54 +01:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
}
/*
* release a write lock on the semaphore
*/
2008-02-08 04:19:55 -08:00
void __up_write ( struct rw_semaphore * sem )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2010-02-24 09:54:54 +01:00
raw_spin_lock_irqsave ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
sem - > activity = 0 ;
if ( ! list_empty ( & sem - > wait_list ) )
sem = __rwsem_do_wake ( sem , 1 ) ;
2010-02-24 09:54:54 +01:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
}
/*
* downgrade a write lock into a read lock
* - just wake up any readers at the front of the queue
*/
2008-02-08 04:19:55 -08:00
void __downgrade_write ( struct rw_semaphore * sem )
2005-04-16 15:20:36 -07:00
{
unsigned long flags ;
2010-02-24 09:54:54 +01:00
raw_spin_lock_irqsave ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
sem - > activity = 1 ;
if ( ! list_empty ( & sem - > wait_list ) )
sem = __rwsem_do_wake ( sem , 0 ) ;
2010-02-24 09:54:54 +01:00
raw_spin_unlock_irqrestore ( & sem - > wait_lock , flags ) ;
2005-04-16 15:20:36 -07:00
}