summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/nfs/nfs_vnops.c84
1 files changed, 48 insertions, 36 deletions
diff --git a/sys/nfs/nfs_vnops.c b/sys/nfs/nfs_vnops.c
index 4449d42e1b3..480f9879897 100644
--- a/sys/nfs/nfs_vnops.c
+++ b/sys/nfs/nfs_vnops.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: nfs_vnops.c,v 1.171 2017/02/22 11:42:46 mpi Exp $ */
+/* $OpenBSD: nfs_vnops.c,v 1.172 2018/04/17 07:45:24 mpi Exp $ */
/* $NetBSD: nfs_vnops.c,v 1.62.4.1 1996/07/08 20:26:52 jtc Exp $ */
/*
@@ -769,6 +769,7 @@ nfs_lookup(void *v)
flags = cnp->cn_flags;
*vpp = NULLVP;
+ newvp = NULLVP;
if ((flags & ISLASTCN) && (dvp->v_mount->mnt_flag & MNT_RDONLY) &&
(cnp->cn_nameiop == DELETE || cnp->cn_nameiop == RENAME))
return (EROFS);
@@ -836,8 +837,10 @@ nfs_lookup(void *v)
if (cnp->cn_nameiop != LOOKUP && (flags & ISLASTCN))
cnp->cn_flags |= SAVENAME;
if ((!lockparent || !(flags & ISLASTCN)) &&
- newvp != dvp)
+ newvp != dvp) {
VOP_UNLOCK(dvp, p);
+ cnp->cn_flags |= PDIRUNLOCK;
+ }
return (0);
}
cache_purge(newvp);
@@ -1282,7 +1285,6 @@ nfs_mknodrpc(struct vnode *dvp, struct vnode **vpp, struct componentname *cnp,
rdev = nfs_xdrneg1;
else {
VOP_ABORTOP(dvp, cnp);
- vput(dvp);
return (EOPNOTSUPP);
}
nfsstats.rpccnt[NFSPROC_MKNOD]++;
@@ -1318,10 +1320,6 @@ nfs_mknodrpc(struct vnode *dvp, struct vnode **vpp, struct componentname *cnp,
if (!error) {
nfsm_mtofh(dvp, newvp, info.nmi_v3, gotvp);
if (!gotvp) {
- if (newvp) {
- vrele(newvp);
- newvp = NULL;
- }
error = nfs_lookitup(dvp, cnp->cn_nameptr,
cnp->cn_namelen, cnp->cn_cred, cnp->cn_proc, &np);
if (!error)
@@ -1335,7 +1333,7 @@ nfs_mknodrpc(struct vnode *dvp, struct vnode **vpp, struct componentname *cnp,
nfsmout:
if (error) {
if (newvp)
- vrele(newvp);
+ vput(newvp);
} else {
if (cnp->cn_flags & MAKEENTRY)
nfs_cache_enter(dvp, newvp, cnp);
@@ -1345,7 +1343,6 @@ nfsmout:
VTONFS(dvp)->n_flag |= NMODIFIED;
if (!wccflag)
NFS_INVALIDATE_ATTRCACHE(VTONFS(dvp));
- vrele(dvp);
return (error);
}
@@ -1362,9 +1359,10 @@ nfs_mknod(void *v)
error = nfs_mknodrpc(ap->a_dvp, &newvp, ap->a_cnp, ap->a_vap);
if (!error)
- vrele(newvp);
+ vput(newvp);
VN_KNOTE(ap->a_dvp, NOTE_WRITE);
+ vput(ap->a_dvp);
return (error);
}
@@ -1390,8 +1388,11 @@ nfs_create(void *v)
/*
* Oops, not for me..
*/
- if (vap->va_type == VSOCK)
- return (nfs_mknodrpc(dvp, ap->a_vpp, cnp, vap));
+ if (vap->va_type == VSOCK) {
+ error = nfs_mknodrpc(dvp, ap->a_vpp, cnp, vap);
+ vput(dvp);
+ return (error);
+ }
if (vap->va_vaflags & VA_EXCLUSIVE)
fmode |= O_EXCL;
@@ -1430,10 +1431,6 @@ again:
if (!error) {
nfsm_mtofh(dvp, newvp, info.nmi_v3, gotvp);
if (!gotvp) {
- if (newvp) {
- vrele(newvp);
- newvp = NULL;
- }
error = nfs_lookitup(dvp, cnp->cn_nameptr,
cnp->cn_namelen, cnp->cn_cred, cnp->cn_proc, &np);
if (!error)
@@ -1446,12 +1443,14 @@ again:
nfsmout:
if (error) {
+ if (newvp) {
+ vput(newvp);
+ newvp = NULL;
+ }
if (info.nmi_v3 && (fmode & O_EXCL) && error == NFSERR_NOTSUPP) {
fmode &= ~O_EXCL;
goto again;
}
- if (newvp)
- vrele(newvp);
} else if (info.nmi_v3 && (fmode & O_EXCL))
error = nfs_setattrrpc(newvp, vap, cnp->cn_cred, cnp->cn_proc);
if (!error) {
@@ -1464,7 +1463,7 @@ nfsmout:
if (!wccflag)
NFS_INVALIDATE_ATTRCACHE(VTONFS(dvp));
VN_KNOTE(ap->a_dvp, NOTE_WRITE);
- vrele(dvp);
+ vput(dvp);
return (error);
}
@@ -1530,12 +1529,13 @@ nfs_remove(void *v)
error = nfs_sillyrename(dvp, vp, cnp);
pool_put(&namei_pool, cnp->cn_pnbuf);
NFS_INVALIDATE_ATTRCACHE(np);
- vrele(dvp);
- vrele(vp);
-
VN_KNOTE(vp, NOTE_DELETE);
VN_KNOTE(dvp, NOTE_WRITE);
-
+ if (vp == dvp)
+ vrele(vp);
+ else
+ vput(vp);
+ vput(dvp);
return (error);
}
@@ -1628,7 +1628,7 @@ nfs_rename(void *v)
if (tvp && tvp->v_usecount > 1 && !VTONFS(tvp)->n_sillyrename &&
tvp->v_type != VDIR && !nfs_sillyrename(tdvp, tvp, tcnp)) {
VN_KNOTE(tvp, NOTE_DELETE);
- vrele(tvp);
+ vput(tvp);
tvp = NULL;
}
@@ -1827,13 +1827,13 @@ nfs_symlink(void *v)
nfsmout:
if (newvp)
- vrele(newvp);
+ vput(newvp);
pool_put(&namei_pool, cnp->cn_pnbuf);
VTONFS(dvp)->n_flag |= NMODIFIED;
if (!wccflag)
NFS_INVALIDATE_ATTRCACHE(VTONFS(dvp));
VN_KNOTE(dvp, NOTE_WRITE);
- vrele(dvp);
+ vput(dvp);
return (error);
}
@@ -1904,7 +1904,7 @@ nfsmout:
}
if (error) {
if (newvp)
- vrele(newvp);
+ vput(newvp);
} else {
VN_KNOTE(dvp, NOTE_WRITE|NOTE_LINK);
if (cnp->cn_flags & MAKEENTRY)
@@ -1912,7 +1912,7 @@ nfsmout:
*ap->a_vpp = newvp;
}
pool_put(&namei_pool, cnp->cn_pnbuf);
- vrele(dvp);
+ vput(dvp);
return (error);
}
@@ -1936,7 +1936,7 @@ nfs_rmdir(void *v)
if (dvp == vp) {
vrele(dvp);
- vrele(dvp);
+ vput(dvp);
pool_put(&namei_pool, cnp->cn_pnbuf);
return (EINVAL);
}
@@ -1964,8 +1964,8 @@ nfsmout:
VN_KNOTE(vp, NOTE_DELETE);
cache_purge(vp);
- vrele(vp);
- vrele(dvp);
+ vput(vp);
+ vput(dvp);
/*
* Kludge: Map ENOENT => 0 assuming that you have a reply to a retry.
*/
@@ -2492,7 +2492,10 @@ nfs_readdirplusrpc(struct vnode *vp, struct uio *uiop, struct ucred *cred,
nfsm_adv(nfsm_rndup(i));
}
if (newvp != NULLVP) {
- vrele(newvp);
+ if (newvp == vp)
+ vrele(newvp);
+ else
+ vput(newvp);
newvp = NULLVP;
}
nfsm_dissect(tl, u_int32_t *, NFSX_UNSIGNED);
@@ -2533,8 +2536,12 @@ nfs_readdirplusrpc(struct vnode *vp, struct uio *uiop, struct ucred *cred,
}
nfsmout:
- if (newvp != NULLVP)
- vrele(newvp);
+ if (newvp != NULLVP) {
+ if (newvp == vp)
+ vrele(newvp);
+ else
+ vput(newvp);
+ }
return (error);
}
@@ -2659,7 +2666,10 @@ nfs_lookitup(struct vnode *dvp, char *name, int len, struct ucred *cred,
nfsm_postop_attr(newvp, attrflag);
if (!attrflag && *npp == NULL) {
m_freem(info.nmi_mrep);
- vrele(newvp);
+ if (newvp == dvp)
+ vrele(newvp);
+ else
+ vput(newvp);
return (ENOENT);
}
} else
@@ -2669,8 +2679,10 @@ nfs_lookitup(struct vnode *dvp, char *name, int len, struct ucred *cred,
nfsmout:
if (npp && *npp == NULL) {
if (error) {
- if (newvp)
+ if (newvp == dvp)
vrele(newvp);
+ else
+ vput(newvp);
} else
*npp = np;
}