svc: Add svc API that queries for a transport instance
Add a new svc function that allows a service to query whether a transport instance has already been created. This is used in lockd to determine whether or not a transport needs to be created when a lockd instance is brought up. Specifying 0 for the address family or port is effectively a wild-card, and will result in matching the first transport in the service's list that has a matching class name. Signed-off-by: Tom Tucker <tom@opengridcomputing.com> Acked-by: Neil Brown <neilb@suse.de> Reviewed-by: Chuck Lever <chuck.lever@oracle.com> Reviewed-by: Greg Banks <gnb@sgi.com> Signed-off-by: J. Bruce Fields <bfields@citi.umich.edu>
This commit is contained in:
Родитель
dc9a16e49d
Коммит
7fcb98d58c
|
@ -219,18 +219,6 @@ lockd(struct svc_rqst *rqstp)
|
|||
module_put_and_exit(0);
|
||||
}
|
||||
|
||||
static int find_xprt(struct svc_serv *serv, char *proto)
|
||||
{
|
||||
struct svc_xprt *xprt;
|
||||
int found = 0;
|
||||
list_for_each_entry(xprt, &serv->sv_permsocks, xpt_list)
|
||||
if (strcmp(xprt->xpt_class->xcl_name, proto) == 0) {
|
||||
found = 1;
|
||||
break;
|
||||
}
|
||||
return found;
|
||||
}
|
||||
|
||||
/*
|
||||
* Make any sockets that are needed but not present.
|
||||
* If nlm_udpport or nlm_tcpport were set as module
|
||||
|
@ -242,11 +230,11 @@ static int make_socks(struct svc_serv *serv, int proto)
|
|||
int err = 0;
|
||||
|
||||
if (proto == IPPROTO_UDP || nlm_udpport)
|
||||
if (!find_xprt(serv, "udp"))
|
||||
if (!svc_find_xprt(serv, "udp", 0, 0))
|
||||
err = svc_create_xprt(serv, "udp", nlm_udpport,
|
||||
SVC_SOCK_DEFAULTS);
|
||||
if (err >= 0 && (proto == IPPROTO_TCP || nlm_tcpport))
|
||||
if (!find_xprt(serv, "tcp"))
|
||||
if (!svc_find_xprt(serv, "tcp", 0, 0))
|
||||
err = svc_create_xprt(serv, "tcp", nlm_tcpport,
|
||||
SVC_SOCK_DEFAULTS);
|
||||
|
||||
|
|
|
@ -80,6 +80,7 @@ void svc_close_xprt(struct svc_xprt *xprt);
|
|||
void svc_delete_xprt(struct svc_xprt *xprt);
|
||||
int svc_port_is_privileged(struct sockaddr *sin);
|
||||
int svc_print_xprts(char *buf, int maxlen);
|
||||
struct svc_xprt *svc_find_xprt(struct svc_serv *, char *, int, int);
|
||||
|
||||
static inline void svc_xprt_get(struct svc_xprt *xprt)
|
||||
{
|
||||
|
|
|
@ -977,3 +977,38 @@ static struct svc_deferred_req *svc_deferred_dequeue(struct svc_xprt *xprt)
|
|||
spin_unlock(&xprt->xpt_lock);
|
||||
return dr;
|
||||
}
|
||||
|
||||
/*
|
||||
* Return the transport instance pointer for the endpoint accepting
|
||||
* connections/peer traffic from the specified transport class,
|
||||
* address family and port.
|
||||
*
|
||||
* Specifying 0 for the address family or port is effectively a
|
||||
* wild-card, and will result in matching the first transport in the
|
||||
* service's list that has a matching class name.
|
||||
*/
|
||||
struct svc_xprt *svc_find_xprt(struct svc_serv *serv, char *xcl_name,
|
||||
int af, int port)
|
||||
{
|
||||
struct svc_xprt *xprt;
|
||||
struct svc_xprt *found = NULL;
|
||||
|
||||
/* Sanity check the args */
|
||||
if (!serv || !xcl_name)
|
||||
return found;
|
||||
|
||||
spin_lock_bh(&serv->sv_lock);
|
||||
list_for_each_entry(xprt, &serv->sv_permsocks, xpt_list) {
|
||||
if (strcmp(xprt->xpt_class->xcl_name, xcl_name))
|
||||
continue;
|
||||
if (af != AF_UNSPEC && af != xprt->xpt_local.ss_family)
|
||||
continue;
|
||||
if (port && port != svc_xprt_local_port(xprt))
|
||||
continue;
|
||||
found = xprt;
|
||||
break;
|
||||
}
|
||||
spin_unlock_bh(&serv->sv_lock);
|
||||
return found;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(svc_find_xprt);
|
||||
|
|
Загрузка…
Ссылка в новой задаче