blob: 87990b7c6b0d693eb4c715f33047e5d8cfa0e5c6 [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
Peter Zijlstra726328d2016-05-26 10:35:03 +020011#include <asm/processor.h>
12#include <asm/barrier.h>
13
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070014/* To get debugging spinlocks which detect and catch
15 * deadlock situations, set CONFIG_DEBUG_SPINLOCK
16 * and rebuild your kernel.
17 */
18
David S. Miller64f2dde2008-10-29 21:25:00 -070019/* Because we play games to save cycles in the non-contention case, we
20 * need to be extra careful about branch targets into the "spinning"
21 * code. They live in their own section, but the newer V9 branches
22 * have a shorter range than the traditional 32-bit sparc branch
23 * variants. The rule is that the branches that go into and out of
24 * the spinner sections must be pre-V9 branches.
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070025 */
26
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010027#define arch_spin_is_locked(lp) ((lp)->lock != 0)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070028
Peter Zijlstra726328d2016-05-26 10:35:03 +020029static inline void arch_spin_unlock_wait(arch_spinlock_t *lock)
30{
31 smp_cond_load_acquire(&lock->lock, !VAL);
32}
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070033
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010034static inline void arch_spin_lock(arch_spinlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070035{
36 unsigned long tmp;
37
38 __asm__ __volatile__(
39"1: ldstub [%1], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070040" brnz,pn %0, 2f\n"
41" nop\n"
42" .subsection 2\n"
43"2: ldub [%1], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070044" brnz,pt %0, 2b\n"
45" nop\n"
46" ba,a,pt %%xcc, 1b\n"
47" .previous"
48 : "=&r" (tmp)
49 : "r" (lock)
50 : "memory");
51}
52
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010053static inline int arch_spin_trylock(arch_spinlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070054{
55 unsigned long result;
56
57 __asm__ __volatile__(
58" ldstub [%1], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070059 : "=r" (result)
60 : "r" (lock)
61 : "memory");
62
63 return (result == 0UL);
64}
65
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010066static inline void arch_spin_unlock(arch_spinlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070067{
68 __asm__ __volatile__(
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070069" stb %%g0, [%0]"
70 : /* No outputs */
71 : "r" (lock)
72 : "memory");
73}
74
Thomas Gleixner0199c4e2009-12-02 20:01:25 +010075static inline void arch_spin_lock_flags(arch_spinlock_t *lock, unsigned long flags)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070076{
77 unsigned long tmp1, tmp2;
78
79 __asm__ __volatile__(
80"1: ldstub [%2], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070081" brnz,pn %0, 2f\n"
82" nop\n"
83" .subsection 2\n"
84"2: rdpr %%pil, %1\n"
85" wrpr %3, %%pil\n"
86"3: ldub [%2], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -070087" brnz,pt %0, 3b\n"
88" nop\n"
89" ba,pt %%xcc, 1b\n"
90" wrpr %1, %%pil\n"
91" .previous"
92 : "=&r" (tmp1), "=&r" (tmp2)
93 : "r"(lock), "r"(flags)
94 : "memory");
95}
96
97/* Multi-reader locks, these are much saner than the 32-bit Sparc ones... */
98
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +010099static void inline arch_read_lock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700100{
101 unsigned long tmp1, tmp2;
102
103 __asm__ __volatile__ (
104"1: ldsw [%2], %0\n"
105" brlz,pn %0, 2f\n"
106"4: add %0, 1, %1\n"
107" cas [%2], %0, %1\n"
108" cmp %0, %1\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700109" bne,pn %%icc, 1b\n"
110" nop\n"
111" .subsection 2\n"
112"2: ldsw [%2], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700113" brlz,pt %0, 2b\n"
114" nop\n"
115" ba,a,pt %%xcc, 4b\n"
116" .previous"
117 : "=&r" (tmp1), "=&r" (tmp2)
118 : "r" (lock)
119 : "memory");
120}
121
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100122static int inline arch_read_trylock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700123{
124 int tmp1, tmp2;
125
126 __asm__ __volatile__ (
127"1: ldsw [%2], %0\n"
128" brlz,a,pn %0, 2f\n"
129" mov 0, %0\n"
130" add %0, 1, %1\n"
131" cas [%2], %0, %1\n"
132" cmp %0, %1\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700133" bne,pn %%icc, 1b\n"
134" mov 1, %0\n"
135"2:"
136 : "=&r" (tmp1), "=&r" (tmp2)
137 : "r" (lock)
138 : "memory");
139
140 return tmp1;
141}
142
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100143static void inline arch_read_unlock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700144{
145 unsigned long tmp1, tmp2;
146
147 __asm__ __volatile__(
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700148"1: lduw [%2], %0\n"
149" sub %0, 1, %1\n"
150" cas [%2], %0, %1\n"
151" cmp %0, %1\n"
152" bne,pn %%xcc, 1b\n"
153" nop"
154 : "=&r" (tmp1), "=&r" (tmp2)
155 : "r" (lock)
156 : "memory");
157}
158
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100159static void inline arch_write_lock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700160{
161 unsigned long mask, tmp1, tmp2;
162
163 mask = 0x80000000UL;
164
165 __asm__ __volatile__(
166"1: lduw [%2], %0\n"
167" brnz,pn %0, 2f\n"
168"4: or %0, %3, %1\n"
169" cas [%2], %0, %1\n"
170" cmp %0, %1\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700171" bne,pn %%icc, 1b\n"
172" nop\n"
173" .subsection 2\n"
174"2: lduw [%2], %0\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700175" brnz,pt %0, 2b\n"
176" nop\n"
177" ba,a,pt %%xcc, 4b\n"
178" .previous"
179 : "=&r" (tmp1), "=&r" (tmp2)
180 : "r" (lock), "r" (mask)
181 : "memory");
182}
183
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100184static void inline arch_write_unlock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700185{
186 __asm__ __volatile__(
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700187" stw %%g0, [%0]"
188 : /* no outputs */
189 : "r" (lock)
190 : "memory");
191}
192
Thomas Gleixnerfb3a6bb2009-12-03 20:01:19 +0100193static int inline arch_write_trylock(arch_rwlock_t *lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700194{
195 unsigned long mask, tmp1, tmp2, result;
196
197 mask = 0x80000000UL;
198
199 __asm__ __volatile__(
200" mov 0, %2\n"
201"1: lduw [%3], %0\n"
202" brnz,pn %0, 2f\n"
203" or %0, %4, %1\n"
204" cas [%3], %0, %1\n"
205" cmp %0, %1\n"
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700206" bne,pn %%icc, 1b\n"
207" nop\n"
208" mov 1, %2\n"
209"2:"
210 : "=&r" (tmp1), "=&r" (tmp2), "=&r" (result)
211 : "r" (lock), "r" (mask)
212 : "memory");
213
214 return result;
215}
216
Thomas Gleixnere5931942009-12-03 20:08:46 +0100217#define arch_read_lock_flags(p, f) arch_read_lock(p)
Thomas Gleixnere5931942009-12-03 20:08:46 +0100218#define arch_write_lock_flags(p, f) arch_write_lock(p)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700219
Thomas Gleixnere5931942009-12-03 20:08:46 +0100220#define arch_read_can_lock(rw) (!((rw)->lock & 0x80000000UL))
221#define arch_write_can_lock(rw) (!(rw)->lock)
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700222
Thomas Gleixner0199c4e2009-12-02 20:01:25 +0100223#define arch_spin_relax(lock) cpu_relax()
224#define arch_read_relax(lock) cpu_relax()
225#define arch_write_relax(lock) cpu_relax()
Sam Ravnborgf5e706a2008-07-17 21:55:51 -0700226
227#endif /* !(__ASSEMBLY__) */
228
229#endif /* !(__SPARC64_SPINLOCK_H) */