blob: 073936a8b2755e1c15e637ff28c5bc607876f4dc [file] [log] [blame]
Sam Ravnborgf5e706a2008-07-17 21:55:51 -07001/* spinlock.h: 64-bit Sparc spinlock support.
2 *
3 * Copyright (C) 1997 David S. Miller (davem@caip.rutgers.edu)
4 */
5
6#ifndef __SPARC64_SPINLOCK_H
7#define __SPARC64_SPINLOCK_H
8
Sam Ravnborgf5e706a2008-07-17 21:55:51 -07009#ifndef __ASSEMBLY__
10
11/* To get debugging spinlocks which detect and catch
12 * deadlock situations, set CONFIG_DEBUG_SPINLOCK
13 * and rebuild your kernel.
14 */
15
David S. Miller64f2dde2008-10-29 21:25:00 -070016/* Because we play games to save cycles in the non-contention case, we
17 * need to be extra careful about branch targets into the "spinning"
18 * code. They live in their own section, but the newer V9 branches
19 * have a shorter range than the traditional 32-bit sparc branch
20 * variants. The rule is that the branches that go into and out of
21 * the spinner sections must be pre-V9 branches.
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070022 */
23
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010024#define arch_spin_is_locked(lp) ((lp)->lock != 0)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070025
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010026#define arch_spin_unlock_wait(lp) \
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070027 do { rmb(); \
28 } while((lp)->lock)
29
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010030static inline void arch_spin_lock(arch_spinlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070031{
32 unsigned long tmp;
33
34 __asm__ __volatile__(
35"1: ldstub [%1], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070036" brnz,pn %0, 2f\n"
37" nop\n"
38" .subsection 2\n"
39"2: ldub [%1], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070040" brnz,pt %0, 2b\n"
41" nop\n"
42" ba,a,pt %%xcc, 1b\n"
43" .previous"
44 : "=&r" (tmp)
45 : "r" (lock)
46 : "memory");
47}
48
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010049static inline int arch_spin_trylock(arch_spinlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070050{
51 unsigned long result;
52
53 __asm__ __volatile__(
54" ldstub [%1], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070055 : "=r" (result)
56 : "r" (lock)
57 : "memory");
58
59 return (result == 0UL);
60}
61
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010062static inline void arch_spin_unlock(arch_spinlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070063{
64 __asm__ __volatile__(
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070065" stb %%g0, [%0]"
66 : /* No outputs */
67 : "r" (lock)
68 : "memory");
69}
70
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010071static inline void arch_spin_lock_flags(arch_spinlock_t *lock, unsigned long flags)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070072{
73 unsigned long tmp1, tmp2;
74
75 __asm__ __volatile__(
76"1: ldstub [%2], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070077" brnz,pn %0, 2f\n"
78" nop\n"
79" .subsection 2\n"
80"2: rdpr %%pil, %1\n"
81" wrpr %3, %%pil\n"
82"3: ldub [%2], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070083" brnz,pt %0, 3b\n"
84" nop\n"
85" ba,pt %%xcc, 1b\n"
86" wrpr %1, %%pil\n"
87" .previous"
88 : "=&r" (tmp1), "=&r" (tmp2)
89 : "r"(lock), "r"(flags)
90 : "memory");
91}
92
93/* Multi-reader locks, these are much saner than the 32-bit Sparc ones... */
94
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +010095static void inline arch_read_lock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070096{
97 unsigned long tmp1, tmp2;
98
99 __asm__ __volatile__ (
100"1: ldsw [%2], %0\n"
101" brlz,pn %0, 2f\n"
102"4: add %0, 1, %1\n"
103" cas [%2], %0, %1\n"
104" cmp %0, %1\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700105" bne,pn %%icc, 1b\n"
106" nop\n"
107" .subsection 2\n"
108"2: ldsw [%2], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700109" brlz,pt %0, 2b\n"
110" nop\n"
111" ba,a,pt %%xcc, 4b\n"
112" .previous"
113 : "=&r" (tmp1), "=&r" (tmp2)
114 : "r" (lock)
115 : "memory");
116}
117
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100118static int inline arch_read_trylock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700119{
120 int tmp1, tmp2;
121
122 __asm__ __volatile__ (
123"1: ldsw [%2], %0\n"
124" brlz,a,pn %0, 2f\n"
125" mov 0, %0\n"
126" add %0, 1, %1\n"
127" cas [%2], %0, %1\n"
128" cmp %0, %1\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700129" bne,pn %%icc, 1b\n"
130" mov 1, %0\n"
131"2:"
132 : "=&r" (tmp1), "=&r" (tmp2)
133 : "r" (lock)
134 : "memory");
135
136 return tmp1;
137}
138
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100139static void inline arch_read_unlock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700140{
141 unsigned long tmp1, tmp2;
142
143 __asm__ __volatile__(
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700144"1: lduw [%2], %0\n"
145" sub %0, 1, %1\n"
146" cas [%2], %0, %1\n"
147" cmp %0, %1\n"
148" bne,pn %%xcc, 1b\n"
149" nop"
150 : "=&r" (tmp1), "=&r" (tmp2)
151 : "r" (lock)
152 : "memory");
153}
154
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100155static void inline arch_write_lock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700156{
157 unsigned long mask, tmp1, tmp2;
158
159 mask = 0x80000000UL;
160
161 __asm__ __volatile__(
162"1: lduw [%2], %0\n"
163" brnz,pn %0, 2f\n"
164"4: or %0, %3, %1\n"
165" cas [%2], %0, %1\n"
166" cmp %0, %1\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700167" bne,pn %%icc, 1b\n"
168" nop\n"
169" .subsection 2\n"
170"2: lduw [%2], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700171" brnz,pt %0, 2b\n"
172" nop\n"
173" ba,a,pt %%xcc, 4b\n"
174" .previous"
175 : "=&r" (tmp1), "=&r" (tmp2)
176 : "r" (lock), "r" (mask)
177 : "memory");
178}
179
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100180static void inline arch_write_unlock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700181{
182 __asm__ __volatile__(
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700183" stw %%g0, [%0]"
184 : /* no outputs */
185 : "r" (lock)
186 : "memory");
187}
188
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100189static int inline arch_write_trylock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700190{
191 unsigned long mask, tmp1, tmp2, result;
192
193 mask = 0x80000000UL;
194
195 __asm__ __volatile__(
196" mov 0, %2\n"
197"1: lduw [%3], %0\n"
198" brnz,pn %0, 2f\n"
199" or %0, %4, %1\n"
200" cas [%3], %0, %1\n"
201" cmp %0, %1\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700202" bne,pn %%icc, 1b\n"
203" nop\n"
204" mov 1, %2\n"
205"2:"
206 : "=&r" (tmp1), "=&r" (tmp2), "=&r" (result)
207 : "r" (lock), "r" (mask)
208 : "memory");
209
210 return result;
211}
212
Thomas Gleixnere5931942009-12-03 20:08:46 +0100213#define arch_read_lock(p) arch_read_lock(p)
214#define arch_read_lock_flags(p, f) arch_read_lock(p)
215#define arch_read_trylock(p) arch_read_trylock(p)
216#define arch_read_unlock(p) arch_read_unlock(p)
217#define arch_write_lock(p) arch_write_lock(p)
218#define arch_write_lock_flags(p, f) arch_write_lock(p)
219#define arch_write_unlock(p) arch_write_unlock(p)
220#define arch_write_trylock(p) arch_write_trylock(p)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700221
Thomas Gleixnere5931942009-12-03 20:08:46 +0100222#define arch_read_can_lock(rw) (!((rw)->lock & 0x80000000UL))
223#define arch_write_can_lock(rw) (!(rw)->lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700224
Thomas Gleixner0199c4e2009-12-02 20:01:25 +0100225#define arch_spin_relax(lock) cpu_relax()
226#define arch_read_relax(lock) cpu_relax()
227#define arch_write_relax(lock) cpu_relax()
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700228
229#endif /* !(__ASSEMBLY__) */
230
231#endif /* !(__SPARC64_SPINLOCK_H) */