diff options
author | Matthew Wilcox <matthew@wil.cx> | 2006-09-02 09:54:58 -0400 |
---|---|---|
committer | Matthew Wilcox <willy@parisc-linux.org> | 2006-10-04 08:47:25 -0400 |
commit | 6e071852a10ec02570c472052f07b5facb5ad857 (patch) | |
tree | 9b90611da8782225df48b12a2c83e0992fef6d4d | |
parent | 9c2c54574e724589858ad656a507be29f9034943 (diff) |
[PARISC] Improve rwlock implementation
Rewrite rwlock implementation to avoid various deadlocks in the current
scheme.
Signed-off-by: Matthew Wilcox <matthew@wil.cx>
Signed-off-by: Kyle McMartin <kyle@parisc-linux.org>
-rw-r--r-- | include/asm-parisc/spinlock.h | 98 |
1 files changed, 61 insertions, 37 deletions
diff --git a/include/asm-parisc/spinlock.h b/include/asm-parisc/spinlock.h index e1825530365d..8980a876cc4e 100644 --- a/include/asm-parisc/spinlock.h +++ b/include/asm-parisc/spinlock.h | |||
@@ -56,50 +56,70 @@ static inline int __raw_spin_trylock(raw_spinlock_t *x) | |||
56 | } | 56 | } |
57 | 57 | ||
58 | /* | 58 | /* |
59 | * Read-write spinlocks, allowing multiple readers | 59 | * Read-write spinlocks, allowing multiple readers but only one writer. |
60 | * but only one writer. | 60 | * The spinlock is held by the writer, preventing any readers or other |
61 | * writers from grabbing the rwlock. Readers use the lock to serialise their | ||
62 | * access to the counter (which records how many readers currently hold the | ||
63 | * lock). Linux rwlocks are unfair to writers; they can be starved for | ||
64 | * an indefinite time by readers. They can also be taken in interrupt context, | ||
65 | * so we have to disable interrupts when acquiring the spin lock to be sure | ||
66 | * that an interrupting reader doesn't get an inconsistent view of the lock. | ||
61 | */ | 67 | */ |
62 | 68 | ||
63 | #define __raw_read_trylock(lock) generic__raw_read_trylock(lock) | ||
64 | |||
65 | /* read_lock, read_unlock are pretty straightforward. Of course it somehow | ||
66 | * sucks we end up saving/restoring flags twice for read_lock_irqsave aso. */ | ||
67 | |||
68 | static __inline__ void __raw_read_lock(raw_rwlock_t *rw) | 69 | static __inline__ void __raw_read_lock(raw_rwlock_t *rw) |
69 | { | 70 | { |
71 | unsigned long flags; | ||
72 | local_irq_save(flags); | ||
70 | __raw_spin_lock(&rw->lock); | 73 | __raw_spin_lock(&rw->lock); |
71 | |||
72 | rw->counter++; | 74 | rw->counter++; |
73 | |||
74 | __raw_spin_unlock(&rw->lock); | 75 | __raw_spin_unlock(&rw->lock); |
76 | local_irq_restore(flags); | ||
75 | } | 77 | } |
76 | 78 | ||
77 | static __inline__ void __raw_read_unlock(raw_rwlock_t *rw) | 79 | static __inline__ void __raw_read_unlock(raw_rwlock_t *rw) |
78 | { | 80 | { |
81 | unsigned long flags; | ||
82 | local_irq_save(flags); | ||
79 | __raw_spin_lock(&rw->lock); | 83 | __raw_spin_lock(&rw->lock); |
80 | |||
81 | rw->counter--; | 84 | rw->counter--; |
82 | |||
83 | __raw_spin_unlock(&rw->lock); | 85 | __raw_spin_unlock(&rw->lock); |
86 | local_irq_restore(flags); | ||
84 | } | 87 | } |
85 | 88 | ||
86 | /* write_lock is less trivial. We optimistically grab the lock and check | 89 | static __inline__ int __raw_read_trylock(raw_rwlock_t *rw) |
87 | * if we surprised any readers. If so we release the lock and wait till | 90 | { |
88 | * they're all gone before trying again | 91 | unsigned long flags; |
89 | * | 92 | retry: |
90 | * Also note that we don't use the _irqsave / _irqrestore suffixes here. | 93 | local_irq_save(flags); |
91 | * If we're called with interrupts enabled and we've got readers (or other | 94 | if (__raw_spin_trylock(&rw->lock)) { |
92 | * writers) in interrupt handlers someone fucked up and we'd dead-lock | 95 | rw->counter++; |
93 | * sooner or later anyway. prumpf */ | 96 | __raw_spin_unlock(&rw->lock); |
97 | local_irq_restore(flags); | ||
98 | return 1; | ||
99 | } | ||
100 | |||
101 | local_irq_restore(flags); | ||
102 | /* If write-locked, we fail to acquire the lock */ | ||
103 | if (rw->counter < 0) | ||
104 | return 0; | ||
105 | |||
106 | /* Wait until we have a realistic chance at the lock */ | ||
107 | while (__raw_spin_is_locked(&rw->lock) && rw->counter >= 0) | ||
108 | cpu_relax(); | ||
109 | |||
110 | goto retry; | ||
111 | } | ||
94 | 112 | ||
95 | static __inline__ void __raw_write_lock(raw_rwlock_t *rw) | 113 | static __inline__ void __raw_write_lock(raw_rwlock_t *rw) |
96 | { | 114 | { |
115 | unsigned long flags; | ||
97 | retry: | 116 | retry: |
117 | local_irq_save(flags); | ||
98 | __raw_spin_lock(&rw->lock); | 118 | __raw_spin_lock(&rw->lock); |
99 | 119 | ||
100 | if(rw->counter != 0) { | 120 | if (rw->counter != 0) { |
101 | /* this basically never happens */ | ||
102 | __raw_spin_unlock(&rw->lock); | 121 | __raw_spin_unlock(&rw->lock); |
122 | local_irq_restore(flags); | ||
103 | 123 | ||
104 | while (rw->counter != 0) | 124 | while (rw->counter != 0) |
105 | cpu_relax(); | 125 | cpu_relax(); |
@@ -107,31 +127,35 @@ retry: | |||
107 | goto retry; | 127 | goto retry; |
108 | } | 128 | } |
109 | 129 | ||
110 | /* got it. now leave without unlocking */ | 130 | rw->counter = -1; /* mark as write-locked */ |
111 | rw->counter = -1; /* remember we are locked */ | 131 | mb(); |
132 | local_irq_restore(flags); | ||
112 | } | 133 | } |
113 | 134 | ||
114 | /* write_unlock is absolutely trivial - we don't have to wait for anything */ | 135 | static __inline__ void __raw_write_unlock(raw_rwlock_t *rw) |
115 | |||
116 | static __inline__ void __raw_write_unlock(raw_rwlock_t *rw) | ||
117 | { | 136 | { |
118 | rw->counter = 0; | 137 | rw->counter = 0; |
119 | __raw_spin_unlock(&rw->lock); | 138 | __raw_spin_unlock(&rw->lock); |
120 | } | 139 | } |
121 | 140 | ||
122 | static __inline__ int __raw_write_trylock(raw_rwlock_t *rw) | 141 | static __inline__ int __raw_write_trylock(raw_rwlock_t *rw) |
123 | { | 142 | { |
124 | __raw_spin_lock(&rw->lock); | 143 | unsigned long flags; |
125 | if (rw->counter != 0) { | 144 | int result = 0; |
126 | /* this basically never happens */ | 145 | |
127 | __raw_spin_unlock(&rw->lock); | 146 | local_irq_save(flags); |
128 | 147 | if (__raw_spin_trylock(&rw->lock)) { | |
129 | return 0; | 148 | if (rw->counter == 0) { |
149 | rw->counter = -1; | ||
150 | result = 1; | ||
151 | } else { | ||
152 | /* Read-locked. Oh well. */ | ||
153 | __raw_spin_unlock(&rw->lock); | ||
154 | } | ||
130 | } | 155 | } |
156 | local_irq_restore(flags); | ||
131 | 157 | ||
132 | /* got it. now leave without unlocking */ | 158 | return result; |
133 | rw->counter = -1; /* remember we are locked */ | ||
134 | return 1; | ||
135 | } | 159 | } |
136 | 160 | ||
137 | /* | 161 | /* |