summaryrefslogtreecommitdiff
path: root/sys
diff options
context:
space:
mode:
Diffstat (limited to 'sys')
-rw-r--r--sys/miscfs/union/union_vnops.c161
1 files changed, 144 insertions, 17 deletions
diff --git a/sys/miscfs/union/union_vnops.c b/sys/miscfs/union/union_vnops.c
index d921b568f8c..fa4482484e5 100644
--- a/sys/miscfs/union/union_vnops.c
+++ b/sys/miscfs/union/union_vnops.c
@@ -201,8 +201,40 @@ union_lookup(ap)
*/
if (upperdvp != NULLVP) {
FIXUP(dun);
+ /*
+ * If we're doing `..' in the underlying filesystem,
+ * we must drop our lock on the union node before
+ * going up the tree in the lower file system--if we block
+ * on the lowervp lock, and that's held by someone else
+ * coming down the tree and who's waiting for our lock,
+ * we would be hosed.
+ */
+ if (cnp->cn_flags & ISDOTDOT) {
+ /* retain lock on underlying VP: */
+ dun->un_flags |= UN_KLOCK;
+ VOP_UNLOCK(dvp);
+ }
uerror = union_lookup1(um->um_uppervp, &upperdvp,
&uppervp, cnp);
+ if (cnp->cn_flags & ISDOTDOT) {
+ if (dun->un_uppervp == upperdvp) {
+ /*
+ * we got the underlying bugger back locked...
+ * now take back the union node lock. Since we
+ * hold the uppervp lock, we can diddle union
+ * locking flags at will. :)
+ */
+ dun->un_flags |= UN_ULOCK;
+ }
+ /*
+ * if upperdvp got swapped out, it means we did
+ * some mount point magic, and we do not have
+ * dun->un_uppervp locked currently--so we get it
+ * locked here (don't set the UN_ULOCK flag).
+ */
+ VOP_LOCK(dvp);
+ }
+
/*if (uppervp == upperdvp)
dun->un_flags |= UN_KLOCK;*/
@@ -248,6 +280,12 @@ union_lookup(ap)
saved_cred = cnp->cn_cred;
cnp->cn_cred = um->um_cred;
}
+ /*
+ * we shouldn't have to worry about locking interactions
+ * between the lower layer and our union layer (w.r.t.
+ * `..' processing) because we don't futz with lowervp
+ * locks in the union-node instantiation code path.
+ */
lerror = union_lookup1(um->um_lowervp, &lowerdvp,
&lowervp, cnp);
if (um->um_op == UNMNT_BELOW)
@@ -319,6 +357,11 @@ union_lookup(ap)
/* case 2. */
if (uerror != 0 /* && (lerror == 0) */ ) {
if (lowervp->v_type == VDIR) { /* case 2b. */
+ /*
+ * We may be racing another process to make the
+ * upper-level shadow directory. Be careful with
+ * locks/etc!
+ */
dun->un_flags &= ~UN_ULOCK;
VOP_UNLOCK(upperdvp);
uerror = union_mkshadow(um, upperdvp, cnp, &uppervp);
@@ -350,6 +393,13 @@ union_lookup(ap)
if (*ap->a_vpp != dvp)
if (!lockparent || !(cnp->cn_flags & ISLASTCN))
VOP_UNLOCK(dvp);
+ if (cnp->cn_namelen == 1 &&
+ cnp->cn_nameptr[0] == '.' &&
+ *ap->a_vpp != dvp) {
+ panic("union_lookup returning. (%x) not same as startdir (%x)",
+ ap->a_vpp, dvp);
+ }
+
}
return (error);
@@ -537,15 +587,14 @@ union_close(ap)
--un->un_openl;
vp = un->un_lowervp;
}
-
-#ifdef DIAGNOSTIC
+#ifdef UNION_DIAGNOSTIC
/*
- * We should never encounter a vnode with both upper and
- * lower vnodes NULL.
+ * A stranded union node may end up here with both vnodes NULL,
+ * in which case we don't do anything.
*/
if (vp == NULLVP) {
- vprint("empty union vnode", vp);
- panic("union_close empty vnode");
+ vprint("empty union vnode", vp);
+ panic("union_close empty vnode");
}
#endif
@@ -941,6 +990,10 @@ union_remove(ap)
return (error);
}
+/* a_vp: directory in which to link
+ a_tdvp: new target of the link
+ a_cnp: name for the link
+ */
int
union_link(ap)
struct vop_link_args /* {
@@ -956,21 +1009,56 @@ union_link(ap)
un = VTOUNION(ap->a_vp);
+#ifdef DIAGNOSTIC
+ if (!(ap->a_cnp->cn_flags & LOCKPARENT)) {
+ printf("union_link called without LOCKPARENT set!\n");
+ error = EIO; /* need some error code for "caller is a bozo" */
+ } else
+#endif
if (ap->a_vp->v_op != ap->a_tdvp->v_op) {
tdvp = ap->a_tdvp;
} else {
struct union_node *tdun = VTOUNION(ap->a_tdvp);
if (tdun->un_uppervp == NULLVP) {
+ /*
+ * needs to be copied up before we can link it.
+ */
VOP_LOCK(ap->a_tdvp);
if (un->un_uppervp == tdun->un_dirvp) {
- un->un_flags &= ~UN_ULOCK;
- VOP_UNLOCK(un->un_uppervp);
+ VOP_UNLOCK(ap->a_vp);
}
error = union_copyup(tdun, 1, ap->a_cnp->cn_cred,
ap->a_cnp->cn_proc);
if (un->un_uppervp == tdun->un_dirvp) {
- VOP_LOCK(un->un_uppervp);
- un->un_flags |= UN_ULOCK;
+ /* During copyup, we dropped the lock on the
+ * dir and invalidated any saved namei lookup
+ * state for the directory we'll be entering
+ * the link in. We need to re-run the lookup
+ * in that directory to reset any state needed
+ * for VOP_LINK.
+ * Call relookup on the union-layer to
+ * reset the state.
+ */
+ vp = NULLVP;
+ if (un->un_uppervp == NULLVP ||
+ /*
+ * relookup starts with an unlocked node,
+ * and since LOCKPARENT is set returns
+ * the starting directory locked.
+ */
+ (error = relookup(ap->a_vp,
+ &vp, ap->a_cnp))) {
+ vrele(ap->a_vp);
+ VOP_UNLOCK(ap->a_tdvp);
+ return EROFS;
+ }
+ if (vp != NULLVP) {
+ /* The name we want to create has
+ mysteriously appeared (a race?) */
+ error = EEXIST;
+ VOP_UNLOCK(ap->a_tdvp);
+ goto croak;
+ }
}
VOP_UNLOCK(ap->a_tdvp);
}
@@ -982,6 +1070,7 @@ union_link(ap)
error = EROFS;
if (error) {
+croak:
vput(ap->a_vp);
return (error);
}
@@ -1011,6 +1100,7 @@ union_rename(ap)
struct vnode *fvp = ap->a_fvp;
struct vnode *tdvp = ap->a_tdvp;
struct vnode *tvp = ap->a_tvp;
+ struct union_node *unfile = (struct union_node *)0;
if (fdvp->v_op == union_vnodeop_p) { /* always true */
struct union_node *un = VTOUNION(fdvp);
@@ -1031,19 +1121,19 @@ union_rename(ap)
}
if (fvp->v_op == union_vnodeop_p) { /* always true */
- struct union_node *un = VTOUNION(fvp);
- if (un->un_uppervp == NULLVP) {
+ unfile = VTOUNION(fvp);
+ if (unfile->un_uppervp == NULLVP) {
/* XXX: should do a copyup */
error = EXDEV;
goto bad;
}
- if (un->un_lowervp != NULLVP)
+ if (unfile->un_lowervp != NULLVP)
ap->a_fcnp->cn_flags |= DOWHITEOUT;
- fvp = un->un_uppervp;
+ fvp = unfile->un_uppervp;
VREF(fvp);
- vrele(ap->a_fvp);
+ /* vrele(ap->a_fvp); */ /* hold for later */
}
if (tdvp->v_op == union_vnodeop_p) {
@@ -1056,7 +1146,7 @@ union_rename(ap)
* directory.
*/
error = EXDEV;
- goto bad;
+ goto badrele;
}
tdvp = un->un_uppervp;
@@ -1073,11 +1163,24 @@ union_rename(ap)
VREF(tvp);
un->un_flags |= UN_KLOCK;
}
+#if 0
+ /* XXX should we toss from the cache? */
+ if (un->un_flags & UN_CACHED) {
+ un->un_flags &= ~UN_CACHED;
+ LIST_REMOVE(un, un_cache);
+ }
+#endif
vput(ap->a_tvp);
}
- return (VOP_RENAME(fdvp, fvp, ap->a_fcnp, tdvp, tvp, ap->a_tcnp));
+ error = VOP_RENAME(fdvp, fvp, ap->a_fcnp, tdvp, tvp, ap->a_tcnp);
+ if (!error && unfile)
+ union_removed_upper(unfile);
+ vrele(ap->a_fvp);
+ return error;
+badrele:
+ vrele(ap->a_fvp);
bad:
vrele(fdvp);
vrele(fvp);
@@ -1358,6 +1461,12 @@ start:
if (un->un_uppervp != NULLVP) {
if (((un->un_flags & UN_ULOCK) == 0) &&
(vp->v_usecount != 0)) {
+ /*
+ * We MUST always use the order of: take upper
+ * vp lock, manipulate union node flags, drop
+ * upper vp lock. This code must not be an
+ * exception.
+ */
VOP_LOCK(un->un_uppervp);
un->un_flags |= UN_ULOCK;
}
@@ -1391,6 +1500,18 @@ start:
return (0);
}
+/*
+ * When operations want to vput() a union node yet retain a lock on
+ * the upper VP (say, to do some further operations like link(),
+ * mkdir(), ...), they set UN_KLOCK on the union node, then call
+ * vput() which calls VOP_UNLOCK() and comes here. union_unlock()
+ * unlocks the union node (leaving the upper VP alone), clears the
+ * KLOCK flag, and then returns to vput(). The caller then does whatever
+ * is left to do with the upper VP, and insures that it gets unlocked.
+ *
+ * If UN_KLOCK isn't set, then the upper VP is unlocked here.
+ */
+
int
union_unlock(ap)
struct vop_lock_args *ap;
@@ -1464,6 +1585,12 @@ union_print(ap)
vprint("uppervp", UPPERVP(vp));
if (LOWERVP(vp))
vprint("lowervp", LOWERVP(vp));
+ if (VTOUNION(vp)->un_dircache) {
+ struct vnode **vpp;
+
+ for (vpp = VTOUNION(vp)->un_dircache; *vpp != NULLVP; vpp++)
+ vprint("dircache:", *vpp);
+ }
return (0);
}