aboutsummaryrefslogtreecommitdiff
path: root/sbin/mount_nfs
diff options
context:
space:
mode:
authorAndrey A. Chernov <ache@FreeBSD.org>2001-03-31 20:45:21 +0000
committerAndrey A. Chernov <ache@FreeBSD.org>2001-03-31 20:45:21 +0000
commitdeffdffafae18cc40beac221cca51ed6615f797c (patch)
tree1f615b8bff0f6f79d05c5c1cf88c0c2a9a104204 /sbin/mount_nfs
parent7ea7bc591b4606e81f1a8fd2a1ddd13e906b0861 (diff)
downloadsrc-deffdffafae18cc40beac221cca51ed6615f797c.tar.gz
src-deffdffafae18cc40beac221cca51ed6615f797c.zip
- Back out the last (wrong) commit, and readd a modified version
of pingnfsserver(). The pingnfsport() function is now called everytime. If we don't get RPC_SUCCESS or RPC_PROGVERSMISMATCH back, there's something wrong with the NFS server and we just exit. - Fix cfs mount on IPv4-only machines - Fixed the looping when we did not run background mode. - Fixed a getnameinfo() call with uninitialized adress. This is a NetBSD bug I didn't notified :-( Thanks Ian ! - Added some #ifdef NFSKERB - Removed some unused variables. - Fixed idention - Remove unnecessary ":" in openlog ident Submitted by: Martin Blapp <mb@imp.ch> Reviewed by: Ian Dowse <iedowse@maths.tcd.ie>
Notes
Notes: svn path=/head/; revision=75046
Diffstat (limited to 'sbin/mount_nfs')
-rw-r--r--sbin/mount_nfs/mount_nfs.c246
1 files changed, 162 insertions, 84 deletions
diff --git a/sbin/mount_nfs/mount_nfs.c b/sbin/mount_nfs/mount_nfs.c
index 19f8fa506938..e020a263c1b0 100644
--- a/sbin/mount_nfs/mount_nfs.c
+++ b/sbin/mount_nfs/mount_nfs.c
@@ -172,7 +172,7 @@ int retrycnt = DEF_RETRY;
int opflags = 0;
int nfsproto = IPPROTO_UDP;
int mnttcp_ok = 1;
-char *port_no = "nfs";
+u_short port_no = 0;
enum {
ANY,
V2,
@@ -199,6 +199,7 @@ int getnfsargs __P((char *, struct nfs_args *));
void usage __P((void)) __dead2;
int xdr_dir __P((XDR *, char *));
int xdr_fh __P((XDR *, struct nfhret *));
+enum clnt_stat pingnfsport(int, struct netconfig *, struct netbuf *);
/*
* Used to set mount flags with getmntopts. Call with dir=TRUE to
@@ -366,11 +367,8 @@ main(argc, argv)
nfsargsp->sotype = SOCK_STREAM;
nfsproto = IPPROTO_TCP;
}
- if(altflags & ALTF_PORT) {
- port_no = strdup(strstr(optarg, "port=") + 5);
- if (port_no == NULL)
- err(1, NULL);
- }
+ if(altflags & ALTF_PORT)
+ port_no = atoi(strstr(optarg, "port=") + 5);
mountmode = ANY;
if(altflags & ALTF_NFSV2)
mountmode = V2;
@@ -486,7 +484,7 @@ main(argc, argv)
(void) close(STDERR_FILENO);
(void) chdir("/");
}
- openlog("mount_nfs:", LOG_PID, LOG_DAEMON);
+ openlog("mount_nfs", LOG_PID, LOG_DAEMON);
nfssvc_flag = NFSSVC_MNTD;
ncd.ncd_dirp = mntpath;
while (nfssvc(nfssvc_flag, (caddr_t)&ncd) < 0) {
@@ -585,7 +583,9 @@ getnfsargs(spec, nfsargsp)
CLIENT *clp;
struct addrinfo hints, *ai_nfs, *ai;
int ecode;
+#ifdef NFSKERB
char host[NI_MAXHOST], serv[NI_MAXSERV];
+#endif
static struct netbuf nfs_nb;
static struct sockaddr_storage nfs_ss;
struct netconfig *nconf;
@@ -644,19 +644,21 @@ getnfsargs(spec, nfsargsp)
memset(&hints, 0, sizeof hints);
hints.ai_flags = AI_NUMERICHOST;
hints.ai_socktype = nfsargsp->sotype;
- if (getaddrinfo(hostp, port_no, &hints, &ai_nfs) == 0) {
+ if (getaddrinfo(hostp, "nfs", &hints, &ai_nfs) == 0) {
+#ifdef NFSKERB
if ((nfsargsp->flags & NFSMNT_KERB)) {
hints.ai_flags = 0;
- if (getnameinfo(ai->ai_addr, ai->ai_addrlen, host,
- sizeof host, serv, sizeof serv, 0) != 0) {
+ if (getnameinfo(ai_nfs->ai_addr, ai_nfs->ai_addrlen,
+ host, sizeof host, serv, sizeof serv, 0) != 0) {
warnx("can't reverse resolve net address");
return (0);
}
- hostp = host;
+ hostp = host;
}
+#endif /* NFSKERB */
} else {
hints.ai_flags = 0;
- if ((ecode = getaddrinfo(hostp, port_no, &hints, &ai_nfs)) != 0) {
+ if ((ecode = getaddrinfo(hostp, "nfs", &hints, &ai_nfs)) != 0) {
warnx("can't get net id for host/nfs: %s",
gai_strerror(ecode));
return (0);
@@ -673,15 +675,6 @@ getnfsargs(spec, nfsargsp)
orgcnt = retrycnt;
- if (mountmode == ANY || mountmode == V3) {
- nfsvers = 3;
- mntvers = 3;
- nfsargsp->flags |= NFSMNT_NFSV3;
- } else {
- nfsvers = 2;
- mntvers = 1;
- nfsargsp->flags &= ~NFSMNT_NFSV3;
- }
nfhret.stat = EACCES; /* Mark not yet successful */
ai = ai_nfs;
while (ai != NULL) {
@@ -706,87 +699,139 @@ getnfsargs(spec, nfsargsp)
tryagain:
retrycnt = orgcnt;
+ if (mountmode == ANY || mountmode == V3) {
+ nfsvers = 3;
+ mntvers = 3;
+ nfsargsp->flags |= NFSMNT_NFSV3;
+ } else {
+ nfsvers = 2;
+ mntvers = 1;
+ nfsargsp->flags &= ~NFSMNT_NFSV3;
+ }
while (retrycnt > 0) {
+ if (port_no != 0) {
+ if (ai_nfs->ai_family == AF_INET6) {
+ ((struct sockaddr_in6 *)ai_nfs->
+ ai_addr)->sin6_port = htons(port_no);
+ nfs_nb.len =
+ sizeof(struct sockaddr_in6);
+ } else {
+ ((struct sockaddr_in *)ai_nfs->
+ ai_addr)->sin_port = htons(port_no);
+ nfs_nb.len = sizeof(struct sockaddr_in);
+ }
+ memset(&nfs_ss, 0, sizeof(nfs_ss));
+ memcpy(&nfs_ss, ai_nfs->ai_addr, nfs_nb.len);
+ }
nfs_nb.buf = &nfs_ss;
nfs_nb.maxlen = sizeof nfs_ss;
- if (!rpcb_getaddr(RPCPROG_NFS, nfsvers, nconf,
- &nfs_nb, hostp)){
- if (rpc_createerr.cf_stat == RPC_SYSTEMERROR) {
- nfhret.stat = rpc_createerr.cf_error.re_errno;
+ if (port_no == 0 && !rpcb_getaddr(RPCPROG_NFS,
+ nfsvers, nconf, &nfs_nb, hostp)) {
+ switch (rpc_createerr.cf_stat) {
+ case RPC_SYSTEMERROR:
+ nfhret.stat =
+ rpc_createerr.cf_error.re_errno;
break;
- }
- if (rpc_createerr.cf_stat == RPC_UNKNOWNPROTO) {
+ case RPC_UNKNOWNPROTO:
nfhret.stat = EPROTONOSUPPORT;
break;
+ case RPC_PROGVERSMISMATCH:
+ if (mountmode == ANY) {
+ mountmode = V2;
+ goto tryagain;
+ } else {
+ errx(1,
+ "can't contact NFS server");
+ }
+ default:
}
if ((opflags & ISBGRND) == 0)
clnt_pcreateerror(
"mount_nfs: rpcbind on server");
- } else {
- pertry.tv_sec = 10;
- pertry.tv_usec = 0;
- /*
- * XXX relies on clnt_tcp_create to bind
- * to a reserved socket.
- */
- clp = clnt_tp_create(hostp, RPCPROG_MNT, mntvers,
- mnttcp_ok ? nconf : getnetconfigent("udp"));
- if (clp == NULL) {
- if ((opflags & ISBGRND) == 0)
- clnt_pcreateerror("Cannot MNT RPC");
+ }
+ /*
+ * Check if the nfs server supports this
+ * version of the protocol we want to use.
+ */
+ clnt_stat = pingnfsport(nfsvers, nconf, &nfs_nb);
+
+ if (clnt_stat == RPC_PROGVERSMISMATCH) {
+ if (mountmode == ANY) {
+ mountmode = V2;
+ goto tryagain;
} else {
- CLNT_CONTROL(clp, CLSET_RETRY_TIMEOUT,
- (char *)&pertry);
- clp->cl_auth = authsys_create_default();
- try.tv_sec = 10;
- try.tv_usec = 0;
- if (nfsargsp->flags & NFSMNT_KERB)
- nfhret.auth = RPCAUTH_KERB4;
- else
- nfhret.auth = RPCAUTH_UNIX;
- nfhret.vers = mntvers;
- clnt_stat = clnt_call(clp, RPCMNT_MOUNT,
- xdr_dir, spec, xdr_fh, &nfhret, try);
- if (clnt_stat != RPC_SUCCESS) {
- if (clnt_stat == RPC_PROGVERSMISMATCH) {
- if (mountmode == ANY) {
- mountmode = V2;
- goto tryagain;
- } else {
- errx(1, "%s",
- clnt_sperror(clp, "MNT RPC"));
- }
+ errx(1, "can't contact NFS server");
+ }
+ }
+ if (clnt_stat != RPC_SUCCESS)
+ errx(1, "can't contact NFS server");
+
+ pertry.tv_sec = 10;
+ pertry.tv_usec = 0;
+ /*
+ * XXX relies on clnt_tcp_create to bind
+ * to a reserved socket.
+ */
+ clp = clnt_tp_create(hostp, RPCPROG_MNT, mntvers,
+ mnttcp_ok ? nconf : getnetconfigent("udp"));
+ if (clp == NULL) {
+ if ((opflags & ISBGRND) == 0)
+ clnt_pcreateerror("Cannot MNT RPC");
+ } else {
+ CLNT_CONTROL(clp, CLSET_RETRY_TIMEOUT,
+ (char *)&pertry);
+ clp->cl_auth = authsys_create_default();
+ try.tv_sec = 10;
+ try.tv_usec = 0;
+ if (nfsargsp->flags & NFSMNT_KERB)
+ nfhret.auth = RPCAUTH_KERB4;
+ else
+ nfhret.auth = RPCAUTH_UNIX;
+ nfhret.vers = mntvers;
+ clnt_stat = clnt_call(clp, RPCMNT_MOUNT,
+ xdr_dir, spec, xdr_fh, &nfhret, try);
+ if (clnt_stat != RPC_SUCCESS) {
+ if (clnt_stat == RPC_PROGVERSMISMATCH) {
+ if (mountmode == ANY) {
+ mountmode = V2;
+ goto tryagain;
+ } else {
+ errx(1, "%s",
+ clnt_sperror(clp,
+ "MNT RPC"));
}
- if ((opflags & ISBGRND) == 0)
- warnx("%s", clnt_sperror(clp,
- "bad MNT RPC"));
- } else {
- retrycnt = 0;
}
- auth_destroy(clp->cl_auth);
- clnt_destroy(clp);
- so = RPC_ANYSOCK;
+ if ((opflags & ISBGRND) == 0)
+ warnx("%s", clnt_sperror(clp,
+ "bad MNT RPC"));
+ } else {
+ retrycnt = 0;
}
+ auth_destroy(clp->cl_auth);
+ clnt_destroy(clp);
+ so = RPC_ANYSOCK;
}
- if (--retrycnt > 0) {
- if (opflags & BGRND) {
- warnx("Cannot immediately mount %s:%s, "
- "backgrounding", hostp, spec);
- opflags &= ~BGRND;
- if ((i = fork())) {
- if (i == -1)
- err(1, "nqnfs 2");
+ }
+ if (--retrycnt > 0) {
+ if (opflags & BGRND) {
+ warnx("Cannot immediately mount %s:%s, "
+ "backgrounding", hostp, spec);
+ opflags &= ~BGRND;
+ if ((i = fork())) {
+ if (i == -1)
+ err(1, "nqnfs 2");
exit(0);
- }
- (void) setsid();
- (void) close(STDIN_FILENO);
- (void) close(STDOUT_FILENO);
- (void) close(STDERR_FILENO);
- (void) chdir("/");
- opflags |= ISBGRND;
}
- sleep(60);
+ (void) setsid();
+ (void) close(STDIN_FILENO);
+ (void) close(STDOUT_FILENO);
+ (void) close(STDERR_FILENO);
+ (void) chdir("/");
+ opflags |= ISBGRND;
+ } else {
+ exit (1);
}
+ sleep(60);
}
if (nfhret.stat == 0)
break;
@@ -865,6 +910,39 @@ xdr_fh(xdrsp, np)
return (0);
}
+/*
+ * Return RPC_SUCCESS if server responds on the port. We have to use
+ * clnt_tli_create and then call the null proc of the nfs-server, to
+ * be sure it works.
+ */
+enum clnt_stat
+pingnfsport(version, nconf, nfs_nb)
+ int version;
+ struct netconfig *nconf;
+ struct netbuf *nfs_nb;
+{
+ CLIENT *clp;
+ enum clnt_stat stat;
+ struct timeval try;
+ int so = RPC_ANYSOCK;
+
+ clp = clnt_tli_create(so, nconf, nfs_nb, RPCPROG_NFS,
+ version, 0, 0);
+
+ if (clp == NULL)
+ return rpc_createerr.cf_stat;
+
+ try.tv_sec = 10;
+ try.tv_usec = 0;
+
+ stat = clnt_call(clp, NFSPROC_NULL,
+ xdr_void, NULL, xdr_void, NULL, try);
+
+ clnt_destroy(clp);
+
+ return stat;
+}
+
void
usage()
{