2005-04-16 15:20:36 -07:00
/*
* linux / fs / lockd / host . c
*
* Management for NLM peer hosts . The nlm_host struct is shared
* between client and server implementation . The only reason to
* do so is to reduce code bloat .
*
* Copyright ( C ) 1996 , Olaf Kirch < okir @ monad . swb . de >
*/
# include <linux/types.h>
# include <linux/slab.h>
# include <linux/in.h>
# include <linux/sunrpc/clnt.h>
# include <linux/sunrpc/svc.h>
# include <linux/lockd/lockd.h>
# include <linux/lockd/sm_inter.h>
2006-03-26 01:37:12 -08:00
# include <linux/mutex.h>
2005-04-16 15:20:36 -07:00
# define NLMDBG_FACILITY NLMDBG_HOSTCACHE
# define NLM_HOST_MAX 64
# define NLM_HOST_NRHASH 32
# define NLM_ADDRHASH(addr) (ntohl(addr) & (NLM_HOST_NRHASH-1))
# define NLM_HOST_REBIND (60 * HZ)
# define NLM_HOST_EXPIRE ((nrhosts > NLM_HOST_MAX)? 300 * HZ : 120 * HZ)
# define NLM_HOST_COLLECT ((nrhosts > NLM_HOST_MAX)? 120 * HZ : 60 * HZ)
2006-10-04 02:15:56 -07:00
static struct hlist_head nlm_hosts [ NLM_HOST_NRHASH ] ;
2005-04-16 15:20:36 -07:00
static unsigned long next_gc ;
static int nrhosts ;
2006-03-26 01:37:12 -08:00
static DEFINE_MUTEX ( nlm_host_mutex ) ;
2005-04-16 15:20:36 -07:00
static void nlm_gc_hosts ( void ) ;
2006-10-04 02:15:53 -07:00
static struct nsm_handle * __nsm_find ( const struct sockaddr_in * ,
const char * , int , int ) ;
2006-12-06 20:38:29 -08:00
static struct nsm_handle * nsm_find ( const struct sockaddr_in * sin ,
const char * hostname ,
int hostname_len ) ;
2005-04-16 15:20:36 -07:00
/*
* Common host lookup routine for server & client
*/
2006-12-06 20:38:29 -08:00
static struct nlm_host *
2006-10-04 02:15:52 -07:00
nlm_lookup_host ( int server , const struct sockaddr_in * sin ,
2007-07-09 22:25:29 +02:00
int proto , int version , const char * hostname ,
int hostname_len , const struct sockaddr_in * ssin )
2005-04-16 15:20:36 -07:00
{
2006-10-04 02:15:56 -07:00
struct hlist_head * chain ;
struct hlist_node * pos ;
struct nlm_host * host ;
2006-10-04 02:15:53 -07:00
struct nsm_handle * nsm = NULL ;
2005-04-16 15:20:36 -07:00
int hash ;
2007-07-09 22:25:29 +02:00
dprintk ( " lockd: nlm_lookup_host( " NIPQUAD_FMT " -> " NIPQUAD_FMT
" , p=%d, v=%d, my role=%s, name=%.*s) \n " ,
NIPQUAD ( ssin - > sin_addr . s_addr ) ,
2006-10-04 02:15:52 -07:00
NIPQUAD ( sin - > sin_addr . s_addr ) , proto , version ,
server ? " server " : " client " ,
hostname_len ,
hostname ? hostname : " <none> " ) ;
2005-04-16 15:20:36 -07:00
hash = NLM_ADDRHASH ( sin - > sin_addr . s_addr ) ;
/* Lock hash table */
2006-03-26 01:37:12 -08:00
mutex_lock ( & nlm_host_mutex ) ;
2005-04-16 15:20:36 -07:00
if ( time_after_eq ( jiffies , next_gc ) )
nlm_gc_hosts ( ) ;
2006-10-04 02:15:53 -07:00
/* We may keep several nlm_host objects for a peer, because each
* nlm_host is identified by
* ( address , protocol , version , server / client )
* We could probably simplify this a little by putting all those
* different NLM rpc_clients into one single nlm_host object .
* This would allow us to have one nlm_host per address .
*/
2006-10-04 02:15:56 -07:00
chain = & nlm_hosts [ hash ] ;
hlist_for_each_entry ( host , pos , chain , h_hash ) {
2006-10-04 02:15:53 -07:00
if ( ! nlm_cmp_addr ( & host - > h_addr , sin ) )
continue ;
/* See if we have an NSM handle for this client */
2006-10-04 02:16:15 -07:00
if ( ! nsm )
nsm = host - > h_nsmhandle ;
2006-10-04 02:15:53 -07:00
2005-04-16 15:20:36 -07:00
if ( host - > h_proto ! = proto )
continue ;
if ( host - > h_version ! = version )
continue ;
if ( host - > h_server ! = server )
continue ;
2007-07-09 22:25:29 +02:00
if ( ! nlm_cmp_addr ( & host - > h_saddr , ssin ) )
continue ;
2005-04-16 15:20:36 -07:00
2006-10-04 02:15:56 -07:00
/* Move to head of hash chain. */
hlist_del ( & host - > h_hash ) ;
hlist_add_head ( & host - > h_hash , chain ) ;
2006-10-04 02:15:54 -07:00
nlm_get_host ( host ) ;
goto out ;
2005-04-16 15:20:36 -07:00
}
2006-10-04 02:16:15 -07:00
if ( nsm )
atomic_inc ( & nsm - > sm_count ) ;
2005-04-16 15:20:36 -07:00
2006-10-04 02:15:56 -07:00
host = NULL ;
2006-10-04 02:15:53 -07:00
/* Sadly, the host isn't in our hash table yet. See if
* we have an NSM handle for it . If not , create one .
*/
if ( ! nsm & & ! ( nsm = nsm_find ( sin , hostname , hostname_len ) ) )
goto out ;
2005-04-16 15:20:36 -07:00
2006-09-27 01:49:37 -07:00
host = kzalloc ( sizeof ( * host ) , GFP_KERNEL ) ;
2006-10-04 02:15:53 -07:00
if ( ! host ) {
nsm_release ( nsm ) ;
goto out ;
}
host - > h_name = nsm - > sm_name ;
2005-04-16 15:20:36 -07:00
host - > h_addr = * sin ;
host - > h_addr . sin_port = 0 ; /* ouch! */
2007-07-09 22:25:29 +02:00
host - > h_saddr = * ssin ;
2005-04-16 15:20:36 -07:00
host - > h_version = version ;
host - > h_proto = proto ;
host - > h_rpcclnt = NULL ;
2006-06-09 09:40:24 -04:00
mutex_init ( & host - > h_mutex ) ;
2005-04-16 15:20:36 -07:00
host - > h_nextrebind = jiffies + NLM_HOST_REBIND ;
host - > h_expires = jiffies + NLM_HOST_EXPIRE ;
atomic_set ( & host - > h_count , 1 ) ;
init_waitqueue_head ( & host - > h_gracewait ) ;
2006-06-09 09:40:27 -04:00
init_rwsem ( & host - > h_rwsem ) ;
2005-04-16 15:20:36 -07:00
host - > h_state = 0 ; /* pseudo NSM state */
host - > h_nsmstate = 0 ; /* real NSM state */
2006-10-04 02:15:53 -07:00
host - > h_nsmhandle = nsm ;
2005-04-16 15:20:36 -07:00
host - > h_server = server ;
2006-10-04 02:15:56 -07:00
hlist_add_head ( & host - > h_hash , chain ) ;
2005-04-16 15:20:36 -07:00
INIT_LIST_HEAD ( & host - > h_lockowners ) ;
spin_lock_init ( & host - > h_lock ) ;
2006-03-20 13:44:40 -05:00
INIT_LIST_HEAD ( & host - > h_granted ) ;
INIT_LIST_HEAD ( & host - > h_reclaim ) ;
2005-04-16 15:20:36 -07:00
if ( + + nrhosts > NLM_HOST_MAX )
next_gc = 0 ;
2006-10-04 02:15:53 -07:00
out :
2006-03-26 01:37:12 -08:00
mutex_unlock ( & nlm_host_mutex ) ;
2005-04-16 15:20:36 -07:00
return host ;
}
2006-10-04 02:16:00 -07:00
/*
* Destroy a host
*/
static void
nlm_destroy_host ( struct nlm_host * host )
{
struct rpc_clnt * clnt ;
BUG_ON ( ! list_empty ( & host - > h_lockowners ) ) ;
BUG_ON ( atomic_read ( & host - > h_count ) ) ;
/*
* Release NSM handle and unmonitor host .
*/
nsm_unmonitor ( host ) ;
2007-06-14 16:40:31 -04:00
clnt = host - > h_rpcclnt ;
if ( clnt ! = NULL )
rpc_shutdown_client ( clnt ) ;
2006-10-04 02:16:00 -07:00
kfree ( host ) ;
}
2006-12-06 20:38:29 -08:00
/*
* Find an NLM server handle in the cache . If there is none , create it .
*/
struct nlm_host *
nlmclnt_lookup_host ( const struct sockaddr_in * sin , int proto , int version ,
const char * hostname , int hostname_len )
{
2007-07-09 22:25:29 +02:00
struct sockaddr_in ssin = { 0 } ;
2006-12-06 20:38:29 -08:00
return nlm_lookup_host ( 0 , sin , proto , version ,
2007-07-09 22:25:29 +02:00
hostname , hostname_len , & ssin ) ;
2006-12-06 20:38:29 -08:00
}
/*
* Find an NLM client handle in the cache . If there is none , create it .
*/
struct nlm_host *
nlmsvc_lookup_host ( struct svc_rqst * rqstp ,
const char * hostname , int hostname_len )
{
2007-07-09 22:25:29 +02:00
struct sockaddr_in ssin = { 0 } ;
ssin . sin_addr = rqstp - > rq_daddr . addr ;
2007-02-12 00:53:34 -08:00
return nlm_lookup_host ( 1 , svc_addr_in ( rqstp ) ,
2006-12-06 20:38:29 -08:00
rqstp - > rq_prot , rqstp - > rq_vers ,
2007-07-09 22:25:29 +02:00
hostname , hostname_len , & ssin ) ;
2006-12-06 20:38:29 -08:00
}
2005-04-16 15:20:36 -07:00
/*
* Create the NLM RPC client for an NLM peer
*/
struct rpc_clnt *
nlm_bind_host ( struct nlm_host * host )
{
struct rpc_clnt * clnt ;
2007-07-09 22:25:29 +02:00
dprintk ( " lockd: nlm_bind_host( " NIPQUAD_FMT " -> " NIPQUAD_FMT " ) \n " ,
NIPQUAD ( host - > h_saddr . sin_addr ) ,
NIPQUAD ( host - > h_addr . sin_addr ) ) ;
2005-04-16 15:20:36 -07:00
/* Lock host handle */
2006-06-09 09:40:24 -04:00
mutex_lock ( & host - > h_mutex ) ;
2005-04-16 15:20:36 -07:00
/* If we've already created an RPC client, check whether
* RPC rebind is required
*/
if ( ( clnt = host - > h_rpcclnt ) ! = NULL ) {
2005-08-25 16:25:49 -07:00
if ( time_after_eq ( jiffies , host - > h_nextrebind ) ) {
2006-01-03 09:55:50 +01:00
rpc_force_rebind ( clnt ) ;
2005-04-16 15:20:36 -07:00
host - > h_nextrebind = jiffies + NLM_HOST_REBIND ;
dprintk ( " lockd: next rebind in %ld jiffies \n " ,
host - > h_nextrebind - jiffies ) ;
}
} else {
2007-05-14 16:50:44 -04:00
unsigned long increment = nlmsvc_timeout ;
2006-08-22 20:06:20 -04:00
struct rpc_timeout timeparms = {
. to_initval = increment ,
. to_increment = increment ,
. to_maxval = increment * 6UL ,
. to_retries = 5U ,
} ;
struct rpc_create_args args = {
. protocol = host - > h_proto ,
. address = ( struct sockaddr * ) & host - > h_addr ,
. addrsize = sizeof ( host - > h_addr ) ,
2007-07-09 22:25:29 +02:00
. saddress = ( struct sockaddr * ) & host - > h_saddr ,
2006-08-22 20:06:20 -04:00
. timeout = & timeparms ,
. servername = host - > h_name ,
. program = & nlm_program ,
. version = host - > h_version ,
. authflavor = RPC_AUTH_UNIX ,
. flags = ( RPC_CLNT_CREATE_HARDRTRY |
RPC_CLNT_CREATE_AUTOBIND ) ,
} ;
clnt = rpc_create ( & args ) ;
if ( ! IS_ERR ( clnt ) )
host - > h_rpcclnt = clnt ;
else {
printk ( " lockd: couldn't create RPC handle for %s \n " , host - > h_name ) ;
clnt = NULL ;
}
2005-04-16 15:20:36 -07:00
}
2006-06-09 09:40:24 -04:00
mutex_unlock ( & host - > h_mutex ) ;
2005-04-16 15:20:36 -07:00
return clnt ;
}
/*
* Force a portmap lookup of the remote lockd port
*/
void
nlm_rebind_host ( struct nlm_host * host )
{
dprintk ( " lockd: rebind host %s \n " , host - > h_name ) ;
if ( host - > h_rpcclnt & & time_after_eq ( jiffies , host - > h_nextrebind ) ) {
2006-01-03 09:55:50 +01:00
rpc_force_rebind ( host - > h_rpcclnt ) ;
2005-04-16 15:20:36 -07:00
host - > h_nextrebind = jiffies + NLM_HOST_REBIND ;
}
}
/*
* Increment NLM host count
*/
struct nlm_host * nlm_get_host ( struct nlm_host * host )
{
if ( host ) {
dprintk ( " lockd: get host %s \n " , host - > h_name ) ;
atomic_inc ( & host - > h_count ) ;
host - > h_expires = jiffies + NLM_HOST_EXPIRE ;
}
return host ;
}
/*
* Release NLM host after use
*/
void nlm_release_host ( struct nlm_host * host )
{
if ( host ! = NULL ) {
dprintk ( " lockd: release host %s \n " , host - > h_name ) ;
BUG_ON ( atomic_read ( & host - > h_count ) < 0 ) ;
2006-03-20 13:44:41 -05:00
if ( atomic_dec_and_test ( & host - > h_count ) ) {
BUG_ON ( ! list_empty ( & host - > h_lockowners ) ) ;
BUG_ON ( ! list_empty ( & host - > h_granted ) ) ;
BUG_ON ( ! list_empty ( & host - > h_reclaim ) ) ;
}
2005-04-16 15:20:36 -07:00
}
}
2006-10-04 02:15:52 -07:00
/*
* We were notified that the host indicated by address & sin
* has rebooted .
* Release all resources held by that peer .
*/
2006-10-04 02:15:55 -07:00
void nlm_host_rebooted ( const struct sockaddr_in * sin ,
const char * hostname , int hostname_len ,
u32 new_state )
2006-10-04 02:15:52 -07:00
{
2006-10-04 02:15:56 -07:00
struct hlist_head * chain ;
struct hlist_node * pos ;
2006-10-04 02:15:55 -07:00
struct nsm_handle * nsm ;
2006-10-04 02:15:56 -07:00
struct nlm_host * host ;
2006-10-04 02:15:52 -07:00
2006-10-04 02:15:55 -07:00
dprintk ( " lockd: nlm_host_rebooted(%s, %u.%u.%u.%u) \n " ,
hostname , NIPQUAD ( sin - > sin_addr ) ) ;
/* Find the NSM handle for this peer */
if ( ! ( nsm = __nsm_find ( sin , hostname , hostname_len , 0 ) ) )
2006-10-04 02:15:52 -07:00
return ;
2006-10-04 02:15:55 -07:00
/* When reclaiming locks on this peer, make sure that
* we set up a new notification */
nsm - > sm_monitored = 0 ;
/* Mark all hosts tied to this NSM state as having rebooted.
* We run the loop repeatedly , because we drop the host table
* lock for this .
* To avoid processing a host several times , we match the nsmstate .
*/
again : mutex_lock ( & nlm_host_mutex ) ;
2006-10-04 02:15:56 -07:00
for ( chain = nlm_hosts ; chain < nlm_hosts + NLM_HOST_NRHASH ; + + chain ) {
hlist_for_each_entry ( host , pos , chain , h_hash ) {
2006-10-04 02:15:55 -07:00
if ( host - > h_nsmhandle = = nsm
& & host - > h_nsmstate ! = new_state ) {
host - > h_nsmstate = new_state ;
host - > h_state + + ;
nlm_get_host ( host ) ;
mutex_unlock ( & nlm_host_mutex ) ;
if ( host - > h_server ) {
/* We're server for this guy, just ditch
* all the locks he held . */
nlmsvc_free_host_resources ( host ) ;
} else {
/* He's the server, initiate lock recovery. */
nlmclnt_recovery ( host ) ;
}
nlm_release_host ( host ) ;
goto again ;
}
}
2006-10-04 02:15:52 -07:00
}
2006-10-04 02:15:55 -07:00
mutex_unlock ( & nlm_host_mutex ) ;
2006-10-04 02:15:52 -07:00
}
2005-04-16 15:20:36 -07:00
/*
* Shut down the hosts module .
* Note that this routine is called only at server shutdown time .
*/
void
nlm_shutdown_hosts ( void )
{
2006-10-04 02:15:56 -07:00
struct hlist_head * chain ;
struct hlist_node * pos ;
2005-04-16 15:20:36 -07:00
struct nlm_host * host ;
dprintk ( " lockd: shutting down host module \n " ) ;
2006-03-26 01:37:12 -08:00
mutex_lock ( & nlm_host_mutex ) ;
2005-04-16 15:20:36 -07:00
/* First, make all hosts eligible for gc */
dprintk ( " lockd: nuking all hosts... \n " ) ;
2006-10-04 02:15:56 -07:00
for ( chain = nlm_hosts ; chain < nlm_hosts + NLM_HOST_NRHASH ; + + chain ) {
hlist_for_each_entry ( host , pos , chain , h_hash )
2005-04-16 15:20:36 -07:00
host - > h_expires = jiffies - 1 ;
}
/* Then, perform a garbage collection pass */
nlm_gc_hosts ( ) ;
2006-03-26 01:37:12 -08:00
mutex_unlock ( & nlm_host_mutex ) ;
2005-04-16 15:20:36 -07:00
/* complain if any hosts are left */
if ( nrhosts ) {
printk ( KERN_WARNING " lockd: couldn't shutdown host module! \n " ) ;
dprintk ( " lockd: %d hosts left: \n " , nrhosts ) ;
2006-10-04 02:15:56 -07:00
for ( chain = nlm_hosts ; chain < nlm_hosts + NLM_HOST_NRHASH ; + + chain ) {
hlist_for_each_entry ( host , pos , chain , h_hash ) {
2005-04-16 15:20:36 -07:00
dprintk ( " %s (cnt %d use %d exp %ld) \n " ,
host - > h_name , atomic_read ( & host - > h_count ) ,
host - > h_inuse , host - > h_expires ) ;
}
}
}
}
/*
* Garbage collect any unused NLM hosts .
* This GC combines reference counting for async operations with
* mark & sweep for resources held by remote clients .
*/
static void
nlm_gc_hosts ( void )
{
2006-10-04 02:15:56 -07:00
struct hlist_head * chain ;
struct hlist_node * pos , * next ;
struct nlm_host * host ;
2005-04-16 15:20:36 -07:00
dprintk ( " lockd: host garbage collection \n " ) ;
2006-10-04 02:15:56 -07:00
for ( chain = nlm_hosts ; chain < nlm_hosts + NLM_HOST_NRHASH ; + + chain ) {
hlist_for_each_entry ( host , pos , chain , h_hash )
2005-04-16 15:20:36 -07:00
host - > h_inuse = 0 ;
}
/* Mark all hosts that hold locks, blocks or shares */
nlmsvc_mark_resources ( ) ;
2006-10-04 02:15:56 -07:00
for ( chain = nlm_hosts ; chain < nlm_hosts + NLM_HOST_NRHASH ; + + chain ) {
hlist_for_each_entry_safe ( host , pos , next , chain , h_hash ) {
2005-04-16 15:20:36 -07:00
if ( atomic_read ( & host - > h_count ) | | host - > h_inuse
| | time_before ( jiffies , host - > h_expires ) ) {
dprintk ( " nlm_gc_hosts skipping %s (cnt %d use %d exp %ld) \n " ,
host - > h_name , atomic_read ( & host - > h_count ) ,
host - > h_inuse , host - > h_expires ) ;
continue ;
}
dprintk ( " lockd: delete host %s \n " , host - > h_name ) ;
2006-10-04 02:15:56 -07:00
hlist_del_init ( & host - > h_hash ) ;
2006-10-04 02:15:51 -07:00
2006-10-04 02:16:00 -07:00
nlm_destroy_host ( host ) ;
2005-04-16 15:20:36 -07:00
nrhosts - - ;
}
}
next_gc = jiffies + NLM_HOST_COLLECT ;
}
2006-10-04 02:15:53 -07:00
/*
* Manage NSM handles
*/
static LIST_HEAD ( nsm_handles ) ;
2006-10-04 02:16:06 -07:00
static DEFINE_MUTEX ( nsm_mutex ) ;
2006-10-04 02:15:53 -07:00
static struct nsm_handle *
__nsm_find ( const struct sockaddr_in * sin ,
const char * hostname , int hostname_len ,
int create )
{
struct nsm_handle * nsm = NULL ;
struct list_head * pos ;
if ( ! sin )
return NULL ;
if ( hostname & & memchr ( hostname , ' / ' , hostname_len ) ! = NULL ) {
if ( printk_ratelimit ( ) ) {
printk ( KERN_WARNING " Invalid hostname \" %.*s \" "
" in NFS lock request \n " ,
hostname_len , hostname ) ;
}
return NULL ;
}
2006-10-04 02:16:06 -07:00
mutex_lock ( & nsm_mutex ) ;
2006-10-04 02:15:53 -07:00
list_for_each ( pos , & nsm_handles ) {
nsm = list_entry ( pos , struct nsm_handle , sm_link ) ;
2006-10-04 02:16:01 -07:00
if ( hostname & & nsm_use_hostnames ) {
if ( strlen ( nsm - > sm_name ) ! = hostname_len
| | memcmp ( nsm - > sm_name , hostname , hostname_len ) )
continue ;
} else if ( ! nlm_cmp_addr ( & nsm - > sm_addr , sin ) )
2006-10-04 02:15:53 -07:00
continue ;
atomic_inc ( & nsm - > sm_count ) ;
goto out ;
}
if ( ! create ) {
nsm = NULL ;
goto out ;
}
nsm = kzalloc ( sizeof ( * nsm ) + hostname_len + 1 , GFP_KERNEL ) ;
if ( nsm ! = NULL ) {
nsm - > sm_addr = * sin ;
nsm - > sm_name = ( char * ) ( nsm + 1 ) ;
memcpy ( nsm - > sm_name , hostname , hostname_len ) ;
nsm - > sm_name [ hostname_len ] = ' \0 ' ;
atomic_set ( & nsm - > sm_count , 1 ) ;
list_add ( & nsm - > sm_link , & nsm_handles ) ;
}
2006-10-04 02:16:06 -07:00
out :
mutex_unlock ( & nsm_mutex ) ;
2006-10-04 02:15:53 -07:00
return nsm ;
}
2006-12-06 20:38:29 -08:00
static struct nsm_handle *
2006-10-04 02:15:53 -07:00
nsm_find ( const struct sockaddr_in * sin , const char * hostname , int hostname_len )
{
return __nsm_find ( sin , hostname , hostname_len , 1 ) ;
}
/*
* Release an NSM handle
*/
void
nsm_release ( struct nsm_handle * nsm )
{
if ( ! nsm )
return ;
if ( atomic_dec_and_test ( & nsm - > sm_count ) ) {
2006-10-04 02:16:06 -07:00
mutex_lock ( & nsm_mutex ) ;
2006-10-04 02:15:53 -07:00
if ( atomic_read ( & nsm - > sm_count ) = = 0 ) {
list_del ( & nsm - > sm_link ) ;
kfree ( nsm ) ;
}
2006-10-04 02:16:06 -07:00
mutex_unlock ( & nsm_mutex ) ;
2006-10-04 02:15:53 -07:00
}
}