The branch stable/12 has been updated by kib:

URL: 
https://cgit.FreeBSD.org/src/commit/?id=b3daf5db761e2e32b8e4384587c593fb6c645e2f

commit b3daf5db761e2e32b8e4384587c593fb6c645e2f
Author:     Konstantin Belousov <[email protected]>
AuthorDate: 2021-01-12 16:10:07 +0000
Commit:     Konstantin Belousov <[email protected]>
CommitDate: 2021-01-21 12:45:32 +0000

    tmpfs_reclaim: detach unlinked node on dereferencing.
    
    (cherry picked from commit 2d1e4220ebd50b9220d3266754425f025c786108)
---
 sys/fs/tmpfs/tmpfs_vnops.c | 33 ++++++++++++++++++++++++++++-----
 1 file changed, 28 insertions(+), 5 deletions(-)

diff --git a/sys/fs/tmpfs/tmpfs_vnops.c b/sys/fs/tmpfs/tmpfs_vnops.c
index 84134e48e83e..bfb84c508c78 100644
--- a/sys/fs/tmpfs/tmpfs_vnops.c
+++ b/sys/fs/tmpfs/tmpfs_vnops.c
@@ -1305,10 +1305,12 @@ tmpfs_reclaim(struct vop_reclaim_args *v)
        struct vnode *vp;
        struct tmpfs_mount *tmp;
        struct tmpfs_node *node;
+       bool unlock, tm_locked;
 
        vp = v->a_vp;
        node = VP_TO_TMPFS_NODE(vp);
        tmp = VFS_TO_TMPFS(vp->v_mount);
+       tm_locked = false;
 
        if (vp->v_type == VREG)
                tmpfs_destroy_vobject(vp, node->tn_reg.tn_aobj);
@@ -1318,7 +1320,15 @@ tmpfs_reclaim(struct vop_reclaim_args *v)
        if (tmpfs_use_nc(vp))
                cache_purge(vp);
 
+relock:
        TMPFS_NODE_LOCK(node);
+       if (!tm_locked && node->tn_links == 0 &&
+           (node->tn_vpstate & TMPFS_VNODE_ALLOCATING) == 0) {
+               TMPFS_NODE_UNLOCK(node);
+               TMPFS_LOCK(tmp);
+               tm_locked = true;
+               goto relock;
+       }
        tmpfs_free_vp(vp);
 
        /*
@@ -1328,11 +1338,18 @@ tmpfs_reclaim(struct vop_reclaim_args *v)
         */
        if (node->tn_links == 0 &&
            (node->tn_vpstate & TMPFS_VNODE_ALLOCATING) == 0) {
+               MPASS(tm_locked);
                node->tn_vpstate = TMPFS_VNODE_DOOMED;
+               unlock = !tmpfs_free_node_locked(tmp, node, true);
+       } else {
+               unlock = true;
+       }
+
+       if (unlock) {
                TMPFS_NODE_UNLOCK(node);
-               tmpfs_free_node(tmp, node);
-       } else
-               TMPFS_NODE_UNLOCK(node);
+               if (tm_locked)
+                       TMPFS_UNLOCK(tmp);
+       }
 
        MPASS(vp->v_data == NULL);
        return (0);
@@ -1543,6 +1560,7 @@ tmpfs_vptocnp(struct vop_vptocnp_args *ap)
        }
 restart:
        TMPFS_LOCK(tm);
+restart_locked:
        LIST_FOREACH_SAFE(tnp, &tm->tm_nodes_used, tn_entries, tnp1) {
                if (tnp->tn_type != VDIR)
                        continue;
@@ -1580,8 +1598,13 @@ restart:
                } else {
                        KASSERT(tnp->tn_refcount > 0,
                            ("node %p refcount zero", tnp));
-                       tnp1 = LIST_NEXT(tnp, tn_entries);
-                       TMPFS_NODE_UNLOCK(tnp);
+                       if (tnp->tn_attached) {
+                               tnp1 = LIST_NEXT(tnp, tn_entries);
+                               TMPFS_NODE_UNLOCK(tnp);
+                       } else {
+                               TMPFS_NODE_UNLOCK(tnp);
+                               goto restart_locked;
+                       }
                }
        }
        TMPFS_UNLOCK(tm);
_______________________________________________
[email protected] mailing list
https://lists.freebsd.org/mailman/listinfo/dev-commits-src-all
To unsubscribe, send any mail to "[email protected]"

Reply via email to