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