Merge 208603, 209946, and 209948 to the new NFS client:
Move attribute cache flushes from VOP_OPEN() to VOP_LOOKUP() to provide more graceful recovery for stale filehandles and eliminate the need for conditionally clearing the attribute cache in the !NMODIFIED case in VOP_OPEN(). Reviewed by: rmacklem MFC after: 2 weeks
This commit is contained in:
parent
b539225cf9
commit
9e1b696f65
@ -505,9 +505,11 @@ nfs_open(struct vop_open_args *ap)
|
|||||||
(void) nfsrpc_close(vp, 0, ap->a_td);
|
(void) nfsrpc_close(vp, 0, ap->a_td);
|
||||||
return (error);
|
return (error);
|
||||||
}
|
}
|
||||||
|
mtx_lock(&np->n_mtx);
|
||||||
np->n_attrstamp = 0;
|
np->n_attrstamp = 0;
|
||||||
if (vp->v_type == VDIR)
|
if (vp->v_type == VDIR)
|
||||||
np->n_direofoffset = 0;
|
np->n_direofoffset = 0;
|
||||||
|
mtx_unlock(&np->n_mtx);
|
||||||
error = VOP_GETATTR(vp, &vattr, ap->a_cred);
|
error = VOP_GETATTR(vp, &vattr, ap->a_cred);
|
||||||
if (error) {
|
if (error) {
|
||||||
if (NFS_ISV4(vp))
|
if (NFS_ISV4(vp))
|
||||||
@ -520,14 +522,6 @@ nfs_open(struct vop_open_args *ap)
|
|||||||
np->n_change = vattr.va_filerev;
|
np->n_change = vattr.va_filerev;
|
||||||
mtx_unlock(&np->n_mtx);
|
mtx_unlock(&np->n_mtx);
|
||||||
} else {
|
} else {
|
||||||
struct thread *td = curthread;
|
|
||||||
|
|
||||||
if (np->n_ac_ts_syscalls != td->td_syscalls ||
|
|
||||||
np->n_ac_ts_tid != td->td_tid ||
|
|
||||||
td->td_proc == NULL ||
|
|
||||||
np->n_ac_ts_pid != td->td_proc->p_pid) {
|
|
||||||
np->n_attrstamp = 0;
|
|
||||||
}
|
|
||||||
mtx_unlock(&np->n_mtx);
|
mtx_unlock(&np->n_mtx);
|
||||||
error = VOP_GETATTR(vp, &vattr, ap->a_cred);
|
error = VOP_GETATTR(vp, &vattr, ap->a_cred);
|
||||||
if (error) {
|
if (error) {
|
||||||
@ -987,7 +981,7 @@ nfs_lookup(struct vop_lookup_args *ap)
|
|||||||
int flags = cnp->cn_flags;
|
int flags = cnp->cn_flags;
|
||||||
struct vnode *newvp;
|
struct vnode *newvp;
|
||||||
struct nfsmount *nmp;
|
struct nfsmount *nmp;
|
||||||
struct nfsnode *np;
|
struct nfsnode *np, *newnp;
|
||||||
int error = 0, attrflag, dattrflag, ltype;
|
int error = 0, attrflag, dattrflag, ltype;
|
||||||
struct thread *td = cnp->cn_thread;
|
struct thread *td = cnp->cn_thread;
|
||||||
struct nfsfh *nfhp;
|
struct nfsfh *nfhp;
|
||||||
@ -1023,11 +1017,27 @@ nfs_lookup(struct vop_lookup_args *ap)
|
|||||||
* change time of the file matches our cached copy.
|
* change time of the file matches our cached copy.
|
||||||
* Otherwise, we discard the cache entry and fallback
|
* Otherwise, we discard the cache entry and fallback
|
||||||
* to doing a lookup RPC.
|
* to doing a lookup RPC.
|
||||||
|
*
|
||||||
|
* To better handle stale file handles and attributes,
|
||||||
|
* clear the attribute cache of this node if it is a
|
||||||
|
* leaf component, part of an open() call, and not
|
||||||
|
* locally modified before fetching the attributes.
|
||||||
|
* This should allow stale file handles to be detected
|
||||||
|
* here where we can fall back to a LOOKUP RPC to
|
||||||
|
* recover rather than having nfs_open() detect the
|
||||||
|
* stale file handle and failing open(2) with ESTALE.
|
||||||
*/
|
*/
|
||||||
newvp = *vpp;
|
newvp = *vpp;
|
||||||
|
newnp = VTONFS(newvp);
|
||||||
|
if ((flags & (ISLASTCN | ISOPEN)) == (ISLASTCN | ISOPEN) &&
|
||||||
|
!(newnp->n_flag & NMODIFIED)) {
|
||||||
|
mtx_lock(&newnp->n_mtx);
|
||||||
|
newnp->n_attrstamp = 0;
|
||||||
|
mtx_unlock(&newnp->n_mtx);
|
||||||
|
}
|
||||||
if (nfscl_nodeleg(newvp, 0) == 0 ||
|
if (nfscl_nodeleg(newvp, 0) == 0 ||
|
||||||
(!VOP_GETATTR(newvp, &vattr, cnp->cn_cred)
|
(VOP_GETATTR(newvp, &vattr, cnp->cn_cred) == 0 &&
|
||||||
&& vattr.va_ctime.tv_sec == VTONFS(newvp)->n_ctime)) {
|
vattr.va_ctime.tv_sec == newnp->n_ctime)) {
|
||||||
NFSINCRGLOBAL(newnfsstats.lookupcache_hits);
|
NFSINCRGLOBAL(newnfsstats.lookupcache_hits);
|
||||||
if (cnp->cn_nameiop != LOOKUP &&
|
if (cnp->cn_nameiop != LOOKUP &&
|
||||||
(flags & ISLASTCN))
|
(flags & ISLASTCN))
|
||||||
@ -1213,6 +1223,18 @@ nfs_lookup(struct vop_lookup_args *ap)
|
|||||||
if (attrflag)
|
if (attrflag)
|
||||||
(void) nfscl_loadattrcache(&newvp, &nfsva, NULL, NULL,
|
(void) nfscl_loadattrcache(&newvp, &nfsva, NULL, NULL,
|
||||||
0, 1);
|
0, 1);
|
||||||
|
else if ((flags & (ISLASTCN | ISOPEN)) == (ISLASTCN | ISOPEN) &&
|
||||||
|
!(np->n_flag & NMODIFIED)) {
|
||||||
|
/*
|
||||||
|
* Flush the attribute cache when opening a
|
||||||
|
* leaf node to ensure that fresh attributes
|
||||||
|
* are fetched in nfs_open() since we did not
|
||||||
|
* fetch attributes from the LOOKUP reply.
|
||||||
|
*/
|
||||||
|
mtx_lock(&np->n_mtx);
|
||||||
|
np->n_attrstamp = 0;
|
||||||
|
mtx_unlock(&np->n_mtx);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (cnp->cn_nameiop != LOOKUP && (flags & ISLASTCN))
|
if (cnp->cn_nameiop != LOOKUP && (flags & ISLASTCN))
|
||||||
cnp->cn_flags |= SAVENAME;
|
cnp->cn_flags |= SAVENAME;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user