xref: /OK3568_Linux_fs/u-boot/drivers/net/smc91111.h (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*------------------------------------------------------------------------
2*4882a593Smuzhiyun  . smc91111.h - macros for the LAN91C111 Ethernet Driver
3*4882a593Smuzhiyun  .
4*4882a593Smuzhiyun  . (C) Copyright 2002
5*4882a593Smuzhiyun  . Sysgo Real-Time Solutions, GmbH <www.elinos.com>
6*4882a593Smuzhiyun  . Rolf Offermanns <rof@sysgo.de>
7*4882a593Smuzhiyun  . Copyright (C) 2001 Standard Microsystems Corporation (SMSC)
8*4882a593Smuzhiyun  .       Developed by Simple Network Magic Corporation (SNMC)
9*4882a593Smuzhiyun  . Copyright (C) 1996 by Erik Stahlman (ES)
10*4882a593Smuzhiyun  .
11*4882a593Smuzhiyun   * SPDX-License-Identifier:	GPL-2.0+
12*4882a593Smuzhiyun  .
13*4882a593Smuzhiyun  . This file contains register information and access macros for
14*4882a593Smuzhiyun  . the LAN91C111 single chip ethernet controller.  It is a modified
15*4882a593Smuzhiyun  . version of the smc9194.h file.
16*4882a593Smuzhiyun  .
17*4882a593Smuzhiyun  . Information contained in this file was obtained from the LAN91C111
18*4882a593Smuzhiyun  . manual from SMC.  To get a copy, if you really want one, you can find
19*4882a593Smuzhiyun  . information under www.smsc.com.
20*4882a593Smuzhiyun  .
21*4882a593Smuzhiyun  . Authors
22*4882a593Smuzhiyun  .	Erik Stahlman				( erik@vt.edu )
23*4882a593Smuzhiyun  .	Daris A Nevil				( dnevil@snmc.com )
24*4882a593Smuzhiyun  .
25*4882a593Smuzhiyun  . History
26*4882a593Smuzhiyun  . 03/16/01		Daris A Nevil	Modified for use with LAN91C111 device
27*4882a593Smuzhiyun  .
28*4882a593Smuzhiyun  ---------------------------------------------------------------------------*/
29*4882a593Smuzhiyun #ifndef _SMC91111_H_
30*4882a593Smuzhiyun #define _SMC91111_H_
31*4882a593Smuzhiyun 
32*4882a593Smuzhiyun #include <asm/types.h>
33*4882a593Smuzhiyun #include <config.h>
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun /*
36*4882a593Smuzhiyun  * This function may be called by the board specific initialisation code
37*4882a593Smuzhiyun  * in order to override the default mac address.
38*4882a593Smuzhiyun  */
39*4882a593Smuzhiyun 
40*4882a593Smuzhiyun void smc_set_mac_addr (const unsigned char *addr);
41*4882a593Smuzhiyun 
42*4882a593Smuzhiyun 
43*4882a593Smuzhiyun /* I want some simple types */
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun typedef unsigned char			byte;
46*4882a593Smuzhiyun typedef unsigned short			word;
47*4882a593Smuzhiyun typedef unsigned long int		dword;
48*4882a593Smuzhiyun 
49*4882a593Smuzhiyun struct smc91111_priv{
50*4882a593Smuzhiyun 	u8 dev_num;
51*4882a593Smuzhiyun };
52*4882a593Smuzhiyun 
53*4882a593Smuzhiyun /*
54*4882a593Smuzhiyun  . DEBUGGING LEVELS
55*4882a593Smuzhiyun  .
56*4882a593Smuzhiyun  . 0 for normal operation
57*4882a593Smuzhiyun  . 1 for slightly more details
58*4882a593Smuzhiyun  . >2 for various levels of increasingly useless information
59*4882a593Smuzhiyun  .    2 for interrupt tracking, status flags
60*4882a593Smuzhiyun  .    3 for packet info
61*4882a593Smuzhiyun  .    4 for complete packet dumps
62*4882a593Smuzhiyun */
63*4882a593Smuzhiyun /*#define SMC_DEBUG 0 */
64*4882a593Smuzhiyun 
65*4882a593Smuzhiyun /* Because of bank switching, the LAN91xxx uses only 16 I/O ports */
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun #define	SMC_IO_EXTENT	16
68*4882a593Smuzhiyun 
69*4882a593Smuzhiyun #ifdef CONFIG_CPU_PXA25X
70*4882a593Smuzhiyun 
71*4882a593Smuzhiyun #ifdef CONFIG_XSENGINE
72*4882a593Smuzhiyun #define	SMC_inl(a,r)	(*((volatile dword *)((a)->iobase+((r)<<1))))
73*4882a593Smuzhiyun #define	SMC_inw(a,r)	(*((volatile word *)((a)->iobase+((r)<<1))))
74*4882a593Smuzhiyun #define SMC_inb(a,p)  ({ \
75*4882a593Smuzhiyun 	unsigned int __p = (unsigned int)((a)->iobase + ((p)<<1)); \
76*4882a593Smuzhiyun 	unsigned int __v = *(volatile unsigned short *)((__p) & ~2); \
77*4882a593Smuzhiyun 	if (__p & 2) __v >>= 8; \
78*4882a593Smuzhiyun 	else __v &= 0xff; \
79*4882a593Smuzhiyun 	__v; })
80*4882a593Smuzhiyun #else
81*4882a593Smuzhiyun #define	SMC_inl(a,r)	(*((volatile dword *)((a)->iobase+(r))))
82*4882a593Smuzhiyun #define	SMC_inw(a,r)	(*((volatile word *)((a)->iobase+(r))))
83*4882a593Smuzhiyun #define SMC_inb(a,p)	({ \
84*4882a593Smuzhiyun 	unsigned int __p = (unsigned int)((a)->iobase + (p)); \
85*4882a593Smuzhiyun 	unsigned int __v = *(volatile unsigned short *)((__p) & ~1); \
86*4882a593Smuzhiyun 	if (__p & 1) __v >>= 8; \
87*4882a593Smuzhiyun 	else __v &= 0xff; \
88*4882a593Smuzhiyun 	__v; })
89*4882a593Smuzhiyun #endif
90*4882a593Smuzhiyun 
91*4882a593Smuzhiyun #ifdef CONFIG_XSENGINE
92*4882a593Smuzhiyun #define	SMC_outl(a,d,r)	(*((volatile dword *)((a)->iobase+(r<<1))) = d)
93*4882a593Smuzhiyun #define	SMC_outw(a,d,r)	(*((volatile word *)((a)->iobase+(r<<1))) = d)
94*4882a593Smuzhiyun #else
95*4882a593Smuzhiyun #define	SMC_outl(a,d,r)	(*((volatile dword *)((a)->iobase+(r))) = d)
96*4882a593Smuzhiyun #define	SMC_outw(a,d,r)	(*((volatile word *)((a)->iobase+(r))) = d)
97*4882a593Smuzhiyun #endif
98*4882a593Smuzhiyun 
99*4882a593Smuzhiyun #define	SMC_outb(a,d,r)	({	word __d = (byte)(d);  \
100*4882a593Smuzhiyun 				word __w = SMC_inw((a),(r)&~1);  \
101*4882a593Smuzhiyun 				__w &= ((r)&1) ? 0x00FF : 0xFF00;  \
102*4882a593Smuzhiyun 				__w |= ((r)&1) ? __d<<8 : __d;  \
103*4882a593Smuzhiyun 				SMC_outw((a),__w,(r)&~1);  \
104*4882a593Smuzhiyun 			})
105*4882a593Smuzhiyun 
106*4882a593Smuzhiyun #define SMC_outsl(a,r,b,l)	({	int __i; \
107*4882a593Smuzhiyun 					dword *__b2; \
108*4882a593Smuzhiyun 					__b2 = (dword *) b; \
109*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) { \
110*4882a593Smuzhiyun 					    SMC_outl((a), *(__b2 + __i), r); \
111*4882a593Smuzhiyun 					} \
112*4882a593Smuzhiyun 				})
113*4882a593Smuzhiyun 
114*4882a593Smuzhiyun #define SMC_outsw(a,r,b,l)	({	int __i; \
115*4882a593Smuzhiyun 					word *__b2; \
116*4882a593Smuzhiyun 					__b2 = (word *) b; \
117*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) { \
118*4882a593Smuzhiyun 					    SMC_outw((a), *(__b2 + __i), r); \
119*4882a593Smuzhiyun 					} \
120*4882a593Smuzhiyun 				})
121*4882a593Smuzhiyun 
122*4882a593Smuzhiyun #define SMC_insl(a,r,b,l)	({	int __i ;  \
123*4882a593Smuzhiyun 					dword *__b2;  \
124*4882a593Smuzhiyun 					__b2 = (dword *) b;  \
125*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) {  \
126*4882a593Smuzhiyun 					  *(__b2 + __i) = SMC_inl((a),(r));  \
127*4882a593Smuzhiyun 					  SMC_inl((a),0);  \
128*4882a593Smuzhiyun 					};  \
129*4882a593Smuzhiyun 				})
130*4882a593Smuzhiyun 
131*4882a593Smuzhiyun #define SMC_insw(a,r,b,l)		({	int __i ;  \
132*4882a593Smuzhiyun 					word *__b2;  \
133*4882a593Smuzhiyun 					__b2 = (word *) b;  \
134*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) {  \
135*4882a593Smuzhiyun 					  *(__b2 + __i) = SMC_inw((a),(r));  \
136*4882a593Smuzhiyun 					  SMC_inw((a),0);  \
137*4882a593Smuzhiyun 					};  \
138*4882a593Smuzhiyun 				})
139*4882a593Smuzhiyun 
140*4882a593Smuzhiyun #define SMC_insb(a,r,b,l)	({	int __i ;  \
141*4882a593Smuzhiyun 					byte *__b2;  \
142*4882a593Smuzhiyun 					__b2 = (byte *) b;  \
143*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) {  \
144*4882a593Smuzhiyun 					  *(__b2 + __i) = SMC_inb((a),(r));  \
145*4882a593Smuzhiyun 					  SMC_inb((a),0);  \
146*4882a593Smuzhiyun 					};  \
147*4882a593Smuzhiyun 				})
148*4882a593Smuzhiyun 
149*4882a593Smuzhiyun #elif defined(CONFIG_LEON)	/* if not CONFIG_CPU_PXA25X */
150*4882a593Smuzhiyun 
151*4882a593Smuzhiyun #define SMC_LEON_SWAP16(_x_) ({ word _x = (_x_); ((_x << 8) | (_x >> 8)); })
152*4882a593Smuzhiyun 
153*4882a593Smuzhiyun #define SMC_LEON_SWAP32(_x_)			\
154*4882a593Smuzhiyun     ({ dword _x = (_x_);			\
155*4882a593Smuzhiyun        ((_x << 24) |				\
156*4882a593Smuzhiyun        ((0x0000FF00UL & _x) <<  8) |		\
157*4882a593Smuzhiyun        ((0x00FF0000UL & _x) >>  8) |		\
158*4882a593Smuzhiyun        (_x  >> 24)); })
159*4882a593Smuzhiyun 
160*4882a593Smuzhiyun #define	SMC_inl(a,r)	(SMC_LEON_SWAP32((*(volatile dword *)((a)->iobase+((r)<<0)))))
161*4882a593Smuzhiyun #define	SMC_inl_nosw(a,r)	((*(volatile dword *)((a)->iobase+((r)<<0))))
162*4882a593Smuzhiyun #define	SMC_inw(a,r)	(SMC_LEON_SWAP16((*(volatile word *)((a)->iobase+((r)<<0)))))
163*4882a593Smuzhiyun #define	SMC_inw_nosw(a,r)	((*(volatile word *)((a)->iobase+((r)<<0))))
164*4882a593Smuzhiyun #define SMC_inb(a,p)	({ \
165*4882a593Smuzhiyun 	word ___v = SMC_inw((a),(p) & ~1); \
166*4882a593Smuzhiyun 	if ((p) & 1) ___v >>= 8; \
167*4882a593Smuzhiyun 	else ___v &= 0xff; \
168*4882a593Smuzhiyun 	___v; })
169*4882a593Smuzhiyun 
170*4882a593Smuzhiyun #define	SMC_outl(a,d,r)	(*(volatile dword *)((a)->iobase+((r)<<0))=SMC_LEON_SWAP32(d))
171*4882a593Smuzhiyun #define	SMC_outl_nosw(a,d,r)	(*(volatile dword *)((a)->iobase+((r)<<0))=(d))
172*4882a593Smuzhiyun #define	SMC_outw(a,d,r)	(*(volatile word *)((a)->iobase+((r)<<0))=SMC_LEON_SWAP16(d))
173*4882a593Smuzhiyun #define	SMC_outw_nosw(a,d,r)	(*(volatile word *)((a)->iobase+((r)<<0))=(d))
174*4882a593Smuzhiyun #define	SMC_outb(a,d,r)	do{	word __d = (byte)(d);  \
175*4882a593Smuzhiyun 				word __w = SMC_inw((a),(r)&~1);  \
176*4882a593Smuzhiyun 				__w &= ((r)&1) ? 0x00FF : 0xFF00;  \
177*4882a593Smuzhiyun 				__w |= ((r)&1) ? __d<<8 : __d;  \
178*4882a593Smuzhiyun 				SMC_outw((a),__w,(r)&~1);  \
179*4882a593Smuzhiyun 			}while(0)
180*4882a593Smuzhiyun #define SMC_outsl(a,r,b,l)	do{	int __i; \
181*4882a593Smuzhiyun 					dword *__b2; \
182*4882a593Smuzhiyun 					__b2 = (dword *) b; \
183*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) { \
184*4882a593Smuzhiyun 					    SMC_outl_nosw((a), *(__b2 + __i), r); \
185*4882a593Smuzhiyun 					} \
186*4882a593Smuzhiyun 				}while(0)
187*4882a593Smuzhiyun #define SMC_outsw(a,r,b,l)	do{	int __i; \
188*4882a593Smuzhiyun 					word *__b2; \
189*4882a593Smuzhiyun 					__b2 = (word *) b; \
190*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) { \
191*4882a593Smuzhiyun 					    SMC_outw_nosw((a), *(__b2 + __i), r); \
192*4882a593Smuzhiyun 					} \
193*4882a593Smuzhiyun 				}while(0)
194*4882a593Smuzhiyun #define SMC_insl(a,r,b,l)	do{	int __i ;  \
195*4882a593Smuzhiyun 					dword *__b2;  \
196*4882a593Smuzhiyun 					__b2 = (dword *) b;  \
197*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) {  \
198*4882a593Smuzhiyun 					  *(__b2 + __i) = SMC_inl_nosw((a),(r));  \
199*4882a593Smuzhiyun 					};  \
200*4882a593Smuzhiyun 				}while(0)
201*4882a593Smuzhiyun 
202*4882a593Smuzhiyun #define SMC_insw(a,r,b,l)		do{	int __i ;  \
203*4882a593Smuzhiyun 					word *__b2;  \
204*4882a593Smuzhiyun 					__b2 = (word *) b;  \
205*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) {  \
206*4882a593Smuzhiyun 					  *(__b2 + __i) = SMC_inw_nosw((a),(r));  \
207*4882a593Smuzhiyun 					};  \
208*4882a593Smuzhiyun 				}while(0)
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun #define SMC_insb(a,r,b,l)		do{	int __i ;  \
211*4882a593Smuzhiyun 					byte *__b2;  \
212*4882a593Smuzhiyun 					__b2 = (byte *) b;  \
213*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) {  \
214*4882a593Smuzhiyun 					  *(__b2 + __i) = SMC_inb((a),(r));  \
215*4882a593Smuzhiyun 					};  \
216*4882a593Smuzhiyun 				}while(0)
217*4882a593Smuzhiyun #elif defined(CONFIG_MS7206SE)
218*4882a593Smuzhiyun #define SWAB7206(x) ({ word __x = x; ((__x << 8)|(__x >> 8)); })
219*4882a593Smuzhiyun #define SMC_inw(a, r) *((volatile word*)((a)->iobase + (r)))
220*4882a593Smuzhiyun #define SMC_inb(a, r) (*((volatile byte*)((a)->iobase + ((r) ^ 0x01))))
221*4882a593Smuzhiyun #define SMC_insw(a, r, b, l) \
222*4882a593Smuzhiyun 	do { \
223*4882a593Smuzhiyun 		int __i; \
224*4882a593Smuzhiyun 		word *__b2 = (word *)(b);		  \
225*4882a593Smuzhiyun 		for (__i = 0; __i < (l); __i++) { \
226*4882a593Smuzhiyun 			*__b2++ = SWAB7206(SMC_inw(a, r));	\
227*4882a593Smuzhiyun 		} \
228*4882a593Smuzhiyun 	} while (0)
229*4882a593Smuzhiyun #define	SMC_outw(a, d, r)	(*((volatile word *)((a)->iobase+(r))) = d)
230*4882a593Smuzhiyun #define	SMC_outb(a, d, r)	({	word __d = (byte)(d);  \
231*4882a593Smuzhiyun 				word __w = SMC_inw((a), ((r)&(~1)));	\
232*4882a593Smuzhiyun 				if (((r) & 1)) \
233*4882a593Smuzhiyun 					__w = (__w & 0x00ff) | (__d << 8); \
234*4882a593Smuzhiyun 				else \
235*4882a593Smuzhiyun 					__w = (__w & 0xff00) | (__d); \
236*4882a593Smuzhiyun 				SMC_outw((a), __w, ((r)&(~1)));	      \
237*4882a593Smuzhiyun 			})
238*4882a593Smuzhiyun #define SMC_outsw(a, r, b, l) \
239*4882a593Smuzhiyun 	do { \
240*4882a593Smuzhiyun 		int __i; \
241*4882a593Smuzhiyun 		word *__b2 = (word *)(b);		  \
242*4882a593Smuzhiyun 		for (__i = 0; __i < (l); __i++) { \
243*4882a593Smuzhiyun 			SMC_outw(a, SWAB7206(*__b2), r);	  \
244*4882a593Smuzhiyun 			__b2++; \
245*4882a593Smuzhiyun 		} \
246*4882a593Smuzhiyun 	} while (0)
247*4882a593Smuzhiyun #else			/* if not CONFIG_CPU_PXA25X and not CONFIG_LEON */
248*4882a593Smuzhiyun 
249*4882a593Smuzhiyun #ifndef CONFIG_SMC_USE_IOFUNCS /* these macros don't work on some boards */
250*4882a593Smuzhiyun /*
251*4882a593Smuzhiyun  * We have only 16 Bit PCMCIA access on Socket 0
252*4882a593Smuzhiyun  */
253*4882a593Smuzhiyun 
254*4882a593Smuzhiyun #ifdef CONFIG_ADNPESC1
255*4882a593Smuzhiyun #define	SMC_inw(a,r)	(*((volatile word *)((a)->iobase+((r)<<1))))
256*4882a593Smuzhiyun #elif CONFIG_ARM64
257*4882a593Smuzhiyun #define	SMC_inw(a, r)	(*((volatile word*)((a)->iobase+((dword)(r)))))
258*4882a593Smuzhiyun #else
259*4882a593Smuzhiyun #define SMC_inw(a, r)	(*((volatile word*)((a)->iobase+(r))))
260*4882a593Smuzhiyun #endif
261*4882a593Smuzhiyun #define  SMC_inb(a,r)	(((r)&1) ? SMC_inw((a),(r)&~1)>>8 : SMC_inw((a),(r)&0xFF))
262*4882a593Smuzhiyun 
263*4882a593Smuzhiyun #ifdef CONFIG_ADNPESC1
264*4882a593Smuzhiyun #define	SMC_outw(a,d,r)	(*((volatile word *)((a)->iobase+((r)<<1))) = d)
265*4882a593Smuzhiyun #elif CONFIG_ARM64
266*4882a593Smuzhiyun #define	SMC_outw(a, d, r)	\
267*4882a593Smuzhiyun 			(*((volatile word*)((a)->iobase+((dword)(r)))) = d)
268*4882a593Smuzhiyun #else
269*4882a593Smuzhiyun #define	SMC_outw(a, d, r)	\
270*4882a593Smuzhiyun 			(*((volatile word*)((a)->iobase+(r))) = d)
271*4882a593Smuzhiyun #endif
272*4882a593Smuzhiyun #define	SMC_outb(a,d,r)	({	word __d = (byte)(d);  \
273*4882a593Smuzhiyun 				word __w = SMC_inw((a),(r)&~1);  \
274*4882a593Smuzhiyun 				__w &= ((r)&1) ? 0x00FF : 0xFF00;  \
275*4882a593Smuzhiyun 				__w |= ((r)&1) ? __d<<8 : __d;  \
276*4882a593Smuzhiyun 				SMC_outw((a),__w,(r)&~1);  \
277*4882a593Smuzhiyun 			})
278*4882a593Smuzhiyun #if 0
279*4882a593Smuzhiyun #define	SMC_outsw(a,r,b,l)	outsw((a)->iobase+(r), (b), (l))
280*4882a593Smuzhiyun #else
281*4882a593Smuzhiyun #define SMC_outsw(a,r,b,l)	({	int __i; \
282*4882a593Smuzhiyun 					word *__b2; \
283*4882a593Smuzhiyun 					__b2 = (word *) b; \
284*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) { \
285*4882a593Smuzhiyun 					    SMC_outw((a), *(__b2 + __i), r); \
286*4882a593Smuzhiyun 					} \
287*4882a593Smuzhiyun 				})
288*4882a593Smuzhiyun #endif
289*4882a593Smuzhiyun 
290*4882a593Smuzhiyun #if 0
291*4882a593Smuzhiyun #define	SMC_insw(a,r,b,l)	insw((a)->iobase+(r), (b), (l))
292*4882a593Smuzhiyun #else
293*4882a593Smuzhiyun #define SMC_insw(a,r,b,l)	({	int __i ;  \
294*4882a593Smuzhiyun 					word *__b2;  \
295*4882a593Smuzhiyun 					__b2 = (word *) b;  \
296*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) {  \
297*4882a593Smuzhiyun 					  *(__b2 + __i) = SMC_inw((a),(r));  \
298*4882a593Smuzhiyun 					  SMC_inw((a),0);  \
299*4882a593Smuzhiyun 					};  \
300*4882a593Smuzhiyun 				})
301*4882a593Smuzhiyun #endif
302*4882a593Smuzhiyun 
303*4882a593Smuzhiyun #endif  /* CONFIG_SMC_USE_IOFUNCS */
304*4882a593Smuzhiyun 
305*4882a593Smuzhiyun #if defined(CONFIG_SMC_USE_32_BIT)
306*4882a593Smuzhiyun 
307*4882a593Smuzhiyun #ifdef CONFIG_XSENGINE
308*4882a593Smuzhiyun #define	SMC_inl(a,r)	(*((volatile dword *)((a)->iobase+(r<<1))))
309*4882a593Smuzhiyun #else
310*4882a593Smuzhiyun #define	SMC_inl(a,r)	(*((volatile dword *)((a)->iobase+(r))))
311*4882a593Smuzhiyun #endif
312*4882a593Smuzhiyun 
313*4882a593Smuzhiyun #define SMC_insl(a,r,b,l)	({	int __i ;  \
314*4882a593Smuzhiyun 					dword *__b2;  \
315*4882a593Smuzhiyun 					__b2 = (dword *) b;  \
316*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) {  \
317*4882a593Smuzhiyun 					  *(__b2 + __i) = SMC_inl((a),(r));  \
318*4882a593Smuzhiyun 					  SMC_inl((a),0);  \
319*4882a593Smuzhiyun 					};  \
320*4882a593Smuzhiyun 				})
321*4882a593Smuzhiyun 
322*4882a593Smuzhiyun #ifdef CONFIG_XSENGINE
323*4882a593Smuzhiyun #define	SMC_outl(a,d,r)	(*((volatile dword *)((a)->iobase+(r<<1))) = d)
324*4882a593Smuzhiyun #else
325*4882a593Smuzhiyun #define	SMC_outl(a,d,r)	(*((volatile dword *)((a)->iobase+(r))) = d)
326*4882a593Smuzhiyun #endif
327*4882a593Smuzhiyun #define SMC_outsl(a,r,b,l)	({	int __i; \
328*4882a593Smuzhiyun 					dword *__b2; \
329*4882a593Smuzhiyun 					__b2 = (dword *) b; \
330*4882a593Smuzhiyun 					for (__i = 0; __i < l; __i++) { \
331*4882a593Smuzhiyun 					    SMC_outl((a), *(__b2 + __i), r); \
332*4882a593Smuzhiyun 					} \
333*4882a593Smuzhiyun 				})
334*4882a593Smuzhiyun 
335*4882a593Smuzhiyun #endif /* CONFIG_SMC_USE_32_BIT */
336*4882a593Smuzhiyun 
337*4882a593Smuzhiyun #endif
338*4882a593Smuzhiyun 
339*4882a593Smuzhiyun /*---------------------------------------------------------------
340*4882a593Smuzhiyun  .
341*4882a593Smuzhiyun  . A description of the SMSC registers is probably in order here,
342*4882a593Smuzhiyun  . although for details, the SMC datasheet is invaluable.
343*4882a593Smuzhiyun  .
344*4882a593Smuzhiyun  . Basically, the chip has 4 banks of registers ( 0 to 3 ), which
345*4882a593Smuzhiyun  . are accessed by writing a number into the BANK_SELECT register
346*4882a593Smuzhiyun  . ( I also use a SMC_SELECT_BANK macro for this ).
347*4882a593Smuzhiyun  .
348*4882a593Smuzhiyun  . The banks are configured so that for most purposes, bank 2 is all
349*4882a593Smuzhiyun  . that is needed for simple run time tasks.
350*4882a593Smuzhiyun  -----------------------------------------------------------------------*/
351*4882a593Smuzhiyun 
352*4882a593Smuzhiyun /*
353*4882a593Smuzhiyun  . Bank Select Register:
354*4882a593Smuzhiyun  .
355*4882a593Smuzhiyun  .		yyyy yyyy 0000 00xx
356*4882a593Smuzhiyun  .		xx		= bank number
357*4882a593Smuzhiyun  .		yyyy yyyy	= 0x33, for identification purposes.
358*4882a593Smuzhiyun */
359*4882a593Smuzhiyun #define	BANK_SELECT		14
360*4882a593Smuzhiyun 
361*4882a593Smuzhiyun /* Transmit Control Register */
362*4882a593Smuzhiyun /* BANK 0  */
363*4882a593Smuzhiyun #define	TCR_REG		0x0000	/* transmit control register */
364*4882a593Smuzhiyun #define TCR_ENABLE	0x0001	/* When 1 we can transmit */
365*4882a593Smuzhiyun #define TCR_LOOP	0x0002	/* Controls output pin LBK */
366*4882a593Smuzhiyun #define TCR_FORCOL	0x0004	/* When 1 will force a collision */
367*4882a593Smuzhiyun #define TCR_PAD_EN	0x0080	/* When 1 will pad tx frames < 64 bytes w/0 */
368*4882a593Smuzhiyun #define TCR_NOCRC	0x0100	/* When 1 will not append CRC to tx frames */
369*4882a593Smuzhiyun #define TCR_MON_CSN	0x0400	/* When 1 tx monitors carrier */
370*4882a593Smuzhiyun #define TCR_FDUPLX	0x0800  /* When 1 enables full duplex operation */
371*4882a593Smuzhiyun #define TCR_STP_SQET	0x1000	/* When 1 stops tx if Signal Quality Error */
372*4882a593Smuzhiyun #define	TCR_EPH_LOOP	0x2000	/* When 1 enables EPH block loopback */
373*4882a593Smuzhiyun #define	TCR_SWFDUP	0x8000	/* When 1 enables Switched Full Duplex mode */
374*4882a593Smuzhiyun 
375*4882a593Smuzhiyun #define	TCR_CLEAR	0	/* do NOTHING */
376*4882a593Smuzhiyun /* the default settings for the TCR register : */
377*4882a593Smuzhiyun /* QUESTION: do I want to enable padding of short packets ? */
378*4882a593Smuzhiyun #define	TCR_DEFAULT	TCR_ENABLE
379*4882a593Smuzhiyun 
380*4882a593Smuzhiyun 
381*4882a593Smuzhiyun /* EPH Status Register */
382*4882a593Smuzhiyun /* BANK 0  */
383*4882a593Smuzhiyun #define EPH_STATUS_REG	0x0002
384*4882a593Smuzhiyun #define ES_TX_SUC	0x0001	/* Last TX was successful */
385*4882a593Smuzhiyun #define ES_SNGL_COL	0x0002	/* Single collision detected for last tx */
386*4882a593Smuzhiyun #define ES_MUL_COL	0x0004	/* Multiple collisions detected for last tx */
387*4882a593Smuzhiyun #define ES_LTX_MULT	0x0008	/* Last tx was a multicast */
388*4882a593Smuzhiyun #define ES_16COL	0x0010	/* 16 Collisions Reached */
389*4882a593Smuzhiyun #define ES_SQET		0x0020	/* Signal Quality Error Test */
390*4882a593Smuzhiyun #define ES_LTXBRD	0x0040	/* Last tx was a broadcast */
391*4882a593Smuzhiyun #define ES_TXDEFR	0x0080	/* Transmit Deferred */
392*4882a593Smuzhiyun #define ES_LATCOL	0x0200	/* Late collision detected on last tx */
393*4882a593Smuzhiyun #define ES_LOSTCARR	0x0400	/* Lost Carrier Sense */
394*4882a593Smuzhiyun #define ES_EXC_DEF	0x0800	/* Excessive Deferral */
395*4882a593Smuzhiyun #define ES_CTR_ROL	0x1000	/* Counter Roll Over indication */
396*4882a593Smuzhiyun #define ES_LINK_OK	0x4000	/* Driven by inverted value of nLNK pin */
397*4882a593Smuzhiyun #define ES_TXUNRN	0x8000	/* Tx Underrun */
398*4882a593Smuzhiyun 
399*4882a593Smuzhiyun 
400*4882a593Smuzhiyun /* Receive Control Register */
401*4882a593Smuzhiyun /* BANK 0  */
402*4882a593Smuzhiyun #define	RCR_REG		0x0004
403*4882a593Smuzhiyun #define	RCR_RX_ABORT	0x0001	/* Set if a rx frame was aborted */
404*4882a593Smuzhiyun #define	RCR_PRMS	0x0002	/* Enable promiscuous mode */
405*4882a593Smuzhiyun #define	RCR_ALMUL	0x0004	/* When set accepts all multicast frames */
406*4882a593Smuzhiyun #define RCR_RXEN	0x0100	/* IFF this is set, we can receive packets */
407*4882a593Smuzhiyun #define	RCR_STRIP_CRC	0x0200	/* When set strips CRC from rx packets */
408*4882a593Smuzhiyun #define	RCR_ABORT_ENB	0x0200	/* When set will abort rx on collision */
409*4882a593Smuzhiyun #define	RCR_FILT_CAR	0x0400	/* When set filters leading 12 bit s of carrier */
410*4882a593Smuzhiyun #define RCR_SOFTRST	0x8000	/* resets the chip */
411*4882a593Smuzhiyun 
412*4882a593Smuzhiyun /* the normal settings for the RCR register : */
413*4882a593Smuzhiyun #define	RCR_DEFAULT	(RCR_STRIP_CRC | RCR_RXEN)
414*4882a593Smuzhiyun #define RCR_CLEAR	0x0	/* set it to a base state */
415*4882a593Smuzhiyun 
416*4882a593Smuzhiyun /* Counter Register */
417*4882a593Smuzhiyun /* BANK 0  */
418*4882a593Smuzhiyun #define	COUNTER_REG	0x0006
419*4882a593Smuzhiyun 
420*4882a593Smuzhiyun /* Memory Information Register */
421*4882a593Smuzhiyun /* BANK 0  */
422*4882a593Smuzhiyun #define	MIR_REG		0x0008
423*4882a593Smuzhiyun 
424*4882a593Smuzhiyun /* Receive/Phy Control Register */
425*4882a593Smuzhiyun /* BANK 0  */
426*4882a593Smuzhiyun #define	RPC_REG		0x000A
427*4882a593Smuzhiyun #define	RPC_SPEED	0x2000	/* When 1 PHY is in 100Mbps mode. */
428*4882a593Smuzhiyun #define	RPC_DPLX	0x1000	/* When 1 PHY is in Full-Duplex Mode */
429*4882a593Smuzhiyun #define	RPC_ANEG	0x0800	/* When 1 PHY is in Auto-Negotiate Mode */
430*4882a593Smuzhiyun #define	RPC_LSXA_SHFT	5	/* Bits to shift LS2A,LS1A,LS0A to lsb */
431*4882a593Smuzhiyun #define	RPC_LSXB_SHFT	2	/* Bits to get LS2B,LS1B,LS0B to lsb */
432*4882a593Smuzhiyun #define RPC_LED_100_10	(0x00)	/* LED = 100Mbps OR's with 10Mbps link detect */
433*4882a593Smuzhiyun #define RPC_LED_RES	(0x01)	/* LED = Reserved */
434*4882a593Smuzhiyun #define RPC_LED_10	(0x02)	/* LED = 10Mbps link detect */
435*4882a593Smuzhiyun #define RPC_LED_FD	(0x03)	/* LED = Full Duplex Mode */
436*4882a593Smuzhiyun #define RPC_LED_TX_RX	(0x04)	/* LED = TX or RX packet occurred */
437*4882a593Smuzhiyun #define RPC_LED_100	(0x05)	/* LED = 100Mbps link dectect */
438*4882a593Smuzhiyun #define RPC_LED_TX	(0x06)	/* LED = TX packet occurred */
439*4882a593Smuzhiyun #define RPC_LED_RX	(0x07)	/* LED = RX packet occurred */
440*4882a593Smuzhiyun #if defined(CONFIG_DK1C20) || defined(CONFIG_DK1S10)
441*4882a593Smuzhiyun /* buggy schematic: LEDa -> yellow, LEDb --> green */
442*4882a593Smuzhiyun #define RPC_DEFAULT	( RPC_SPEED | RPC_DPLX | RPC_ANEG	\
443*4882a593Smuzhiyun 			| (RPC_LED_TX_RX << RPC_LSXA_SHFT)	\
444*4882a593Smuzhiyun 			| (RPC_LED_100_10 << RPC_LSXB_SHFT)	)
445*4882a593Smuzhiyun #elif defined(CONFIG_ADNPESC1)
446*4882a593Smuzhiyun /* SSV ADNP/ESC1 has only one LED: LEDa -> Rx/Tx indicator */
447*4882a593Smuzhiyun #define RPC_DEFAULT	( RPC_SPEED | RPC_DPLX | RPC_ANEG	\
448*4882a593Smuzhiyun 			| (RPC_LED_TX_RX << RPC_LSXA_SHFT)	\
449*4882a593Smuzhiyun 			| (RPC_LED_100_10 << RPC_LSXB_SHFT)	)
450*4882a593Smuzhiyun #else
451*4882a593Smuzhiyun /* SMSC reference design: LEDa --> green, LEDb --> yellow */
452*4882a593Smuzhiyun #define RPC_DEFAULT	( RPC_SPEED | RPC_DPLX | RPC_ANEG	\
453*4882a593Smuzhiyun 			| (RPC_LED_100_10 << RPC_LSXA_SHFT)	\
454*4882a593Smuzhiyun 			| (RPC_LED_TX_RX << RPC_LSXB_SHFT)	)
455*4882a593Smuzhiyun #endif
456*4882a593Smuzhiyun 
457*4882a593Smuzhiyun /* Bank 0 0x000C is reserved */
458*4882a593Smuzhiyun 
459*4882a593Smuzhiyun /* Bank Select Register */
460*4882a593Smuzhiyun /* All Banks */
461*4882a593Smuzhiyun #define BSR_REG	0x000E
462*4882a593Smuzhiyun 
463*4882a593Smuzhiyun 
464*4882a593Smuzhiyun /* Configuration Reg */
465*4882a593Smuzhiyun /* BANK 1 */
466*4882a593Smuzhiyun #define CONFIG_REG	0x0000
467*4882a593Smuzhiyun #define CONFIG_EXT_PHY	0x0200	/* 1=external MII, 0=internal Phy */
468*4882a593Smuzhiyun #define CONFIG_GPCNTRL	0x0400	/* Inverse value drives pin nCNTRL */
469*4882a593Smuzhiyun #define CONFIG_NO_WAIT	0x1000	/* When 1 no extra wait states on ISA bus */
470*4882a593Smuzhiyun #define CONFIG_EPH_POWER_EN 0x8000 /* When 0 EPH is placed into low power mode. */
471*4882a593Smuzhiyun 
472*4882a593Smuzhiyun /* Default is powered-up, Internal Phy, Wait States, and pin nCNTRL=low */
473*4882a593Smuzhiyun #define CONFIG_DEFAULT	(CONFIG_EPH_POWER_EN)
474*4882a593Smuzhiyun 
475*4882a593Smuzhiyun 
476*4882a593Smuzhiyun /* Base Address Register */
477*4882a593Smuzhiyun /* BANK 1 */
478*4882a593Smuzhiyun #define	BASE_REG	0x0002
479*4882a593Smuzhiyun 
480*4882a593Smuzhiyun 
481*4882a593Smuzhiyun /* Individual Address Registers */
482*4882a593Smuzhiyun /* BANK 1 */
483*4882a593Smuzhiyun #define	ADDR0_REG	0x0004
484*4882a593Smuzhiyun #define	ADDR1_REG	0x0006
485*4882a593Smuzhiyun #define	ADDR2_REG	0x0008
486*4882a593Smuzhiyun 
487*4882a593Smuzhiyun 
488*4882a593Smuzhiyun /* General Purpose Register */
489*4882a593Smuzhiyun /* BANK 1 */
490*4882a593Smuzhiyun #define	GP_REG		0x000A
491*4882a593Smuzhiyun 
492*4882a593Smuzhiyun 
493*4882a593Smuzhiyun /* Control Register */
494*4882a593Smuzhiyun /* BANK 1 */
495*4882a593Smuzhiyun #define	CTL_REG		0x000C
496*4882a593Smuzhiyun #define CTL_RCV_BAD	0x4000 /* When 1 bad CRC packets are received */
497*4882a593Smuzhiyun #define CTL_AUTO_RELEASE 0x0800 /* When 1 tx pages are released automatically */
498*4882a593Smuzhiyun #define	CTL_LE_ENABLE	0x0080 /* When 1 enables Link Error interrupt */
499*4882a593Smuzhiyun #define	CTL_CR_ENABLE	0x0040 /* When 1 enables Counter Rollover interrupt */
500*4882a593Smuzhiyun #define	CTL_TE_ENABLE	0x0020 /* When 1 enables Transmit Error interrupt */
501*4882a593Smuzhiyun #define	CTL_EEPROM_SELECT 0x0004 /* Controls EEPROM reload & store */
502*4882a593Smuzhiyun #define	CTL_RELOAD	0x0002 /* When set reads EEPROM into registers */
503*4882a593Smuzhiyun #define	CTL_STORE	0x0001 /* When set stores registers into EEPROM */
504*4882a593Smuzhiyun #define CTL_DEFAULT     (0x1A10) /* Autorelease enabled*/
505*4882a593Smuzhiyun 
506*4882a593Smuzhiyun /* MMU Command Register */
507*4882a593Smuzhiyun /* BANK 2 */
508*4882a593Smuzhiyun #define MMU_CMD_REG	0x0000
509*4882a593Smuzhiyun #define MC_BUSY		1	/* When 1 the last release has not completed */
510*4882a593Smuzhiyun #define MC_NOP		(0<<5)	/* No Op */
511*4882a593Smuzhiyun #define	MC_ALLOC	(1<<5)	/* OR with number of 256 byte packets */
512*4882a593Smuzhiyun #define	MC_RESET	(2<<5)	/* Reset MMU to initial state */
513*4882a593Smuzhiyun #define	MC_REMOVE	(3<<5)	/* Remove the current rx packet */
514*4882a593Smuzhiyun #define MC_RELEASE	(4<<5)	/* Remove and release the current rx packet */
515*4882a593Smuzhiyun #define MC_FREEPKT	(5<<5)	/* Release packet in PNR register */
516*4882a593Smuzhiyun #define MC_ENQUEUE	(6<<5)	/* Enqueue the packet for transmit */
517*4882a593Smuzhiyun #define MC_RSTTXFIFO	(7<<5)	/* Reset the TX FIFOs */
518*4882a593Smuzhiyun 
519*4882a593Smuzhiyun 
520*4882a593Smuzhiyun /* Packet Number Register */
521*4882a593Smuzhiyun /* BANK 2 */
522*4882a593Smuzhiyun #define	PN_REG		0x0002
523*4882a593Smuzhiyun 
524*4882a593Smuzhiyun 
525*4882a593Smuzhiyun /* Allocation Result Register */
526*4882a593Smuzhiyun /* BANK 2 */
527*4882a593Smuzhiyun #define	AR_REG		0x0003
528*4882a593Smuzhiyun #define AR_FAILED	0x80	/* Alocation Failed */
529*4882a593Smuzhiyun 
530*4882a593Smuzhiyun 
531*4882a593Smuzhiyun /* RX FIFO Ports Register */
532*4882a593Smuzhiyun /* BANK 2 */
533*4882a593Smuzhiyun #define RXFIFO_REG	0x0004	/* Must be read as a word */
534*4882a593Smuzhiyun #define RXFIFO_REMPTY	0x8000	/* RX FIFO Empty */
535*4882a593Smuzhiyun 
536*4882a593Smuzhiyun 
537*4882a593Smuzhiyun /* TX FIFO Ports Register */
538*4882a593Smuzhiyun /* BANK 2 */
539*4882a593Smuzhiyun #define TXFIFO_REG	RXFIFO_REG	/* Must be read as a word */
540*4882a593Smuzhiyun #define TXFIFO_TEMPTY	0x80	/* TX FIFO Empty */
541*4882a593Smuzhiyun 
542*4882a593Smuzhiyun 
543*4882a593Smuzhiyun /* Pointer Register */
544*4882a593Smuzhiyun /* BANK 2 */
545*4882a593Smuzhiyun #define PTR_REG		0x0006
546*4882a593Smuzhiyun #define	PTR_RCV		0x8000 /* 1=Receive area, 0=Transmit area */
547*4882a593Smuzhiyun #define	PTR_AUTOINC	0x4000 /* Auto increment the pointer on each access */
548*4882a593Smuzhiyun #define PTR_READ	0x2000 /* When 1 the operation is a read */
549*4882a593Smuzhiyun #define PTR_NOTEMPTY	0x0800 /* When 1 _do not_ write fifo DATA REG */
550*4882a593Smuzhiyun 
551*4882a593Smuzhiyun 
552*4882a593Smuzhiyun /* Data Register */
553*4882a593Smuzhiyun /* BANK 2 */
554*4882a593Smuzhiyun #define	SMC91111_DATA_REG	0x0008
555*4882a593Smuzhiyun 
556*4882a593Smuzhiyun 
557*4882a593Smuzhiyun /* Interrupt Status/Acknowledge Register */
558*4882a593Smuzhiyun /* BANK 2 */
559*4882a593Smuzhiyun #define	SMC91111_INT_REG	0x000C
560*4882a593Smuzhiyun 
561*4882a593Smuzhiyun 
562*4882a593Smuzhiyun /* Interrupt Mask Register */
563*4882a593Smuzhiyun /* BANK 2 */
564*4882a593Smuzhiyun #define IM_REG		0x000D
565*4882a593Smuzhiyun #define	IM_MDINT	0x80 /* PHY MI Register 18 Interrupt */
566*4882a593Smuzhiyun #define	IM_ERCV_INT	0x40 /* Early Receive Interrupt */
567*4882a593Smuzhiyun #define	IM_EPH_INT	0x20 /* Set by Etheret Protocol Handler section */
568*4882a593Smuzhiyun #define	IM_RX_OVRN_INT	0x10 /* Set by Receiver Overruns */
569*4882a593Smuzhiyun #define	IM_ALLOC_INT	0x08 /* Set when allocation request is completed */
570*4882a593Smuzhiyun #define	IM_TX_EMPTY_INT	0x04 /* Set if the TX FIFO goes empty */
571*4882a593Smuzhiyun #define	IM_TX_INT	0x02 /* Transmit Interrrupt */
572*4882a593Smuzhiyun #define IM_RCV_INT	0x01 /* Receive Interrupt */
573*4882a593Smuzhiyun 
574*4882a593Smuzhiyun 
575*4882a593Smuzhiyun /* Multicast Table Registers */
576*4882a593Smuzhiyun /* BANK 3 */
577*4882a593Smuzhiyun #define	MCAST_REG1	0x0000
578*4882a593Smuzhiyun #define	MCAST_REG2	0x0002
579*4882a593Smuzhiyun #define	MCAST_REG3	0x0004
580*4882a593Smuzhiyun #define	MCAST_REG4	0x0006
581*4882a593Smuzhiyun 
582*4882a593Smuzhiyun 
583*4882a593Smuzhiyun /* Management Interface Register (MII) */
584*4882a593Smuzhiyun /* BANK 3 */
585*4882a593Smuzhiyun #define	MII_REG		0x0008
586*4882a593Smuzhiyun #define MII_MSK_CRS100	0x4000 /* Disables CRS100 detection during tx half dup */
587*4882a593Smuzhiyun #define MII_MDOE	0x0008 /* MII Output Enable */
588*4882a593Smuzhiyun #define MII_MCLK	0x0004 /* MII Clock, pin MDCLK */
589*4882a593Smuzhiyun #define MII_MDI		0x0002 /* MII Input, pin MDI */
590*4882a593Smuzhiyun #define MII_MDO		0x0001 /* MII Output, pin MDO */
591*4882a593Smuzhiyun 
592*4882a593Smuzhiyun 
593*4882a593Smuzhiyun /* Revision Register */
594*4882a593Smuzhiyun /* BANK 3 */
595*4882a593Smuzhiyun #define	REV_REG		0x000A /* ( hi: chip id   low: rev # ) */
596*4882a593Smuzhiyun 
597*4882a593Smuzhiyun 
598*4882a593Smuzhiyun /* Early RCV Register */
599*4882a593Smuzhiyun /* BANK 3 */
600*4882a593Smuzhiyun /* this is NOT on SMC9192 */
601*4882a593Smuzhiyun #define	ERCV_REG	0x000C
602*4882a593Smuzhiyun #define ERCV_RCV_DISCRD	0x0080 /* When 1 discards a packet being received */
603*4882a593Smuzhiyun #define ERCV_THRESHOLD	0x001F /* ERCV Threshold Mask */
604*4882a593Smuzhiyun 
605*4882a593Smuzhiyun /* External Register */
606*4882a593Smuzhiyun /* BANK 7 */
607*4882a593Smuzhiyun #define	EXT_REG		0x0000
608*4882a593Smuzhiyun 
609*4882a593Smuzhiyun 
610*4882a593Smuzhiyun #define CHIP_9192	3
611*4882a593Smuzhiyun #define CHIP_9194	4
612*4882a593Smuzhiyun #define CHIP_9195	5
613*4882a593Smuzhiyun #define CHIP_9196	6
614*4882a593Smuzhiyun #define CHIP_91100	7
615*4882a593Smuzhiyun #define CHIP_91100FD	8
616*4882a593Smuzhiyun #define CHIP_91111FD	9
617*4882a593Smuzhiyun 
618*4882a593Smuzhiyun #if 0
619*4882a593Smuzhiyun static const char * chip_ids[ 15 ] =  {
620*4882a593Smuzhiyun 	NULL, NULL, NULL,
621*4882a593Smuzhiyun 	/* 3 */ "SMC91C90/91C92",
622*4882a593Smuzhiyun 	/* 4 */ "SMC91C94",
623*4882a593Smuzhiyun 	/* 5 */ "SMC91C95",
624*4882a593Smuzhiyun 	/* 6 */ "SMC91C96",
625*4882a593Smuzhiyun 	/* 7 */ "SMC91C100",
626*4882a593Smuzhiyun 	/* 8 */ "SMC91C100FD",
627*4882a593Smuzhiyun 	/* 9 */ "SMC91C111",
628*4882a593Smuzhiyun 	NULL, NULL,
629*4882a593Smuzhiyun 	NULL, NULL, NULL};
630*4882a593Smuzhiyun #endif
631*4882a593Smuzhiyun 
632*4882a593Smuzhiyun /*
633*4882a593Smuzhiyun  . Transmit status bits
634*4882a593Smuzhiyun */
635*4882a593Smuzhiyun #define TS_SUCCESS 0x0001
636*4882a593Smuzhiyun #define TS_LOSTCAR 0x0400
637*4882a593Smuzhiyun #define TS_LATCOL  0x0200
638*4882a593Smuzhiyun #define TS_16COL   0x0010
639*4882a593Smuzhiyun 
640*4882a593Smuzhiyun /*
641*4882a593Smuzhiyun  . Receive status bits
642*4882a593Smuzhiyun */
643*4882a593Smuzhiyun #define RS_ALGNERR	0x8000
644*4882a593Smuzhiyun #define RS_BRODCAST	0x4000
645*4882a593Smuzhiyun #define RS_BADCRC	0x2000
646*4882a593Smuzhiyun #define RS_ODDFRAME	0x1000	/* bug: the LAN91C111 never sets this on receive */
647*4882a593Smuzhiyun #define RS_TOOLONG	0x0800
648*4882a593Smuzhiyun #define RS_TOOSHORT	0x0400
649*4882a593Smuzhiyun #define RS_MULTICAST	0x0001
650*4882a593Smuzhiyun #define RS_ERRORS	(RS_ALGNERR | RS_BADCRC | RS_TOOLONG | RS_TOOSHORT)
651*4882a593Smuzhiyun 
652*4882a593Smuzhiyun 
653*4882a593Smuzhiyun /* PHY Types */
654*4882a593Smuzhiyun enum {
655*4882a593Smuzhiyun 	PHY_LAN83C183 = 1,	/* LAN91C111 Internal PHY */
656*4882a593Smuzhiyun 	PHY_LAN83C180
657*4882a593Smuzhiyun };
658*4882a593Smuzhiyun 
659*4882a593Smuzhiyun 
660*4882a593Smuzhiyun /* PHY Register Addresses (LAN91C111 Internal PHY) */
661*4882a593Smuzhiyun 
662*4882a593Smuzhiyun /* PHY Control Register */
663*4882a593Smuzhiyun #define PHY_CNTL_REG		0x00
664*4882a593Smuzhiyun #define PHY_CNTL_RST		0x8000	/* 1=PHY Reset */
665*4882a593Smuzhiyun #define PHY_CNTL_LPBK		0x4000	/* 1=PHY Loopback */
666*4882a593Smuzhiyun #define PHY_CNTL_SPEED		0x2000	/* 1=100Mbps, 0=10Mpbs */
667*4882a593Smuzhiyun #define PHY_CNTL_ANEG_EN	0x1000 /* 1=Enable Auto negotiation */
668*4882a593Smuzhiyun #define PHY_CNTL_PDN		0x0800	/* 1=PHY Power Down mode */
669*4882a593Smuzhiyun #define PHY_CNTL_MII_DIS	0x0400	/* 1=MII 4 bit interface disabled */
670*4882a593Smuzhiyun #define PHY_CNTL_ANEG_RST	0x0200 /* 1=Reset Auto negotiate */
671*4882a593Smuzhiyun #define PHY_CNTL_DPLX		0x0100	/* 1=Full Duplex, 0=Half Duplex */
672*4882a593Smuzhiyun #define PHY_CNTL_COLTST		0x0080	/* 1= MII Colision Test */
673*4882a593Smuzhiyun 
674*4882a593Smuzhiyun /* PHY Status Register */
675*4882a593Smuzhiyun #define PHY_STAT_REG		0x01
676*4882a593Smuzhiyun #define PHY_STAT_CAP_T4		0x8000	/* 1=100Base-T4 capable */
677*4882a593Smuzhiyun #define PHY_STAT_CAP_TXF	0x4000	/* 1=100Base-X full duplex capable */
678*4882a593Smuzhiyun #define PHY_STAT_CAP_TXH	0x2000	/* 1=100Base-X half duplex capable */
679*4882a593Smuzhiyun #define PHY_STAT_CAP_TF		0x1000	/* 1=10Mbps full duplex capable */
680*4882a593Smuzhiyun #define PHY_STAT_CAP_TH		0x0800	/* 1=10Mbps half duplex capable */
681*4882a593Smuzhiyun #define PHY_STAT_CAP_SUPR	0x0040	/* 1=recv mgmt frames with not preamble */
682*4882a593Smuzhiyun #define PHY_STAT_ANEG_ACK	0x0020	/* 1=ANEG has completed */
683*4882a593Smuzhiyun #define PHY_STAT_REM_FLT	0x0010	/* 1=Remote Fault detected */
684*4882a593Smuzhiyun #define PHY_STAT_CAP_ANEG	0x0008	/* 1=Auto negotiate capable */
685*4882a593Smuzhiyun #define PHY_STAT_LINK		0x0004	/* 1=valid link */
686*4882a593Smuzhiyun #define PHY_STAT_JAB		0x0002	/* 1=10Mbps jabber condition */
687*4882a593Smuzhiyun #define PHY_STAT_EXREG		0x0001	/* 1=extended registers implemented */
688*4882a593Smuzhiyun 
689*4882a593Smuzhiyun /* PHY Identifier Registers */
690*4882a593Smuzhiyun #define PHY_ID1_REG		0x02	/* PHY Identifier 1 */
691*4882a593Smuzhiyun #define PHY_ID2_REG		0x03	/* PHY Identifier 2 */
692*4882a593Smuzhiyun 
693*4882a593Smuzhiyun /* PHY Auto-Negotiation Advertisement Register */
694*4882a593Smuzhiyun #define PHY_AD_REG		0x04
695*4882a593Smuzhiyun #define PHY_AD_NP		0x8000	/* 1=PHY requests exchange of Next Page */
696*4882a593Smuzhiyun #define PHY_AD_ACK		0x4000	/* 1=got link code word from remote */
697*4882a593Smuzhiyun #define PHY_AD_RF		0x2000	/* 1=advertise remote fault */
698*4882a593Smuzhiyun #define PHY_AD_T4		0x0200	/* 1=PHY is capable of 100Base-T4 */
699*4882a593Smuzhiyun #define PHY_AD_TX_FDX		0x0100	/* 1=PHY is capable of 100Base-TX FDPLX */
700*4882a593Smuzhiyun #define PHY_AD_TX_HDX		0x0080	/* 1=PHY is capable of 100Base-TX HDPLX */
701*4882a593Smuzhiyun #define PHY_AD_10_FDX		0x0040	/* 1=PHY is capable of 10Base-T FDPLX */
702*4882a593Smuzhiyun #define PHY_AD_10_HDX		0x0020	/* 1=PHY is capable of 10Base-T HDPLX */
703*4882a593Smuzhiyun #define PHY_AD_CSMA		0x0001	/* 1=PHY is capable of 802.3 CMSA */
704*4882a593Smuzhiyun 
705*4882a593Smuzhiyun /* PHY Auto-negotiation Remote End Capability Register */
706*4882a593Smuzhiyun #define PHY_RMT_REG		0x05
707*4882a593Smuzhiyun /* Uses same bit definitions as PHY_AD_REG */
708*4882a593Smuzhiyun 
709*4882a593Smuzhiyun /* PHY Configuration Register 1 */
710*4882a593Smuzhiyun #define PHY_CFG1_REG		0x10
711*4882a593Smuzhiyun #define PHY_CFG1_LNKDIS		0x8000	/* 1=Rx Link Detect Function disabled */
712*4882a593Smuzhiyun #define PHY_CFG1_XMTDIS		0x4000	/* 1=TP Transmitter Disabled */
713*4882a593Smuzhiyun #define PHY_CFG1_XMTPDN		0x2000	/* 1=TP Transmitter Powered Down */
714*4882a593Smuzhiyun #define PHY_CFG1_BYPSCR		0x0400	/* 1=Bypass scrambler/descrambler */
715*4882a593Smuzhiyun #define PHY_CFG1_UNSCDS		0x0200	/* 1=Unscramble Idle Reception Disable */
716*4882a593Smuzhiyun #define PHY_CFG1_EQLZR		0x0100	/* 1=Rx Equalizer Disabled */
717*4882a593Smuzhiyun #define PHY_CFG1_CABLE		0x0080	/* 1=STP(150ohm), 0=UTP(100ohm) */
718*4882a593Smuzhiyun #define PHY_CFG1_RLVL0		0x0040	/* 1=Rx Squelch level reduced by 4.5db */
719*4882a593Smuzhiyun #define PHY_CFG1_TLVL_SHIFT	2	/* Transmit Output Level Adjust */
720*4882a593Smuzhiyun #define PHY_CFG1_TLVL_MASK	0x003C
721*4882a593Smuzhiyun #define PHY_CFG1_TRF_MASK	0x0003	/* Transmitter Rise/Fall time */
722*4882a593Smuzhiyun 
723*4882a593Smuzhiyun 
724*4882a593Smuzhiyun /* PHY Configuration Register 2 */
725*4882a593Smuzhiyun #define PHY_CFG2_REG		0x11
726*4882a593Smuzhiyun #define PHY_CFG2_APOLDIS	0x0020	/* 1=Auto Polarity Correction disabled */
727*4882a593Smuzhiyun #define PHY_CFG2_JABDIS		0x0010	/* 1=Jabber disabled */
728*4882a593Smuzhiyun #define PHY_CFG2_MREG		0x0008	/* 1=Multiple register access (MII mgt) */
729*4882a593Smuzhiyun #define PHY_CFG2_INTMDIO	0x0004	/* 1=Interrupt signaled with MDIO pulseo */
730*4882a593Smuzhiyun 
731*4882a593Smuzhiyun /* PHY Status Output (and Interrupt status) Register */
732*4882a593Smuzhiyun #define PHY_INT_REG		0x12	/* Status Output (Interrupt Status) */
733*4882a593Smuzhiyun #define PHY_INT_INT		0x8000	/* 1=bits have changed since last read */
734*4882a593Smuzhiyun #define	PHY_INT_LNKFAIL		0x4000	/* 1=Link Not detected */
735*4882a593Smuzhiyun #define PHY_INT_LOSSSYNC	0x2000	/* 1=Descrambler has lost sync */
736*4882a593Smuzhiyun #define PHY_INT_CWRD		0x1000	/* 1=Invalid 4B5B code detected on rx */
737*4882a593Smuzhiyun #define PHY_INT_SSD		0x0800	/* 1=No Start Of Stream detected on rx */
738*4882a593Smuzhiyun #define PHY_INT_ESD		0x0400	/* 1=No End Of Stream detected on rx */
739*4882a593Smuzhiyun #define PHY_INT_RPOL		0x0200	/* 1=Reverse Polarity detected */
740*4882a593Smuzhiyun #define PHY_INT_JAB		0x0100	/* 1=Jabber detected */
741*4882a593Smuzhiyun #define PHY_INT_SPDDET		0x0080	/* 1=100Base-TX mode, 0=10Base-T mode */
742*4882a593Smuzhiyun #define PHY_INT_DPLXDET		0x0040	/* 1=Device in Full Duplex */
743*4882a593Smuzhiyun 
744*4882a593Smuzhiyun /* PHY Interrupt/Status Mask Register */
745*4882a593Smuzhiyun #define PHY_MASK_REG		0x13	/* Interrupt Mask */
746*4882a593Smuzhiyun /* Uses the same bit definitions as PHY_INT_REG */
747*4882a593Smuzhiyun 
748*4882a593Smuzhiyun 
749*4882a593Smuzhiyun /*-------------------------------------------------------------------------
750*4882a593Smuzhiyun  .  I define some macros to make it easier to do somewhat common
751*4882a593Smuzhiyun  . or slightly complicated, repeated tasks.
752*4882a593Smuzhiyun  --------------------------------------------------------------------------*/
753*4882a593Smuzhiyun 
754*4882a593Smuzhiyun /* select a register bank, 0 to 3  */
755*4882a593Smuzhiyun 
756*4882a593Smuzhiyun #define SMC_SELECT_BANK(a,x)  { SMC_outw((a), (x), BANK_SELECT ); }
757*4882a593Smuzhiyun 
758*4882a593Smuzhiyun /* this enables an interrupt in the interrupt mask register */
759*4882a593Smuzhiyun #define SMC_ENABLE_INT(a,x) {\
760*4882a593Smuzhiyun 		unsigned char mask;\
761*4882a593Smuzhiyun 		SMC_SELECT_BANK((a),2);\
762*4882a593Smuzhiyun 		mask = SMC_inb((a), IM_REG );\
763*4882a593Smuzhiyun 		mask |= (x);\
764*4882a593Smuzhiyun 		SMC_outb( (a), mask, IM_REG ); \
765*4882a593Smuzhiyun }
766*4882a593Smuzhiyun 
767*4882a593Smuzhiyun /* this disables an interrupt from the interrupt mask register */
768*4882a593Smuzhiyun 
769*4882a593Smuzhiyun #define SMC_DISABLE_INT(a,x) {\
770*4882a593Smuzhiyun 		unsigned char mask;\
771*4882a593Smuzhiyun 		SMC_SELECT_BANK(2);\
772*4882a593Smuzhiyun 		mask = SMC_inb( (a), IM_REG );\
773*4882a593Smuzhiyun 		mask &= ~(x);\
774*4882a593Smuzhiyun 		SMC_outb( (a), mask, IM_REG ); \
775*4882a593Smuzhiyun }
776*4882a593Smuzhiyun 
777*4882a593Smuzhiyun /*----------------------------------------------------------------------
778*4882a593Smuzhiyun  . Define the interrupts that I want to receive from the card
779*4882a593Smuzhiyun  .
780*4882a593Smuzhiyun  . I want:
781*4882a593Smuzhiyun  .  IM_EPH_INT, for nasty errors
782*4882a593Smuzhiyun  .  IM_RCV_INT, for happy received packets
783*4882a593Smuzhiyun  .  IM_RX_OVRN_INT, because I have to kick the receiver
784*4882a593Smuzhiyun  .  IM_MDINT, for PHY Register 18 Status Changes
785*4882a593Smuzhiyun  --------------------------------------------------------------------------*/
786*4882a593Smuzhiyun #define SMC_INTERRUPT_MASK   (IM_EPH_INT | IM_RX_OVRN_INT | IM_RCV_INT | \
787*4882a593Smuzhiyun 	IM_MDINT)
788*4882a593Smuzhiyun 
789*4882a593Smuzhiyun #endif  /* _SMC_91111_H_ */
790