2014-02-03 16:18:49 +04:00
/*
* Queue read / write lock
*
* 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 .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* ( C ) Copyright 2013 - 2014 Hewlett - Packard Development Company , L . P .
*
* Authors : Waiman Long < waiman . long @ hp . com >
*/
# ifndef __ASM_GENERIC_QRWLOCK_H
# define __ASM_GENERIC_QRWLOCK_H
# include <linux/atomic.h>
# include <asm/barrier.h>
# include <asm/processor.h>
# include <asm-generic/qrwlock_types.h>
/*
2016-07-18 12:47:39 +03:00
* Writer states & reader shift and bias .
*
* | + 0 | + 1 | + 2 | + 3 |
* - - - - + - - - - + - - - - + - - - - + - - - - +
* LE | 78 | 56 | 34 | 12 | 0x12345678
* - - - - + - - - - + - - - - + - - - - + - - - - +
* | wr | rd |
* + - - - - + - - - - + - - - - + - - - - +
*
* - - - - + - - - - + - - - - + - - - - + - - - - +
* BE | 12 | 34 | 56 | 78 | 0x12345678
* - - - - + - - - - + - - - - + - - - - + - - - - +
* | rd | wr |
* + - - - - + - - - - + - - - - + - - - - +
2014-02-03 16:18:49 +04:00
*/
# define _QW_WAITING 1 /* A writer is waiting */
# define _QW_LOCKED 0xff /* A writer holds the lock */
# define _QW_WMASK 0xff /* Writer mask */
# define _QR_SHIFT 8 /* Reader count shift */
# define _QR_BIAS (1U << _QR_SHIFT)
/*
* External function declarations
*/
2015-06-19 18:50:01 +03:00
extern void queued_read_lock_slowpath ( struct qrwlock * lock , u32 cnts ) ;
2015-06-19 18:50:00 +03:00
extern void queued_write_lock_slowpath ( struct qrwlock * lock ) ;
2014-02-03 16:18:49 +04:00
/**
2015-06-19 18:50:00 +03:00
* queued_read_can_lock - would read_trylock ( ) succeed ?
2014-02-03 16:18:49 +04:00
* @ lock : Pointer to queue rwlock structure
*/
2015-06-19 18:50:00 +03:00
static inline int queued_read_can_lock ( struct qrwlock * lock )
2014-02-03 16:18:49 +04:00
{
return ! ( atomic_read ( & lock - > cnts ) & _QW_WMASK ) ;
}
/**
2015-06-19 18:50:00 +03:00
* queued_write_can_lock - would write_trylock ( ) succeed ?
2014-02-03 16:18:49 +04:00
* @ lock : Pointer to queue rwlock structure
*/
2015-06-19 18:50:00 +03:00
static inline int queued_write_can_lock ( struct qrwlock * lock )
2014-02-03 16:18:49 +04:00
{
return ! atomic_read ( & lock - > cnts ) ;
}
/**
2015-06-19 18:50:00 +03:00
* queued_read_trylock - try to acquire read lock of a queue rwlock
2014-02-03 16:18:49 +04:00
* @ lock : Pointer to queue rwlock structure
* Return : 1 if lock acquired , 0 if failed
*/
2015-06-19 18:50:00 +03:00
static inline int queued_read_trylock ( struct qrwlock * lock )
2014-02-03 16:18:49 +04:00
{
u32 cnts ;
cnts = atomic_read ( & lock - > cnts ) ;
if ( likely ( ! ( cnts & _QW_WMASK ) ) ) {
2015-08-06 19:54:42 +03:00
cnts = ( u32 ) atomic_add_return_acquire ( _QR_BIAS , & lock - > cnts ) ;
2014-02-03 16:18:49 +04:00
if ( likely ( ! ( cnts & _QW_WMASK ) ) )
return 1 ;
atomic_sub ( _QR_BIAS , & lock - > cnts ) ;
}
return 0 ;
}
/**
2015-06-19 18:50:00 +03:00
* queued_write_trylock - try to acquire write lock of a queue rwlock
2014-02-03 16:18:49 +04:00
* @ lock : Pointer to queue rwlock structure
* Return : 1 if lock acquired , 0 if failed
*/
2015-06-19 18:50:00 +03:00
static inline int queued_write_trylock ( struct qrwlock * lock )
2014-02-03 16:18:49 +04:00
{
u32 cnts ;
cnts = atomic_read ( & lock - > cnts ) ;
if ( unlikely ( cnts ) )
return 0 ;
2015-08-06 19:54:42 +03:00
return likely ( atomic_cmpxchg_acquire ( & lock - > cnts ,
cnts , cnts | _QW_LOCKED ) = = cnts ) ;
2014-02-03 16:18:49 +04:00
}
/**
2015-06-19 18:50:00 +03:00
* queued_read_lock - acquire read lock of a queue rwlock
2014-02-03 16:18:49 +04:00
* @ lock : Pointer to queue rwlock structure
*/
2015-06-19 18:50:00 +03:00
static inline void queued_read_lock ( struct qrwlock * lock )
2014-02-03 16:18:49 +04:00
{
u32 cnts ;
2015-08-06 19:54:42 +03:00
cnts = atomic_add_return_acquire ( _QR_BIAS , & lock - > cnts ) ;
2014-02-03 16:18:49 +04:00
if ( likely ( ! ( cnts & _QW_WMASK ) ) )
return ;
/* The slowpath will decrement the reader count, if necessary. */
2015-06-19 18:50:01 +03:00
queued_read_lock_slowpath ( lock , cnts ) ;
2014-02-03 16:18:49 +04:00
}
/**
2015-06-19 18:50:00 +03:00
* queued_write_lock - acquire write lock of a queue rwlock
2014-02-03 16:18:49 +04:00
* @ lock : Pointer to queue rwlock structure
*/
2015-06-19 18:50:00 +03:00
static inline void queued_write_lock ( struct qrwlock * lock )
2014-02-03 16:18:49 +04:00
{
/* Optimize for the unfair lock case where the fair flag is 0. */
2015-08-06 19:54:42 +03:00
if ( atomic_cmpxchg_acquire ( & lock - > cnts , 0 , _QW_LOCKED ) = = 0 )
2014-02-03 16:18:49 +04:00
return ;
2015-06-19 18:50:00 +03:00
queued_write_lock_slowpath ( lock ) ;
2014-02-03 16:18:49 +04:00
}
/**
2015-06-19 18:50:00 +03:00
* queued_read_unlock - release read lock of a queue rwlock
2014-02-03 16:18:49 +04:00
* @ lock : Pointer to queue rwlock structure
*/
2015-06-19 18:50:00 +03:00
static inline void queued_read_unlock ( struct qrwlock * lock )
2014-02-03 16:18:49 +04:00
{
/*
* Atomically decrement the reader count
*/
2015-08-06 19:54:42 +03:00
( void ) atomic_sub_return_release ( _QR_BIAS , & lock - > cnts ) ;
2014-02-03 16:18:49 +04:00
}
2016-07-18 12:47:39 +03:00
/**
* __qrwlock_write_byte - retrieve the write byte address of a queue rwlock
* @ lock : Pointer to queue rwlock structure
* Return : the write byte address of a queue rwlock
*/
static inline u8 * __qrwlock_write_byte ( struct qrwlock * lock )
{
return ( u8 * ) lock + 3 * IS_BUILTIN ( CONFIG_CPU_BIG_ENDIAN ) ;
}
2014-02-03 16:18:49 +04:00
/**
2015-06-19 18:50:00 +03:00
* queued_write_unlock - release write lock of a queue rwlock
2014-02-03 16:18:49 +04:00
* @ lock : Pointer to queue rwlock structure
*/
2015-06-19 18:50:00 +03:00
static inline void queued_write_unlock ( struct qrwlock * lock )
2014-02-03 16:18:49 +04:00
{
2016-07-18 12:47:39 +03:00
smp_store_release ( __qrwlock_write_byte ( lock ) , 0 ) ;
2014-02-03 16:18:49 +04:00
}
/*
* Remapping rwlock architecture specific functions to the corresponding
* queue rwlock functions .
*/
2015-06-19 18:50:00 +03:00
# define arch_read_can_lock(l) queued_read_can_lock(l)
# define arch_write_can_lock(l) queued_write_can_lock(l)
# define arch_read_lock(l) queued_read_lock(l)
# define arch_write_lock(l) queued_write_lock(l)
# define arch_read_trylock(l) queued_read_trylock(l)
# define arch_write_trylock(l) queued_write_trylock(l)
# define arch_read_unlock(l) queued_read_unlock(l)
# define arch_write_unlock(l) queued_write_unlock(l)
2014-02-03 16:18:49 +04:00
# endif /* __ASM_GENERIC_QRWLOCK_H */