2005-04-16 15:20:36 -07:00
/*
* linux / fs / lockd / clntlock . c
*
* Lock handling for the client side NLM implementation
*
* Copyright ( C ) 1996 , Olaf Kirch < okir @ monad . swb . de >
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/time.h>
# include <linux/nfs_fs.h>
# include <linux/sunrpc/clnt.h>
# include <linux/sunrpc/svc.h>
# include <linux/lockd/lockd.h>
# include <linux/smp_lock.h>
# define NLMDBG_FACILITY NLMDBG_CLIENT
/*
* Local function prototypes
*/
static int reclaimer ( void * ptr ) ;
/*
* The following functions handle blocking and granting from the
* client perspective .
*/
/*
* This is the representation of a blocked client lock .
*/
struct nlm_wait {
2005-06-22 17:16:31 +00:00
struct list_head b_list ; /* linked list */
2005-04-16 15:20:36 -07:00
wait_queue_head_t b_wait ; /* where to wait on */
struct nlm_host * b_host ;
struct file_lock * b_lock ; /* local file lock */
unsigned short b_reclaim ; /* got to reclaim lock */
2006-12-13 00:35:03 -08:00
__be32 b_status ; /* grant callback status */
2005-04-16 15:20:36 -07:00
} ;
2005-06-22 17:16:31 +00:00
static LIST_HEAD ( nlm_blocked ) ;
2005-04-16 15:20:36 -07:00
2008-01-11 17:09:44 -05:00
/**
* nlmclnt_init - Set up per - NFS mount point lockd data structures
2008-01-15 16:04:20 -05:00
* @ nlm_init : pointer to arguments structure
2008-01-11 17:09:44 -05:00
*
* Returns pointer to an appropriate nlm_host struct ,
* or an ERR_PTR value .
*/
2008-01-15 16:04:20 -05:00
struct nlm_host * nlmclnt_init ( const struct nlmclnt_initdata * nlm_init )
2008-01-11 17:09:44 -05:00
{
struct nlm_host * host ;
2008-01-15 16:04:20 -05:00
u32 nlm_version = ( nlm_init - > nfs_version = = 2 ) ? 1 : 4 ;
2008-01-11 17:09:44 -05:00
int status ;
2008-01-15 16:04:20 -05:00
status = lockd_up ( nlm_init - > protocol ) ;
2008-01-11 17:09:44 -05:00
if ( status < 0 )
return ERR_PTR ( status ) ;
2008-01-15 16:04:20 -05:00
host = nlmclnt_lookup_host ( ( struct sockaddr_in * ) nlm_init - > address ,
nlm_init - > protocol , nlm_version ,
nlm_init - > hostname ,
strlen ( nlm_init - > hostname ) ) ;
2008-01-11 17:09:44 -05:00
if ( host = = NULL ) {
lockd_down ( ) ;
return ERR_PTR ( - ENOLCK ) ;
}
return host ;
}
EXPORT_SYMBOL_GPL ( nlmclnt_init ) ;
/**
* nlmclnt_done - Release resources allocated by nlmclnt_init ( )
* @ host : nlm_host structure reserved by nlmclnt_init ( )
*
*/
void nlmclnt_done ( struct nlm_host * host )
{
nlm_release_host ( host ) ;
lockd_down ( ) ;
}
EXPORT_SYMBOL_GPL ( nlmclnt_done ) ;
2005-04-16 15:20:36 -07:00
/*
2005-06-22 17:16:31 +00:00
* Queue up a lock for blocking so that the GRANTED request can see it
2005-04-16 15:20:36 -07:00
*/
2006-03-20 13:44:44 -05:00
struct nlm_wait * nlmclnt_prepare_block ( struct nlm_host * host , struct file_lock * fl )
2005-06-22 17:16:31 +00:00
{
struct nlm_wait * block ;
block = kmalloc ( sizeof ( * block ) , GFP_KERNEL ) ;
2006-03-20 13:44:44 -05:00
if ( block ! = NULL ) {
block - > b_host = host ;
block - > b_lock = fl ;
init_waitqueue_head ( & block - > b_wait ) ;
2006-12-13 00:35:03 -08:00
block - > b_status = nlm_lck_blocked ;
2006-03-20 13:44:44 -05:00
list_add ( & block - > b_list , & nlm_blocked ) ;
}
return block ;
2005-06-22 17:16:31 +00:00
}
2006-03-20 13:44:44 -05:00
void nlmclnt_finish_block ( struct nlm_wait * block )
2005-04-16 15:20:36 -07:00
{
2005-06-22 17:16:31 +00:00
if ( block = = NULL )
return ;
list_del ( & block - > b_list ) ;
kfree ( block ) ;
}
2005-04-16 15:20:36 -07:00
2005-06-22 17:16:31 +00:00
/*
* Block on a lock
*/
2006-03-20 13:44:44 -05:00
int nlmclnt_block ( struct nlm_wait * block , struct nlm_rqst * req , long timeout )
2005-06-22 17:16:31 +00:00
{
long ret ;
2005-04-16 15:20:36 -07:00
2005-06-22 17:16:31 +00:00
/* A borken server might ask us to block even if we didn't
* request it . Just say no !
*/
2006-03-20 13:44:44 -05:00
if ( block = = NULL )
2005-06-22 17:16:31 +00:00
return - EAGAIN ;
2005-04-16 15:20:36 -07:00
/* Go to sleep waiting for GRANT callback. Some servers seem
* to lose callbacks , however , so we ' re going to poll from
* time to time just to make sure .
*
* For now , the retry frequency is pretty high ; normally
* a 1 minute timeout would do . See the comment before
* nlmclnt_lock for an explanation .
*/
2005-06-22 17:16:31 +00:00
ret = wait_event_interruptible_timeout ( block - > b_wait ,
2006-12-13 00:35:03 -08:00
block - > b_status ! = nlm_lck_blocked ,
2005-06-22 17:16:31 +00:00
timeout ) ;
2006-03-20 13:44:44 -05:00
if ( ret < 0 )
return - ERESTARTSYS ;
req - > a_res . status = block - > b_status ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
/*
* The server lockd has called us back to tell us the lock was granted
*/
2006-10-19 23:28:46 -07:00
__be32 nlmclnt_grant ( const struct sockaddr_in * addr , const struct nlm_lock * lock )
2005-04-16 15:20:36 -07:00
{
2006-02-14 13:53:04 -08:00
const struct file_lock * fl = & lock - > fl ;
const struct nfs_fh * fh = & lock - > fh ;
2005-04-16 15:20:36 -07:00
struct nlm_wait * block ;
2006-10-19 23:28:46 -07:00
__be32 res = nlm_lck_denied ;
2005-04-16 15:20:36 -07:00
/*
* Look up blocked request based on arguments .
* Warning : must not use cookie to match it !
*/
2005-06-22 17:16:31 +00:00
list_for_each_entry ( block , & nlm_blocked , b_list ) {
2006-02-14 13:53:04 -08:00
struct file_lock * fl_blocked = block - > b_lock ;
2006-03-20 13:44:06 -05:00
if ( fl_blocked - > fl_start ! = fl - > fl_start )
continue ;
if ( fl_blocked - > fl_end ! = fl - > fl_end )
continue ;
/*
* Careful ! The NLM server will return the 32 - bit " pid " that
* we put on the wire : in this case the lockowner " pid " .
*/
if ( fl_blocked - > fl_u . nfs_fl . owner - > pid ! = lock - > svid )
2006-02-14 13:53:04 -08:00
continue ;
if ( ! nlm_cmp_addr ( & block - > b_host - > h_addr , addr ) )
continue ;
2006-12-08 02:37:18 -08:00
if ( nfs_compare_fh ( NFS_FH ( fl_blocked - > fl_file - > f_path . dentry - > d_inode ) , fh ) ! = 0 )
2006-02-14 13:53:04 -08:00
continue ;
/* Alright, we found a lock. Set the return status
* and wake up the caller
*/
2006-12-13 00:35:03 -08:00
block - > b_status = nlm_granted ;
2006-02-14 13:53:04 -08:00
wake_up ( & block - > b_wait ) ;
res = nlm_granted ;
2005-04-16 15:20:36 -07:00
}
2005-06-22 17:16:31 +00:00
return res ;
2005-04-16 15:20:36 -07:00
}
/*
* The following procedures deal with the recovery of locks after a
* server crash .
*/
/*
* Reclaim all locks on server host . We do this by spawning a separate
* reclaimer thread .
*/
void
2006-10-04 02:15:55 -07:00
nlmclnt_recovery ( struct nlm_host * host )
2005-04-16 15:20:36 -07:00
{
2006-06-09 09:40:27 -04:00
if ( ! host - > h_reclaiming + + ) {
2005-04-16 15:20:36 -07:00
nlm_get_host ( host ) ;
__module_get ( THIS_MODULE ) ;
2007-05-10 22:55:07 -07:00
if ( kernel_thread ( reclaimer , host , CLONE_FS | CLONE_FILES ) < 0 )
2005-04-16 15:20:36 -07:00
module_put ( THIS_MODULE ) ;
}
}
static int
reclaimer ( void * ptr )
{
struct nlm_host * host = ( struct nlm_host * ) ptr ;
struct nlm_wait * block ;
2006-03-20 13:44:40 -05:00
struct file_lock * fl , * next ;
2006-06-09 09:40:27 -04:00
u32 nsmstate ;
2005-04-16 15:20:36 -07:00
daemonize ( " %s-reclaim " , host - > h_name ) ;
allow_signal ( SIGKILL ) ;
2006-10-04 02:15:55 -07:00
down_write ( & host - > h_rwsem ) ;
2005-04-16 15:20:36 -07:00
/* This one ensures that our parent doesn't terminate while the
* reclaim is in progress */
lock_kernel ( ) ;
2006-10-02 02:17:53 -07:00
lockd_up ( 0 ) ; /* note: this cannot fail as lockd is already running */
2005-04-16 15:20:36 -07:00
2007-01-29 13:19:51 -08:00
dprintk ( " lockd: reclaiming locks for host %s \n " , host - > h_name ) ;
2006-10-04 02:15:55 -07:00
2005-04-16 15:20:36 -07:00
restart :
2006-06-09 09:40:27 -04:00
nsmstate = host - > h_nsmstate ;
2006-10-04 02:15:55 -07:00
/* Force a portmap getport - the peer's lockd will
* most likely end up on a different port .
*/
2006-10-04 02:16:04 -07:00
host - > h_nextrebind = jiffies ;
2006-10-04 02:15:55 -07:00
nlm_rebind_host ( host ) ;
/* First, reclaim all locks that have been granted. */
list_splice_init ( & host - > h_granted , & host - > h_reclaim ) ;
2006-03-20 13:44:40 -05:00
list_for_each_entry_safe ( fl , next , & host - > h_reclaim , fl_u . nfs_fl . list ) {
2006-03-20 13:44:41 -05:00
list_del_init ( & fl - > fl_u . nfs_fl . list ) ;
2005-04-16 15:20:36 -07:00
2006-10-04 02:15:55 -07:00
/* Why are we leaking memory here? --okir */
2005-04-16 15:20:36 -07:00
if ( signalled ( ) )
2006-03-20 13:44:41 -05:00
continue ;
2006-06-09 09:40:27 -04:00
if ( nlmclnt_reclaim ( host , fl ) ! = 0 )
continue ;
list_add_tail ( & fl - > fl_u . nfs_fl . list , & host - > h_granted ) ;
if ( host - > h_nsmstate ! = nsmstate ) {
/* Argh! The server rebooted again! */
goto restart ;
}
2005-04-16 15:20:36 -07:00
}
2006-10-04 02:15:55 -07:00
host - > h_reclaiming = 0 ;
up_write ( & host - > h_rwsem ) ;
2007-01-29 13:19:51 -08:00
dprintk ( " NLM: done reclaiming locks for host %s \n " , host - > h_name ) ;
2005-04-16 15:20:36 -07:00
/* Now, wake up all processes that sleep on a blocked lock */
2005-06-22 17:16:31 +00:00
list_for_each_entry ( block , & nlm_blocked , b_list ) {
2005-04-16 15:20:36 -07:00
if ( block - > b_host = = host ) {
2006-12-13 00:35:03 -08:00
block - > b_status = nlm_lck_denied_grace_period ;
2005-04-16 15:20:36 -07:00
wake_up ( & block - > b_wait ) ;
}
}
/* Release host handle after use */
nlm_release_host ( host ) ;
lockd_down ( ) ;
unlock_kernel ( ) ;
module_put_and_exit ( 0 ) ;
}