diff options
author | Justus Winter <4winter@informatik.uni-hamburg.de> | 2014-05-14 11:19:35 +0200 |
---|---|---|
committer | Justus Winter <4winter@informatik.uni-hamburg.de> | 2015-04-17 14:06:46 +0200 |
commit | c16eed2cb64089bf7d958db0fe85352f4ceefb4d (patch) | |
tree | 3240f6e025fa4c4fb47d7d270f285a7f0f9612e4 /libdiskfs/node-nrele.c | |
parent | 8c050fb080c6e1981dc8e5a97a2313cd24e9b4b4 (diff) |
libdiskfs: lock-less reference counting of nodes
* libdiskfs/diskfs.h (struct node): Use refcounts_t for reference counting.
(diskfs_node_refcnt_lock): Remove.
(diskfs_node_norefs,diskfs_drop_node): Change comments accordingly.
* libdiskfs/init-init.c: Adjust accordingly.
* libdiskfs/node-drop.c: Likewise.
* libdiskfs/node-make.c: Likewise.
* libdiskfs/node-nput.c: Likewise.
* libdiskfs/node-nputl.c: Likewise.
* libdiskfs/node-nref.c: Likewise.
* libdiskfs/node-nrefl.c: Likewise.
* libdiskfs/node-nrele.c: Likewise.
* libdiskfs/node-nrelel.c: Likewise.
* ext2fs/inode.c: Likewise.
* fatfs/inode.c: Likewise.
* isofs/inode.c: Likewise.
* tmpfs/node.c: Likewise.
* doc/hurd.texi: Likewise.
Diffstat (limited to 'libdiskfs/node-nrele.c')
-rw-r--r-- | libdiskfs/node-nrele.c | 48 |
1 files changed, 23 insertions, 25 deletions
diff --git a/libdiskfs/node-nrele.c b/libdiskfs/node-nrele.c index cc680893..d9628469 100644 --- a/libdiskfs/node-nrele.c +++ b/libdiskfs/node-nrele.c @@ -28,38 +28,36 @@ void diskfs_nrele (struct node *np) { - int tried_drop_softrefs = 0; + struct references result; - loop: - pthread_spin_lock (&diskfs_node_refcnt_lock); - assert (np->references); - np->references--; - if (np->references + np->light_references == 0) - { - pthread_mutex_lock (&np->lock); - diskfs_drop_node (np); - } - else if (np->references == 0) + /* While we call the diskfs_try_dropping_softrefs, we need to hold + one reference. We use a weak reference for this purpose, which + we acquire by demoting our hard reference to a weak one. */ + refcounts_demote (&np->refcounts, &result); + + if (result.hard == 0) { pthread_mutex_lock (&np->lock); - pthread_spin_unlock (&diskfs_node_refcnt_lock); diskfs_lost_hardrefs (np); - if (!np->dn_stat.st_nlink && !tried_drop_softrefs) + if (!np->dn_stat.st_nlink) { - /* Same issue here as in nput; see that for explanation */ - pthread_spin_lock (&diskfs_node_refcnt_lock); - np->references++; - pthread_spin_unlock (&diskfs_node_refcnt_lock); - + /* There are no links. If there are soft references that + can be dropped, we can't let them postpone deallocation. + So attempt to drop them. But that's a user-supplied + routine, which might result in further recursive calls to + the ref-counting system. This is not a problem, as we + hold a weak reference ourselves. */ diskfs_try_dropping_softrefs (np); - tried_drop_softrefs = 1; - - /* Now we can drop the reference back... */ - pthread_mutex_unlock (&np->lock); - goto loop; } pthread_mutex_unlock (&np->lock); } - else - pthread_spin_unlock (&diskfs_node_refcnt_lock); + + /* Finally get rid of our reference. */ + refcounts_deref_weak (&np->refcounts, &result); + + if (result.hard == 0 && result.weak == 0) + { + pthread_mutex_lock (&np->lock); + diskfs_drop_node (np); + } } |