aboutsummaryrefslogtreecommitdiffstats
path: root/arch/blackfin
diff options
context:
space:
mode:
Diffstat (limited to 'arch/blackfin')
-rw-r--r--arch/blackfin/lib/ins.S429
1 files changed, 84 insertions, 345 deletions
diff --git a/arch/blackfin/lib/ins.S b/arch/blackfin/lib/ins.S
index 1b84b21ca7d1..1863a6ba507c 100644
--- a/arch/blackfin/lib/ins.S
+++ b/arch/blackfin/lib/ins.S
@@ -1,31 +1,9 @@
1/* 1/*
2 * File: arch/blackfin/lib/ins.S 2 * arch/blackfin/lib/ins.S - ins{bwl} using hardware loops
3 * Based on:
4 * Author: Bas Vermeulen <bas@buyways.nl>
5 * 3 *
6 * Created: Tue Mar 22 15:27:24 CEST 2005 4 * Copyright 2004-2008 Analog Devices Inc.
7 * Description: Implementation of ins{bwl} for BlackFin processors using zero overhead loops. 5 * Copyright (C) 2005 Bas Vermeulen, BuyWays BV <bas@buyways.nl>
8 * 6 * Licensed under the GPL-2 or later.
9 * Modified:
10 * Copyright 2004-2008 Analog Devices Inc.
11 * Copyright (C) 2005 Bas Vermeulen, BuyWays BV <bas@buyways.nl>
12 *
13 * Bugs: Enter bugs at http://blackfin.uclinux.org/
14 *
15 * This program is free software; you can redistribute it and/or modify
16 * it under the terms of the GNU General Public License as published by
17 * the Free Software Foundation; either version 2 of the License, or
18 * (at your option) any later version.
19 *
20 * This program is distributed in the hope that it will be useful,
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23 * GNU General Public License for more details.
24 *
25 * You should have received a copy of the GNU General Public License
26 * along with this program; if not, see the file COPYING, or write
27 * to the Free Software Foundation, Inc.,
28 * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
29 */ 7 */
30 8
31#include <linux/linkage.h> 9#include <linux/linkage.h>
@@ -33,6 +11,46 @@
33 11
34.align 2 12.align 2
35 13
14#ifdef CONFIG_IPIPE
15# define DO_CLI \
16 [--sp] = rets; \
17 [--sp] = (P5:0); \
18 sp += -12; \
19 call ___ipipe_stall_root_raw; \
20 sp += 12; \
21 (P5:0) = [sp++];
22# define CLI_INNER_NOP
23#else
24# define DO_CLI cli R3;
25# define CLI_INNER_NOP nop; nop; nop;
26#endif
27
28#ifdef CONFIG_IPIPE
29# define DO_STI \
30 sp += -12; \
31 call ___ipipe_unstall_root_raw; \
32 sp += 12; \
332: rets = [sp++];
34#else
35# define DO_STI 2: sti R3;
36#endif
37
38#ifdef CONFIG_BFIN_INS_LOWOVERHEAD
39# define CLI_OUTER DO_CLI;
40# define STI_OUTER DO_STI;
41# define CLI_INNER 1:
42# if ANOMALY_05000416
43# define STI_INNER nop; 2: nop;
44# else
45# define STI_INNER 2:
46# endif
47#else
48# define CLI_OUTER
49# define STI_OUTER
50# define CLI_INNER 1: DO_CLI; CLI_INNER_NOP;
51# define STI_INNER DO_STI;
52#endif
53
36/* 54/*
37 * Reads on the Blackfin are speculative. In Blackfin terms, this means they 55 * Reads on the Blackfin are speculative. In Blackfin terms, this means they
38 * can be interrupted at any time (even after they have been issued on to the 56 * can be interrupted at any time (even after they have been issued on to the
@@ -53,327 +71,48 @@
53 * buffers in/out of FIFOs. 71 * buffers in/out of FIFOs.
54 */ 72 */
55 73
56ENTRY(_insl) 74#define COMMON_INS(func, ops) \
57#ifdef CONFIG_BFIN_INS_LOWOVERHEAD 75ENTRY(_ins##func) \
58 P0 = R0; /* P0 = port */ 76 P0 = R0; /* P0 = port */ \
59#ifdef CONFIG_IPIPE 77 CLI_OUTER; /* 3 instructions before first read access */ \
60 [--sp] = rets 78 P1 = R1; /* P1 = address */ \
61 [--sp] = (P5:0); 79 P2 = R2; /* P2 = count */ \
62 sp += -12 80 SSYNC; \
63 call ___ipipe_stall_root_raw 81 \
64 sp += 12 82 LSETUP(1f, 2f) LC0 = P2; \
65 (P5:0) = [sp++]; 83 CLI_INNER; \
66#else 84 ops; \
67 cli R3; 85 STI_INNER; \
68#endif 86 \
69 P1 = R1; /* P1 = address */ 87 STI_OUTER; \
70 P2 = R2; /* P2 = count */ 88 RTS; \
71 SSYNC; 89ENDPROC(_ins##func)
72 LSETUP( .Llong_loop_s, .Llong_loop_e) LC0 = P2;
73.Llong_loop_s: R0 = [P0];
74 [P1++] = R0;
75 NOP;
76.Llong_loop_e: NOP;
77#ifdef CONFIG_IPIPE
78 sp += -12
79 call ___ipipe_unstall_root_raw
80 sp += 12
81 rets = [sp++]
82#else
83 sti R3;
84#endif
85 RTS;
86#else
87 P0 = R0; /* P0 = port */
88 P1 = R1; /* P1 = address */
89 P2 = R2; /* P2 = count */
90 SSYNC;
91 LSETUP( .Llong_loop_s, .Llong_loop_e) LC0 = P2;
92.Llong_loop_s:
93#ifdef CONFIG_IPIPE
94 [--sp] = rets
95 [--sp] = (P5:0);
96 sp += -12
97 call ___ipipe_stall_root_raw
98 sp += 12
99 (P5:0) = [sp++];
100#else
101 CLI R3;
102#endif
103 NOP; NOP; NOP;
104 R0 = [P0];
105 [P1++] = R0;
106.Llong_loop_e:
107#ifdef CONFIG_IPIPE
108 sp += -12
109 call ___ipipe_unstall_root_raw
110 sp += 12
111 rets = [sp++]
112#else
113 STI R3;
114#endif
115 RTS;
116#endif
117ENDPROC(_insl)
118 90
119ENTRY(_insw) 91COMMON_INS(l, \
120#ifdef CONFIG_BFIN_INS_LOWOVERHEAD 92 R0 = [P0]; \
121 P0 = R0; /* P0 = port */ 93 [P1++] = R0; \
122#ifdef CONFIG_IPIPE 94)
123 [--sp] = rets
124 [--sp] = (P5:0);
125 sp += -12
126 call ___ipipe_stall_root_raw
127 sp += 12
128 (P5:0) = [sp++];
129#else
130 cli R3;
131#endif
132 P1 = R1; /* P1 = address */
133 P2 = R2; /* P2 = count */
134 SSYNC;
135 LSETUP( .Lword_loop_s, .Lword_loop_e) LC0 = P2;
136.Lword_loop_s: R0 = W[P0];
137 W[P1++] = R0;
138 NOP;
139.Lword_loop_e: NOP;
140#ifdef CONFIG_IPIPE
141 sp += -12
142 call ___ipipe_unstall_root_raw
143 sp += 12
144 rets = [sp++]
145#else
146 sti R3;
147#endif
148 RTS;
149#else
150 P0 = R0; /* P0 = port */
151 P1 = R1; /* P1 = address */
152 P2 = R2; /* P2 = count */
153 SSYNC;
154 LSETUP( .Lword_loop_s, .Lword_loop_e) LC0 = P2;
155.Lword_loop_s:
156#ifdef CONFIG_IPIPE
157 [--sp] = rets
158 [--sp] = (P5:0);
159 sp += -12
160 call ___ipipe_stall_root_raw
161 sp += 12
162 (P5:0) = [sp++];
163#else
164 CLI R3;
165#endif
166 NOP; NOP; NOP;
167 R0 = W[P0];
168 W[P1++] = R0;
169.Lword_loop_e:
170#ifdef CONFIG_IPIPE
171 sp += -12
172 call ___ipipe_unstall_root_raw
173 sp += 12
174 rets = [sp++]
175#else
176 STI R3;
177#endif
178 RTS;
179 95
180#endif 96COMMON_INS(w, \
181ENDPROC(_insw) 97 R0 = W[P0]; \
98 W[P1++] = R0; \
99)
182 100
183ENTRY(_insw_8) 101COMMON_INS(w_8, \
184#ifdef CONFIG_BFIN_INS_LOWOVERHEAD 102 R0 = W[P0]; \
185 P0 = R0; /* P0 = port */ 103 B[P1++] = R0; \
186#ifdef CONFIG_IPIPE 104 R0 = R0 >> 8; \
187 [--sp] = rets 105 B[P1++] = R0; \
188 [--sp] = (P5:0); 106)
189 sp += -12
190 call ___ipipe_stall_root_raw
191 sp += 12
192 (P5:0) = [sp++];
193#else
194 cli R3;
195#endif
196 P1 = R1; /* P1 = address */
197 P2 = R2; /* P2 = count */
198 SSYNC;
199 LSETUP( .Lword8_loop_s, .Lword8_loop_e) LC0 = P2;
200.Lword8_loop_s: R0 = W[P0];
201 B[P1++] = R0;
202 R0 = R0 >> 8;
203 B[P1++] = R0;
204 NOP;
205.Lword8_loop_e: NOP;
206#ifdef CONFIG_IPIPE
207 sp += -12
208 call ___ipipe_unstall_root_raw
209 sp += 12
210 rets = [sp++]
211#else
212 sti R3;
213#endif
214 RTS;
215#else
216 P0 = R0; /* P0 = port */
217 P1 = R1; /* P1 = address */
218 P2 = R2; /* P2 = count */
219 SSYNC;
220 LSETUP( .Lword8_loop_s, .Lword8_loop_e) LC0 = P2;
221.Lword8_loop_s:
222#ifdef CONFIG_IPIPE
223 [--sp] = rets
224 [--sp] = (P5:0);
225 sp += -12
226 call ___ipipe_stall_root_raw
227 sp += 12
228 (P5:0) = [sp++];
229#else
230 CLI R3;
231#endif
232 NOP; NOP; NOP;
233 R0 = W[P0];
234 B[P1++] = R0;
235 R0 = R0 >> 8;
236 B[P1++] = R0;
237 NOP;
238.Lword8_loop_e:
239#ifdef CONFIG_IPIPE
240 sp += -12
241 call ___ipipe_unstall_root_raw
242 sp += 12
243 rets = [sp++]
244#else
245 STI R3;
246#endif
247 RTS;
248#endif
249ENDPROC(_insw_8)
250 107
251ENTRY(_insb) 108COMMON_INS(b, \
252#ifdef CONFIG_BFIN_INS_LOWOVERHEAD 109 R0 = B[P0]; \
253 P0 = R0; /* P0 = port */ 110 B[P1++] = R0; \
254#ifdef CONFIG_IPIPE 111)
255 [--sp] = rets
256 [--sp] = (P5:0);
257 sp += -12
258 call ___ipipe_stall_root_raw
259 sp += 12
260 (P5:0) = [sp++];
261#else
262 cli R3;
263#endif
264 P1 = R1; /* P1 = address */
265 P2 = R2; /* P2 = count */
266 SSYNC;
267 LSETUP( .Lbyte_loop_s, .Lbyte_loop_e) LC0 = P2;
268.Lbyte_loop_s: R0 = B[P0];
269 B[P1++] = R0;
270 NOP;
271.Lbyte_loop_e: NOP;
272#ifdef CONFIG_IPIPE
273 sp += -12
274 call ___ipipe_unstall_root_raw
275 sp += 12
276 rets = [sp++]
277#else
278 sti R3;
279#endif
280 RTS;
281#else
282 P0 = R0; /* P0 = port */
283 P1 = R1; /* P1 = address */
284 P2 = R2; /* P2 = count */
285 SSYNC;
286 LSETUP( .Lbyte_loop_s, .Lbyte_loop_e) LC0 = P2;
287.Lbyte_loop_s:
288#ifdef CONFIG_IPIPE
289 [--sp] = rets
290 [--sp] = (P5:0);
291 sp += -12
292 call ___ipipe_stall_root_raw
293 sp += 12
294 (P5:0) = [sp++];
295#else
296 CLI R3;
297#endif
298 NOP; NOP; NOP;
299 R0 = B[P0];
300 B[P1++] = R0;
301.Lbyte_loop_e:
302#ifdef CONFIG_IPIPE
303 sp += -12
304 call ___ipipe_unstall_root_raw
305 sp += 12
306 rets = [sp++]
307#else
308 STI R3;
309#endif
310 RTS;
311#endif
312ENDPROC(_insb)
313 112
314ENTRY(_insl_16) 113COMMON_INS(l_16, \
315#ifdef CONFIG_BFIN_INS_LOWOVERHEAD 114 R0 = [P0]; \
316 P0 = R0; /* P0 = port */ 115 W[P1++] = R0; \
317#ifdef CONFIG_IPIPE 116 R0 = R0 >> 16; \
318 [--sp] = rets 117 W[P1++] = R0; \
319 [--sp] = (P5:0); 118)
320 sp += -12
321 call ___ipipe_stall_root_raw
322 sp += 12
323 (P5:0) = [sp++];
324#else
325 cli R3;
326#endif
327 P1 = R1; /* P1 = address */
328 P2 = R2; /* P2 = count */
329 SSYNC;
330 LSETUP( .Llong16_loop_s, .Llong16_loop_e) LC0 = P2;
331.Llong16_loop_s: R0 = [P0];
332 W[P1++] = R0;
333 R0 = R0 >> 16;
334 W[P1++] = R0;
335 NOP;
336.Llong16_loop_e: NOP;
337#ifdef CONFIG_IPIPE
338 sp += -12
339 call ___ipipe_unstall_root_raw
340 sp += 12
341 rets = [sp++]
342#else
343 sti R3;
344#endif
345 RTS;
346#else
347 P0 = R0; /* P0 = port */
348 P1 = R1; /* P1 = address */
349 P2 = R2; /* P2 = count */
350 SSYNC;
351 LSETUP( .Llong16_loop_s, .Llong16_loop_e) LC0 = P2;
352.Llong16_loop_s:
353#ifdef CONFIG_IPIPE
354 [--sp] = rets
355 [--sp] = (P5:0);
356 sp += -12
357 call ___ipipe_stall_root_raw
358 sp += 12
359 (P5:0) = [sp++];
360#else
361 CLI R3;
362#endif
363 NOP; NOP; NOP;
364 R0 = [P0];
365 W[P1++] = R0;
366 R0 = R0 >> 16;
367 W[P1++] = R0;
368.Llong16_loop_e:
369#ifdef CONFIG_IPIPE
370 sp += -12
371 call ___ipipe_unstall_root_raw
372 sp += 12
373 rets = [sp++]
374#else
375 STI R3;
376#endif
377 RTS;
378#endif
379ENDPROC(_insl_16)