xref: /freebsd/sys/x86/include/bus.h (revision 884a2a699669ec61e2366e3e358342dbc94be24a)
1 /*-
2  * Copyright (c) KATO Takenori, 1999.
3  *
4  * All rights reserved.  Unpublished rights reserved under the copyright
5  * laws of Japan.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  *
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice, this list of conditions and the following disclaimer as
13  *    the first lines of this file unmodified.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in the
16  *    documentation and/or other materials provided with the distribution.
17  * 3. The name of the author may not be used to endorse or promote products
18  *    derived from this software without specific prior written permission.
19  *
20  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
21  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
22  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
23  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
24  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
25  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
29  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30  *
31  * $FreeBSD$
32  */
33 
34 /*	$NetBSD: bus.h,v 1.12 1997/10/01 08:25:15 fvdl Exp $	*/
35 
36 /*-
37  * Copyright (c) 1996, 1997 The NetBSD Foundation, Inc.
38  * All rights reserved.
39  *
40  * This code is derived from software contributed to The NetBSD Foundation
41  * by Jason R. Thorpe of the Numerical Aerospace Simulation Facility,
42  * NASA Ames Research Center.
43  *
44  * Redistribution and use in source and binary forms, with or without
45  * modification, are permitted provided that the following conditions
46  * are met:
47  * 1. Redistributions of source code must retain the above copyright
48  *    notice, this list of conditions and the following disclaimer.
49  * 2. Redistributions in binary form must reproduce the above copyright
50  *    notice, this list of conditions and the following disclaimer in the
51  *    documentation and/or other materials provided with the distribution.
52  * 3. All advertising materials mentioning features or use of this software
53  *    must display the following acknowledgement:
54  *	This product includes software developed by the NetBSD
55  *	Foundation, Inc. and its contributors.
56  * 4. Neither the name of The NetBSD Foundation nor the names of its
57  *    contributors may be used to endorse or promote products derived
58  *    from this software without specific prior written permission.
59  *
60  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
61  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
62  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
63  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
64  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
65  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
66  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
67  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
68  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
69  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
70  * POSSIBILITY OF SUCH DAMAGE.
71  */
72 
73 /*-
74  * Copyright (c) 1996 Charles M. Hannum.  All rights reserved.
75  * Copyright (c) 1996 Christopher G. Demetriou.  All rights reserved.
76  *
77  * Redistribution and use in source and binary forms, with or without
78  * modification, are permitted provided that the following conditions
79  * are met:
80  * 1. Redistributions of source code must retain the above copyright
81  *    notice, this list of conditions and the following disclaimer.
82  * 2. Redistributions in binary form must reproduce the above copyright
83  *    notice, this list of conditions and the following disclaimer in the
84  *    documentation and/or other materials provided with the distribution.
85  * 3. All advertising materials mentioning features or use of this software
86  *    must display the following acknowledgement:
87  *      This product includes software developed by Christopher G. Demetriou
88  *	for the NetBSD Project.
89  * 4. The name of the author may not be used to endorse or promote products
90  *    derived from this software without specific prior written permission
91  *
92  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
93  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
94  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
95  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
96  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
97  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
98  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
99  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
100  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
101  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
102  */
103 
104 #ifndef _X86_BUS_H_
105 #define _X86_BUS_H_
106 
107 #include <machine/_bus.h>
108 #include <machine/cpufunc.h>
109 
110 #ifndef __GNUCLIKE_ASM
111 # ifndef lint
112 #  error "no assembler code for your compiler"
113 # endif
114 #endif
115 
116 /*
117  * Values for the x86 bus space tag, not to be used directly by MI code.
118  */
119 #define	X86_BUS_SPACE_IO	0	/* space is i/o space */
120 #define	X86_BUS_SPACE_MEM	1	/* space is mem space */
121 
122 #define BUS_SPACE_MAXSIZE_24BIT	0xFFFFFF
123 #define BUS_SPACE_MAXSIZE_32BIT 0xFFFFFFFF
124 #define BUS_SPACE_MAXSIZE	0xFFFFFFFF
125 #define BUS_SPACE_MAXADDR_24BIT	0xFFFFFF
126 #define BUS_SPACE_MAXADDR_32BIT 0xFFFFFFFF
127 #if defined(__amd64__) || defined(PAE)
128 #define BUS_SPACE_MAXADDR	0xFFFFFFFFFFFFFFFFULL
129 #else
130 #define BUS_SPACE_MAXADDR	0xFFFFFFFF
131 #endif
132 
133 #define BUS_SPACE_UNRESTRICTED	(~0)
134 
135 /*
136  * Map a region of device bus space into CPU virtual address space.
137  */
138 
139 static __inline int bus_space_map(bus_space_tag_t t, bus_addr_t addr,
140 				  bus_size_t size, int flags,
141 				  bus_space_handle_t *bshp);
142 
143 static __inline int
144 bus_space_map(bus_space_tag_t t __unused, bus_addr_t addr,
145 	      bus_size_t size __unused, int flags __unused,
146 	      bus_space_handle_t *bshp)
147 {
148 
149 	*bshp = addr;
150 	return (0);
151 }
152 
153 /*
154  * Unmap a region of device bus space.
155  */
156 
157 static __inline void bus_space_unmap(bus_space_tag_t t, bus_space_handle_t bsh,
158 				     bus_size_t size);
159 
160 static __inline void
161 bus_space_unmap(bus_space_tag_t t __unused, bus_space_handle_t bsh __unused,
162 		bus_size_t size __unused)
163 {
164 }
165 
166 /*
167  * Get a new handle for a subregion of an already-mapped area of bus space.
168  */
169 
170 static __inline int bus_space_subregion(bus_space_tag_t t,
171 					bus_space_handle_t bsh,
172 					bus_size_t offset, bus_size_t size,
173 					bus_space_handle_t *nbshp);
174 
175 static __inline int
176 bus_space_subregion(bus_space_tag_t t __unused, bus_space_handle_t bsh,
177 		    bus_size_t offset, bus_size_t size __unused,
178 		    bus_space_handle_t *nbshp)
179 {
180 
181 	*nbshp = bsh + offset;
182 	return (0);
183 }
184 
185 /*
186  * Allocate a region of memory that is accessible to devices in bus space.
187  */
188 
189 int	bus_space_alloc(bus_space_tag_t t, bus_addr_t rstart,
190 			bus_addr_t rend, bus_size_t size, bus_size_t align,
191 			bus_size_t boundary, int flags, bus_addr_t *addrp,
192 			bus_space_handle_t *bshp);
193 
194 /*
195  * Free a region of bus space accessible memory.
196  */
197 
198 static __inline void bus_space_free(bus_space_tag_t t, bus_space_handle_t bsh,
199 				    bus_size_t size);
200 
201 static __inline void
202 bus_space_free(bus_space_tag_t t __unused, bus_space_handle_t bsh __unused,
203 	       bus_size_t size __unused)
204 {
205 }
206 
207 
208 /*
209  * Read a 1, 2, 4, or 8 byte quantity from bus space
210  * described by tag/handle/offset.
211  */
212 static __inline u_int8_t bus_space_read_1(bus_space_tag_t tag,
213 					  bus_space_handle_t handle,
214 					  bus_size_t offset);
215 
216 static __inline u_int16_t bus_space_read_2(bus_space_tag_t tag,
217 					   bus_space_handle_t handle,
218 					   bus_size_t offset);
219 
220 static __inline u_int32_t bus_space_read_4(bus_space_tag_t tag,
221 					   bus_space_handle_t handle,
222 					   bus_size_t offset);
223 
224 static __inline u_int8_t
225 bus_space_read_1(bus_space_tag_t tag, bus_space_handle_t handle,
226 		 bus_size_t offset)
227 {
228 
229 	if (tag == X86_BUS_SPACE_IO)
230 		return (inb(handle + offset));
231 	return (*(volatile u_int8_t *)(handle + offset));
232 }
233 
234 static __inline u_int16_t
235 bus_space_read_2(bus_space_tag_t tag, bus_space_handle_t handle,
236 		 bus_size_t offset)
237 {
238 
239 	if (tag == X86_BUS_SPACE_IO)
240 		return (inw(handle + offset));
241 	return (*(volatile u_int16_t *)(handle + offset));
242 }
243 
244 static __inline u_int32_t
245 bus_space_read_4(bus_space_tag_t tag, bus_space_handle_t handle,
246 		 bus_size_t offset)
247 {
248 
249 	if (tag == X86_BUS_SPACE_IO)
250 		return (inl(handle + offset));
251 	return (*(volatile u_int32_t *)(handle + offset));
252 }
253 
254 #if 0	/* Cause a link error for bus_space_read_8 */
255 #define	bus_space_read_8(t, h, o)	!!! bus_space_read_8 unimplemented !!!
256 #endif
257 
258 /*
259  * Read `count' 1, 2, 4, or 8 byte quantities from bus space
260  * described by tag/handle/offset and copy into buffer provided.
261  */
262 static __inline void bus_space_read_multi_1(bus_space_tag_t tag,
263 					    bus_space_handle_t bsh,
264 					    bus_size_t offset, u_int8_t *addr,
265 					    size_t count);
266 
267 static __inline void bus_space_read_multi_2(bus_space_tag_t tag,
268 					    bus_space_handle_t bsh,
269 					    bus_size_t offset, u_int16_t *addr,
270 					    size_t count);
271 
272 static __inline void bus_space_read_multi_4(bus_space_tag_t tag,
273 					    bus_space_handle_t bsh,
274 					    bus_size_t offset, u_int32_t *addr,
275 					    size_t count);
276 
277 static __inline void
278 bus_space_read_multi_1(bus_space_tag_t tag, bus_space_handle_t bsh,
279 		       bus_size_t offset, u_int8_t *addr, size_t count)
280 {
281 
282 	if (tag == X86_BUS_SPACE_IO)
283 		insb(bsh + offset, addr, count);
284 	else {
285 #ifdef __GNUCLIKE_ASM
286 		__asm __volatile("				\n\
287 			cld					\n\
288 		1:	movb (%2),%%al				\n\
289 			stosb					\n\
290 			loop 1b"				:
291 		    "=D" (addr), "=c" (count)			:
292 		    "r" (bsh + offset), "0" (addr), "1" (count)	:
293 		    "%eax", "memory");
294 #endif
295 	}
296 }
297 
298 static __inline void
299 bus_space_read_multi_2(bus_space_tag_t tag, bus_space_handle_t bsh,
300 		       bus_size_t offset, u_int16_t *addr, size_t count)
301 {
302 
303 	if (tag == X86_BUS_SPACE_IO)
304 		insw(bsh + offset, addr, count);
305 	else {
306 #ifdef __GNUCLIKE_ASM
307 		__asm __volatile("				\n\
308 			cld					\n\
309 		1:	movw (%2),%%ax				\n\
310 			stosw					\n\
311 			loop 1b"				:
312 		    "=D" (addr), "=c" (count)			:
313 		    "r" (bsh + offset), "0" (addr), "1" (count)	:
314 		    "%eax", "memory");
315 #endif
316 	}
317 }
318 
319 static __inline void
320 bus_space_read_multi_4(bus_space_tag_t tag, bus_space_handle_t bsh,
321 		       bus_size_t offset, u_int32_t *addr, size_t count)
322 {
323 
324 	if (tag == X86_BUS_SPACE_IO)
325 		insl(bsh + offset, addr, count);
326 	else {
327 #ifdef __GNUCLIKE_ASM
328 		__asm __volatile("				\n\
329 			cld					\n\
330 		1:	movl (%2),%%eax				\n\
331 			stosl					\n\
332 			loop 1b"				:
333 		    "=D" (addr), "=c" (count)			:
334 		    "r" (bsh + offset), "0" (addr), "1" (count)	:
335 		    "%eax", "memory");
336 #endif
337 	}
338 }
339 
340 #if 0	/* Cause a link error for bus_space_read_multi_8 */
341 #define	bus_space_read_multi_8	!!! bus_space_read_multi_8 unimplemented !!!
342 #endif
343 
344 /*
345  * Read `count' 1, 2, 4, or 8 byte quantities from bus space
346  * described by tag/handle and starting at `offset' and copy into
347  * buffer provided.
348  */
349 static __inline void bus_space_read_region_1(bus_space_tag_t tag,
350 					     bus_space_handle_t bsh,
351 					     bus_size_t offset, u_int8_t *addr,
352 					     size_t count);
353 
354 static __inline void bus_space_read_region_2(bus_space_tag_t tag,
355 					     bus_space_handle_t bsh,
356 					     bus_size_t offset, u_int16_t *addr,
357 					     size_t count);
358 
359 static __inline void bus_space_read_region_4(bus_space_tag_t tag,
360 					     bus_space_handle_t bsh,
361 					     bus_size_t offset, u_int32_t *addr,
362 					     size_t count);
363 
364 
365 static __inline void
366 bus_space_read_region_1(bus_space_tag_t tag, bus_space_handle_t bsh,
367 			bus_size_t offset, u_int8_t *addr, size_t count)
368 {
369 
370 	if (tag == X86_BUS_SPACE_IO) {
371 		int _port_ = bsh + offset;
372 #ifdef __GNUCLIKE_ASM
373 		__asm __volatile("				\n\
374 			cld					\n\
375 		1:	inb %w2,%%al				\n\
376 			stosb					\n\
377 			incl %2					\n\
378 			loop 1b"				:
379 		    "=D" (addr), "=c" (count), "=d" (_port_)	:
380 		    "0" (addr), "1" (count), "2" (_port_)	:
381 		    "%eax", "memory", "cc");
382 #endif
383 	} else {
384 		bus_space_handle_t _port_ = bsh + offset;
385 #ifdef __GNUCLIKE_ASM
386 		__asm __volatile("				\n\
387 			cld					\n\
388 			repne					\n\
389 			movsb"					:
390 		    "=D" (addr), "=c" (count), "=S" (_port_)	:
391 		    "0" (addr), "1" (count), "2" (_port_)	:
392 		    "memory", "cc");
393 #endif
394 	}
395 }
396 
397 static __inline void
398 bus_space_read_region_2(bus_space_tag_t tag, bus_space_handle_t bsh,
399 			bus_size_t offset, u_int16_t *addr, size_t count)
400 {
401 
402 	if (tag == X86_BUS_SPACE_IO) {
403 		int _port_ = bsh + offset;
404 #ifdef __GNUCLIKE_ASM
405 		__asm __volatile("				\n\
406 			cld					\n\
407 		1:	inw %w2,%%ax				\n\
408 			stosw					\n\
409 			addl $2,%2				\n\
410 			loop 1b"				:
411 		    "=D" (addr), "=c" (count), "=d" (_port_)	:
412 		    "0" (addr), "1" (count), "2" (_port_)	:
413 		    "%eax", "memory", "cc");
414 #endif
415 	} else {
416 		bus_space_handle_t _port_ = bsh + offset;
417 #ifdef __GNUCLIKE_ASM
418 		__asm __volatile("				\n\
419 			cld					\n\
420 			repne					\n\
421 			movsw"					:
422 		    "=D" (addr), "=c" (count), "=S" (_port_)	:
423 		    "0" (addr), "1" (count), "2" (_port_)	:
424 		    "memory", "cc");
425 #endif
426 	}
427 }
428 
429 static __inline void
430 bus_space_read_region_4(bus_space_tag_t tag, bus_space_handle_t bsh,
431 			bus_size_t offset, u_int32_t *addr, size_t count)
432 {
433 
434 	if (tag == X86_BUS_SPACE_IO) {
435 		int _port_ = bsh + offset;
436 #ifdef __GNUCLIKE_ASM
437 		__asm __volatile("				\n\
438 			cld					\n\
439 		1:	inl %w2,%%eax				\n\
440 			stosl					\n\
441 			addl $4,%2				\n\
442 			loop 1b"				:
443 		    "=D" (addr), "=c" (count), "=d" (_port_)	:
444 		    "0" (addr), "1" (count), "2" (_port_)	:
445 		    "%eax", "memory", "cc");
446 #endif
447 	} else {
448 		bus_space_handle_t _port_ = bsh + offset;
449 #ifdef __GNUCLIKE_ASM
450 		__asm __volatile("				\n\
451 			cld					\n\
452 			repne					\n\
453 			movsl"					:
454 		    "=D" (addr), "=c" (count), "=S" (_port_)	:
455 		    "0" (addr), "1" (count), "2" (_port_)	:
456 		    "memory", "cc");
457 #endif
458 	}
459 }
460 
461 #if 0	/* Cause a link error for bus_space_read_region_8 */
462 #define	bus_space_read_region_8	!!! bus_space_read_region_8 unimplemented !!!
463 #endif
464 
465 /*
466  * Write the 1, 2, 4, or 8 byte value `value' to bus space
467  * described by tag/handle/offset.
468  */
469 
470 static __inline void bus_space_write_1(bus_space_tag_t tag,
471 				       bus_space_handle_t bsh,
472 				       bus_size_t offset, u_int8_t value);
473 
474 static __inline void bus_space_write_2(bus_space_tag_t tag,
475 				       bus_space_handle_t bsh,
476 				       bus_size_t offset, u_int16_t value);
477 
478 static __inline void bus_space_write_4(bus_space_tag_t tag,
479 				       bus_space_handle_t bsh,
480 				       bus_size_t offset, u_int32_t value);
481 
482 static __inline void
483 bus_space_write_1(bus_space_tag_t tag, bus_space_handle_t bsh,
484 		       bus_size_t offset, u_int8_t value)
485 {
486 
487 	if (tag == X86_BUS_SPACE_IO)
488 		outb(bsh + offset, value);
489 	else
490 		*(volatile u_int8_t *)(bsh + offset) = value;
491 }
492 
493 static __inline void
494 bus_space_write_2(bus_space_tag_t tag, bus_space_handle_t bsh,
495 		       bus_size_t offset, u_int16_t value)
496 {
497 
498 	if (tag == X86_BUS_SPACE_IO)
499 		outw(bsh + offset, value);
500 	else
501 		*(volatile u_int16_t *)(bsh + offset) = value;
502 }
503 
504 static __inline void
505 bus_space_write_4(bus_space_tag_t tag, bus_space_handle_t bsh,
506 		       bus_size_t offset, u_int32_t value)
507 {
508 
509 	if (tag == X86_BUS_SPACE_IO)
510 		outl(bsh + offset, value);
511 	else
512 		*(volatile u_int32_t *)(bsh + offset) = value;
513 }
514 
515 #if 0	/* Cause a link error for bus_space_write_8 */
516 #define	bus_space_write_8	!!! bus_space_write_8 not implemented !!!
517 #endif
518 
519 /*
520  * Write `count' 1, 2, 4, or 8 byte quantities from the buffer
521  * provided to bus space described by tag/handle/offset.
522  */
523 
524 static __inline void bus_space_write_multi_1(bus_space_tag_t tag,
525 					     bus_space_handle_t bsh,
526 					     bus_size_t offset,
527 					     const u_int8_t *addr,
528 					     size_t count);
529 static __inline void bus_space_write_multi_2(bus_space_tag_t tag,
530 					     bus_space_handle_t bsh,
531 					     bus_size_t offset,
532 					     const u_int16_t *addr,
533 					     size_t count);
534 
535 static __inline void bus_space_write_multi_4(bus_space_tag_t tag,
536 					     bus_space_handle_t bsh,
537 					     bus_size_t offset,
538 					     const u_int32_t *addr,
539 					     size_t count);
540 
541 static __inline void
542 bus_space_write_multi_1(bus_space_tag_t tag, bus_space_handle_t bsh,
543 			bus_size_t offset, const u_int8_t *addr, size_t count)
544 {
545 
546 	if (tag == X86_BUS_SPACE_IO)
547 		outsb(bsh + offset, addr, count);
548 	else {
549 #ifdef __GNUCLIKE_ASM
550 		__asm __volatile("				\n\
551 			cld					\n\
552 		1:	lodsb					\n\
553 			movb %%al,(%2)				\n\
554 			loop 1b"				:
555 		    "=S" (addr), "=c" (count)			:
556 		    "r" (bsh + offset), "0" (addr), "1" (count)	:
557 		    "%eax", "memory", "cc");
558 #endif
559 	}
560 }
561 
562 static __inline void
563 bus_space_write_multi_2(bus_space_tag_t tag, bus_space_handle_t bsh,
564 			bus_size_t offset, const u_int16_t *addr, size_t count)
565 {
566 
567 	if (tag == X86_BUS_SPACE_IO)
568 		outsw(bsh + offset, addr, count);
569 	else {
570 #ifdef __GNUCLIKE_ASM
571 		__asm __volatile("				\n\
572 			cld					\n\
573 		1:	lodsw					\n\
574 			movw %%ax,(%2)				\n\
575 			loop 1b"				:
576 		    "=S" (addr), "=c" (count)			:
577 		    "r" (bsh + offset), "0" (addr), "1" (count)	:
578 		    "%eax", "memory", "cc");
579 #endif
580 	}
581 }
582 
583 static __inline void
584 bus_space_write_multi_4(bus_space_tag_t tag, bus_space_handle_t bsh,
585 			bus_size_t offset, const u_int32_t *addr, size_t count)
586 {
587 
588 	if (tag == X86_BUS_SPACE_IO)
589 		outsl(bsh + offset, addr, count);
590 	else {
591 #ifdef __GNUCLIKE_ASM
592 		__asm __volatile("				\n\
593 			cld					\n\
594 		1:	lodsl					\n\
595 			movl %%eax,(%2)				\n\
596 			loop 1b"				:
597 		    "=S" (addr), "=c" (count)			:
598 		    "r" (bsh + offset), "0" (addr), "1" (count)	:
599 		    "%eax", "memory", "cc");
600 #endif
601 	}
602 }
603 
604 #if 0	/* Cause a link error for bus_space_write_multi_8 */
605 #define	bus_space_write_multi_8(t, h, o, a, c)				\
606 			!!! bus_space_write_multi_8 unimplemented !!!
607 #endif
608 
609 /*
610  * Write `count' 1, 2, 4, or 8 byte quantities from the buffer provided
611  * to bus space described by tag/handle starting at `offset'.
612  */
613 
614 static __inline void bus_space_write_region_1(bus_space_tag_t tag,
615 					      bus_space_handle_t bsh,
616 					      bus_size_t offset,
617 					      const u_int8_t *addr,
618 					      size_t count);
619 static __inline void bus_space_write_region_2(bus_space_tag_t tag,
620 					      bus_space_handle_t bsh,
621 					      bus_size_t offset,
622 					      const u_int16_t *addr,
623 					      size_t count);
624 static __inline void bus_space_write_region_4(bus_space_tag_t tag,
625 					      bus_space_handle_t bsh,
626 					      bus_size_t offset,
627 					      const u_int32_t *addr,
628 					      size_t count);
629 
630 static __inline void
631 bus_space_write_region_1(bus_space_tag_t tag, bus_space_handle_t bsh,
632 			 bus_size_t offset, const u_int8_t *addr, size_t count)
633 {
634 
635 	if (tag == X86_BUS_SPACE_IO) {
636 		int _port_ = bsh + offset;
637 #ifdef __GNUCLIKE_ASM
638 		__asm __volatile("				\n\
639 			cld					\n\
640 		1:	lodsb					\n\
641 			outb %%al,%w0				\n\
642 			incl %0					\n\
643 			loop 1b"				:
644 		    "=d" (_port_), "=S" (addr), "=c" (count)	:
645 		    "0" (_port_), "1" (addr), "2" (count)	:
646 		    "%eax", "memory", "cc");
647 #endif
648 	} else {
649 		bus_space_handle_t _port_ = bsh + offset;
650 #ifdef __GNUCLIKE_ASM
651 		__asm __volatile("				\n\
652 			cld					\n\
653 			repne					\n\
654 			movsb"					:
655 		    "=D" (_port_), "=S" (addr), "=c" (count)	:
656 		    "0" (_port_), "1" (addr), "2" (count)	:
657 		    "memory", "cc");
658 #endif
659 	}
660 }
661 
662 static __inline void
663 bus_space_write_region_2(bus_space_tag_t tag, bus_space_handle_t bsh,
664 			 bus_size_t offset, const u_int16_t *addr, size_t count)
665 {
666 
667 	if (tag == X86_BUS_SPACE_IO) {
668 		int _port_ = bsh + offset;
669 #ifdef __GNUCLIKE_ASM
670 		__asm __volatile("				\n\
671 			cld					\n\
672 		1:	lodsw					\n\
673 			outw %%ax,%w0				\n\
674 			addl $2,%0				\n\
675 			loop 1b"				:
676 		    "=d" (_port_), "=S" (addr), "=c" (count)	:
677 		    "0" (_port_), "1" (addr), "2" (count)	:
678 		    "%eax", "memory", "cc");
679 #endif
680 	} else {
681 		bus_space_handle_t _port_ = bsh + offset;
682 #ifdef __GNUCLIKE_ASM
683 		__asm __volatile("				\n\
684 			cld					\n\
685 			repne					\n\
686 			movsw"					:
687 		    "=D" (_port_), "=S" (addr), "=c" (count)	:
688 		    "0" (_port_), "1" (addr), "2" (count)	:
689 		    "memory", "cc");
690 #endif
691 	}
692 }
693 
694 static __inline void
695 bus_space_write_region_4(bus_space_tag_t tag, bus_space_handle_t bsh,
696 			 bus_size_t offset, const u_int32_t *addr, size_t count)
697 {
698 
699 	if (tag == X86_BUS_SPACE_IO) {
700 		int _port_ = bsh + offset;
701 #ifdef __GNUCLIKE_ASM
702 		__asm __volatile("				\n\
703 			cld					\n\
704 		1:	lodsl					\n\
705 			outl %%eax,%w0				\n\
706 			addl $4,%0				\n\
707 			loop 1b"				:
708 		    "=d" (_port_), "=S" (addr), "=c" (count)	:
709 		    "0" (_port_), "1" (addr), "2" (count)	:
710 		    "%eax", "memory", "cc");
711 #endif
712 	} else {
713 		bus_space_handle_t _port_ = bsh + offset;
714 #ifdef __GNUCLIKE_ASM
715 		__asm __volatile("				\n\
716 			cld					\n\
717 			repne					\n\
718 			movsl"					:
719 		    "=D" (_port_), "=S" (addr), "=c" (count)	:
720 		    "0" (_port_), "1" (addr), "2" (count)	:
721 		    "memory", "cc");
722 #endif
723 	}
724 }
725 
726 #if 0	/* Cause a link error for bus_space_write_region_8 */
727 #define	bus_space_write_region_8					\
728 			!!! bus_space_write_region_8 unimplemented !!!
729 #endif
730 
731 /*
732  * Write the 1, 2, 4, or 8 byte value `val' to bus space described
733  * by tag/handle/offset `count' times.
734  */
735 
736 static __inline void bus_space_set_multi_1(bus_space_tag_t tag,
737 					   bus_space_handle_t bsh,
738 					   bus_size_t offset,
739 					   u_int8_t value, size_t count);
740 static __inline void bus_space_set_multi_2(bus_space_tag_t tag,
741 					   bus_space_handle_t bsh,
742 					   bus_size_t offset,
743 					   u_int16_t value, size_t count);
744 static __inline void bus_space_set_multi_4(bus_space_tag_t tag,
745 					   bus_space_handle_t bsh,
746 					   bus_size_t offset,
747 					   u_int32_t value, size_t count);
748 
749 static __inline void
750 bus_space_set_multi_1(bus_space_tag_t tag, bus_space_handle_t bsh,
751 		      bus_size_t offset, u_int8_t value, size_t count)
752 {
753 	bus_space_handle_t addr = bsh + offset;
754 
755 	if (tag == X86_BUS_SPACE_IO)
756 		while (count--)
757 			outb(addr, value);
758 	else
759 		while (count--)
760 			*(volatile u_int8_t *)(addr) = value;
761 }
762 
763 static __inline void
764 bus_space_set_multi_2(bus_space_tag_t tag, bus_space_handle_t bsh,
765 		     bus_size_t offset, u_int16_t value, size_t count)
766 {
767 	bus_space_handle_t addr = bsh + offset;
768 
769 	if (tag == X86_BUS_SPACE_IO)
770 		while (count--)
771 			outw(addr, value);
772 	else
773 		while (count--)
774 			*(volatile u_int16_t *)(addr) = value;
775 }
776 
777 static __inline void
778 bus_space_set_multi_4(bus_space_tag_t tag, bus_space_handle_t bsh,
779 		      bus_size_t offset, u_int32_t value, size_t count)
780 {
781 	bus_space_handle_t addr = bsh + offset;
782 
783 	if (tag == X86_BUS_SPACE_IO)
784 		while (count--)
785 			outl(addr, value);
786 	else
787 		while (count--)
788 			*(volatile u_int32_t *)(addr) = value;
789 }
790 
791 #if 0	/* Cause a link error for bus_space_set_multi_8 */
792 #define	bus_space_set_multi_8 !!! bus_space_set_multi_8 unimplemented !!!
793 #endif
794 
795 /*
796  * Write `count' 1, 2, 4, or 8 byte value `val' to bus space described
797  * by tag/handle starting at `offset'.
798  */
799 
800 static __inline void bus_space_set_region_1(bus_space_tag_t tag,
801 					    bus_space_handle_t bsh,
802 					    bus_size_t offset, u_int8_t value,
803 					    size_t count);
804 static __inline void bus_space_set_region_2(bus_space_tag_t tag,
805 					    bus_space_handle_t bsh,
806 					    bus_size_t offset, u_int16_t value,
807 					    size_t count);
808 static __inline void bus_space_set_region_4(bus_space_tag_t tag,
809 					    bus_space_handle_t bsh,
810 					    bus_size_t offset, u_int32_t value,
811 					    size_t count);
812 
813 static __inline void
814 bus_space_set_region_1(bus_space_tag_t tag, bus_space_handle_t bsh,
815 		       bus_size_t offset, u_int8_t value, size_t count)
816 {
817 	bus_space_handle_t addr = bsh + offset;
818 
819 	if (tag == X86_BUS_SPACE_IO)
820 		for (; count != 0; count--, addr++)
821 			outb(addr, value);
822 	else
823 		for (; count != 0; count--, addr++)
824 			*(volatile u_int8_t *)(addr) = value;
825 }
826 
827 static __inline void
828 bus_space_set_region_2(bus_space_tag_t tag, bus_space_handle_t bsh,
829 		       bus_size_t offset, u_int16_t value, size_t count)
830 {
831 	bus_space_handle_t addr = bsh + offset;
832 
833 	if (tag == X86_BUS_SPACE_IO)
834 		for (; count != 0; count--, addr += 2)
835 			outw(addr, value);
836 	else
837 		for (; count != 0; count--, addr += 2)
838 			*(volatile u_int16_t *)(addr) = value;
839 }
840 
841 static __inline void
842 bus_space_set_region_4(bus_space_tag_t tag, bus_space_handle_t bsh,
843 		       bus_size_t offset, u_int32_t value, size_t count)
844 {
845 	bus_space_handle_t addr = bsh + offset;
846 
847 	if (tag == X86_BUS_SPACE_IO)
848 		for (; count != 0; count--, addr += 4)
849 			outl(addr, value);
850 	else
851 		for (; count != 0; count--, addr += 4)
852 			*(volatile u_int32_t *)(addr) = value;
853 }
854 
855 #if 0	/* Cause a link error for bus_space_set_region_8 */
856 #define	bus_space_set_region_8	!!! bus_space_set_region_8 unimplemented !!!
857 #endif
858 
859 /*
860  * Copy `count' 1, 2, 4, or 8 byte values from bus space starting
861  * at tag/bsh1/off1 to bus space starting at tag/bsh2/off2.
862  */
863 
864 static __inline void bus_space_copy_region_1(bus_space_tag_t tag,
865 					     bus_space_handle_t bsh1,
866 					     bus_size_t off1,
867 					     bus_space_handle_t bsh2,
868 					     bus_size_t off2, size_t count);
869 
870 static __inline void bus_space_copy_region_2(bus_space_tag_t tag,
871 					     bus_space_handle_t bsh1,
872 					     bus_size_t off1,
873 					     bus_space_handle_t bsh2,
874 					     bus_size_t off2, size_t count);
875 
876 static __inline void bus_space_copy_region_4(bus_space_tag_t tag,
877 					     bus_space_handle_t bsh1,
878 					     bus_size_t off1,
879 					     bus_space_handle_t bsh2,
880 					     bus_size_t off2, size_t count);
881 
882 static __inline void
883 bus_space_copy_region_1(bus_space_tag_t tag, bus_space_handle_t bsh1,
884 			bus_size_t off1, bus_space_handle_t bsh2,
885 			bus_size_t off2, size_t count)
886 {
887 	bus_space_handle_t addr1 = bsh1 + off1;
888 	bus_space_handle_t addr2 = bsh2 + off2;
889 
890 	if (tag == X86_BUS_SPACE_IO) {
891 		if (addr1 >= addr2) {
892 			/* src after dest: copy forward */
893 			for (; count != 0; count--, addr1++, addr2++)
894 				outb(addr2, inb(addr1));
895 		} else {
896 			/* dest after src: copy backwards */
897 			for (addr1 += (count - 1), addr2 += (count - 1);
898 			    count != 0; count--, addr1--, addr2--)
899 				outb(addr2, inb(addr1));
900 		}
901 	} else {
902 		if (addr1 >= addr2) {
903 			/* src after dest: copy forward */
904 			for (; count != 0; count--, addr1++, addr2++)
905 				*(volatile u_int8_t *)(addr2) =
906 				    *(volatile u_int8_t *)(addr1);
907 		} else {
908 			/* dest after src: copy backwards */
909 			for (addr1 += (count - 1), addr2 += (count - 1);
910 			    count != 0; count--, addr1--, addr2--)
911 				*(volatile u_int8_t *)(addr2) =
912 				    *(volatile u_int8_t *)(addr1);
913 		}
914 	}
915 }
916 
917 static __inline void
918 bus_space_copy_region_2(bus_space_tag_t tag, bus_space_handle_t bsh1,
919 			bus_size_t off1, bus_space_handle_t bsh2,
920 			bus_size_t off2, size_t count)
921 {
922 	bus_space_handle_t addr1 = bsh1 + off1;
923 	bus_space_handle_t addr2 = bsh2 + off2;
924 
925 	if (tag == X86_BUS_SPACE_IO) {
926 		if (addr1 >= addr2) {
927 			/* src after dest: copy forward */
928 			for (; count != 0; count--, addr1 += 2, addr2 += 2)
929 				outw(addr2, inw(addr1));
930 		} else {
931 			/* dest after src: copy backwards */
932 			for (addr1 += 2 * (count - 1), addr2 += 2 * (count - 1);
933 			    count != 0; count--, addr1 -= 2, addr2 -= 2)
934 				outw(addr2, inw(addr1));
935 		}
936 	} else {
937 		if (addr1 >= addr2) {
938 			/* src after dest: copy forward */
939 			for (; count != 0; count--, addr1 += 2, addr2 += 2)
940 				*(volatile u_int16_t *)(addr2) =
941 				    *(volatile u_int16_t *)(addr1);
942 		} else {
943 			/* dest after src: copy backwards */
944 			for (addr1 += 2 * (count - 1), addr2 += 2 * (count - 1);
945 			    count != 0; count--, addr1 -= 2, addr2 -= 2)
946 				*(volatile u_int16_t *)(addr2) =
947 				    *(volatile u_int16_t *)(addr1);
948 		}
949 	}
950 }
951 
952 static __inline void
953 bus_space_copy_region_4(bus_space_tag_t tag, bus_space_handle_t bsh1,
954 			bus_size_t off1, bus_space_handle_t bsh2,
955 			bus_size_t off2, size_t count)
956 {
957 	bus_space_handle_t addr1 = bsh1 + off1;
958 	bus_space_handle_t addr2 = bsh2 + off2;
959 
960 	if (tag == X86_BUS_SPACE_IO) {
961 		if (addr1 >= addr2) {
962 			/* src after dest: copy forward */
963 			for (; count != 0; count--, addr1 += 4, addr2 += 4)
964 				outl(addr2, inl(addr1));
965 		} else {
966 			/* dest after src: copy backwards */
967 			for (addr1 += 4 * (count - 1), addr2 += 4 * (count - 1);
968 			    count != 0; count--, addr1 -= 4, addr2 -= 4)
969 				outl(addr2, inl(addr1));
970 		}
971 	} else {
972 		if (addr1 >= addr2) {
973 			/* src after dest: copy forward */
974 			for (; count != 0; count--, addr1 += 4, addr2 += 4)
975 				*(volatile u_int32_t *)(addr2) =
976 				    *(volatile u_int32_t *)(addr1);
977 		} else {
978 			/* dest after src: copy backwards */
979 			for (addr1 += 4 * (count - 1), addr2 += 4 * (count - 1);
980 			    count != 0; count--, addr1 -= 4, addr2 -= 4)
981 				*(volatile u_int32_t *)(addr2) =
982 				    *(volatile u_int32_t *)(addr1);
983 		}
984 	}
985 }
986 
987 #if 0	/* Cause a link error for bus_space_copy_8 */
988 #define	bus_space_copy_region_8	!!! bus_space_copy_region_8 unimplemented !!!
989 #endif
990 
991 /*
992  * Bus read/write barrier methods.
993  *
994  *	void bus_space_barrier(bus_space_tag_t tag, bus_space_handle_t bsh,
995  *			       bus_size_t offset, bus_size_t len, int flags);
996  *
997  *
998  * Note that BUS_SPACE_BARRIER_WRITE doesn't do anything other than
999  * prevent reordering by the compiler; all Intel x86 processors currently
1000  * retire operations outside the CPU in program order.
1001  */
1002 #define	BUS_SPACE_BARRIER_READ	0x01		/* force read barrier */
1003 #define	BUS_SPACE_BARRIER_WRITE	0x02		/* force write barrier */
1004 
1005 static __inline void
1006 bus_space_barrier(bus_space_tag_t tag __unused, bus_space_handle_t bsh __unused,
1007 		  bus_size_t offset __unused, bus_size_t len __unused, int flags)
1008 {
1009 #ifdef __GNUCLIKE_ASM
1010 	if (flags & BUS_SPACE_BARRIER_READ)
1011 #ifdef __amd64__
1012 		__asm __volatile("lock; addl $0,0(%%rsp)" : : : "memory");
1013 #else
1014 		__asm __volatile("lock; addl $0,0(%%esp)" : : : "memory");
1015 #endif
1016 	else
1017 		__asm __volatile("" : : : "memory");
1018 #endif
1019 }
1020 
1021 #ifdef BUS_SPACE_NO_LEGACY
1022 #undef inb
1023 #undef outb
1024 #define inb(a) compiler_error
1025 #define inw(a) compiler_error
1026 #define inl(a) compiler_error
1027 #define outb(a, b) compiler_error
1028 #define outw(a, b) compiler_error
1029 #define outl(a, b) compiler_error
1030 #endif
1031 
1032 #include <machine/bus_dma.h>
1033 
1034 /*
1035  * Stream accesses are the same as normal accesses on x86; there are no
1036  * supported bus systems with an endianess different from the host one.
1037  */
1038 #define	bus_space_read_stream_1(t, h, o)	bus_space_read_1((t), (h), (o))
1039 #define	bus_space_read_stream_2(t, h, o)	bus_space_read_2((t), (h), (o))
1040 #define	bus_space_read_stream_4(t, h, o)	bus_space_read_4((t), (h), (o))
1041 
1042 #define	bus_space_read_multi_stream_1(t, h, o, a, c) \
1043 	bus_space_read_multi_1((t), (h), (o), (a), (c))
1044 #define	bus_space_read_multi_stream_2(t, h, o, a, c) \
1045 	bus_space_read_multi_2((t), (h), (o), (a), (c))
1046 #define	bus_space_read_multi_stream_4(t, h, o, a, c) \
1047 	bus_space_read_multi_4((t), (h), (o), (a), (c))
1048 
1049 #define	bus_space_write_stream_1(t, h, o, v) \
1050 	bus_space_write_1((t), (h), (o), (v))
1051 #define	bus_space_write_stream_2(t, h, o, v) \
1052 	bus_space_write_2((t), (h), (o), (v))
1053 #define	bus_space_write_stream_4(t, h, o, v) \
1054 	bus_space_write_4((t), (h), (o), (v))
1055 
1056 #define	bus_space_write_multi_stream_1(t, h, o, a, c) \
1057 	bus_space_write_multi_1((t), (h), (o), (a), (c))
1058 #define	bus_space_write_multi_stream_2(t, h, o, a, c) \
1059 	bus_space_write_multi_2((t), (h), (o), (a), (c))
1060 #define	bus_space_write_multi_stream_4(t, h, o, a, c) \
1061 	bus_space_write_multi_4((t), (h), (o), (a), (c))
1062 
1063 #define	bus_space_set_multi_stream_1(t, h, o, v, c) \
1064 	bus_space_set_multi_1((t), (h), (o), (v), (c))
1065 #define	bus_space_set_multi_stream_2(t, h, o, v, c) \
1066 	bus_space_set_multi_2((t), (h), (o), (v), (c))
1067 #define	bus_space_set_multi_stream_4(t, h, o, v, c) \
1068 	bus_space_set_multi_4((t), (h), (o), (v), (c))
1069 
1070 #define	bus_space_read_region_stream_1(t, h, o, a, c) \
1071 	bus_space_read_region_1((t), (h), (o), (a), (c))
1072 #define	bus_space_read_region_stream_2(t, h, o, a, c) \
1073 	bus_space_read_region_2((t), (h), (o), (a), (c))
1074 #define	bus_space_read_region_stream_4(t, h, o, a, c) \
1075 	bus_space_read_region_4((t), (h), (o), (a), (c))
1076 
1077 #define	bus_space_write_region_stream_1(t, h, o, a, c) \
1078 	bus_space_write_region_1((t), (h), (o), (a), (c))
1079 #define	bus_space_write_region_stream_2(t, h, o, a, c) \
1080 	bus_space_write_region_2((t), (h), (o), (a), (c))
1081 #define	bus_space_write_region_stream_4(t, h, o, a, c) \
1082 	bus_space_write_region_4((t), (h), (o), (a), (c))
1083 
1084 #define	bus_space_set_region_stream_1(t, h, o, v, c) \
1085 	bus_space_set_region_1((t), (h), (o), (v), (c))
1086 #define	bus_space_set_region_stream_2(t, h, o, v, c) \
1087 	bus_space_set_region_2((t), (h), (o), (v), (c))
1088 #define	bus_space_set_region_stream_4(t, h, o, v, c) \
1089 	bus_space_set_region_4((t), (h), (o), (v), (c))
1090 
1091 #define	bus_space_copy_region_stream_1(t, h1, o1, h2, o2, c) \
1092 	bus_space_copy_region_1((t), (h1), (o1), (h2), (o2), (c))
1093 #define	bus_space_copy_region_stream_2(t, h1, o1, h2, o2, c) \
1094 	bus_space_copy_region_2((t), (h1), (o1), (h2), (o2), (c))
1095 #define	bus_space_copy_region_stream_4(t, h1, o1, h2, o2, c) \
1096 	bus_space_copy_region_4((t), (h1), (o1), (h2), (o2), (c))
1097 
1098 #endif /* _X86_BUS_H_ */
1099