- 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>
This commit is contained in:
Andrey A. Chernov 2001-03-31 20:45:21 +00:00
parent 7ea7bc591b
commit deffdffafa

View File

@ -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()
{