X-Git-Url: https://git.decadent.org.uk/gitweb/?p=nfs-utils.git;a=blobdiff_plain;f=utils%2Fmount%2Fnetwork.c;h=c997c4cc0ee30e86d482bd23c4cc7646d53c5a49;hp=21cf4efe72c646b2d26acca4ba038dde23962ca5;hb=7b763f58bb882caf4fbe44d9668237c4d3e77728;hpb=a3c8371b4f4cf9f0ff93ca040cac13dc4806ded9 diff --git a/utils/mount/network.c b/utils/mount/network.c index 21cf4ef..c997c4c 100644 --- a/utils/mount/network.c +++ b/utils/mount/network.c @@ -121,42 +121,66 @@ int nfs_gethostbyname(const char *hostname, struct sockaddr_in *saddr) } /* - * getport() is very similar to pmap_getport() with - * the exception this version uses a non-reserve ports - * instead of reserve ports since reserve ports - * are not needed for pmap requests. + * getport() is very similar to pmap_getport() with the exception that + * this version tries to use an ephemeral port, since reserved ports are + * not needed for GETPORT queries. This conserves the very limited + * reserved port space, which helps reduce failed socket binds + * during mount storms. + * + * A side effect of calling this function is that rpccreateerr is set. */ -unsigned short getport(struct sockaddr_in *saddr, unsigned long prog, - unsigned long vers, unsigned int prot) +static unsigned short getport(struct sockaddr_in *saddr, + unsigned long program, + unsigned long version, + unsigned int proto) { unsigned short port = 0; int socket; CLIENT *clnt = NULL; - struct pmap parms; enum clnt_stat stat; - saddr->sin_port = htons (PMAPPORT); - socket = get_socket(saddr, prot, FALSE, FALSE); + saddr->sin_port = htons(PMAPPORT); + + /* + * Try to get a socket with a non-privileged port. + * clnt*create() will create one anyway if this + * fails. + */ + socket = get_socket(saddr, proto, FALSE, FALSE); + if (socket == RPC_ANYSOCK) { + if (proto == IPPROTO_TCP && errno == ETIMEDOUT) { + /* + * TCP SYN timed out, so exit now. + */ + rpc_createerr.cf_stat = RPC_TIMEDOUT; + } + return 0; + } - switch (prot) { + switch (proto) { case IPPROTO_UDP: clnt = clntudp_bufcreate(saddr, - PMAPPROG, PMAPVERS, TIMEOUT, &socket, - UDPMSGSIZE, UDPMSGSIZE); + PMAPPROG, PMAPVERS, + RETRY_TIMEOUT, &socket, + RPCSMALLMSGSIZE, + RPCSMALLMSGSIZE); break; case IPPROTO_TCP: - clnt = clnttcp_create(saddr, - PMAPPROG, PMAPVERS, &socket, 50, 500); + clnt = clnttcp_create(saddr, PMAPPROG, PMAPVERS, &socket, + RPCSMALLMSGSIZE, RPCSMALLMSGSIZE); break; } if (clnt != NULL) { - parms.pm_prog = prog; - parms.pm_vers = vers; - parms.pm_prot = prot; - parms.pm_port = 0; /* not needed or used */ - - stat = clnt_call(clnt, PMAPPROC_GETPORT, (xdrproc_t)xdr_pmap, - (caddr_t)&parms, (xdrproc_t)xdr_u_short, (caddr_t)&port, TIMEOUT); + struct pmap parms = { + .pm_prog = program, + .pm_vers = version, + .pm_prot = proto, + }; + + stat = clnt_call(clnt, PMAPPROC_GETPORT, + (xdrproc_t)xdr_pmap, (caddr_t)&parms, + (xdrproc_t)xdr_u_short, (caddr_t)&port, + TIMEOUT); if (stat) { clnt_geterr(clnt, &rpc_createerr.cf_error); rpc_createerr.cf_stat = stat;