• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**************************************************************************
2 ETHERBOOT -  BOOTP/TFTP Bootstrap Program
3 
4 Author: Martin Renters
5   Date: May/94
6 
7  This code is based heavily on David Greenman's if_ed.c driver
8 
9  Copyright (C) 1993-1994, David Greenman, Martin Renters.
10   This software may be used, modified, copied, distributed, and sold, in
11   both source and binary form provided that the above copyright and these
12   terms are retained. Under no circumstances are the authors responsible for
13   the proper functioning of this software, nor do the authors assume any
14   responsibility for damages incurred with its use.
15 
16 3c503 support added by Bill Paul (wpaul@ctr.columbia.edu) on 11/15/94
17 SMC8416 support added by Bill Paul (wpaul@ctr.columbia.edu) on 12/25/94
18 3c503 PIO support added by Jim Hague (jim.hague@acm.org) on 2/17/98
19 RX overrun by Klaus Espenlaub (espenlaub@informatik.uni-ulm.de) on 3/10/99
20   parts taken from the Linux 8390 driver (by Donald Becker and Paul Gortmaker)
21 
22 **************************************************************************/
23 
24 #include "etherboot.h"
25 #include "nic.h"
26 #include "ns8390.h"
27 #ifdef	INCLUDE_NS8390
28 #include "pci.h"
29 #endif
30 #include "cards.h"
31 
32 static unsigned char	eth_vendor, eth_flags, eth_laar;
33 static unsigned short	eth_nic_base, eth_asic_base;
34 static unsigned char	eth_memsize, eth_rx_start, eth_tx_start;
35 static Address		eth_bmem, eth_rmem;
36 static unsigned char	eth_drain_receiver;
37 
38 #ifdef	INCLUDE_WD
39 static struct wd_board {
40 	const char *name;
41 	char id;
42 	char flags;
43 	char memsize;
44 } wd_boards[] = {
45 	{"WD8003S",	TYPE_WD8003S,	0,			MEM_8192},
46 	{"WD8003E",	TYPE_WD8003E,	0,			MEM_8192},
47 	{"WD8013EBT",	TYPE_WD8013EBT,	FLAG_16BIT,		MEM_16384},
48 	{"WD8003W",	TYPE_WD8003W,	0,			MEM_8192},
49 	{"WD8003EB",	TYPE_WD8003EB,	0,			MEM_8192},
50 	{"WD8013W",	TYPE_WD8013W,	FLAG_16BIT,		MEM_16384},
51 	{"WD8003EP/WD8013EP",
52 			TYPE_WD8013EP,	0,			MEM_8192},
53 	{"WD8013WC",	TYPE_WD8013WC,	FLAG_16BIT,		MEM_16384},
54 	{"WD8013EPC",	TYPE_WD8013EPC,	FLAG_16BIT,		MEM_16384},
55 	{"SMC8216T",	TYPE_SMC8216T,	FLAG_16BIT | FLAG_790,	MEM_16384},
56 	{"SMC8216C",	TYPE_SMC8216C,	FLAG_16BIT | FLAG_790,	MEM_16384},
57 	{"SMC8416T",	TYPE_SMC8416T,	FLAG_16BIT | FLAG_790,	MEM_8192},
58 	{"SMC8416C/BT",	TYPE_SMC8416C,	FLAG_16BIT | FLAG_790,	MEM_8192},
59 	{"SMC8013EBP",	TYPE_SMC8013EBP,FLAG_16BIT,		MEM_16384},
60 	{NULL,		0,		0,			0}
61 };
62 #endif
63 
64 #ifdef	INCLUDE_3C503
65 static unsigned char	t503_output;	/* AUI or internal xcvr (Thinnet) */
66 #endif
67 
68 #if	defined(INCLUDE_WD)
69 #define	eth_probe	wd_probe
70 #if	defined(INCLUDE_3C503) || defined(INCLUDE_NE) || defined(INCLUDE_NS8390)
71 Error you must only define one of INCLUDE_WD, INCLUDE_3C503, INCLUDE_NE, INCLUDE_NS8390
72 #endif
73 #endif
74 
75 #if	defined(INCLUDE_3C503)
76 #define	eth_probe	t503_probe
77 #if	defined(INCLUDE_NE) || defined(INCLUDE_NS8390) || defined(INCLUDE_WD)
78 Error you must only define one of INCLUDE_WD, INCLUDE_3C503, INCLUDE_NE, INCLUDE_NS8390
79 #endif
80 #endif
81 
82 #if	defined(INCLUDE_NE)
83 #define	eth_probe	ne_probe
84 #if	defined(INCLUDE_NS8390) || defined(INCLUDE_3C503) || defined(INCLUDE_WD)
85 Error you must only define one of INCLUDE_WD, INCLUDE_3C503, INCLUDE_NE, INCLUDE_NS8390
86 #endif
87 #endif
88 
89 #if	defined(INCLUDE_NS8390)
90 #define	eth_probe	nepci_probe
91 #if	defined(INCLUDE_NE) || defined(INCLUDE_3C503) || defined(INCLUDE_WD)
92 Error you must only define one of INCLUDE_WD, INCLUDE_3C503, INCLUDE_NE, INCLUDE_NS8390
93 #endif
94 #endif
95 
96 #if	defined(INCLUDE_3C503)
97 #define	ASIC_PIO	_3COM_RFMSB
98 #else
99 #if	defined(INCLUDE_NE) || defined(INCLUDE_NS8390)
100 #define	ASIC_PIO	NE_DATA
101 #endif
102 #endif
103 
104 #if	defined(INCLUDE_NE) || defined(INCLUDE_NS8390) || (defined(INCLUDE_3C503) && !defined(T503_SHMEM))
105 /**************************************************************************
106 ETH_PIO_READ - Read a frame via Programmed I/O
107 **************************************************************************/
108 static void eth_pio_read(unsigned int src, unsigned char *dst, unsigned int cnt)
109 {
110 	if (eth_flags & FLAG_16BIT) { ++cnt; cnt &= ~1; }
111 	outb(D8390_COMMAND_RD2 |
112 		D8390_COMMAND_STA, eth_nic_base + D8390_P0_COMMAND);
113 	outb(cnt, eth_nic_base + D8390_P0_RBCR0);
114 	outb(cnt>>8, eth_nic_base + D8390_P0_RBCR1);
115 	outb(src, eth_nic_base + D8390_P0_RSAR0);
116 	outb(src>>8, eth_nic_base + D8390_P0_RSAR1);
117 	outb(D8390_COMMAND_RD0 |
118 		D8390_COMMAND_STA, eth_nic_base + D8390_P0_COMMAND);
119 
120 #ifdef	INCLUDE_3C503
121 	outb(src & 0xff, eth_asic_base + _3COM_DALSB);
122 	outb(src >> 8, eth_asic_base + _3COM_DAMSB);
123 	outb(t503_output | _3COM_CR_START, eth_asic_base + _3COM_CR);
124 #endif
125 
126 	if (eth_flags & FLAG_16BIT)
127 		cnt >>= 1;
128 
129 	while(cnt--) {
130 #ifdef	INCLUDE_3C503
131 		while((inb(eth_asic_base + _3COM_STREG) & _3COM_STREG_DPRDY) == 0)
132 			;
133 #endif
134 
135 		if (eth_flags & FLAG_16BIT) {
136 			*((unsigned short *)dst) = inw(eth_asic_base + ASIC_PIO);
137 			dst += 2;
138 		}
139 		else
140 			*(dst++) = inb(eth_asic_base + ASIC_PIO);
141 	}
142 
143 #ifdef	INCLUDE_3C503
144 	outb(t503_output, eth_asic_base + _3COM_CR);
145 #endif
146 }
147 
148 /**************************************************************************
149 ETH_PIO_WRITE - Write a frame via Programmed I/O
150 **************************************************************************/
eth_pio_write(const unsigned char * src,unsigned int dst,unsigned int cnt)151 static void eth_pio_write(const unsigned char *src, unsigned int dst, unsigned int cnt)
152 {
153 #ifdef	COMPEX_RL2000_FIX
154 	unsigned int x;
155 #endif	/* COMPEX_RL2000_FIX */
156 	if (eth_flags & FLAG_16BIT) { ++cnt; cnt &= ~1; }
157 	outb(D8390_COMMAND_RD2 |
158 		D8390_COMMAND_STA, eth_nic_base + D8390_P0_COMMAND);
159 	outb(D8390_ISR_RDC, eth_nic_base + D8390_P0_ISR);
160 	outb(cnt, eth_nic_base + D8390_P0_RBCR0);
161 	outb(cnt>>8, eth_nic_base + D8390_P0_RBCR1);
162 	outb(dst, eth_nic_base + D8390_P0_RSAR0);
163 	outb(dst>>8, eth_nic_base + D8390_P0_RSAR1);
164 	outb(D8390_COMMAND_RD1 |
165 		D8390_COMMAND_STA, eth_nic_base + D8390_P0_COMMAND);
166 
167 #ifdef	INCLUDE_3C503
168 	outb(dst & 0xff, eth_asic_base + _3COM_DALSB);
169 	outb(dst >> 8, eth_asic_base + _3COM_DAMSB);
170 
171 	outb(t503_output | _3COM_CR_DDIR | _3COM_CR_START, eth_asic_base + _3COM_CR);
172 #endif
173 
174 	if (eth_flags & FLAG_16BIT)
175 		cnt >>= 1;
176 
177 	while(cnt--)
178 	{
179 #ifdef	INCLUDE_3C503
180 		while((inb(eth_asic_base + _3COM_STREG) & _3COM_STREG_DPRDY) == 0)
181 			;
182 #endif
183 
184 		if (eth_flags & FLAG_16BIT) {
185 			outw(*((unsigned short *)src), eth_asic_base + ASIC_PIO);
186 			src += 2;
187 		}
188 		else
189 			outb(*(src++), eth_asic_base + ASIC_PIO);
190 	}
191 
192 #ifdef	INCLUDE_3C503
193 	outb(t503_output, eth_asic_base + _3COM_CR);
194 #else
195 #ifdef	COMPEX_RL2000_FIX
196 	for (x = 0;
197 		x < COMPEX_RL2000_TRIES &&
198 		(inb(eth_nic_base + D8390_P0_ISR) & D8390_ISR_RDC)
199 		!= D8390_ISR_RDC;
200 		++x);
201 	if (x >= COMPEX_RL2000_TRIES)
202 		printf("Warning: Compex RL2000 aborted wait!\n");
203 #endif	/* COMPEX_RL2000_FIX */
204 	while((inb(eth_nic_base + D8390_P0_ISR) & D8390_ISR_RDC)
205 		!= D8390_ISR_RDC);
206 #endif
207 }
208 #else
209 /**************************************************************************
210 ETH_PIO_READ - Dummy routine when NE2000 not compiled in
211 **************************************************************************/
212 static void eth_pio_read(unsigned int src, unsigned char *dst, unsigned int cnt) {}
213 #endif
214 
215 /**************************************************************************
216 NS8390_RESET - Reset adapter
217 **************************************************************************/
ns8390_reset(struct nic * nic)218 static void ns8390_reset(struct nic *nic)
219 {
220 	int i;
221 
222 	eth_drain_receiver = 0;
223 #ifdef	INCLUDE_WD
224 	if (eth_flags & FLAG_790)
225 		outb(D8390_COMMAND_PS0 | D8390_COMMAND_STP, eth_nic_base+D8390_P0_COMMAND);
226 	else
227 #endif
228 		outb(D8390_COMMAND_PS0 | D8390_COMMAND_RD2 |
229 			D8390_COMMAND_STP, eth_nic_base+D8390_P0_COMMAND);
230 	if (eth_flags & FLAG_16BIT)
231 		outb(0x49, eth_nic_base+D8390_P0_DCR);
232 	else
233 		outb(0x48, eth_nic_base+D8390_P0_DCR);
234 	outb(0, eth_nic_base+D8390_P0_RBCR0);
235 	outb(0, eth_nic_base+D8390_P0_RBCR1);
236 	outb(0x20, eth_nic_base+D8390_P0_RCR);	/* monitor mode */
237 	outb(2, eth_nic_base+D8390_P0_TCR);
238 	outb(eth_tx_start, eth_nic_base+D8390_P0_TPSR);
239 	outb(eth_rx_start, eth_nic_base+D8390_P0_PSTART);
240 #ifdef	INCLUDE_WD
241 	if (eth_flags & FLAG_790) outb(0, eth_nic_base + 0x09);
242 #endif
243 	outb(eth_memsize, eth_nic_base+D8390_P0_PSTOP);
244 	outb(eth_memsize - 1, eth_nic_base+D8390_P0_BOUND);
245 	outb(0xFF, eth_nic_base+D8390_P0_ISR);
246 	outb(0, eth_nic_base+D8390_P0_IMR);
247 #ifdef	INCLUDE_WD
248 	if (eth_flags & FLAG_790)
249 		outb(D8390_COMMAND_PS1 |
250 			D8390_COMMAND_STP, eth_nic_base+D8390_P0_COMMAND);
251 	else
252 #endif
253 		outb(D8390_COMMAND_PS1 |
254 			D8390_COMMAND_RD2 | D8390_COMMAND_STP, eth_nic_base+D8390_P0_COMMAND);
255 	for (i=0; i<ETH_ALEN; i++)
256 		outb(nic->node_addr[i], eth_nic_base+D8390_P1_PAR0+i);
257 	for (i=0; i<ETH_ALEN; i++)
258 		outb(0xFF, eth_nic_base+D8390_P1_MAR0+i);
259 	outb(eth_rx_start, eth_nic_base+D8390_P1_CURR);
260 #ifdef	INCLUDE_WD
261 	if (eth_flags & FLAG_790)
262 		outb(D8390_COMMAND_PS0 |
263 			D8390_COMMAND_STA, eth_nic_base+D8390_P0_COMMAND);
264 	else
265 #endif
266 		outb(D8390_COMMAND_PS0 |
267 			D8390_COMMAND_RD2 | D8390_COMMAND_STA, eth_nic_base+D8390_P0_COMMAND);
268 	outb(0xFF, eth_nic_base+D8390_P0_ISR);
269 	outb(0, eth_nic_base+D8390_P0_TCR);
270 	outb(4, eth_nic_base+D8390_P0_RCR);	/* allow broadcast frames */
271 
272 #ifdef	INCLUDE_3C503
273         /*
274          * No way to tell whether or not we're supposed to use
275          * the 3Com's transceiver unless the user tells us.
276          * 'flags' should have some compile time default value
277          * which can be changed from the command menu.
278          */
279 	t503_output = (nic->flags) ? 0 : _3COM_CR_XSEL;
280 	outb(t503_output, eth_asic_base + _3COM_CR);
281 #endif
282 }
283 
284 static int ns8390_poll(struct nic *nic);
285 
286 #ifndef	INCLUDE_3C503
287 /**************************************************************************
288 ETH_RX_OVERRUN - Bring adapter back to work after an RX overrun
289 **************************************************************************/
eth_rx_overrun(struct nic * nic)290 static void eth_rx_overrun(struct nic *nic)
291 {
292 	int start_time;
293 
294 #ifdef	INCLUDE_WD
295 	if (eth_flags & FLAG_790)
296 		outb(D8390_COMMAND_PS0 | D8390_COMMAND_STP, eth_nic_base+D8390_P0_COMMAND);
297 	else
298 #endif
299 		outb(D8390_COMMAND_PS0 | D8390_COMMAND_RD2 |
300 			D8390_COMMAND_STP, eth_nic_base+D8390_P0_COMMAND);
301 
302 	/* wait for at least 1.6ms - we wait one timer tick */
303 	start_time = currticks();
304 	while (currticks() - start_time <= 1)
305 		/* Nothing */;
306 
307 	outb(0, eth_nic_base+D8390_P0_RBCR0);	/* reset byte counter */
308 	outb(0, eth_nic_base+D8390_P0_RBCR1);
309 
310 	/*
311 	 * Linux driver checks for interrupted TX here. This is not necessary,
312 	 * because the transmit routine waits until the frame is sent.
313 	 */
314 
315 	/* enter loopback mode and restart NIC */
316 	outb(2, eth_nic_base+D8390_P0_TCR);
317 #ifdef	INCLUDE_WD
318 	if (eth_flags & FLAG_790)
319 		outb(D8390_COMMAND_PS0 | D8390_COMMAND_STA, eth_nic_base+D8390_P0_COMMAND);
320 	else
321 #endif
322 		outb(D8390_COMMAND_PS0 | D8390_COMMAND_RD2 |
323 			D8390_COMMAND_STA, eth_nic_base+D8390_P0_COMMAND);
324 
325 	/* clear the RX ring, acknowledge overrun interrupt */
326 	eth_drain_receiver = 1;
327 	while (ns8390_poll(nic))
328 		/* Nothing */;
329 	eth_drain_receiver = 0;
330 	outb(D8390_ISR_OVW, eth_nic_base+D8390_P0_ISR);
331 
332 	/* leave loopback mode - no packets to be resent (see Linux driver) */
333 	outb(0, eth_nic_base+D8390_P0_TCR);
334 }
335 #endif	/* INCLUDE_3C503 */
336 
337 /**************************************************************************
338 NS8390_TRANSMIT - Transmit a frame
339 **************************************************************************/
ns8390_transmit(struct nic * nic,const char * d,unsigned int t,unsigned int s,const char * p)340 static void ns8390_transmit(
341 	struct nic *nic,
342 	const char *d,			/* Destination */
343 	unsigned int t,			/* Type */
344 	unsigned int s,			/* size */
345 	const char *p)			/* Packet */
346 {
347 #ifdef	INCLUDE_3C503
348         if (!(eth_flags & FLAG_PIO)) {
349                 memcpy((char *)eth_bmem, d, ETH_ALEN);	/* dst */
350                 memcpy((char *)eth_bmem+ETH_ALEN, nic->node_addr, ETH_ALEN); /* src */
351                 *((char *)eth_bmem+12) = t>>8;		/* type */
352                 *((char *)eth_bmem+13) = t;
353                 memcpy((char *)eth_bmem+ETH_HLEN, p, s);
354                 s += ETH_HLEN;
355                 while (s < ETH_ZLEN) *((char *)eth_bmem+(s++)) = 0;
356         }
357 #endif
358 
359 #ifdef	INCLUDE_WD
360 	/* Memory interface */
361 	if (eth_flags & FLAG_16BIT) {
362 		outb(eth_laar | WD_LAAR_M16EN, eth_asic_base + WD_LAAR);
363 		inb(0x84);
364 	}
365 	if (eth_flags & FLAG_790) {
366 		outb(WD_MSR_MENB, eth_asic_base + WD_MSR);
367 		inb(0x84);
368 	}
369 	inb(0x84);
370 	memcpy((char *)eth_bmem, d, ETH_ALEN);	/* dst */
371 	memcpy((char *)eth_bmem+ETH_ALEN, nic->node_addr, ETH_ALEN); /* src */
372 	*((char *)eth_bmem+12) = t>>8;		/* type */
373 	*((char *)eth_bmem+13) = t;
374 	memcpy((char *)eth_bmem+ETH_HLEN, p, s);
375 	s += ETH_HLEN;
376 	while (s < ETH_ZLEN) *((char *)eth_bmem+(s++)) = 0;
377 	if (eth_flags & FLAG_790) {
378 		outb(0, eth_asic_base + WD_MSR);
379 		inb(0x84);
380 	}
381 	if (eth_flags & FLAG_16BIT) {
382 		outb(eth_laar & ~WD_LAAR_M16EN, eth_asic_base + WD_LAAR);
383 		inb(0x84);
384 	}
385 #endif
386 
387 #if	defined(INCLUDE_3C503)
388 	if (eth_flags & FLAG_PIO) {
389 #endif
390 #if	defined(INCLUDE_NE) || defined(INCLUDE_NS8390) || (defined(INCLUDE_3C503) && !defined(T503_SHMEM))
391 		/* Programmed I/O */
392 		unsigned short type;
393 		type = (t >> 8) | (t << 8);
394 		eth_pio_write(d, eth_tx_start<<8, ETH_ALEN);
395 		eth_pio_write(nic->node_addr, (eth_tx_start<<8)+ETH_ALEN, ETH_ALEN);
396 		/* bcc generates worse code without (const+const) below */
397 		eth_pio_write((unsigned char *)&type, (eth_tx_start<<8)+(ETH_ALEN+ETH_ALEN), 2);
398 		eth_pio_write(p, (eth_tx_start<<8)+ETH_HLEN, s);
399 		s += ETH_HLEN;
400 		if (s < ETH_ZLEN) s = ETH_ZLEN;
401 #endif
402 #if	defined(INCLUDE_3C503)
403 	}
404 #endif
405 
406 #ifdef	INCLUDE_WD
407 	if (eth_flags & FLAG_790)
408 		outb(D8390_COMMAND_PS0 |
409 			D8390_COMMAND_STA, eth_nic_base+D8390_P0_COMMAND);
410 	else
411 #endif
412 		outb(D8390_COMMAND_PS0 |
413 			D8390_COMMAND_RD2 | D8390_COMMAND_STA, eth_nic_base+D8390_P0_COMMAND);
414 	outb(eth_tx_start, eth_nic_base+D8390_P0_TPSR);
415 	outb(s, eth_nic_base+D8390_P0_TBCR0);
416 	outb(s>>8, eth_nic_base+D8390_P0_TBCR1);
417 #ifdef	INCLUDE_WD
418 	if (eth_flags & FLAG_790)
419 		outb(D8390_COMMAND_PS0 |
420 			D8390_COMMAND_TXP | D8390_COMMAND_STA, eth_nic_base+D8390_P0_COMMAND);
421 	else
422 #endif
423 		outb(D8390_COMMAND_PS0 |
424 			D8390_COMMAND_TXP | D8390_COMMAND_RD2 |
425 			D8390_COMMAND_STA, eth_nic_base+D8390_P0_COMMAND);
426 }
427 
428 /**************************************************************************
429 NS8390_POLL - Wait for a frame
430 **************************************************************************/
ns8390_poll(struct nic * nic)431 static int ns8390_poll(struct nic *nic)
432 {
433 	int ret = 0;
434 	unsigned char rstat, curr, next;
435 	unsigned short len, frag;
436 	unsigned short pktoff;
437 	unsigned char *p;
438 	struct ringbuffer pkthdr;
439 
440 #ifndef	INCLUDE_3C503
441 	/* avoid infinite recursion: see eth_rx_overrun() */
442 	if (!eth_drain_receiver && (inb(eth_nic_base+D8390_P0_ISR) & D8390_ISR_OVW)) {
443 		eth_rx_overrun(nic);
444 		return(0);
445 	}
446 #endif	/* INCLUDE_3C503 */
447 	rstat = inb(eth_nic_base+D8390_P0_RSR);
448 	if (!(rstat & D8390_RSTAT_PRX)) return(0);
449 	next = inb(eth_nic_base+D8390_P0_BOUND)+1;
450 	if (next >= eth_memsize) next = eth_rx_start;
451 	outb(D8390_COMMAND_PS1, eth_nic_base+D8390_P0_COMMAND);
452 	curr = inb(eth_nic_base+D8390_P1_CURR);
453 	outb(D8390_COMMAND_PS0, eth_nic_base+D8390_P0_COMMAND);
454 	if (curr >= eth_memsize) curr=eth_rx_start;
455 	if (curr == next) return(0);
456 #ifdef	INCLUDE_WD
457 	if (eth_flags & FLAG_16BIT) {
458 		outb(eth_laar | WD_LAAR_M16EN, eth_asic_base + WD_LAAR);
459 		inb(0x84);
460 	}
461 	if (eth_flags & FLAG_790) {
462 		outb(WD_MSR_MENB, eth_asic_base + WD_MSR);
463 		inb(0x84);
464 	}
465 	inb(0x84);
466 #endif
467 	pktoff = next << 8;
468 	if (eth_flags & FLAG_PIO)
469 		eth_pio_read(pktoff, (char *)&pkthdr, 4);
470 	else
471 		memcpy(&pkthdr, (char *)eth_rmem + pktoff, 4);
472 	pktoff += sizeof(pkthdr);
473 	/* incoming length includes FCS so must sub 4 */
474 	len = pkthdr.len - 4;
475 	if ((pkthdr.status & D8390_RSTAT_PRX) == 0 || len < ETH_ZLEN
476 		|| len > ETH_FRAME_LEN) {
477 		printf("Bogus packet, ignoring\n");
478 		return (0);
479 	}
480 	else {
481 		p = nic->packet;
482 		nic->packetlen = len;		/* available to caller */
483 		frag = (eth_memsize << 8) - pktoff;
484 		if (len > frag) {		/* We have a wrap-around */
485 			/* read first part */
486 			if (eth_flags & FLAG_PIO)
487 				eth_pio_read(pktoff, p, frag);
488 			else
489 				memcpy(p, (char *)eth_rmem + pktoff, frag);
490 			pktoff = eth_rx_start << 8;
491 			p += frag;
492 			len -= frag;
493 		}
494 		/* read second part */
495 		if (eth_flags & FLAG_PIO)
496 			eth_pio_read(pktoff, p, len);
497 		else
498 			memcpy(p, (char *)eth_rmem + pktoff, len);
499 		ret = 1;
500 	}
501 #ifdef	INCLUDE_WD
502 	if (eth_flags & FLAG_790) {
503 		outb(0, eth_asic_base + WD_MSR);
504 		inb(0x84);
505 	}
506 	if (eth_flags & FLAG_16BIT) {
507 		outb(eth_laar & ~WD_LAAR_M16EN, eth_asic_base + WD_LAAR);
508 		inb(0x84);
509 	}
510 	inb(0x84);
511 #endif
512 	next = pkthdr.next;		/* frame number of next packet */
513 	if (next == eth_rx_start)
514 		next = eth_memsize;
515 	outb(next-1, eth_nic_base+D8390_P0_BOUND);
516 	return(ret);
517 }
518 
519 /**************************************************************************
520 NS8390_DISABLE - Turn off adapter
521 **************************************************************************/
ns8390_disable(struct nic * nic)522 static void ns8390_disable(struct nic *nic)
523 {
524 }
525 
526 /**************************************************************************
527 ETH_PROBE - Look for an adapter
528 **************************************************************************/
529 #ifdef	INCLUDE_NS8390
eth_probe(struct nic * nic,unsigned short * probe_addrs,struct pci_device * pci)530 struct nic *eth_probe(struct nic *nic, unsigned short *probe_addrs,
531 		      struct pci_device *pci)
532 #else
533 struct nic *eth_probe(struct nic *nic, unsigned short *probe_addrs)
534 #endif
535 {
536 	int i;
537 	struct wd_board *brd;
538 	unsigned short chksum;
539 	unsigned char c;
540 	eth_vendor = VENDOR_NONE;
541 	eth_drain_receiver = 0;
542 
543 #ifdef	INCLUDE_WD
544 	/******************************************************************
545 	Search for WD/SMC cards
546 	******************************************************************/
547 	for (eth_asic_base = WD_LOW_BASE; eth_asic_base <= WD_HIGH_BASE;
548 		eth_asic_base += 0x20) {
549 		chksum = 0;
550 		for (i=8; i<16; i++)
551 			chksum += inb(eth_asic_base+i);
552 		/* Extra checks to avoid soundcard */
553 		if ((chksum & 0xFF) == 0xFF &&
554 			inb(eth_asic_base+8) != 0xFF &&
555 			inb(eth_asic_base+9) != 0xFF)
556 			break;
557 	}
558 	if (eth_asic_base > WD_HIGH_BASE)
559 		return (0);
560 	/* We've found a board */
561 	eth_vendor = VENDOR_WD;
562 	eth_nic_base = eth_asic_base + WD_NIC_ADDR;
563 	c = inb(eth_asic_base+WD_BID);	/* Get board id */
564 	for (brd = wd_boards; brd->name; brd++)
565 		if (brd->id == c) break;
566 	if (!brd->name) {
567 		printf("Unknown WD/SMC NIC type %hhX\n", c);
568 		return (0);	/* Unknown type */
569 	}
570 	eth_flags = brd->flags;
571 	eth_memsize = brd->memsize;
572 	eth_tx_start = 0;
573 	eth_rx_start = D8390_TXBUF_SIZE;
574 	if ((c == TYPE_WD8013EP) &&
575 		(inb(eth_asic_base + WD_ICR) & WD_ICR_16BIT)) {
576 			eth_flags = FLAG_16BIT;
577 			eth_memsize = MEM_16384;
578 	}
579 	if ((c & WD_SOFTCONFIG) && (!(eth_flags & FLAG_790))) {
580 		eth_bmem = (0x80000 |
581 		 ((inb(eth_asic_base + WD_MSR) & 0x3F) << 13));
582 	} else
583 		eth_bmem = WD_DEFAULT_MEM;
584 	if (brd->id == TYPE_SMC8216T || brd->id == TYPE_SMC8216C) {
585 		*((unsigned int *)(eth_bmem + 8192)) = (unsigned int)0;
586 		if (*((unsigned int *)(eth_bmem + 8192))) {
587 			brd += 2;
588 			eth_memsize = brd->memsize;
589 		}
590 	}
591 	outb(0x80, eth_asic_base + WD_MSR);	/* Reset */
592 	for (i=0; i<ETH_ALEN; i++) {
593 		nic->node_addr[i] = inb(i+eth_asic_base+WD_LAR);
594 	}
595 	printf("\n%s base %#hx, memory %#hx, addr %!\n",
596 		brd->name, eth_asic_base, eth_bmem, nic->node_addr);
597 	if (eth_flags & FLAG_790) {
598 		outb(WD_MSR_MENB, eth_asic_base+WD_MSR);
599 		outb((inb(eth_asic_base+0x04) |
600 			0x80), eth_asic_base+0x04);
601 		outb((((unsigned)eth_bmem >> 13) & 0x0F) |
602 			(((unsigned)eth_bmem >> 11) & 0x40) |
603 			(inb(eth_asic_base+0x0B) & 0xB0), eth_asic_base+0x0B);
604 		outb((inb(eth_asic_base+0x04) &
605 			~0x80), eth_asic_base+0x04);
606 	} else {
607 		outb((((unsigned)eth_bmem >> 13) & 0x3F) | 0x40, eth_asic_base+WD_MSR);
608 	}
609 	if (eth_flags & FLAG_16BIT) {
610 		if (eth_flags & FLAG_790) {
611 			eth_laar = inb(eth_asic_base + WD_LAAR);
612 			outb(WD_LAAR_M16EN, eth_asic_base + WD_LAAR);
613 		} else {
614 			outb((eth_laar =
615 				WD_LAAR_L16EN | 1), eth_asic_base + WD_LAAR);
616 /*
617 	The previous line used to be
618 				WD_LAAR_M16EN | WD_LAAR_L16EN | 1));
619 	jluke@deakin.edu.au reported that removing WD_LAAR_M16EN made
620 	it work for WD8013s.  This seems to work for my 8013 boards. I
621 	don't know what is really happening.  I wish I had data sheets
622 	or more time to decode the Linux driver. - Ken
623 */
624 		}
625 		inb(0x84);
626 	}
627 #endif
628 #ifdef	INCLUDE_3C503
629         /******************************************************************
630         Search for 3Com 3c503 if no WD/SMC cards
631         ******************************************************************/
632 	if (eth_vendor == VENDOR_NONE) {
633 		int	idx;
634 		int	iobase_reg, membase_reg;
635 		static unsigned short	base[] = {
636 			0x300, 0x310, 0x330, 0x350,
637 			0x250, 0x280, 0x2A0, 0x2E0, 0 };
638 
639 		/* Loop through possible addresses checking each one */
640 
641 		for (idx = 0; (eth_nic_base = base[idx]) != 0; ++idx) {
642 
643 			eth_asic_base = eth_nic_base + _3COM_ASIC_OFFSET;
644 /*
645  * Note that we use the same settings for both 8 and 16 bit cards:
646  * both have an 8K bank of memory at page 1 while only the 16 bit
647  * cards have a bank at page 0.
648  */
649 			eth_memsize = MEM_16384;
650 			eth_tx_start = 32;
651 			eth_rx_start = 32 + D8390_TXBUF_SIZE;
652 
653 		/* Check our base address. iobase and membase should */
654 		/* both have a maximum of 1 bit set or be 0. */
655 
656 			iobase_reg = inb(eth_asic_base + _3COM_BCFR);
657 			membase_reg = inb(eth_asic_base + _3COM_PCFR);
658 
659 			if ((iobase_reg & (iobase_reg - 1)) ||
660 				(membase_reg & (membase_reg - 1)))
661 				continue;		/* nope */
662 
663 		/* Now get the shared memory address */
664 
665 			eth_flags = 0;
666 
667 			switch (membase_reg) {
668 				case _3COM_PCFR_DC000:
669 					eth_bmem = 0xdc000;
670 					break;
671 				case _3COM_PCFR_D8000:
672 					eth_bmem = 0xd8000;
673 					break;
674 				case _3COM_PCFR_CC000:
675 					eth_bmem = 0xcc000;
676 					break;
677 				case _3COM_PCFR_C8000:
678 					eth_bmem = 0xc8000;
679 					break;
680 				case _3COM_PCFR_PIO:
681 					eth_flags |= FLAG_PIO;
682 					eth_bmem = 0;
683 					break;
684 				default:
685 					continue;	/* nope */
686 				}
687 			break;
688 		}
689 
690 		if (base[idx] == 0)		/* not found */
691 			return (0);
692 #ifndef	T503_SHMEM
693 		eth_flags |= FLAG_PIO;		/* force PIO mode */
694 		eth_bmem = 0;
695 #endif
696 		eth_vendor = VENDOR_3COM;
697 
698 
699         /* Need this to make ns8390_poll() happy. */
700 
701                 eth_rmem = eth_bmem - 0x2000;
702 
703         /* Reset NIC and ASIC */
704 
705                 outb(_3COM_CR_RST | _3COM_CR_XSEL, eth_asic_base + _3COM_CR );
706                 outb(_3COM_CR_XSEL, eth_asic_base + _3COM_CR );
707 
708         /* Get our ethernet address */
709 
710                 outb(_3COM_CR_EALO | _3COM_CR_XSEL, eth_asic_base + _3COM_CR);
711                 printf("\n3Com 3c503 base %#hx, ", eth_nic_base);
712                 if (eth_flags & FLAG_PIO)
713 			printf("PIO mode");
714                 else
715 			printf("memory %#hx", eth_bmem);
716                 for (i=0; i<ETH_ALEN; i++) {
717                         nic->node_addr[i] = inb(eth_nic_base+i);
718                 }
719                 printf(", %s, addr %!\n", nic->flags ? "AUI" : "internal xcvr",
720 			nic->node_addr);
721                 outb(_3COM_CR_XSEL, eth_asic_base + _3COM_CR);
722         /*
723          * Initialize GA configuration register. Set bank and enable shared
724          * mem. We always use bank 1. Disable interrupts.
725          */
726                 outb(_3COM_GACFR_RSEL |
727 			_3COM_GACFR_MBS0 | _3COM_GACFR_TCM | _3COM_GACFR_NIM, eth_asic_base + _3COM_GACFR);
728 
729                 outb(0xff, eth_asic_base + _3COM_VPTR2);
730                 outb(0xff, eth_asic_base + _3COM_VPTR1);
731                 outb(0x00, eth_asic_base + _3COM_VPTR0);
732         /*
733          * Clear memory and verify that it worked (we use only 8K)
734          */
735 
736 		if (!(eth_flags & FLAG_PIO)) {
737 			memset((char *)eth_bmem, 0, 0x2000);
738 			for(i = 0; i < 0x2000; ++i)
739 				if (*(((char *)eth_bmem)+i)) {
740 					printf ("Failed to clear 3c503 shared mem.\n");
741 					return (0);
742 				}
743 		}
744         /*
745          * Initialize GA page/start/stop registers.
746          */
747                 outb(eth_tx_start, eth_asic_base + _3COM_PSTR);
748                 outb(eth_memsize, eth_asic_base + _3COM_PSPR);
749         }
750 #endif
751 #if	defined(INCLUDE_NE) || defined(INCLUDE_NS8390)
752 	/******************************************************************
753 	Search for NE1000/2000 if no WD/SMC or 3com cards
754 	******************************************************************/
755 	if (eth_vendor == VENDOR_NONE) {
756 		char romdata[16], testbuf[32];
757 		int idx;
758 		static char test[] = "NE*000 memory";
759 		static unsigned short base[] = {
760 #ifdef	NE_SCAN
761 			NE_SCAN,
762 #endif
763 			0 };
764 		/* if no addresses supplied, fall back on defaults */
765 		if (probe_addrs == 0 || probe_addrs[0] == 0)
766 			probe_addrs = base;
767 		eth_bmem = 0;		/* No shared memory */
768 		for (idx = 0; (eth_nic_base = probe_addrs[idx]) != 0; ++idx) {
769 			eth_flags = FLAG_PIO;
770 			eth_asic_base = eth_nic_base + NE_ASIC_OFFSET;
771 			eth_memsize = MEM_16384;
772 			eth_tx_start = 32;
773 			eth_rx_start = 32 + D8390_TXBUF_SIZE;
774 			c = inb(eth_asic_base + NE_RESET);
775 			outb(c, eth_asic_base + NE_RESET);
776 			inb(0x84);
777 			outb(D8390_COMMAND_STP |
778 				D8390_COMMAND_RD2, eth_nic_base + D8390_P0_COMMAND);
779 			outb(D8390_RCR_MON, eth_nic_base + D8390_P0_RCR);
780 			outb(D8390_DCR_FT1 | D8390_DCR_LS, eth_nic_base + D8390_P0_DCR);
781 			outb(MEM_8192, eth_nic_base + D8390_P0_PSTART);
782 			outb(MEM_16384, eth_nic_base + D8390_P0_PSTOP);
783 #ifdef	NS8390_FORCE_16BIT
784 			eth_flags |= FLAG_16BIT;	/* force 16-bit mode */
785 #endif
786 
787 			eth_pio_write(test, 8192, sizeof(test));
788 			eth_pio_read(8192, testbuf, sizeof(test));
789 			if (!memcmp(test, testbuf, sizeof(test)))
790 				break;
791 			eth_flags |= FLAG_16BIT;
792 			eth_memsize = MEM_32768;
793 			eth_tx_start = 64;
794 			eth_rx_start = 64 + D8390_TXBUF_SIZE;
795 			outb(D8390_DCR_WTS |
796 				D8390_DCR_FT1 | D8390_DCR_LS, eth_nic_base + D8390_P0_DCR);
797 			outb(MEM_16384, eth_nic_base + D8390_P0_PSTART);
798 			outb(MEM_32768, eth_nic_base + D8390_P0_PSTOP);
799 			eth_pio_write(test, 16384, sizeof(test));
800 			eth_pio_read(16384, testbuf, sizeof(test));
801 			if (!memcmp(testbuf, test, sizeof(test)))
802 				break;
803 		}
804 		if (eth_nic_base == 0)
805 			return (0);
806 		if (eth_nic_base > ISA_MAX_ADDR)	/* PCI probably */
807 			eth_flags |= FLAG_16BIT;
808 		eth_vendor = VENDOR_NOVELL;
809 		eth_pio_read(0, romdata, sizeof(romdata));
810 		for (i=0; i<ETH_ALEN; i++) {
811 			nic->node_addr[i] = romdata[i + ((eth_flags & FLAG_16BIT) ? i : 0)];
812 		}
813 		printf("\nNE%c000 base %#hx, addr %!\n",
814 			(eth_flags & FLAG_16BIT) ? '2' : '1', eth_nic_base,
815 			nic->node_addr);
816 	}
817 #endif
818 	if (eth_vendor == VENDOR_NONE)
819 		return(0);
820         if (eth_vendor != VENDOR_3COM)
821 		eth_rmem = eth_bmem;
822 	ns8390_reset(nic);
823 	nic->reset = ns8390_reset;
824 	nic->poll = ns8390_poll;
825 	nic->transmit = ns8390_transmit;
826 	nic->disable = ns8390_disable;
827 	return(nic);
828 }
829 
830 /*
831  * Local variables:
832  *  c-basic-offset: 8
833  * End:
834  */
835 
836