aboutsummaryrefslogtreecommitdiffstats
path: root/lib
diff options
context:
space:
mode:
authorTejun Heo <tj@kernel.org>2013-06-12 23:52:35 -0400
committerTejun Heo <tj@kernel.org>2013-06-13 14:08:27 -0400
commitbc497bd33b2d6a6f07bc8574b4764edbd7fdffa8 (patch)
tree2425880399cae042dcb7bf9c4fefdbdaf67f6ab5 /lib
parentacac7883ee7bcc32476963bce7baf73d44574dd1 (diff)
percpu-refcount: implement percpu_ref_cancel_init()
Normally, percpu_ref_init() initializes and percpu_ref_kill() initiates destruction which completes asynchronously. The asynchronous destruction can be problematic in init failure path where the caller wants to destroy half-constructed object - distinguishing half-constructed objects from the usual release method can be painful for complex objects. This patch implements percpu_ref_cancel_init() which synchronously destroys the percpu_ref without invoking release. To avoid unintentional misuses, the function requires the ref to have finished percpu_ref_init() but never used and triggers WARN otherwise. v2: Explain the weird name and usage restriction in the function comment. Signed-off-by: Tejun Heo <tj@kernel.org> Acked-by: Kent Overstreet <koverstreet@google.com>
Diffstat (limited to 'lib')
-rw-r--r--lib/percpu-refcount.c31
1 files changed, 31 insertions, 0 deletions
diff --git a/lib/percpu-refcount.c b/lib/percpu-refcount.c
index b35eaac2954f..ebeaac274cb9 100644
--- a/lib/percpu-refcount.c
+++ b/lib/percpu-refcount.c
@@ -54,6 +54,37 @@ int percpu_ref_init(struct percpu_ref *ref, percpu_ref_func_t *release)
54 return 0; 54 return 0;
55} 55}
56 56
57/**
58 * percpu_ref_cancel_init - cancel percpu_ref_init()
59 * @ref: percpu_ref to cancel init for
60 *
61 * Once a percpu_ref is initialized, its destruction is initiated by
62 * percpu_ref_kill() and completes asynchronously, which can be painful to
63 * do when destroying a half-constructed object in init failure path.
64 *
65 * This function destroys @ref without invoking @ref->release and the
66 * memory area containing it can be freed immediately on return. To
67 * prevent accidental misuse, it's required that @ref has finished
68 * percpu_ref_init(), whether successful or not, but never used.
69 *
70 * The weird name and usage restriction are to prevent people from using
71 * this function by mistake for normal shutdown instead of
72 * percpu_ref_kill().
73 */
74void percpu_ref_cancel_init(struct percpu_ref *ref)
75{
76 unsigned __percpu *pcpu_count = ref->pcpu_count;
77 int cpu;
78
79 WARN_ON_ONCE(atomic_read(&ref->count) != 1 + PCPU_COUNT_BIAS);
80
81 if (pcpu_count) {
82 for_each_possible_cpu(cpu)
83 WARN_ON_ONCE(*per_cpu_ptr(pcpu_count, cpu));
84 free_percpu(ref->pcpu_count);
85 }
86}
87
57static void percpu_ref_kill_rcu(struct rcu_head *rcu) 88static void percpu_ref_kill_rcu(struct rcu_head *rcu)
58{ 89{
59 struct percpu_ref *ref = container_of(rcu, struct percpu_ref, rcu); 90 struct percpu_ref *ref = container_of(rcu, struct percpu_ref, rcu);