X-Git-Url: https://git.decadent.org.uk/gitweb/?p=nfs-utils.git;a=blobdiff_plain;f=support%2Fnfs%2Fconn.c;h=29dbb82c33dd102d3e90d0d0dcb223f7f7bc9c25;hp=a020394c86b7dc1541b58b3dcd823eaeefe8d6ff;hb=7f9343a38c031d9cc61a6bc7e06df519b16d1293;hpb=4e2bae795e5eaf9922f0b966ab5df64994c836a2 diff --git a/support/nfs/conn.c b/support/nfs/conn.c index a020394..29dbb82 100644 --- a/support/nfs/conn.c +++ b/support/nfs/conn.c @@ -6,7 +6,9 @@ * */ +#include "config.h" #include +#include #include #include #include @@ -17,6 +19,10 @@ #include "conn.h" +#if SIZEOF_SOCKLEN_T - 0 == 0 +#define socklen_t int +#endif + extern int verbose; /* Map an NFS version into the corresponding Mountd version */ @@ -43,7 +49,7 @@ u_long mntvers_to_nfs(const u_long vers) * RPC_ANYSOCK is returned which will cause * the RPC code to create the socket instead. */ -int get_socket(struct sockaddr_in *saddr, u_int p_prot, int resvp) +int get_socket(struct sockaddr_in *saddr, u_int p_prot, int resvp, int conn) { int so, cc, type; struct sockaddr_in laddr; @@ -92,7 +98,7 @@ int get_socket(struct sockaddr_in *saddr, u_int p_prot, int resvp) return RPC_ANYSOCK; } } - if (type == SOCK_STREAM) { + if (type == SOCK_STREAM || (conn && type == SOCK_DGRAM)) { cc = connect(so, (struct sockaddr *)saddr, namelen); if (cc < 0) { rpc_createerr.cf_stat = RPC_SYSTEMERROR; @@ -118,25 +124,42 @@ int get_socket(struct sockaddr_in *saddr, u_int p_prot, int resvp) */ int clnt_ping(struct sockaddr_in *saddr, const u_long prog, const u_long vers, - const u_int prot) + const u_int prot, struct sockaddr_in *caddr) { CLIENT *clnt=NULL; int sock, stat; static char clnt_res; + struct sockaddr dissolve; rpc_createerr.cf_stat = stat = errno = 0; - sock = get_socket(saddr, prot, FALSE); - if (sock == RPC_ANYSOCK && errno == ETIMEDOUT) { - /* - * TCP timeout. Bubble up the error to see - * how it should be handled. - */ - rpc_createerr.cf_stat = RPC_TIMEDOUT; - goto out_bad; + sock = get_socket(saddr, prot, FALSE, TRUE); + if (sock == RPC_ANYSOCK) { + if (errno == ETIMEDOUT) { + /* + * TCP timeout. Bubble up the error to see + * how it should be handled. + */ + rpc_createerr.cf_stat = RPC_TIMEDOUT; + } + return 0; + } + + if (caddr) { + /* Get the address of our end of this connection */ + socklen_t len = sizeof(*caddr); + if (getsockname(sock, caddr, &len) != 0) + caddr->sin_family = 0; } switch(prot) { case IPPROTO_UDP: + /* The socket is connected (so we could getsockname successfully), + * but some servers on multi-homed hosts reply from + * the wrong address, so if we stay connected, we lose the reply. + */ + dissolve.sa_family = AF_UNSPEC; + connect(sock, &dissolve, sizeof(dissolve)); + clnt = clntudp_bufcreate(saddr, prog, vers, RETRY_TIMEOUT, &sock, RPCSMALLMSGSIZE, RPCSMALLMSGSIZE); @@ -145,11 +168,11 @@ clnt_ping(struct sockaddr_in *saddr, const u_long prog, const u_long vers, clnt = clnttcp_create(saddr, prog, vers, &sock, RPCSMALLMSGSIZE, RPCSMALLMSGSIZE); break; - default: - goto out_bad; } - if (!clnt) - goto out_bad; + if (!clnt) { + close(sock); + return 0; + } memset(&clnt_res, 0, sizeof(clnt_res)); stat = clnt_call(clnt, NULLPROC, (xdrproc_t)xdr_void, (caddr_t)NULL, @@ -160,14 +183,12 @@ clnt_ping(struct sockaddr_in *saddr, const u_long prog, const u_long vers, rpc_createerr.cf_stat = stat; } clnt_destroy(clnt); - if (sock != -1) - close(sock); + close(sock); if (stat == RPC_SUCCESS) return 1; - - out_bad: - return 0; + else + return 0; } CLIENT *mnt_openclnt(clnt_addr_t *mnt_server, int *msock) @@ -178,7 +199,7 @@ CLIENT *mnt_openclnt(clnt_addr_t *mnt_server, int *msock) /* contact the mount daemon via TCP */ mnt_saddr->sin_port = htons((u_short)mnt_pmap->pm_port); - *msock = get_socket(mnt_saddr, mnt_pmap->pm_prot, TRUE); + *msock = get_socket(mnt_saddr, mnt_pmap->pm_prot, TRUE, FALSE); switch (mnt_pmap->pm_prot) { case IPPROTO_UDP: