2005-04-17 02:20:36 +04:00
/*
* linux / fs / lockd / svcproc . c
*
* Lockd server procedures . We don ' t implement the NLM_ * _RES
* procedures because we don ' t use the async procedures .
*
* Copyright ( C ) 1996 , Olaf Kirch < okir @ monad . swb . de >
*/
# include <linux/types.h>
# include <linux/time.h>
# include <linux/lockd/lockd.h>
# include <linux/lockd/share.h>
2012-07-25 16:57:22 +04:00
# include <linux/sunrpc/svc_xprt.h>
2005-04-17 02:20:36 +04:00
# define NLMDBG_FACILITY NLMDBG_CLIENT
# ifdef CONFIG_LOCKD_V4
2006-10-20 10:28:46 +04:00
static __be32
cast_to_nlm ( __be32 status , u32 vers )
2005-04-17 02:20:36 +04:00
{
/* Note: status is assumed to be in network byte order !!! */
if ( vers ! = 4 ) {
switch ( status ) {
case nlm_granted :
case nlm_lck_denied :
case nlm_lck_denied_nolocks :
case nlm_lck_blocked :
case nlm_lck_denied_grace_period :
2006-11-29 00:27:06 +03:00
case nlm_drop_reply :
2005-04-17 02:20:36 +04:00
break ;
case nlm4_deadlock :
status = nlm_lck_denied ;
break ;
default :
status = nlm_lck_denied_nolocks ;
}
}
return ( status ) ;
}
# define cast_status(status) (cast_to_nlm(status, rqstp->rq_vers))
# else
# define cast_status(status) (status)
# endif
/*
* Obtain client and file from arguments
*/
2006-10-20 10:28:46 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_retrieve_args ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_host * * hostp , struct nlm_file * * filp )
{
struct nlm_host * host = NULL ;
struct nlm_file * file = NULL ;
struct nlm_lock * lock = & argp - > lock ;
2006-10-20 10:28:46 +04:00
__be32 error = 0 ;
2005-04-17 02:20:36 +04:00
/* nfsd callbacks must have been installed for this procedure */
if ( ! nlmsvc_ops )
return nlm_lck_denied_nolocks ;
/* Obtain host handle */
2006-10-04 13:15:52 +04:00
if ( ! ( host = nlmsvc_lookup_host ( rqstp , lock - > caller , lock - > len ) )
2006-10-04 13:15:51 +04:00
| | ( argp - > monitor & & nsm_monitor ( host ) < 0 ) )
2005-04-17 02:20:36 +04:00
goto no_locks ;
* hostp = host ;
/* Obtain file pointer. Not used by FREE_ALL call. */
if ( filp ! = NULL ) {
2012-10-13 08:30:28 +04:00
error = cast_status ( nlm_lookup_file ( rqstp , & file , & lock - > fh ) ) ;
if ( error ! = 0 )
2005-04-17 02:20:36 +04:00
goto no_locks ;
* filp = file ;
/* Set up the missing parts of the file_lock structure */
lock - > fl . fl_file = file - > f_file ;
lock - > fl . fl_owner = ( fl_owner_t ) host ;
lock - > fl . fl_lmops = & nlmsvc_lock_operations ;
}
return 0 ;
no_locks :
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( host ) ;
2006-10-17 11:10:18 +04:00
if ( error )
return error ;
2005-04-17 02:20:36 +04:00
return nlm_lck_denied_nolocks ;
}
/*
* NULL : Test for presence of service
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_null ( struct svc_rqst * rqstp , void * argp , void * resp )
{
dprintk ( " lockd: NULL called \n " ) ;
return rpc_success ;
}
/*
* TEST : Check for conflicting lock
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_test ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_res * resp )
{
struct nlm_host * host ;
struct nlm_file * file ;
2008-07-21 10:41:24 +04:00
__be32 rc = rpc_success ;
2005-04-17 02:20:36 +04:00
dprintk ( " lockd: TEST called \n " ) ;
resp - > cookie = argp - > cookie ;
/* Obtain client and file */
if ( ( resp - > status = nlmsvc_retrieve_args ( rqstp , argp , & host , & file ) ) )
2006-10-17 11:10:18 +04:00
return resp - > status = = nlm_drop_reply ? rpc_drop_reply : rpc_success ;
2005-04-17 02:20:36 +04:00
/* Now check for conflicting locks */
2008-07-15 22:06:48 +04:00
resp - > status = cast_status ( nlmsvc_testlock ( rqstp , file , host , & argp - > lock , & resp - > lock , & resp - > cookie ) ) ;
2006-11-29 00:27:06 +03:00
if ( resp - > status = = nlm_drop_reply )
2007-11-26 21:35:11 +03:00
rc = rpc_drop_reply ;
else
dprintk ( " lockd: TEST status %d vers %d \n " ,
ntohl ( resp - > status ) , rqstp - > rq_vers ) ;
2005-04-17 02:20:36 +04:00
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( host ) ;
2005-04-17 02:20:36 +04:00
nlm_release_file ( file ) ;
2007-11-26 21:35:11 +03:00
return rc ;
2005-04-17 02:20:36 +04:00
}
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_lock ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_res * resp )
{
struct nlm_host * host ;
struct nlm_file * file ;
2008-07-21 10:41:24 +04:00
__be32 rc = rpc_success ;
2005-04-17 02:20:36 +04:00
dprintk ( " lockd: LOCK called \n " ) ;
resp - > cookie = argp - > cookie ;
/* Obtain client and file */
if ( ( resp - > status = nlmsvc_retrieve_args ( rqstp , argp , & host , & file ) ) )
2006-10-17 11:10:18 +04:00
return resp - > status = = nlm_drop_reply ? rpc_drop_reply : rpc_success ;
2005-04-17 02:20:36 +04:00
#if 0
/* If supplied state doesn't match current state, we assume it's
* an old request that time - warped somehow . Any error return would
* do in this case because it ' s irrelevant anyway .
*
* NB : We don ' t retrieve the remote host ' s state yet .
*/
if ( host - > h_nsmstate & & host - > h_nsmstate ! = argp - > state ) {
resp - > status = nlm_lck_denied_nolocks ;
} else
# endif
/* Now try to lock the file */
2008-07-15 22:26:17 +04:00
resp - > status = cast_status ( nlmsvc_lock ( rqstp , file , host , & argp - > lock ,
2008-02-06 21:59:23 +03:00
argp - > block , & argp - > cookie ,
argp - > reclaim ) ) ;
2006-11-29 00:27:06 +03:00
if ( resp - > status = = nlm_drop_reply )
2007-11-26 21:35:11 +03:00
rc = rpc_drop_reply ;
else
dprintk ( " lockd: LOCK status %d \n " , ntohl ( resp - > status ) ) ;
2005-04-17 02:20:36 +04:00
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( host ) ;
2005-04-17 02:20:36 +04:00
nlm_release_file ( file ) ;
2007-11-26 21:35:11 +03:00
return rc ;
2005-04-17 02:20:36 +04:00
}
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_cancel ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_res * resp )
{
struct nlm_host * host ;
struct nlm_file * file ;
2012-07-25 16:57:22 +04:00
struct net * net = SVC_NET ( rqstp ) ;
2005-04-17 02:20:36 +04:00
dprintk ( " lockd: CANCEL called \n " ) ;
resp - > cookie = argp - > cookie ;
/* Don't accept requests during grace period */
2012-07-25 16:57:22 +04:00
if ( locks_in_grace ( net ) ) {
2005-04-17 02:20:36 +04:00
resp - > status = nlm_lck_denied_grace_period ;
return rpc_success ;
}
/* Obtain client and file */
if ( ( resp - > status = nlmsvc_retrieve_args ( rqstp , argp , & host , & file ) ) )
2006-10-17 11:10:18 +04:00
return resp - > status = = nlm_drop_reply ? rpc_drop_reply : rpc_success ;
2005-04-17 02:20:36 +04:00
/* Try to cancel request. */
2012-07-25 16:57:22 +04:00
resp - > status = cast_status ( nlmsvc_cancel_blocked ( net , file , & argp - > lock ) ) ;
2005-04-17 02:20:36 +04:00
dprintk ( " lockd: CANCEL status %d \n " , ntohl ( resp - > status ) ) ;
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( host ) ;
2005-04-17 02:20:36 +04:00
nlm_release_file ( file ) ;
return rpc_success ;
}
/*
* UNLOCK : release a lock
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_unlock ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_res * resp )
{
struct nlm_host * host ;
struct nlm_file * file ;
2012-07-25 16:57:22 +04:00
struct net * net = SVC_NET ( rqstp ) ;
2005-04-17 02:20:36 +04:00
dprintk ( " lockd: UNLOCK called \n " ) ;
resp - > cookie = argp - > cookie ;
/* Don't accept new lock requests during grace period */
2012-07-25 16:57:22 +04:00
if ( locks_in_grace ( net ) ) {
2005-04-17 02:20:36 +04:00
resp - > status = nlm_lck_denied_grace_period ;
return rpc_success ;
}
/* Obtain client and file */
if ( ( resp - > status = nlmsvc_retrieve_args ( rqstp , argp , & host , & file ) ) )
2006-10-17 11:10:18 +04:00
return resp - > status = = nlm_drop_reply ? rpc_drop_reply : rpc_success ;
2005-04-17 02:20:36 +04:00
/* Now try to remove the lock */
2012-07-25 16:57:22 +04:00
resp - > status = cast_status ( nlmsvc_unlock ( net , file , & argp - > lock ) ) ;
2005-04-17 02:20:36 +04:00
dprintk ( " lockd: UNLOCK status %d \n " , ntohl ( resp - > status ) ) ;
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( host ) ;
2005-04-17 02:20:36 +04:00
nlm_release_file ( file ) ;
return rpc_success ;
}
/*
* GRANTED : A server calls us to tell that a process ' lock request
* was granted
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_granted ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_res * resp )
{
resp - > cookie = argp - > cookie ;
dprintk ( " lockd: GRANTED called \n " ) ;
2008-10-03 20:50:36 +04:00
resp - > status = nlmclnt_grant ( svc_addr ( rqstp ) , & argp - > lock ) ;
2005-04-17 02:20:36 +04:00
dprintk ( " lockd: GRANTED status %d \n " , ntohl ( resp - > status ) ) ;
return rpc_success ;
}
2006-03-20 21:44:45 +03:00
/*
* This is the generic lockd callback for async RPC calls
*/
static void nlmsvc_callback_exit ( struct rpc_task * task , void * data )
{
2006-12-06 00:36:03 +03:00
dprintk ( " lockd: %5u callback returned %d \n " , task - > tk_pid ,
2006-03-20 21:44:45 +03:00
- task - > tk_status ) ;
}
2010-12-14 18:05:42 +03:00
void nlmsvc_release_call ( struct nlm_rqst * call )
{
if ( ! atomic_dec_and_test ( & call - > a_count ) )
return ;
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( call - > a_host ) ;
2010-12-14 18:05:42 +03:00
kfree ( call ) ;
}
2006-03-20 21:44:45 +03:00
static void nlmsvc_callback_release ( void * data )
{
2010-12-14 18:05:42 +03:00
nlmsvc_release_call ( data ) ;
2006-03-20 21:44:45 +03:00
}
static const struct rpc_call_ops nlmsvc_callback_ops = {
. rpc_call_done = nlmsvc_callback_exit ,
. rpc_release = nlmsvc_callback_release ,
} ;
2005-04-17 02:20:36 +04:00
/*
* ` Async ' versions of the above service routines . They aren ' t really ,
* because we send the callback before the reply proper . I hope this
* doesn ' t break any clients .
*/
2006-10-20 10:28:45 +04:00
static __be32 nlmsvc_callback ( struct svc_rqst * rqstp , u32 proc , struct nlm_args * argp ,
__be32 ( * func ) ( struct svc_rqst * , struct nlm_args * , struct nlm_res * ) )
2005-04-17 02:20:36 +04:00
{
2006-03-20 21:44:45 +03:00
struct nlm_host * host ;
struct nlm_rqst * call ;
2006-10-20 10:28:45 +04:00
__be32 stat ;
2005-04-17 02:20:36 +04:00
2006-10-04 13:15:52 +04:00
host = nlmsvc_lookup_host ( rqstp ,
argp - > lock . caller ,
argp - > lock . len ) ;
2006-03-20 21:44:45 +03:00
if ( host = = NULL )
return rpc_system_err ;
call = nlm_alloc_call ( host ) ;
2012-07-26 00:39:50 +04:00
nlmsvc_release_host ( host ) ;
2006-03-20 21:44:45 +03:00
if ( call = = NULL )
return rpc_system_err ;
stat = func ( rqstp , argp , & call - > a_res ) ;
if ( stat ! = 0 ) {
2010-12-14 18:05:42 +03:00
nlmsvc_release_call ( call ) ;
2006-03-20 21:44:45 +03:00
return stat ;
}
2005-04-17 02:20:36 +04:00
2006-03-20 21:44:45 +03:00
call - > a_flags = RPC_TASK_ASYNC ;
if ( nlm_async_reply ( call , proc , & nlmsvc_callback_ops ) < 0 )
return rpc_system_err ;
return rpc_success ;
2005-04-17 02:20:36 +04:00
}
2006-10-20 10:28:45 +04:00
static __be32 nlmsvc_proc_test_msg ( struct svc_rqst * rqstp , struct nlm_args * argp ,
2005-04-17 02:20:36 +04:00
void * resp )
{
2006-03-20 21:44:45 +03:00
dprintk ( " lockd: TEST_MSG called \n " ) ;
return nlmsvc_callback ( rqstp , NLMPROC_TEST_RES , argp , nlmsvc_proc_test ) ;
}
2005-04-17 02:20:36 +04:00
2006-10-20 10:28:45 +04:00
static __be32 nlmsvc_proc_lock_msg ( struct svc_rqst * rqstp , struct nlm_args * argp ,
2006-03-20 21:44:45 +03:00
void * resp )
{
2005-04-17 02:20:36 +04:00
dprintk ( " lockd: LOCK_MSG called \n " ) ;
2006-03-20 21:44:45 +03:00
return nlmsvc_callback ( rqstp , NLMPROC_LOCK_RES , argp , nlmsvc_proc_lock ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-20 10:28:45 +04:00
static __be32 nlmsvc_proc_cancel_msg ( struct svc_rqst * rqstp , struct nlm_args * argp ,
2005-04-17 02:20:36 +04:00
void * resp )
{
dprintk ( " lockd: CANCEL_MSG called \n " ) ;
2006-03-20 21:44:45 +03:00
return nlmsvc_callback ( rqstp , NLMPROC_CANCEL_RES , argp , nlmsvc_proc_cancel ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_unlock_msg ( struct svc_rqst * rqstp , struct nlm_args * argp ,
void * resp )
{
dprintk ( " lockd: UNLOCK_MSG called \n " ) ;
2006-03-20 21:44:45 +03:00
return nlmsvc_callback ( rqstp , NLMPROC_UNLOCK_RES , argp , nlmsvc_proc_unlock ) ;
2005-04-17 02:20:36 +04:00
}
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_granted_msg ( struct svc_rqst * rqstp , struct nlm_args * argp ,
void * resp )
{
dprintk ( " lockd: GRANTED_MSG called \n " ) ;
2006-03-20 21:44:45 +03:00
return nlmsvc_callback ( rqstp , NLMPROC_GRANTED_RES , argp , nlmsvc_proc_granted ) ;
2005-04-17 02:20:36 +04:00
}
/*
* SHARE : create a DOS share or alter existing share .
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_share ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_res * resp )
{
struct nlm_host * host ;
struct nlm_file * file ;
dprintk ( " lockd: SHARE called \n " ) ;
resp - > cookie = argp - > cookie ;
/* Don't accept new lock requests during grace period */
2012-07-25 16:57:22 +04:00
if ( locks_in_grace ( SVC_NET ( rqstp ) ) & & ! argp - > reclaim ) {
2005-04-17 02:20:36 +04:00
resp - > status = nlm_lck_denied_grace_period ;
return rpc_success ;
}
/* Obtain client and file */
if ( ( resp - > status = nlmsvc_retrieve_args ( rqstp , argp , & host , & file ) ) )
2006-10-17 11:10:18 +04:00
return resp - > status = = nlm_drop_reply ? rpc_drop_reply : rpc_success ;
2005-04-17 02:20:36 +04:00
/* Now try to create the share */
resp - > status = cast_status ( nlmsvc_share_file ( host , file , argp ) ) ;
dprintk ( " lockd: SHARE status %d \n " , ntohl ( resp - > status ) ) ;
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( host ) ;
2005-04-17 02:20:36 +04:00
nlm_release_file ( file ) ;
return rpc_success ;
}
/*
* UNSHARE : Release a DOS share .
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_unshare ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_res * resp )
{
struct nlm_host * host ;
struct nlm_file * file ;
dprintk ( " lockd: UNSHARE called \n " ) ;
resp - > cookie = argp - > cookie ;
/* Don't accept requests during grace period */
2012-07-25 16:57:22 +04:00
if ( locks_in_grace ( SVC_NET ( rqstp ) ) ) {
2005-04-17 02:20:36 +04:00
resp - > status = nlm_lck_denied_grace_period ;
return rpc_success ;
}
/* Obtain client and file */
if ( ( resp - > status = nlmsvc_retrieve_args ( rqstp , argp , & host , & file ) ) )
2006-10-17 11:10:18 +04:00
return resp - > status = = nlm_drop_reply ? rpc_drop_reply : rpc_success ;
2005-04-17 02:20:36 +04:00
/* Now try to unshare the file */
resp - > status = cast_status ( nlmsvc_unshare_file ( host , file , argp ) ) ;
dprintk ( " lockd: UNSHARE status %d \n " , ntohl ( resp - > status ) ) ;
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( host ) ;
2005-04-17 02:20:36 +04:00
nlm_release_file ( file ) ;
return rpc_success ;
}
/*
* NM_LOCK : Create an unmonitored lock
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_nm_lock ( struct svc_rqst * rqstp , struct nlm_args * argp ,
struct nlm_res * resp )
{
dprintk ( " lockd: NM_LOCK called \n " ) ;
argp - > monitor = 0 ; /* just clean the monitor flag */
return nlmsvc_proc_lock ( rqstp , argp , resp ) ;
}
/*
* FREE_ALL : Release all locks and shares held by client
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_free_all ( struct svc_rqst * rqstp , struct nlm_args * argp ,
void * resp )
{
struct nlm_host * host ;
/* Obtain client */
if ( nlmsvc_retrieve_args ( rqstp , argp , & host , NULL ) )
return rpc_success ;
nlmsvc_free_host_resources ( host ) ;
2010-12-14 18:06:12 +03:00
nlmsvc_release_host ( host ) ;
2005-04-17 02:20:36 +04:00
return rpc_success ;
}
/*
* SM_NOTIFY : private callback from statd ( not part of official NLM proto )
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_sm_notify ( struct svc_rqst * rqstp , struct nlm_reboot * argp ,
void * resp )
{
dprintk ( " lockd: SM_NOTIFY called \n " ) ;
2008-10-03 20:50:44 +04:00
if ( ! nlm_privileged_requester ( rqstp ) ) {
2007-02-12 11:53:32 +03:00
char buf [ RPC_MAX_ADDRBUFLEN ] ;
printk ( KERN_WARNING " lockd: rejected NSM callback from %s \n " ,
svc_print_addr ( rqstp , buf , sizeof ( buf ) ) ) ;
2005-04-17 02:20:36 +04:00
return rpc_system_err ;
}
2008-12-06 03:03:31 +03:00
nlm_host_rebooted ( argp ) ;
2005-04-17 02:20:36 +04:00
return rpc_success ;
}
/*
* client sent a GRANTED_RES , let ' s remove the associated block
*/
2006-10-20 10:28:45 +04:00
static __be32
2005-04-17 02:20:36 +04:00
nlmsvc_proc_granted_res ( struct svc_rqst * rqstp , struct nlm_res * argp ,
void * resp )
{
if ( ! nlmsvc_ops )
return rpc_success ;
dprintk ( " lockd: GRANTED_RES called \n " ) ;
2006-10-04 13:16:03 +04:00
nlmsvc_grant_reply ( & argp - > cookie , argp - > status ) ;
2005-04-17 02:20:36 +04:00
return rpc_success ;
}
/*
* NLM Server procedures .
*/
# define nlmsvc_encode_norep nlmsvc_encode_void
# define nlmsvc_decode_norep nlmsvc_decode_void
# define nlmsvc_decode_testres nlmsvc_decode_void
# define nlmsvc_decode_lockres nlmsvc_decode_void
# define nlmsvc_decode_unlockres nlmsvc_decode_void
# define nlmsvc_decode_cancelres nlmsvc_decode_void
# define nlmsvc_decode_grantedres nlmsvc_decode_void
# define nlmsvc_proc_none nlmsvc_proc_null
# define nlmsvc_proc_test_res nlmsvc_proc_null
# define nlmsvc_proc_lock_res nlmsvc_proc_null
# define nlmsvc_proc_cancel_res nlmsvc_proc_null
# define nlmsvc_proc_unlock_res nlmsvc_proc_null
struct nlm_void { int dummy ; } ;
# define PROC(name, xargt, xrest, argt, rest, respsize) \
{ . pc_func = ( svc_procfunc ) nlmsvc_proc_ # # name , \
. pc_decode = ( kxdrproc_t ) nlmsvc_decode_ # # xargt , \
. pc_encode = ( kxdrproc_t ) nlmsvc_encode_ # # xrest , \
. pc_release = NULL , \
. pc_argsize = sizeof ( struct nlm_ # # argt ) , \
. pc_ressize = sizeof ( struct nlm_ # # rest ) , \
. pc_xdrressize = respsize , \
}
# define Ck (1+XDR_QUADLEN(NLM_MAXCOOKIELEN)) /* cookie */
# define St 1 /* status */
# define No (1+1024 / 4) /* Net Obj */
# define Rg 2 /* range - offset + size */
struct svc_procedure nlmsvc_procedures [ ] = {
PROC ( null , void , void , void , void , 1 ) ,
PROC ( test , testargs , testres , args , res , Ck + St + 2 + No + Rg ) ,
PROC ( lock , lockargs , res , args , res , Ck + St ) ,
PROC ( cancel , cancargs , res , args , res , Ck + St ) ,
PROC ( unlock , unlockargs , res , args , res , Ck + St ) ,
PROC ( granted , testargs , res , args , res , Ck + St ) ,
PROC ( test_msg , testargs , norep , args , void , 1 ) ,
PROC ( lock_msg , lockargs , norep , args , void , 1 ) ,
PROC ( cancel_msg , cancargs , norep , args , void , 1 ) ,
PROC ( unlock_msg , unlockargs , norep , args , void , 1 ) ,
PROC ( granted_msg , testargs , norep , args , void , 1 ) ,
PROC ( test_res , testres , norep , res , void , 1 ) ,
PROC ( lock_res , lockres , norep , res , void , 1 ) ,
PROC ( cancel_res , cancelres , norep , res , void , 1 ) ,
PROC ( unlock_res , unlockres , norep , res , void , 1 ) ,
PROC ( granted_res , res , norep , res , void , 1 ) ,
/* statd callback */
PROC ( sm_notify , reboot , void , reboot , void , 1 ) ,
PROC ( none , void , void , void , void , 1 ) ,
PROC ( none , void , void , void , void , 1 ) ,
PROC ( none , void , void , void , void , 1 ) ,
PROC ( share , shareargs , shareres , args , res , Ck + St + 1 ) ,
PROC ( unshare , shareargs , shareres , args , res , Ck + St + 1 ) ,
PROC ( nm_lock , lockargs , res , args , res , Ck + St ) ,
PROC ( free_all , notify , void , args , void , 0 ) ,
} ;