REFCNT(9) | Kernel Developer's Manual | REFCNT(9) |
struct refcnt;
void
refcnt_init(struct refcnt *refcnt);
void
refcnt_fini(struct refcnt *refcnt);
int
refcnt_acquire(struct refcnt *refcnt);
bool
refcnt_release_tryenter(struct refcnt *refcnt, kmutex_t *interlock);
void
refcnt_release_signal(struct refcnt *refcnt, kmutex_t *interlock, kcondvar_t *cv);
void
refcnt_release_broadcast(struct refcnt *refcnt, kmutex_t *interlock, kcondvar_t *cv);
void
refcnt_release_wait(struct refcnt *refcnt, kmutex_t *interlock, kcondvar_t *cv);
void
refcnt_release_wait_unlocked(struct refcnt *refcnt, kmutex_t *interlock, kcondvar_t *cv);
struct refcnt
object, which users should treat as opaque and should not inspect or copy. The struct refcnt
object must be initialized with refcnt_init() before use, and finalized with refcnt_fini() when done. References are acquired with refcnt_acquire(), and released with one of the refcnt_release_*() routines.The REFCNT abstraction admits multiple approaches to freeing objects when they are no longer referenced:
In this case, users of the object should use refcnt_release_signal() or refcnt_release_broadcast() to notify the free operation when they are done, and the free operation should use refcnt_release_wait() or refcnt_release_wait_unlocked() to wait until there are no users left.
In this case, each user of the object should use refcnt_release_tryenter(), which returns true and enters a mutex if and only if that user is the last one, so that the last user can remove it from any global tables and free it.
The transition from nonzero references to zero references is guaranteed to happen under an interlock, except in the case of refcnt_release_wait_unlocked() whose caller must guarantee that no new references to the object can be created.
May enter interlock even if it returns false, in case another thread acquired a reference at the same time.
May enter interlock even if it does not signal cv, in case another thread acquired a reference at the same time.
May enter interlock even if it does not broadcast cv, in case another thread acquired a reference at the same time.
The caller must hold interlock.
May sleep.
The caller must guarantee no new references can be made before calling refcnt_release_wait_unlocked(), e.g. by setting a flag in the object causing prospective users to skip it before attempting refcnt_acquire().
May enter interlock and exit it on return. May sleep.
As an optimization, refcnt_release_wait_unlocked() may avoid interprocessor synchronization if the caller holds the only reference.
May be used only under DIAGNOSTIC for assertions. Do not make run-time decisions on the basis of refcnt_referenced_p().
May be used only under DIAGNOSTIC for assertions. Do not make run-time decisions on the basis of refcnt_exclusive_p().
The implementation is untested.
April 11, 2015 | NetBSD 6.1_STABLE |