X-Git-Url: https://git.liburcu.org/?p=urcu.git;a=blobdiff_plain;f=urcu-pointer.h;h=dc1a0dac60718b15d1f8eb844b3feca315dd4068;hp=b10aa13a4ad5c9689f7364476b263531e86e4278;hb=7dea28e7c3827cfcfaea71caa81648ea8a19cfe5;hpb=2b5554c9169ea1ef93dffacb7671a00f9e7de511 diff --git a/urcu-pointer.h b/urcu-pointer.h index b10aa13..dc1a0da 100644 --- a/urcu-pointer.h +++ b/urcu-pointer.h @@ -6,7 +6,7 @@ * * Userspace RCU header. Operations on pointers. * - * Copyright (c) 2009 Mathieu Desnoyers + * Copyright (c) 2009 Mathieu Desnoyers * Copyright (c) 2009 Paul E. McKenney, IBM Corporation. * * This library is free software; you can redistribute it and/or @@ -28,11 +28,15 @@ #include #include -#include +#include -#ifdef _LGPL_SOURCE +#ifdef __cplusplus +extern "C" { +#endif -#include +#if defined(_LGPL_SOURCE) || defined(URCU_INLINE_SMALL_FUNCTIONS) + +#include /* * rcu_dereference(ptr) @@ -43,9 +47,9 @@ #define rcu_dereference _rcu_dereference /* - * rcu_cmpxchg_pointer(type **ptr, type *new, type *old) + * type *rcu_cmpxchg_pointer(type **ptr, type *new, type *old) * type *rcu_xchg_pointer(type **ptr, type *new) - * type *rcu_set_pointer(type **ptr, type *new) + * void rcu_set_pointer(type **ptr, type *new) * * RCU pointer updates. * @ptr: address of the pointer to modify @@ -58,48 +62,59 @@ #define rcu_xchg_pointer _rcu_xchg_pointer #define rcu_set_pointer _rcu_set_pointer -#else /* !_LGPL_SOURCE */ +#else /* !(defined(_LGPL_SOURCE) || defined(URCU_INLINE_SMALL_FUNCTIONS)) */ extern void *rcu_dereference_sym(void *p); #define rcu_dereference(p) \ + __extension__ \ ({ \ - typeof(p) _________p1 = \ - rcu_dereference_sym((void *)(p)); \ + __typeof__(p) _________p1 = URCU_FORCE_CAST(__typeof__(p), \ + rcu_dereference_sym(URCU_FORCE_CAST(void *, p))); \ (_________p1); \ }) extern void *rcu_cmpxchg_pointer_sym(void **p, void *old, void *_new); #define rcu_cmpxchg_pointer(p, old, _new) \ + __extension__ \ ({ \ - typeof(*p) _________pold = (old); \ - typeof(*p) _________pnew = (_new); \ - typeof(*p) _________p1 = \ - rcu_cmpxchg_pointer_sym((void **)(p), _________pold, \ - _________pnew); \ + __typeof__(*(p)) _________pold = (old); \ + __typeof__(*(p)) _________pnew = (_new); \ + __typeof__(*(p)) _________p1 = URCU_FORCE_CAST(__typeof__(*(p)), \ + rcu_cmpxchg_pointer_sym(URCU_FORCE_CAST(void **, p), \ + _________pold, \ + _________pnew)); \ (_________p1); \ }) extern void *rcu_xchg_pointer_sym(void **p, void *v); #define rcu_xchg_pointer(p, v) \ + __extension__ \ ({ \ - typeof(*p) _________pv = (v); \ - typeof(*p) _________p1 = \ - rcu_xchg_pointer_sym((void **)(p), _________pv); \ + __typeof__(*(p)) _________pv = (v); \ + __typeof__(*(p)) _________p1 = URCU_FORCE_CAST(__typeof__(*(p)), \ + rcu_xchg_pointer_sym(URCU_FORCE_CAST(void **, p), \ + _________pv)); \ (_________p1); \ }) +/* + * Note: rcu_set_pointer_sym returns @v because we don't want to break + * the ABI. At the API level, rcu_set_pointer() now returns void. Use of + * the return value is therefore deprecated, and will cause a build + * error. + */ extern void *rcu_set_pointer_sym(void **p, void *v); #define rcu_set_pointer(p, v) \ - ({ \ - typeof(*p) _________pv = (v); \ - typeof(*p) _________p1 = \ - rcu_set_pointer_sym((void **)(p), _________pv); \ - }) + do { \ + __typeof__(*(p)) _________pv = (v); \ + (void) rcu_set_pointer_sym(URCU_FORCE_CAST(void **, p), \ + _________pv); \ + } while (0) -#endif /* !_LGPL_SOURCE */ +#endif /* !(defined(_LGPL_SOURCE) || defined(URCU_INLINE_SMALL_FUNCTIONS)) */ /* - * rcu_assign_pointer(type *ptr, type *new) + * void rcu_assign_pointer(type *ptr, type *new) * * Same as rcu_set_pointer, but takes the pointer to assign to rather than its * address as first parameter. Provided for compatibility with the Linux kernel @@ -107,4 +122,8 @@ extern void *rcu_set_pointer_sym(void **p, void *v); */ #define rcu_assign_pointer(p, v) rcu_set_pointer((&p), (v)) +#ifdef __cplusplus +} +#endif + #endif /* _URCU_POINTER_H */