diff options
author | Justus Winter <4winter@informatik.uni-hamburg.de> | 2014-05-03 01:02:35 +0200 |
---|---|---|
committer | Justus Winter <4winter@informatik.uni-hamburg.de> | 2014-10-05 23:31:08 +0200 |
commit | 78343e97c68fda079c66f2443bab6d1ee16f0b67 (patch) | |
tree | c2293ad70042bf6d839ab6ef5436e4c9fdf2cdcc /libports/port-deref.c | |
parent | 9a043bf1792be16a045dd2e2e37c2c6354425b9b (diff) |
libports: lock-less reference counting for port_info objects
* libports/ports.h (struct port_info): Use the new type.
* libports/lookup-port.c: No need to lock _ports_lock anymore.
* libports/bucket-iterate.c: Likewise.
* libports/complete-deallocate.c: Check if someone reacquired a
reference through a hash table lookup.
* libports/create-internal.c: Use the new reference counting primitives.
* libports/get-right.c: Likewise.
* libports/import-port.c: Likewise.
* libports/port-deref-weak.c: Likewise.
* libports/port-deref.c: Likewise.
* libports/port-ref-weak.c: Likewise.
* libports/port-ref.c: Likewise.
* libports/reallocate-from-external.c: Likewise.
* libports/transfer-right.c: Likewise.
* utils/rpctrace.c: Likewise.
Diffstat (limited to 'libports/port-deref.c')
-rw-r--r-- | libports/port-deref.c | 34 |
1 files changed, 16 insertions, 18 deletions
diff --git a/libports/port-deref.c b/libports/port-deref.c index cf9b2383..b97dd13d 100644 --- a/libports/port-deref.c +++ b/libports/port-deref.c @@ -25,26 +25,24 @@ void ports_port_deref (void *portstruct) { struct port_info *pi = portstruct; - int trieddroppingweakrefs = 0; - - retry: - - pthread_mutex_lock (&_ports_lock); - - if (pi->refcnt == 1 && pi->weakrefcnt - && pi->class->dropweak_routine && !trieddroppingweakrefs) + struct references result; + + if (pi->class->dropweak_routine) { - pthread_mutex_unlock (&_ports_lock); - (*pi->class->dropweak_routine) (pi); - trieddroppingweakrefs = 1; - goto retry; + /* If we need to call the dropweak routine, we need to hold one + reference while doing so. We use a weak reference for this + purpose, which we acquire by demoting our hard reference to a + weak one. */ + refcounts_demote (&pi->refcounts, &result); + + if (result.hard == 0 && result.weak > 1) + (*pi->class->dropweak_routine) (pi); + + refcounts_deref_weak (&pi->refcounts, &result); } - - assert (pi->refcnt); + else + refcounts_deref (&pi->refcounts, &result); - pi->refcnt--; - if (pi->refcnt == 0 && pi->weakrefcnt == 0) + if (result.hard == 0 && result.weak == 0) _ports_complete_deallocate (pi); - else - pthread_mutex_unlock (&_ports_lock); } |