avoid endless loops in lib/swiotlb.c
[wrt350n-kernel.git] / arch / sparc64 / lib / PeeCeeI.c
blob3c6cfbb20360fe7fb8a38a8105b356b88865928d
1 /* $Id: PeeCeeI.c,v 1.4 1999/09/06 01:17:35 davem Exp $
2 * PeeCeeI.c: The emerging standard...
4 * Copyright (C) 1997 David S. Miller (davem@caip.rutgers.edu)
5 */
7 #include <asm/io.h>
8 #include <asm/byteorder.h>
10 void outsb(unsigned long __addr, const void *src, unsigned long count)
12 void __iomem *addr = (void __iomem *) __addr;
13 const u8 *p = src;
15 while (count--)
16 outb(*p++, addr);
19 void outsw(unsigned long __addr, const void *src, unsigned long count)
21 void __iomem *addr = (void __iomem *) __addr;
23 if (count) {
24 u16 *ps = (u16 *)src;
25 u32 *pi;
27 if (((u64)src) & 0x2) {
28 u16 val = le16_to_cpup(ps);
29 outw(val, addr);
30 ps++;
31 count--;
33 pi = (u32 *)ps;
34 while (count >= 2) {
35 u32 w = le32_to_cpup(pi);
37 pi++;
38 outw(w >> 0, addr);
39 outw(w >> 16, addr);
40 count -= 2;
42 ps = (u16 *)pi;
43 if (count) {
44 u16 val = le16_to_cpup(ps);
45 outw(val, addr);
50 void outsl(unsigned long __addr, const void *src, unsigned long count)
52 void __iomem *addr = (void __iomem *) __addr;
54 if (count) {
55 if ((((u64)src) & 0x3) == 0) {
56 u32 *p = (u32 *)src;
57 while (count--) {
58 u32 val = cpu_to_le32p(p);
59 outl(val, addr);
60 p++;
62 } else {
63 u8 *pb;
64 u16 *ps = (u16 *)src;
65 u32 l = 0, l2;
66 u32 *pi;
68 switch (((u64)src) & 0x3) {
69 case 0x2:
70 count -= 1;
71 l = cpu_to_le16p(ps) << 16;
72 ps++;
73 pi = (u32 *)ps;
74 while (count--) {
75 l2 = cpu_to_le32p(pi);
76 pi++;
77 outl(((l >> 16) | (l2 << 16)), addr);
78 l = l2;
80 ps = (u16 *)pi;
81 l2 = cpu_to_le16p(ps);
82 outl(((l >> 16) | (l2 << 16)), addr);
83 break;
85 case 0x1:
86 count -= 1;
87 pb = (u8 *)src;
88 l = (*pb++ << 8);
89 ps = (u16 *)pb;
90 l2 = cpu_to_le16p(ps);
91 ps++;
92 l |= (l2 << 16);
93 pi = (u32 *)ps;
94 while (count--) {
95 l2 = cpu_to_le32p(pi);
96 pi++;
97 outl(((l >> 8) | (l2 << 24)), addr);
98 l = l2;
100 pb = (u8 *)pi;
101 outl(((l >> 8) | (*pb << 24)), addr);
102 break;
104 case 0x3:
105 count -= 1;
106 pb = (u8 *)src;
107 l = (*pb++ << 24);
108 pi = (u32 *)pb;
109 while (count--) {
110 l2 = cpu_to_le32p(pi);
111 pi++;
112 outl(((l >> 24) | (l2 << 8)), addr);
113 l = l2;
115 ps = (u16 *)pi;
116 l2 = cpu_to_le16p(ps);
117 ps++;
118 pb = (u8 *)ps;
119 l2 |= (*pb << 16);
120 outl(((l >> 24) | (l2 << 8)), addr);
121 break;
127 void insb(unsigned long __addr, void *dst, unsigned long count)
129 void __iomem *addr = (void __iomem *) __addr;
131 if (count) {
132 u32 *pi;
133 u8 *pb = dst;
135 while ((((unsigned long)pb) & 0x3) && count--)
136 *pb++ = inb(addr);
137 pi = (u32 *)pb;
138 while (count >= 4) {
139 u32 w;
141 w = (inb(addr) << 24);
142 w |= (inb(addr) << 16);
143 w |= (inb(addr) << 8);
144 w |= (inb(addr) << 0);
145 *pi++ = w;
146 count -= 4;
148 pb = (u8 *)pi;
149 while (count--)
150 *pb++ = inb(addr);
154 void insw(unsigned long __addr, void *dst, unsigned long count)
156 void __iomem *addr = (void __iomem *) __addr;
158 if (count) {
159 u16 *ps = dst;
160 u32 *pi;
162 if (((unsigned long)ps) & 0x2) {
163 *ps++ = le16_to_cpu(inw(addr));
164 count--;
166 pi = (u32 *)ps;
167 while (count >= 2) {
168 u32 w;
170 w = (le16_to_cpu(inw(addr)) << 16);
171 w |= (le16_to_cpu(inw(addr)) << 0);
172 *pi++ = w;
173 count -= 2;
175 ps = (u16 *)pi;
176 if (count)
177 *ps = le16_to_cpu(inw(addr));
181 void insl(unsigned long __addr, void *dst, unsigned long count)
183 void __iomem *addr = (void __iomem *) __addr;
185 if (count) {
186 if ((((unsigned long)dst) & 0x3) == 0) {
187 u32 *pi = dst;
188 while (count--)
189 *pi++ = le32_to_cpu(inl(addr));
190 } else {
191 u32 l = 0, l2, *pi;
192 u16 *ps;
193 u8 *pb;
195 switch (((unsigned long)dst) & 3) {
196 case 0x2:
197 ps = dst;
198 count -= 1;
199 l = le32_to_cpu(inl(addr));
200 *ps++ = l;
201 pi = (u32 *)ps;
202 while (count--) {
203 l2 = le32_to_cpu(inl(addr));
204 *pi++ = (l << 16) | (l2 >> 16);
205 l = l2;
207 ps = (u16 *)pi;
208 *ps = l;
209 break;
211 case 0x1:
212 pb = dst;
213 count -= 1;
214 l = le32_to_cpu(inl(addr));
215 *pb++ = l >> 24;
216 ps = (u16 *)pb;
217 *ps++ = ((l >> 8) & 0xffff);
218 pi = (u32 *)ps;
219 while (count--) {
220 l2 = le32_to_cpu(inl(addr));
221 *pi++ = (l << 24) | (l2 >> 8);
222 l = l2;
224 pb = (u8 *)pi;
225 *pb = l;
226 break;
228 case 0x3:
229 pb = (u8 *)dst;
230 count -= 1;
231 l = le32_to_cpu(inl(addr));
232 *pb++ = l >> 24;
233 pi = (u32 *)pb;
234 while (count--) {
235 l2 = le32_to_cpu(inl(addr));
236 *pi++ = (l << 8) | (l2 >> 24);
237 l = l2;
239 ps = (u16 *)pi;
240 *ps++ = ((l >> 8) & 0xffff);
241 pb = (u8 *)ps;
242 *pb = l;
243 break;