diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2015-02-09 21:55:45 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2015-02-09 21:55:45 -0500 |
commit | c2189e3a3d67e94941c568ee8fab1b86983cd024 (patch) | |
tree | 57622099fc953f33309c8059050cb3e0517c2ebc | |
parent | ed824a625006e47d2584ee0bba9227a34cc06671 (diff) | |
parent | 4c907baf36d8339f393bb576d0bab29194d0e6ad (diff) |
Merge branch 'for-3.20' of git://git.kernel.org/pub/scm/linux/kernel/git/tj/percpu
Pull percpu changes from Tejun Heo:
"Nothing too interesting. One cleanup patch and another to add a
trivial state check function"
* 'for-3.20' of git://git.kernel.org/pub/scm/linux/kernel/git/tj/percpu:
percpu_ref: implement percpu_ref_is_dying()
percpu_ref: remove unnecessary ACCESS_ONCE() in percpu_ref_tryget_live()
-rw-r--r-- | include/linux/percpu-refcount.h | 34 |
1 files changed, 31 insertions, 3 deletions
diff --git a/include/linux/percpu-refcount.h b/include/linux/percpu-refcount.h index b4337646388b..12c9b485beb7 100644 --- a/include/linux/percpu-refcount.h +++ b/include/linux/percpu-refcount.h | |||
@@ -128,8 +128,22 @@ static inline void percpu_ref_kill(struct percpu_ref *ref) | |||
128 | static inline bool __ref_is_percpu(struct percpu_ref *ref, | 128 | static inline bool __ref_is_percpu(struct percpu_ref *ref, |
129 | unsigned long __percpu **percpu_countp) | 129 | unsigned long __percpu **percpu_countp) |
130 | { | 130 | { |
131 | /* paired with smp_store_release() in percpu_ref_reinit() */ | 131 | unsigned long percpu_ptr; |
132 | unsigned long percpu_ptr = lockless_dereference(ref->percpu_count_ptr); | 132 | |
133 | /* | ||
134 | * The value of @ref->percpu_count_ptr is tested for | ||
135 | * !__PERCPU_REF_ATOMIC, which may be set asynchronously, and then | ||
136 | * used as a pointer. If the compiler generates a separate fetch | ||
137 | * when using it as a pointer, __PERCPU_REF_ATOMIC may be set in | ||
138 | * between contaminating the pointer value, meaning that | ||
139 | * ACCESS_ONCE() is required when fetching it. | ||
140 | * | ||
141 | * Also, we need a data dependency barrier to be paired with | ||
142 | * smp_store_release() in __percpu_ref_switch_to_percpu(). | ||
143 | * | ||
144 | * Use lockless deref which contains both. | ||
145 | */ | ||
146 | percpu_ptr = lockless_dereference(ref->percpu_count_ptr); | ||
133 | 147 | ||
134 | /* | 148 | /* |
135 | * Theoretically, the following could test just ATOMIC; however, | 149 | * Theoretically, the following could test just ATOMIC; however, |
@@ -233,7 +247,7 @@ static inline bool percpu_ref_tryget_live(struct percpu_ref *ref) | |||
233 | if (__ref_is_percpu(ref, &percpu_count)) { | 247 | if (__ref_is_percpu(ref, &percpu_count)) { |
234 | this_cpu_inc(*percpu_count); | 248 | this_cpu_inc(*percpu_count); |
235 | ret = true; | 249 | ret = true; |
236 | } else if (!(ACCESS_ONCE(ref->percpu_count_ptr) & __PERCPU_REF_DEAD)) { | 250 | } else if (!(ref->percpu_count_ptr & __PERCPU_REF_DEAD)) { |
237 | ret = atomic_long_inc_not_zero(&ref->count); | 251 | ret = atomic_long_inc_not_zero(&ref->count); |
238 | } | 252 | } |
239 | 253 | ||
@@ -281,6 +295,20 @@ static inline void percpu_ref_put(struct percpu_ref *ref) | |||
281 | } | 295 | } |
282 | 296 | ||
283 | /** | 297 | /** |
298 | * percpu_ref_is_dying - test whether a percpu refcount is dying or dead | ||
299 | * @ref: percpu_ref to test | ||
300 | * | ||
301 | * Returns %true if @ref is dying or dead. | ||
302 | * | ||
303 | * This function is safe to call as long as @ref is between init and exit | ||
304 | * and the caller is responsible for synchronizing against state changes. | ||
305 | */ | ||
306 | static inline bool percpu_ref_is_dying(struct percpu_ref *ref) | ||
307 | { | ||
308 | return ref->percpu_count_ptr & __PERCPU_REF_DEAD; | ||
309 | } | ||
310 | |||
311 | /** | ||
284 | * percpu_ref_is_zero - test whether a percpu refcount reached zero | 312 | * percpu_ref_is_zero - test whether a percpu refcount reached zero |
285 | * @ref: percpu_ref to test | 313 | * @ref: percpu_ref to test |
286 | * | 314 | * |