diff options
| author | Tejun Heo <tj@kernel.org> | 2013-06-12 23:52:35 -0400 |
|---|---|---|
| committer | Tejun Heo <tj@kernel.org> | 2013-06-13 14:08:27 -0400 |
| commit | bc497bd33b2d6a6f07bc8574b4764edbd7fdffa8 (patch) | |
| tree | 2425880399cae042dcb7bf9c4fefdbdaf67f6ab5 /lib | |
| parent | acac7883ee7bcc32476963bce7baf73d44574dd1 (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.c | 31 |
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 | */ | ||
| 74 | void 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 | |||
| 57 | static void percpu_ref_kill_rcu(struct rcu_head *rcu) | 88 | static 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); |
