git: b3daf5db761e - stable/12 - tmpfs_reclaim: detach unlinked node on dereferencing.
Konstantin Belousov
kib at FreeBSD.org
Thu Jan 21 12:47:21 UTC 2021
The branch stable/12 has been updated by kib:
URL: https://cgit.FreeBSD.org/src/commit/?id=b3daf5db761e2e32b8e4384587c593fb6c645e2f
commit b3daf5db761e2e32b8e4384587c593fb6c645e2f
Author: Konstantin Belousov <kib at FreeBSD.org>
AuthorDate: 2021-01-12 16:10:07 +0000
Commit: Konstantin Belousov <kib at FreeBSD.org>
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);
More information about the dev-commits-src-all
mailing list