diff options
author | Steffen Klassert <steffen.klassert@secunet.com> | 2012-03-09 01:20:49 -0500 |
---|---|---|
committer | Herbert Xu <herbert@gondor.apana.org.au> | 2012-03-14 05:25:56 -0400 |
commit | 2dc9b5dbdef09840de852a4f0cc6a9c9eece7220 (patch) | |
tree | 4164272ddddfc2477005866ce3261c76470c0d1a /include | |
parent | 3047817b894ddae62be07787bc8735a616104398 (diff) |
padata: Fix race on sequence number wrap
When padata_do_parallel() is called from multiple cpus for the same
padata instance, we can get object reordering on sequence number wrap
because testing for sequence number wrap and reseting the sequence
number must happen atomically but is implemented with two atomic
operations. This patch fixes this by converting the sequence number
from atomic_t to an unsigned int and protect the access with a
spin_lock. As a side effect, we get rid of the sequence number wrap
handling because the seqence number wraps back to null now without
the need to do anything.
Signed-off-by: Steffen Klassert <steffen.klassert@secunet.com>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
Diffstat (limited to 'include')
-rw-r--r-- | include/linux/padata.h | 6 |
1 files changed, 2 insertions, 4 deletions
diff --git a/include/linux/padata.h b/include/linux/padata.h index 4633b2f726b6..86292beebfe2 100644 --- a/include/linux/padata.h +++ b/include/linux/padata.h | |||
@@ -46,7 +46,6 @@ struct padata_priv { | |||
46 | struct list_head list; | 46 | struct list_head list; |
47 | struct parallel_data *pd; | 47 | struct parallel_data *pd; |
48 | int cb_cpu; | 48 | int cb_cpu; |
49 | int seq_nr; | ||
50 | int info; | 49 | int info; |
51 | void (*parallel)(struct padata_priv *padata); | 50 | void (*parallel)(struct padata_priv *padata); |
52 | void (*serial)(struct padata_priv *padata); | 51 | void (*serial)(struct padata_priv *padata); |
@@ -116,7 +115,6 @@ struct padata_cpumask { | |||
116 | * @pinst: padata instance. | 115 | * @pinst: padata instance. |
117 | * @pqueue: percpu padata queues used for parallelization. | 116 | * @pqueue: percpu padata queues used for parallelization. |
118 | * @squeue: percpu padata queues used for serialuzation. | 117 | * @squeue: percpu padata queues used for serialuzation. |
119 | * @seq_nr: The sequence number that will be attached to the next object. | ||
120 | * @reorder_objects: Number of objects waiting in the reorder queues. | 118 | * @reorder_objects: Number of objects waiting in the reorder queues. |
121 | * @refcnt: Number of objects holding a reference on this parallel_data. | 119 | * @refcnt: Number of objects holding a reference on this parallel_data. |
122 | * @max_seq_nr: Maximal used sequence number. | 120 | * @max_seq_nr: Maximal used sequence number. |
@@ -129,12 +127,12 @@ struct parallel_data { | |||
129 | struct padata_instance *pinst; | 127 | struct padata_instance *pinst; |
130 | struct padata_parallel_queue __percpu *pqueue; | 128 | struct padata_parallel_queue __percpu *pqueue; |
131 | struct padata_serial_queue __percpu *squeue; | 129 | struct padata_serial_queue __percpu *squeue; |
132 | atomic_t seq_nr; | ||
133 | atomic_t reorder_objects; | 130 | atomic_t reorder_objects; |
134 | atomic_t refcnt; | 131 | atomic_t refcnt; |
135 | unsigned int max_seq_nr; | ||
136 | struct padata_cpumask cpumask; | 132 | struct padata_cpumask cpumask; |
137 | spinlock_t lock ____cacheline_aligned; | 133 | spinlock_t lock ____cacheline_aligned; |
134 | spinlock_t seq_lock; | ||
135 | unsigned int seq_nr; | ||
138 | unsigned int processed; | 136 | unsigned int processed; |
139 | struct timer_list timer; | 137 | struct timer_list timer; |
140 | }; | 138 | }; |