Home | History | Annotate | Line # | Download | only in include
pio.h revision 1.17
      1 /*	$NetBSD: pio.h,v 1.17 2001/04/30 01:17:31 lukem Exp $	*/
      2 
      3 /*-
      4  * Copyright (c) 1998 The NetBSD Foundation, Inc.
      5  * All rights reserved.
      6  *
      7  * This code is derived from software contributed to The NetBSD Foundation
      8  * by Charles M. Hannum.
      9  *
     10  * Redistribution and use in source and binary forms, with or without
     11  * modification, are permitted provided that the following conditions
     12  * are met:
     13  * 1. Redistributions of source code must retain the above copyright
     14  *    notice, this list of conditions and the following disclaimer.
     15  * 2. Redistributions in binary form must reproduce the above copyright
     16  *    notice, this list of conditions and the following disclaimer in the
     17  *    documentation and/or other materials provided with the distribution.
     18  * 3. All advertising materials mentioning features or use of this software
     19  *    must display the following acknowledgement:
     20  *        This product includes software developed by the NetBSD
     21  *        Foundation, Inc. and its contributors.
     22  * 4. Neither the name of The NetBSD Foundation nor the names of its
     23  *    contributors may be used to endorse or promote products derived
     24  *    from this software without specific prior written permission.
     25  *
     26  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     27  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     28  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     29  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     30  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     31  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     32  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     33  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     34  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     35  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     36  * POSSIBILITY OF SUCH DAMAGE.
     37  */
     38 
     39 #ifndef _I386_PIO_H_
     40 #define _I386_PIO_H_
     41 
     42 /*
     43  * Functions to provide access to i386 programmed I/O instructions.
     44  *
     45  * The in[bwl]() and out[bwl]() functions are split into two varieties: one to
     46  * use a small, constant, 8-bit port number, and another to use a large or
     47  * variable port number.  The former can be compiled as a smaller instruction.
     48  */
     49 
     50 
     51 #ifdef __OPTIMIZE__
     52 
     53 #define	__use_immediate_port(port) \
     54 	(__builtin_constant_p((port)) && (port) < 0x100)
     55 
     56 #else
     57 
     58 #define	__use_immediate_port(port)	0
     59 
     60 #endif
     61 
     62 
     63 #define	inb(port) \
     64     (/* CONSTCOND */ __use_immediate_port(port) ? __inbc(port) : __inb(port))
     65 
     66 static __inline u_int8_t
     67 __inbc(int port)
     68 {
     69 	u_int8_t data;
     70 	__asm __volatile("inb %1,%0" : "=a" (data) : "id" (port));
     71 	return data;
     72 }
     73 
     74 static __inline u_int8_t
     75 __inb(int port)
     76 {
     77 	u_int8_t data;
     78 	__asm __volatile("inb %w1,%0" : "=a" (data) : "d" (port));
     79 	return data;
     80 }
     81 
     82 static __inline void
     83 insb(int port, void *addr, int cnt)
     84 {
     85 	void *dummy1;
     86 	int dummy2;
     87 	__asm __volatile("cld\n\trepne\n\tinsb"			:
     88 			 "=D" (dummy1), "=c" (dummy2) 		:
     89 			 "d" (port), "0" (addr), "1" (cnt)	:
     90 			 "memory");
     91 }
     92 
     93 #define	inw(port) \
     94     (/* CONSTCOND */ __use_immediate_port(port) ? __inwc(port) : __inw(port))
     95 
     96 static __inline u_int16_t
     97 __inwc(int port)
     98 {
     99 	u_int16_t data;
    100 	__asm __volatile("inw %1,%0" : "=a" (data) : "id" (port));
    101 	return data;
    102 }
    103 
    104 static __inline u_int16_t
    105 __inw(int port)
    106 {
    107 	u_int16_t data;
    108 	__asm __volatile("inw %w1,%0" : "=a" (data) : "d" (port));
    109 	return data;
    110 }
    111 
    112 static __inline void
    113 insw(int port, void *addr, int cnt)
    114 {
    115 	void *dummy1;
    116 	int dummy2;
    117 	__asm __volatile("cld\n\trepne\n\tinsw"			:
    118 			 "=D" (dummy1), "=c" (dummy2)		:
    119 			 "d" (port), "0" (addr), "1" (cnt)	:
    120 			 "memory");
    121 }
    122 
    123 #define	inl(port) \
    124     (/* CONSTCOND */ __use_immediate_port(port) ? __inlc(port) : __inl(port))
    125 
    126 static __inline u_int32_t
    127 __inlc(int port)
    128 {
    129 	u_int32_t data;
    130 	__asm __volatile("inl %1,%0" : "=a" (data) : "id" (port));
    131 	return data;
    132 }
    133 
    134 static __inline u_int32_t
    135 __inl(int port)
    136 {
    137 	u_int32_t data;
    138 	__asm __volatile("inl %w1,%0" : "=a" (data) : "d" (port));
    139 	return data;
    140 }
    141 
    142 static __inline void
    143 insl(int port, void *addr, int cnt)
    144 {
    145 	void *dummy1;
    146 	int dummy2;
    147 	__asm __volatile("cld\n\trepne\n\tinsl"			:
    148 			 "=D" (dummy1), "=c" (dummy2)		:
    149 			 "d" (port), "0" (addr), "1" (cnt)	:
    150 			 "memory");
    151 }
    152 
    153 #define	outb(port, data) \
    154     (/* CONSTCOND */ __use_immediate_port(port) ? __outbc(port, data) : \
    155 						__outb(port, data))
    156 
    157 static __inline void
    158 __outbc(int port, u_int8_t data)
    159 {
    160 	__asm __volatile("outb %0,%1" : : "a" (data), "id" (port));
    161 }
    162 
    163 static __inline void
    164 __outb(int port, u_int8_t data)
    165 {
    166 	__asm __volatile("outb %0,%w1" : : "a" (data), "d" (port));
    167 }
    168 
    169 static __inline void
    170 outsb(int port, const void *addr, int cnt)
    171 {
    172 	void *dummy1;
    173 	int dummy2;
    174 	__asm __volatile("cld\n\trepne\n\toutsb"		:
    175 			 "=S" (dummy1), "=c" (dummy2)		:
    176 			 "d" (port), "0" (addr), "1" (cnt));
    177 }
    178 
    179 #define	outw(port, data) \
    180     (/* CONSTCOND */ __use_immediate_port(port) ? __outwc(port, data) : \
    181 						__outw(port, data))
    182 
    183 static __inline void
    184 __outwc(int port, u_int16_t data)
    185 {
    186 	__asm __volatile("outw %0,%1" : : "a" (data), "id" (port));
    187 }
    188 
    189 static __inline void
    190 __outw(int port, u_int16_t data)
    191 {
    192 	__asm __volatile("outw %0,%w1" : : "a" (data), "d" (port));
    193 }
    194 
    195 static __inline void
    196 outsw(int port, const void *addr, int cnt)
    197 {
    198 	void *dummy1;
    199 	int dummy2;
    200 	__asm __volatile("cld\n\trepne\n\toutsw"		:
    201 			 "=S" (dummy1), "=c" (dummy2)		:
    202 			 "d" (port), "0" (addr), "1" (cnt));
    203 }
    204 
    205 #define	outl(port, data) \
    206     (/* CONSTCOND */ __use_immediate_port(port) ? __outlc(port, data) : \
    207 						__outl(port, data))
    208 
    209 static __inline void
    210 __outlc(int port, u_int32_t data)
    211 {
    212 	__asm __volatile("outl %0,%1" : : "a" (data), "id" (port));
    213 }
    214 
    215 static __inline void
    216 __outl(int port, u_int32_t data)
    217 {
    218 	__asm __volatile("outl %0,%w1" : : "a" (data), "d" (port));
    219 }
    220 
    221 static __inline void
    222 outsl(int port, const void *addr, int cnt)
    223 {
    224 	void *dummy1;
    225 	int dummy2;
    226 	__asm __volatile("cld\n\trepne\n\toutsl"		:
    227 			 "=S" (dummy1), "=c" (dummy2)		:
    228 			 "d" (port), "0" (addr), "1" (cnt));
    229 }
    230 
    231 #endif /* _I386_PIO_H_ */
    232