aboutsummaryrefslogtreecommitdiffstats
path: root/include
diff options
context:
space:
mode:
Diffstat (limited to 'include')
-rw-r--r--include/linux/percpu-refcount.h19
1 files changed, 19 insertions, 0 deletions
diff --git a/include/linux/percpu-refcount.h b/include/linux/percpu-refcount.h
index 0ddd2839ca84..3dfbf237cd8f 100644
--- a/include/linux/percpu-refcount.h
+++ b/include/linux/percpu-refcount.h
@@ -67,6 +67,7 @@ struct percpu_ref {
67 67
68int __must_check percpu_ref_init(struct percpu_ref *ref, 68int __must_check percpu_ref_init(struct percpu_ref *ref,
69 percpu_ref_func_t *release); 69 percpu_ref_func_t *release);
70void percpu_ref_reinit(struct percpu_ref *ref);
70void percpu_ref_exit(struct percpu_ref *ref); 71void percpu_ref_exit(struct percpu_ref *ref);
71void percpu_ref_kill_and_confirm(struct percpu_ref *ref, 72void percpu_ref_kill_and_confirm(struct percpu_ref *ref,
72 percpu_ref_func_t *confirm_kill); 73 percpu_ref_func_t *confirm_kill);
@@ -99,6 +100,9 @@ static inline bool __pcpu_ref_alive(struct percpu_ref *ref,
99{ 100{
100 unsigned long pcpu_ptr = ACCESS_ONCE(ref->pcpu_count_ptr); 101 unsigned long pcpu_ptr = ACCESS_ONCE(ref->pcpu_count_ptr);
101 102
103 /* paired with smp_store_release() in percpu_ref_reinit() */
104 smp_read_barrier_depends();
105
102 if (unlikely(pcpu_ptr & PCPU_REF_DEAD)) 106 if (unlikely(pcpu_ptr & PCPU_REF_DEAD))
103 return false; 107 return false;
104 108
@@ -206,4 +210,19 @@ static inline void percpu_ref_put(struct percpu_ref *ref)
206 rcu_read_unlock_sched(); 210 rcu_read_unlock_sched();
207} 211}
208 212
213/**
214 * percpu_ref_is_zero - test whether a percpu refcount reached zero
215 * @ref: percpu_ref to test
216 *
217 * Returns %true if @ref reached zero.
218 */
219static inline bool percpu_ref_is_zero(struct percpu_ref *ref)
220{
221 unsigned __percpu *pcpu_count;
222
223 if (__pcpu_ref_alive(ref, &pcpu_count))
224 return false;
225 return !atomic_read(&ref->count);
226}
227
209#endif 228#endif