xref: /titanic_50/usr/src/uts/intel/ia32/os/ddi_i86.c (revision 554ff184129088135ad2643c1c9832174a17be88)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 #pragma ident	"%Z%%M%	%I%	%E% SMI"
28 
29 #include <sys/conf.h>
30 #include <sys/kmem.h>
31 #include <sys/ddi_impldefs.h>
32 #include <sys/ddi.h>
33 #include <sys/sunddi.h>
34 #include <sys/ddifm.h>
35 
36 
37 /*
38  * DDI DMA Engine functions for x86.
39  * These functions are more naturally generic, but do not apply to SPARC.
40  */
41 
42 int
43 ddi_dmae_alloc(dev_info_t *dip, int chnl, int (*dmae_waitfp)(), caddr_t arg)
44 {
45 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_ACQUIRE,
46 	    (off_t *)dmae_waitfp, (size_t *)arg,
47 	    (caddr_t *)(uintptr_t)chnl, 0));
48 }
49 
50 int
51 ddi_dmae_release(dev_info_t *dip, int chnl)
52 {
53 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_FREE, 0, 0,
54 	    (caddr_t *)(uintptr_t)chnl, 0));
55 }
56 
57 int
58 ddi_dmae_getlim(dev_info_t *dip, ddi_dma_lim_t *limitsp)
59 {
60 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_GETLIM, 0, 0,
61 	    (caddr_t *)limitsp, 0));
62 }
63 
64 int
65 ddi_dmae_getattr(dev_info_t *dip, ddi_dma_attr_t *attrp)
66 {
67 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_GETATTR, 0, 0,
68 	    (caddr_t *)attrp, 0));
69 }
70 
71 int
72 ddi_dmae_1stparty(dev_info_t *dip, int chnl)
73 {
74 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_1STPTY, 0, 0,
75 	    (caddr_t *)(uintptr_t)chnl, 0));
76 }
77 
78 int
79 ddi_dmae_prog(dev_info_t *dip, struct ddi_dmae_req *dmaereqp,
80 	ddi_dma_cookie_t *cookiep, int chnl)
81 {
82 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_PROG, (off_t *)dmaereqp,
83 	    (size_t *)cookiep, (caddr_t *)(uintptr_t)chnl, 0));
84 }
85 
86 int
87 ddi_dmae_swsetup(dev_info_t *dip, struct ddi_dmae_req *dmaereqp,
88 	ddi_dma_cookie_t *cookiep, int chnl)
89 {
90 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_SWSETUP, (off_t *)dmaereqp,
91 	    (size_t *)cookiep, (caddr_t *)(uintptr_t)chnl, 0));
92 }
93 
94 int
95 ddi_dmae_swstart(dev_info_t *dip, int chnl)
96 {
97 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_SWSTART, 0, 0,
98 	    (caddr_t *)(uintptr_t)chnl, 0));
99 }
100 
101 int
102 ddi_dmae_stop(dev_info_t *dip, int chnl)
103 {
104 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_STOP, 0, 0,
105 	    (caddr_t *)(uintptr_t)chnl, 0));
106 }
107 
108 int
109 ddi_dmae_enable(dev_info_t *dip, int chnl)
110 {
111 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_ENABLE, 0, 0,
112 	    (caddr_t *)(uintptr_t)chnl, 0));
113 }
114 
115 int
116 ddi_dmae_disable(dev_info_t *dip, int chnl)
117 {
118 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_DISABLE, 0, 0,
119 	    (caddr_t *)(uintptr_t)chnl, 0));
120 }
121 
122 int
123 ddi_dmae_getcnt(dev_info_t *dip, int chnl, int *countp)
124 {
125 	return (ddi_dma_mctl(dip, dip, 0, DDI_DMA_E_GETCNT, 0, (size_t *)countp,
126 	    (caddr_t *)(uintptr_t)chnl, 0));
127 }
128 
129 /*
130  * implementation specific access handle and routines:
131  */
132 
133 static uintptr_t impl_acc_hdl_id = 0;
134 
135 /*
136  * access handle allocator
137  */
138 ddi_acc_hdl_t *
139 impl_acc_hdl_get(ddi_acc_handle_t hdl)
140 {
141 	/*
142 	 * recast to ddi_acc_hdl_t instead of
143 	 * casting to ddi_acc_impl_t and then return the ah_platform_private
144 	 *
145 	 * this optimization based on the ddi_acc_hdl_t is the
146 	 * first member of the ddi_acc_impl_t.
147 	 */
148 	return ((ddi_acc_hdl_t *)hdl);
149 }
150 
151 ddi_acc_handle_t
152 impl_acc_hdl_alloc(int (*waitfp)(caddr_t), caddr_t arg)
153 {
154 	ddi_acc_impl_t *hp;
155 	int sleepflag;
156 
157 	sleepflag = ((waitfp == (int (*)())KM_SLEEP) ? KM_SLEEP : KM_NOSLEEP);
158 	/*
159 	 * Allocate and initialize the data access handle.
160 	 */
161 	hp = kmem_zalloc(sizeof (ddi_acc_impl_t), sleepflag);
162 	if (!hp) {
163 		if ((waitfp != (int (*)())KM_SLEEP) &&
164 		    (waitfp != (int (*)())KM_NOSLEEP))
165 			ddi_set_callback(waitfp, arg, &impl_acc_hdl_id);
166 		return (NULL);
167 	}
168 	hp->ahi_common.ah_platform_private = (void *)hp;
169 	return ((ddi_acc_handle_t)hp);
170 }
171 
172 void
173 impl_acc_hdl_free(ddi_acc_handle_t handle)
174 {
175 	ddi_acc_impl_t *hp;
176 
177 	hp = (ddi_acc_impl_t *)handle;
178 	if (hp) {
179 		kmem_free(hp, sizeof (*hp));
180 		if (impl_acc_hdl_id)
181 			ddi_run_callback(&impl_acc_hdl_id);
182 	}
183 }
184 
185 void
186 impl_acc_err_init(ddi_acc_hdl_t *handlep)
187 {
188 	/* Error handling not supported */
189 	handlep->ah_acc.devacc_attr_access = DDI_DEFAULT_ACC;
190 }
191 
192 void
193 impl_acc_hdl_init(ddi_acc_hdl_t *handlep)
194 {
195 	ddi_acc_impl_t *hp;
196 
197 	if (!handlep)
198 		return;
199 	hp = (ddi_acc_impl_t *)handlep->ah_platform_private;
200 
201 	if (hp->ahi_acc_attr & DDI_ACCATTR_IO_SPACE) {
202 		hp->ahi_get8 = i_ddi_io_get8;
203 		hp->ahi_put8 = i_ddi_io_put8;
204 		hp->ahi_rep_get8 = i_ddi_io_rep_get8;
205 		hp->ahi_rep_put8 = i_ddi_io_rep_put8;
206 
207 		/* temporary set these 64 functions to no-ops */
208 		hp->ahi_get64 = i_ddi_io_get64;
209 		hp->ahi_put64 = i_ddi_io_put64;
210 		hp->ahi_rep_get64 = i_ddi_io_rep_get64;
211 		hp->ahi_rep_put64 = i_ddi_io_rep_put64;
212 
213 		/*
214 		 * check for BIG endian access
215 		 */
216 		if (handlep->ah_acc.devacc_attr_endian_flags ==
217 			DDI_STRUCTURE_BE_ACC) {
218 			hp->ahi_get16 = i_ddi_io_swap_get16;
219 			hp->ahi_get32 = i_ddi_io_swap_get32;
220 			hp->ahi_put16 = i_ddi_io_swap_put16;
221 			hp->ahi_put32 = i_ddi_io_swap_put32;
222 			hp->ahi_rep_get16 = i_ddi_io_swap_rep_get16;
223 			hp->ahi_rep_get32 = i_ddi_io_swap_rep_get32;
224 			hp->ahi_rep_put16 = i_ddi_io_swap_rep_put16;
225 			hp->ahi_rep_put32 = i_ddi_io_swap_rep_put32;
226 		} else {
227 			hp->ahi_acc_attr |= DDI_ACCATTR_DIRECT;
228 			hp->ahi_get16 = i_ddi_io_get16;
229 			hp->ahi_get32 = i_ddi_io_get32;
230 			hp->ahi_put16 = i_ddi_io_put16;
231 			hp->ahi_put32 = i_ddi_io_put32;
232 			hp->ahi_rep_get16 = i_ddi_io_rep_get16;
233 			hp->ahi_rep_get32 = i_ddi_io_rep_get32;
234 			hp->ahi_rep_put16 = i_ddi_io_rep_put16;
235 			hp->ahi_rep_put32 = i_ddi_io_rep_put32;
236 		}
237 
238 	} else if (hp->ahi_acc_attr & DDI_ACCATTR_CPU_VADDR) {
239 
240 		hp->ahi_get8 = i_ddi_vaddr_get8;
241 		hp->ahi_put8 = i_ddi_vaddr_put8;
242 		hp->ahi_rep_get8 = i_ddi_vaddr_rep_get8;
243 		hp->ahi_rep_put8 = i_ddi_vaddr_rep_put8;
244 
245 		/*
246 		 * check for BIG endian access
247 		 */
248 		if (handlep->ah_acc.devacc_attr_endian_flags ==
249 			DDI_STRUCTURE_BE_ACC) {
250 
251 			hp->ahi_get16 = i_ddi_vaddr_swap_get16;
252 			hp->ahi_get32 = i_ddi_vaddr_swap_get32;
253 			hp->ahi_get64 = i_ddi_vaddr_swap_get64;
254 			hp->ahi_put16 = i_ddi_vaddr_swap_put16;
255 			hp->ahi_put32 = i_ddi_vaddr_swap_put32;
256 			hp->ahi_put64 = i_ddi_vaddr_swap_put64;
257 			hp->ahi_rep_get16 = i_ddi_vaddr_swap_rep_get16;
258 			hp->ahi_rep_get32 = i_ddi_vaddr_swap_rep_get32;
259 			hp->ahi_rep_get64 = i_ddi_vaddr_swap_rep_get64;
260 			hp->ahi_rep_put16 = i_ddi_vaddr_swap_rep_put16;
261 			hp->ahi_rep_put32 = i_ddi_vaddr_swap_rep_put32;
262 			hp->ahi_rep_put64 = i_ddi_vaddr_swap_rep_put64;
263 		} else {
264 			hp->ahi_acc_attr |= DDI_ACCATTR_DIRECT;
265 			hp->ahi_get16 = i_ddi_vaddr_get16;
266 			hp->ahi_get32 = i_ddi_vaddr_get32;
267 			hp->ahi_get64 = i_ddi_vaddr_get64;
268 			hp->ahi_put16 = i_ddi_vaddr_put16;
269 			hp->ahi_put32 = i_ddi_vaddr_put32;
270 			hp->ahi_put64 = i_ddi_vaddr_put64;
271 			hp->ahi_rep_get16 = i_ddi_vaddr_rep_get16;
272 			hp->ahi_rep_get32 = i_ddi_vaddr_rep_get32;
273 			hp->ahi_rep_get64 = i_ddi_vaddr_rep_get64;
274 			hp->ahi_rep_put16 = i_ddi_vaddr_rep_put16;
275 			hp->ahi_rep_put32 = i_ddi_vaddr_rep_put32;
276 			hp->ahi_rep_put64 = i_ddi_vaddr_rep_put64;
277 		}
278 	}
279 	hp->ahi_fault_check = i_ddi_acc_fault_check;
280 	hp->ahi_fault_notify = i_ddi_acc_fault_notify;
281 	hp->ahi_fault = 0;
282 	impl_acc_err_init(handlep);
283 }
284 
285 /*
286  * The followings are low-level routines for data access.
287  *
288  * All of these routines should be implemented in assembly. Those
289  * that have been rewritten be found in ~ml/ddi_i86_asm.s
290  */
291 
292 /*ARGSUSED*/
293 uint16_t
294 i_ddi_vaddr_swap_get16(ddi_acc_impl_t *hdlp, uint16_t *addr)
295 {
296 	return (ddi_swap16(*addr));
297 }
298 
299 /*ARGSUSED*/
300 uint16_t
301 i_ddi_io_swap_get16(ddi_acc_impl_t *hdlp, uint16_t *addr)
302 {
303 	return (ddi_swap16(inw((uintptr_t)addr)));
304 }
305 
306 /*ARGSUSED*/
307 uint32_t
308 i_ddi_vaddr_swap_get32(ddi_acc_impl_t *hdlp, uint32_t *addr)
309 {
310 	return (ddi_swap32(*addr));
311 }
312 
313 /*ARGSUSED*/
314 uint32_t
315 i_ddi_io_swap_get32(ddi_acc_impl_t *hdlp, uint32_t *addr)
316 {
317 	return (ddi_swap32(inl((uintptr_t)addr)));
318 }
319 
320 /*ARGSUSED*/
321 uint64_t
322 i_ddi_vaddr_swap_get64(ddi_acc_impl_t *hdlp, uint64_t *addr)
323 {
324 	return (ddi_swap64(*addr));
325 }
326 
327 /*ARGSUSED*/
328 void
329 i_ddi_vaddr_swap_put16(ddi_acc_impl_t *hdlp, uint16_t *addr, uint16_t value)
330 {
331 	*addr = ddi_swap16(value);
332 }
333 
334 /*ARGSUSED*/
335 void
336 i_ddi_io_swap_put16(ddi_acc_impl_t *hdlp, uint16_t *addr, uint16_t value)
337 {
338 	outw((uintptr_t)addr, ddi_swap16(value));
339 }
340 
341 /*ARGSUSED*/
342 void
343 i_ddi_vaddr_swap_put32(ddi_acc_impl_t *hdlp, uint32_t *addr, uint32_t value)
344 {
345 	*addr = ddi_swap32(value);
346 }
347 
348 /*ARGSUSED*/
349 void
350 i_ddi_io_swap_put32(ddi_acc_impl_t *hdlp, uint32_t *addr, uint32_t value)
351 {
352 	outl((uintptr_t)addr, ddi_swap32(value));
353 }
354 
355 /*ARGSUSED*/
356 void
357 i_ddi_vaddr_swap_put64(ddi_acc_impl_t *hdlp, uint64_t *addr, uint64_t value)
358 {
359 	*addr = ddi_swap64(value);
360 }
361 
362 /*ARGSUSED*/
363 void
364 i_ddi_vaddr_rep_get8(ddi_acc_impl_t *hdlp, uint8_t *host_addr,
365 	uint8_t *dev_addr, size_t repcount, uint_t flags)
366 {
367 	uint8_t	*h, *d;
368 
369 	h = host_addr;
370 	d = dev_addr;
371 
372 	if (flags == DDI_DEV_AUTOINCR)
373 		for (; repcount; repcount--)
374 			*h++ = *d++;
375 	else
376 		for (; repcount; repcount--)
377 			*h++ = *d;
378 }
379 
380 /*ARGSUSED*/
381 void
382 i_ddi_vaddr_rep_get16(ddi_acc_impl_t *hdlp, uint16_t *host_addr,
383 	uint16_t *dev_addr, size_t repcount, uint_t flags)
384 {
385 	uint16_t *h, *d;
386 
387 	h = host_addr;
388 	d = dev_addr;
389 
390 	if (flags == DDI_DEV_AUTOINCR)
391 		for (; repcount; repcount--)
392 			*h++ = *d++;
393 	else
394 		for (; repcount; repcount--)
395 			*h++ = *d;
396 }
397 
398 /*ARGSUSED*/
399 void
400 i_ddi_vaddr_swap_rep_get16(ddi_acc_impl_t *hdlp, uint16_t *host_addr,
401 	uint16_t *dev_addr, size_t repcount, uint_t flags)
402 {
403 	uint16_t *h, *d;
404 
405 	h = host_addr;
406 	d = dev_addr;
407 
408 	if (flags == DDI_DEV_AUTOINCR)
409 		for (; repcount; repcount--)
410 			*h++ = ddi_swap16(*d++);
411 	else
412 		for (; repcount; repcount--)
413 			*h++ = ddi_swap16(*d);
414 }
415 
416 /*ARGSUSED*/
417 void
418 i_ddi_io_swap_rep_get16(ddi_acc_impl_t *hdlp, uint16_t *host_addr,
419 	uint16_t *dev_addr, size_t repcount, uint_t flags)
420 {
421 	uint16_t *h;
422 	uintptr_t port;
423 
424 	h = host_addr;
425 	port = (uintptr_t)dev_addr;
426 
427 	if (flags == DDI_DEV_AUTOINCR)
428 		for (; repcount; repcount--, port += 2)
429 			*h++ = ddi_swap16(inw(port));
430 	else
431 		for (; repcount; repcount--)
432 			*h++ = ddi_swap16(inw(port));
433 }
434 
435 /*ARGSUSED*/
436 void
437 i_ddi_vaddr_rep_get32(ddi_acc_impl_t *hdlp, uint32_t *host_addr,
438 	uint32_t *dev_addr, size_t repcount, uint_t flags)
439 {
440 	uint32_t *h, *d;
441 
442 	h = host_addr;
443 	d = dev_addr;
444 
445 	if (flags == DDI_DEV_AUTOINCR)
446 		for (; repcount; repcount--)
447 			*h++ = *d++;
448 	else
449 		for (; repcount; repcount--)
450 			*h++ = *d;
451 }
452 
453 /*ARGSUSED*/
454 void
455 i_ddi_vaddr_swap_rep_get32(ddi_acc_impl_t *hdlp, uint32_t *host_addr,
456 	uint32_t *dev_addr, size_t repcount, uint_t flags)
457 {
458 	uint32_t *h, *d;
459 
460 	h = host_addr;
461 	d = dev_addr;
462 
463 	if (flags == DDI_DEV_AUTOINCR)
464 		for (; repcount; repcount--)
465 			*h++ = ddi_swap32(*d++);
466 	else
467 		for (; repcount; repcount--)
468 			*h++ = ddi_swap32(*d);
469 }
470 
471 /*ARGSUSED*/
472 void
473 i_ddi_io_swap_rep_get32(ddi_acc_impl_t *hdlp, uint32_t *host_addr,
474 	uint32_t *dev_addr, size_t repcount, uint_t flags)
475 {
476 	uint32_t *h;
477 	uintptr_t port;
478 
479 	h = host_addr;
480 	port = (uintptr_t)dev_addr;
481 
482 	if (flags == DDI_DEV_AUTOINCR)
483 		for (; repcount; repcount--, port += 4)
484 			*h++ = ddi_swap32(inl(port));
485 	else
486 		for (; repcount; repcount--)
487 			*h++ = ddi_swap32(inl(port));
488 }
489 
490 /*ARGSUSED*/
491 void
492 i_ddi_vaddr_rep_get64(ddi_acc_impl_t *hdlp, uint64_t *host_addr,
493 	uint64_t *dev_addr, size_t repcount, uint_t flags)
494 {
495 	uint64_t *h, *d;
496 
497 	h = host_addr;
498 	d = dev_addr;
499 
500 	if (flags == DDI_DEV_AUTOINCR)
501 		for (; repcount; repcount--)
502 			*h++ = *d++;
503 	else
504 		for (; repcount; repcount--)
505 			*h++ = *d;
506 }
507 
508 /*ARGSUSED*/
509 void
510 i_ddi_vaddr_swap_rep_get64(ddi_acc_impl_t *hdlp, uint64_t *host_addr,
511 	uint64_t *dev_addr, size_t repcount, uint_t flags)
512 {
513 	uint64_t *h, *d;
514 
515 	h = host_addr;
516 	d = dev_addr;
517 
518 	if (flags == DDI_DEV_AUTOINCR)
519 		for (; repcount; repcount--)
520 			*h++ = ddi_swap64(*d++);
521 	else
522 		for (; repcount; repcount--)
523 			*h++ = ddi_swap64(*d);
524 }
525 
526 /*ARGSUSED*/
527 void
528 i_ddi_vaddr_rep_put8(ddi_acc_impl_t *hdlp, uint8_t *host_addr,
529 	uint8_t *dev_addr, size_t repcount, uint_t flags)
530 {
531 	uint8_t	*h, *d;
532 
533 	h = host_addr;
534 	d = dev_addr;
535 
536 	if (flags == DDI_DEV_AUTOINCR)
537 		for (; repcount; repcount--)
538 			*d++ = *h++;
539 	else
540 		for (; repcount; repcount--)
541 			*d = *h++;
542 }
543 
544 /*ARGSUSED*/
545 void
546 i_ddi_vaddr_rep_put16(ddi_acc_impl_t *hdlp, uint16_t *host_addr,
547 	uint16_t *dev_addr, size_t repcount, uint_t flags)
548 {
549 	uint16_t *h, *d;
550 
551 	h = host_addr;
552 	d = dev_addr;
553 
554 	if (flags == DDI_DEV_AUTOINCR)
555 		for (; repcount; repcount--)
556 			*d++ = *h++;
557 	else
558 		for (; repcount; repcount--)
559 			*d = *h++;
560 }
561 
562 /*ARGSUSED*/
563 void
564 i_ddi_vaddr_swap_rep_put16(ddi_acc_impl_t *hdlp, uint16_t *host_addr,
565 	uint16_t *dev_addr, size_t repcount, uint_t flags)
566 {
567 	uint16_t *h, *d;
568 
569 	h = host_addr;
570 	d = dev_addr;
571 
572 	if (flags == DDI_DEV_AUTOINCR)
573 		for (; repcount; repcount--)
574 			*d++ = ddi_swap16(*h++);
575 	else
576 		for (; repcount; repcount--)
577 			*d = ddi_swap16(*h++);
578 }
579 
580 /*ARGSUSED*/
581 void
582 i_ddi_io_swap_rep_put16(ddi_acc_impl_t *hdlp, uint16_t *host_addr,
583 	uint16_t *dev_addr, size_t repcount, uint_t flags)
584 {
585 	uint16_t *h;
586 	uintptr_t port;
587 
588 	h = host_addr;
589 	port = (uintptr_t)dev_addr;
590 
591 	if (flags == DDI_DEV_AUTOINCR)
592 		for (; repcount; repcount--, port += 2)
593 			outw(port, ddi_swap16(*h++));
594 	else
595 		for (; repcount; repcount--)
596 			outw(port, ddi_swap16(*h++));
597 }
598 
599 /*ARGSUSED*/
600 void
601 i_ddi_vaddr_rep_put32(ddi_acc_impl_t *hdlp, uint32_t *host_addr,
602 	uint32_t *dev_addr, size_t repcount, uint_t flags)
603 {
604 	uint32_t *h, *d;
605 
606 	h = host_addr;
607 	d = dev_addr;
608 
609 	if (flags == DDI_DEV_AUTOINCR)
610 		for (; repcount; repcount--)
611 			*d++ = *h++;
612 	else
613 		for (; repcount; repcount--)
614 			*d = *h++;
615 }
616 
617 /*ARGSUSED*/
618 void
619 i_ddi_vaddr_swap_rep_put32(ddi_acc_impl_t *hdlp, uint32_t *host_addr,
620 	uint32_t *dev_addr, size_t repcount, uint_t flags)
621 {
622 	uint32_t *h, *d;
623 
624 	h = host_addr;
625 	d = dev_addr;
626 
627 	if (flags == DDI_DEV_AUTOINCR)
628 		for (; repcount; repcount--)
629 			*d++ = ddi_swap32(*h++);
630 	else
631 		for (; repcount; repcount--)
632 			*d = ddi_swap32(*h++);
633 }
634 
635 /*ARGSUSED*/
636 void
637 i_ddi_io_swap_rep_put32(ddi_acc_impl_t *hdlp, uint32_t *host_addr,
638 	uint32_t *dev_addr, size_t repcount, uint_t flags)
639 {
640 	uint32_t *h;
641 	uintptr_t port;
642 
643 	h = host_addr;
644 	port = (uintptr_t)dev_addr;
645 
646 	if (flags == DDI_DEV_AUTOINCR)
647 		for (; repcount; repcount--, port += 4)
648 			outl(port, ddi_swap32(*h++));
649 	else
650 		for (; repcount; repcount--)
651 			outl(port, ddi_swap32(*h++));
652 }
653 
654 /*ARGSUSED*/
655 void
656 i_ddi_vaddr_rep_put64(ddi_acc_impl_t *hdlp, uint64_t *host_addr,
657 	uint64_t *dev_addr, size_t repcount, uint_t flags)
658 {
659 	uint64_t *h, *d;
660 
661 	h = host_addr;
662 	d = dev_addr;
663 
664 	if (flags == DDI_DEV_AUTOINCR)
665 		for (; repcount; repcount--)
666 			*d++ = *h++;
667 	else
668 		for (; repcount; repcount--)
669 			*d = *h++;
670 }
671 
672 /*ARGSUSED*/
673 void
674 i_ddi_vaddr_swap_rep_put64(ddi_acc_impl_t *hdlp, uint64_t *host_addr,
675 	uint64_t *dev_addr, size_t repcount, uint_t flags)
676 {
677 	uint64_t *h, *d;
678 
679 	h = host_addr;
680 	d = dev_addr;
681 
682 	if (flags == DDI_DEV_AUTOINCR)
683 		for (; repcount; repcount--)
684 			*d++ = ddi_swap64(*h++);
685 	else
686 		for (; repcount; repcount--)
687 			*d = ddi_swap64(*h++);
688 }
689 
690 /*ARGSUSED*/
691 uint64_t
692 i_ddi_io_get64(ddi_acc_impl_t *hdlp, uint64_t *addr)
693 {
694 	panic("ddi_get64 from i/o space");
695 	/*NOTREACHED*/
696 	return (0);
697 }
698 
699 /*ARGSUSED*/
700 void
701 i_ddi_io_put64(ddi_acc_impl_t *hdlp, uint64_t *host_addr, uint64_t value)
702 {
703 	panic("ddi_put64 to i/o space");
704 	/*NOTREACHED*/
705 }
706 
707 #ifdef _LP64
708 void
709 ddi_io_rep_get8(ddi_acc_handle_t handle,
710 	uint8_t *host_addr, uint8_t *dev_addr, size_t repcount)
711 #else /* _ILP32 */
712 void
713 ddi_io_rep_getb(ddi_acc_handle_t handle,
714 	uint8_t *host_addr, uint8_t *dev_addr, size_t repcount)
715 #endif
716 {
717 	(((ddi_acc_impl_t *)handle)->ahi_rep_get8)
718 		((ddi_acc_impl_t *)handle, host_addr, dev_addr,
719 		repcount, DDI_DEV_NO_AUTOINCR);
720 }
721 
722 #ifdef _LP64
723 void
724 ddi_io_rep_get16(ddi_acc_handle_t handle,
725 	uint16_t *host_addr, uint16_t *dev_addr, size_t repcount)
726 #else /* _ILP32 */
727 void
728 ddi_io_rep_getw(ddi_acc_handle_t handle,
729 	uint16_t *host_addr, uint16_t *dev_addr, size_t repcount)
730 #endif
731 {
732 	(((ddi_acc_impl_t *)handle)->ahi_rep_get16)
733 		((ddi_acc_impl_t *)handle, host_addr, dev_addr,
734 		repcount, DDI_DEV_NO_AUTOINCR);
735 }
736 
737 #ifdef _LP64
738 void
739 ddi_io_rep_get32(ddi_acc_handle_t handle,
740 	uint32_t *host_addr, uint32_t *dev_addr, size_t repcount)
741 #else /* _ILP32 */
742 void
743 ddi_io_rep_getl(ddi_acc_handle_t handle,
744 	uint32_t *host_addr, uint32_t *dev_addr, size_t repcount)
745 #endif
746 {
747 	(((ddi_acc_impl_t *)handle)->ahi_rep_get32)
748 		((ddi_acc_impl_t *)handle, host_addr, dev_addr,
749 		repcount, DDI_DEV_NO_AUTOINCR);
750 }
751 
752 /*ARGSUSED*/
753 void
754 i_ddi_io_rep_get64(ddi_acc_impl_t *hdlp, uint64_t *host_addr,
755 	uint64_t *dev_addr, size_t repcount, uint_t flags)
756 {
757 	cmn_err(CE_PANIC, "ddi_rep_get64 from i/o space");
758 }
759 
760 #ifdef _LP64
761 void
762 ddi_io_rep_put8(ddi_acc_handle_t handle,
763 	uint8_t *host_addr, uint8_t *dev_addr, size_t repcount)
764 #else /* _ILP32 */
765 void
766 ddi_io_rep_putb(ddi_acc_handle_t handle,
767 	uint8_t *host_addr, uint8_t *dev_addr, size_t repcount)
768 #endif
769 {
770 	(((ddi_acc_impl_t *)handle)->ahi_rep_put8)
771 		((ddi_acc_impl_t *)handle, host_addr, dev_addr,
772 		repcount, DDI_DEV_NO_AUTOINCR);
773 }
774 
775 #ifdef _LP64
776 void
777 ddi_io_rep_put16(ddi_acc_handle_t handle,
778 	uint16_t *host_addr, uint16_t *dev_addr, size_t repcount)
779 #else /* _ILP32 */
780 void
781 ddi_io_rep_putw(ddi_acc_handle_t handle,
782 	uint16_t *host_addr, uint16_t *dev_addr, size_t repcount)
783 #endif
784 {
785 	(((ddi_acc_impl_t *)handle)->ahi_rep_put16)
786 		((ddi_acc_impl_t *)handle, host_addr, dev_addr,
787 		repcount, DDI_DEV_NO_AUTOINCR);
788 }
789 
790 #ifdef _LP64
791 void
792 ddi_io_rep_put32(ddi_acc_handle_t handle,
793 	uint32_t *host_addr, uint32_t *dev_addr, size_t repcount)
794 #else /* _ILP32 */
795 void
796 ddi_io_rep_putl(ddi_acc_handle_t handle,
797 	uint32_t *host_addr, uint32_t *dev_addr, size_t repcount)
798 #endif
799 {
800 	(((ddi_acc_impl_t *)handle)->ahi_rep_put32)
801 		((ddi_acc_impl_t *)handle, host_addr, dev_addr,
802 		repcount, DDI_DEV_NO_AUTOINCR);
803 }
804 
805 /*ARGSUSED*/
806 void
807 i_ddi_io_rep_put64(ddi_acc_impl_t *hdlp, uint64_t *host_addr,
808 	uint64_t *dev_addr, size_t repcount, uint_t flags)
809 {
810 	cmn_err(CE_PANIC, "ddi_rep_put64 to i/o space");
811 }
812 
813 /*
814  * These next two functions could be translated into assembler someday
815  */
816 int
817 ddi_check_acc_handle(ddi_acc_handle_t handle)
818 {
819 	ddi_acc_impl_t *hdlp = (ddi_acc_impl_t *)handle;
820 	return (((*hdlp->ahi_fault_check)(hdlp) == DDI_SUCCESS) ? DDI_SUCCESS :
821 	    DDI_FAILURE);
822 }
823 
824 int
825 i_ddi_acc_fault_check(ddi_acc_impl_t *hdlp)
826 {
827 	/* Default version, just returns flag value */
828 	return (hdlp->ahi_fault);
829 }
830 
831 /*ARGSUSED*/
832 void
833 i_ddi_acc_fault_notify(ddi_acc_impl_t *hdlp)
834 {
835 	/* Default version, does nothing for now */
836 }
837 
838 void
839 i_ddi_acc_set_fault(ddi_acc_handle_t handle)
840 {
841 	ddi_acc_impl_t *hdlp = (ddi_acc_impl_t *)handle;
842 
843 	if (!hdlp->ahi_fault) {
844 		hdlp->ahi_fault = 1;
845 		(*hdlp->ahi_fault_notify)(hdlp);
846 	}
847 }
848 
849 void
850 i_ddi_acc_clr_fault(ddi_acc_handle_t handle)
851 {
852 	ddi_acc_impl_t *hdlp = (ddi_acc_impl_t *)handle;
853 
854 	if (hdlp->ahi_fault) {
855 		hdlp->ahi_fault = 0;
856 		(*hdlp->ahi_fault_notify)(hdlp);
857 	}
858 }
859