1*74a4d8c2SCharles.Forsyth #include "u.h"
2*74a4d8c2SCharles.Forsyth #include "lib.h"
3*74a4d8c2SCharles.Forsyth #include "mem.h"
4*74a4d8c2SCharles.Forsyth #include "dat.h"
5*74a4d8c2SCharles.Forsyth #include "fns.h"
6*74a4d8c2SCharles.Forsyth #include "io.h"
7*74a4d8c2SCharles.Forsyth
8*74a4d8c2SCharles.Forsyth #include "etherif.h"
9*74a4d8c2SCharles.Forsyth #include "ether8390.h"
10*74a4d8c2SCharles.Forsyth
11*74a4d8c2SCharles.Forsyth /*
12*74a4d8c2SCharles.Forsyth * Driver written for the 'Notebook Computer Ethernet LAN Adapter',
13*74a4d8c2SCharles.Forsyth * a plug-in to the bus-slot on the rear of the Gateway NOMAD 425DXL
14*74a4d8c2SCharles.Forsyth * laptop. The manual says NE2000 compatible.
15*74a4d8c2SCharles.Forsyth * The interface appears to be pretty well described in the National
16*74a4d8c2SCharles.Forsyth * Semiconductor Local Area Network Databook (1992) as one of the
17*74a4d8c2SCharles.Forsyth * AT evaluation cards.
18*74a4d8c2SCharles.Forsyth *
19*74a4d8c2SCharles.Forsyth * The NE2000 is really just a DP8390[12] plus a data port
20*74a4d8c2SCharles.Forsyth * and a reset port.
21*74a4d8c2SCharles.Forsyth */
22*74a4d8c2SCharles.Forsyth enum {
23*74a4d8c2SCharles.Forsyth Data = 0x10, /* offset from I/O base of data port */
24*74a4d8c2SCharles.Forsyth Reset = 0x1F, /* offset from I/O base of reset port */
25*74a4d8c2SCharles.Forsyth };
26*74a4d8c2SCharles.Forsyth
27*74a4d8c2SCharles.Forsyth int
ne2000reset(Ether * ether)28*74a4d8c2SCharles.Forsyth ne2000reset(Ether* ether)
29*74a4d8c2SCharles.Forsyth {
30*74a4d8c2SCharles.Forsyth ushort buf[16];
31*74a4d8c2SCharles.Forsyth ulong port;
32*74a4d8c2SCharles.Forsyth Dp8390 *ctlr;
33*74a4d8c2SCharles.Forsyth int i;
34*74a4d8c2SCharles.Forsyth uchar ea[Eaddrlen];
35*74a4d8c2SCharles.Forsyth
36*74a4d8c2SCharles.Forsyth /*
37*74a4d8c2SCharles.Forsyth * Set up the software configuration.
38*74a4d8c2SCharles.Forsyth * Use defaults for port, irq, mem and size
39*74a4d8c2SCharles.Forsyth * if not specified.
40*74a4d8c2SCharles.Forsyth */
41*74a4d8c2SCharles.Forsyth if(ether->port == 0)
42*74a4d8c2SCharles.Forsyth ether->port = 0x300;
43*74a4d8c2SCharles.Forsyth if(ether->irq == 0)
44*74a4d8c2SCharles.Forsyth ether->irq = 2;
45*74a4d8c2SCharles.Forsyth if(ether->mem == 0)
46*74a4d8c2SCharles.Forsyth ether->mem = 0x4000;
47*74a4d8c2SCharles.Forsyth if(ether->size == 0)
48*74a4d8c2SCharles.Forsyth ether->size = 16*1024;
49*74a4d8c2SCharles.Forsyth port = ether->port;
50*74a4d8c2SCharles.Forsyth
51*74a4d8c2SCharles.Forsyth ether->ctlr = malloc(sizeof(Dp8390));
52*74a4d8c2SCharles.Forsyth ctlr = ether->ctlr;
53*74a4d8c2SCharles.Forsyth ctlr->width = 2;
54*74a4d8c2SCharles.Forsyth ctlr->ram = 0;
55*74a4d8c2SCharles.Forsyth
56*74a4d8c2SCharles.Forsyth ctlr->port = port;
57*74a4d8c2SCharles.Forsyth ctlr->data = port+Data;
58*74a4d8c2SCharles.Forsyth
59*74a4d8c2SCharles.Forsyth ctlr->tstart = HOWMANY(ether->mem, Dp8390BufSz);
60*74a4d8c2SCharles.Forsyth ctlr->pstart = ctlr->tstart + HOWMANY(sizeof(Etherpkt), Dp8390BufSz);
61*74a4d8c2SCharles.Forsyth ctlr->pstop = ctlr->tstart + HOWMANY(ether->size, Dp8390BufSz);
62*74a4d8c2SCharles.Forsyth
63*74a4d8c2SCharles.Forsyth ctlr->dummyrr = 1;
64*74a4d8c2SCharles.Forsyth for(i = 0; i < ether->nopt; i++){
65*74a4d8c2SCharles.Forsyth if(strcmp(ether->opt[i], "nodummyrr"))
66*74a4d8c2SCharles.Forsyth continue;
67*74a4d8c2SCharles.Forsyth ctlr->dummyrr = 0;
68*74a4d8c2SCharles.Forsyth break;
69*74a4d8c2SCharles.Forsyth }
70*74a4d8c2SCharles.Forsyth
71*74a4d8c2SCharles.Forsyth /*
72*74a4d8c2SCharles.Forsyth * Reset the board. This is done by doing a read
73*74a4d8c2SCharles.Forsyth * followed by a write to the Reset address.
74*74a4d8c2SCharles.Forsyth */
75*74a4d8c2SCharles.Forsyth buf[0] = inb(port+Reset);
76*74a4d8c2SCharles.Forsyth delay(2);
77*74a4d8c2SCharles.Forsyth outb(port+Reset, buf[0]);
78*74a4d8c2SCharles.Forsyth delay(2);
79*74a4d8c2SCharles.Forsyth
80*74a4d8c2SCharles.Forsyth /*
81*74a4d8c2SCharles.Forsyth * Init the (possible) chip, then use the (possible)
82*74a4d8c2SCharles.Forsyth * chip to read the (possible) PROM for ethernet address
83*74a4d8c2SCharles.Forsyth * and a marker byte.
84*74a4d8c2SCharles.Forsyth * Could just look at the DP8390 command register after
85*74a4d8c2SCharles.Forsyth * initialisation has been tried, but that wouldn't be
86*74a4d8c2SCharles.Forsyth * enough, there are other ethernet boards which could
87*74a4d8c2SCharles.Forsyth * match.
88*74a4d8c2SCharles.Forsyth */
89*74a4d8c2SCharles.Forsyth dp8390reset(ether);
90*74a4d8c2SCharles.Forsyth memset(buf, 0, sizeof(buf));
91*74a4d8c2SCharles.Forsyth dp8390read(ctlr, buf, 0, sizeof(buf));
92*74a4d8c2SCharles.Forsyth if((buf[0x0E] & 0xFF) != 0x57 || (buf[0x0F] & 0xFF) != 0x57){
93*74a4d8c2SCharles.Forsyth free(ether->ctlr);
94*74a4d8c2SCharles.Forsyth return -1;
95*74a4d8c2SCharles.Forsyth }
96*74a4d8c2SCharles.Forsyth
97*74a4d8c2SCharles.Forsyth /*
98*74a4d8c2SCharles.Forsyth * Stupid machine. Shorts were asked for,
99*74a4d8c2SCharles.Forsyth * shorts were delivered, although the PROM is a byte array.
100*74a4d8c2SCharles.Forsyth * Set the ethernet address.
101*74a4d8c2SCharles.Forsyth */
102*74a4d8c2SCharles.Forsyth memset(ea, 0, Eaddrlen);
103*74a4d8c2SCharles.Forsyth if(memcmp(ea, ether->ea, Eaddrlen) == 0){
104*74a4d8c2SCharles.Forsyth for(i = 0; i < sizeof(ether->ea); i++)
105*74a4d8c2SCharles.Forsyth ether->ea[i] = buf[i];
106*74a4d8c2SCharles.Forsyth }
107*74a4d8c2SCharles.Forsyth dp8390setea(ether);
108*74a4d8c2SCharles.Forsyth
109*74a4d8c2SCharles.Forsyth return 0;
110*74a4d8c2SCharles.Forsyth }
111