diff options
author | Tobias Brunner <tobias@strongswan.org> | 2014-04-11 15:13:22 +0200 |
---|---|---|
committer | Tobias Brunner <tobias@strongswan.org> | 2014-04-24 17:53:42 +0200 |
commit | efedd0d21e4caf6edae6872f29c470a464e1917a (patch) | |
tree | 74b0784a419225a74f0ec73d148bfcd04c4a7345 /src | |
parent | 1dfd11fd9249704ca29675c5ee11059a79aac203 (diff) | |
download | strongswan-efedd0d21e4caf6edae6872f29c470a464e1917a.tar.bz2 strongswan-efedd0d21e4caf6edae6872f29c470a464e1917a.tar.xz |
utils: Add ref_cur() to retrieve the current value of a reference counter
On many architectures it is safe to read the value directly (those
using cache coherency protocols, and with atomic loads for 32-bit
values) but it is not if that's not the case or if we ever decide to
make refcount_t 64-bit (load not atomic on x86).
So make sure the operation is actually atomic and that users do not
have to care about the size of refcount_t.
Diffstat (limited to 'src')
-rw-r--r-- | src/libstrongswan/utils/utils.c | 14 | ||||
-rw-r--r-- | src/libstrongswan/utils/utils.h | 13 |
2 files changed, 24 insertions, 3 deletions
diff --git a/src/libstrongswan/utils/utils.c b/src/libstrongswan/utils/utils.c index fe80edb82..e4da6ddb9 100644 --- a/src/libstrongswan/utils/utils.c +++ b/src/libstrongswan/utils/utils.c @@ -528,7 +528,6 @@ refcount_t ref_get(refcount_t *ref) pthread_mutex_lock(&ref_mutex); current = ++(*ref); pthread_mutex_unlock(&ref_mutex); - return current; } @@ -546,6 +545,19 @@ bool ref_put(refcount_t *ref) } /** + * Current refcount + */ +refcount_t ref_cur(refcount_t *ref) +{ + refcount_t current; + + pthread_mutex_lock(&ref_mutex); + current = *ref; + pthread_mutex_unlock(&ref_mutex); + return current; +} + +/** * Single mutex for all compare and swap operations. */ static pthread_mutex_t cas_mutex = PTHREAD_MUTEX_INITIALIZER; diff --git a/src/libstrongswan/utils/utils.h b/src/libstrongswan/utils/utils.h index a55e7d831..4b2990371 100644 --- a/src/libstrongswan/utils/utils.h +++ b/src/libstrongswan/utils/utils.h @@ -752,6 +752,7 @@ typedef u_int refcount_t; #define ref_get(ref) __sync_add_and_fetch(ref, 1) #define ref_put(ref) (!__sync_sub_and_fetch(ref, 1)) +#define ref_cur(ref) __sync_fetch_and_add(ref, 0) #define cas_bool(ptr, oldval, newval) \ (__sync_bool_compare_and_swap(ptr, oldval, newval)) @@ -763,7 +764,7 @@ typedef u_int refcount_t; /** * Get a new reference. * - * Increments the reference counter atomic. + * Increments the reference counter atomically. * * @param ref pointer to ref counter * @return new value of ref @@ -773,7 +774,7 @@ refcount_t ref_get(refcount_t *ref); /** * Put back a unused reference. * - * Decrements the reference counter atomic and + * Decrements the reference counter atomically and * says if more references available. * * @param ref pointer to ref counter @@ -782,6 +783,14 @@ refcount_t ref_get(refcount_t *ref); bool ref_put(refcount_t *ref); /** + * Get the current value of the reference counter. + * + * @param ref pointer to ref counter + * @return current value of ref + */ +refcount_t ref_cur(refcount_t *ref); + +/** * Atomically replace value of ptr with newval if it currently equals oldval. * * @param ptr pointer to variable |