xref: /plan9/sys/src/9/pc/etherec2t.c (revision aa72973a2891ccbd3fb042462446761159389e19)
17dd7cddfSDavid du Colombier /*
2223a736eSDavid du Colombier  * Supposed NE2000 PCMCIA clones, see the comments in ether2000.c
37dd7cddfSDavid du Colombier  */
47dd7cddfSDavid du Colombier #include "u.h"
57dd7cddfSDavid du Colombier #include "../port/lib.h"
67dd7cddfSDavid du Colombier #include "mem.h"
77dd7cddfSDavid du Colombier #include "dat.h"
87dd7cddfSDavid du Colombier #include "fns.h"
97dd7cddfSDavid du Colombier #include "io.h"
107dd7cddfSDavid du Colombier #include "../port/error.h"
117dd7cddfSDavid du Colombier #include "../port/netif.h"
127dd7cddfSDavid du Colombier 
137dd7cddfSDavid du Colombier #include "etherif.h"
147dd7cddfSDavid du Colombier #include "ether8390.h"
157dd7cddfSDavid du Colombier 
167dd7cddfSDavid du Colombier enum {
177dd7cddfSDavid du Colombier 	Data		= 0x10,		/* offset from I/O base of data port */
187dd7cddfSDavid du Colombier 	Reset		= 0x1F,		/* offset from I/O base of reset port */
197dd7cddfSDavid du Colombier };
207dd7cddfSDavid du Colombier 
219a747e4fSDavid du Colombier typedef struct Ec2t {
229a747e4fSDavid du Colombier 	char*	name;
239a747e4fSDavid du Colombier 	int	iochecksum;
249a747e4fSDavid du Colombier } Ec2t;
259a747e4fSDavid du Colombier 
269a747e4fSDavid du Colombier static Ec2t ec2tpcmcia[] = {
279a747e4fSDavid du Colombier 	{ "EC2T", 0, },			/* Linksys Combo PCMCIA EthernetCard */
289a747e4fSDavid du Colombier 	{ "PCMPC100", 1, },		/* EtherFast 10/100 PC Card */
29b7b24591SDavid du Colombier 	{ "PCM100", 1, },		/* EtherFast PCM100 Card */
309a747e4fSDavid du Colombier 	{ "EN2216", 0, },		/* Accton EtherPair-PCMCIA */
319a747e4fSDavid du Colombier 	{ "FA410TX", 1, },		/* Netgear FA410TX */
32b2495906SDavid du Colombier 	{ "FA411", 0 },			/* Netgear FA411 PCMCIA */
339a747e4fSDavid du Colombier 	{ "Network Everywhere", 0, },	/* Linksys NP10T 10BaseT Card */
349a747e4fSDavid du Colombier 	{ "10/100 Port Attached", 1, },	/* SMC 8040TX */
3532b747daSDavid du Colombier 	{ "8041TX-10/100-PC-Card-V2", 0 }, /* SMC 8041TX */
36b2495906SDavid du Colombier 	{ "SMC8022", 0},		/* SMC 8022 / EZCard-10-PCMCIA */
379a747e4fSDavid du Colombier 	{ nil, 0, },
387dd7cddfSDavid du Colombier };
397dd7cddfSDavid du Colombier 
407dd7cddfSDavid du Colombier static int
reset(Ether * ether)417dd7cddfSDavid du Colombier reset(Ether* ether)
427dd7cddfSDavid du Colombier {
437dd7cddfSDavid du Colombier 	ushort buf[16];
447dd7cddfSDavid du Colombier 	ulong port;
457dd7cddfSDavid du Colombier 	Dp8390 *ctlr;
467dd7cddfSDavid du Colombier 	int i, slot;
477dd7cddfSDavid du Colombier 	uchar ea[Eaddrlen], sum, x;
489a747e4fSDavid du Colombier 	Ec2t *ec2t, tmpec2t;
497dd7cddfSDavid du Colombier 
507dd7cddfSDavid du Colombier 	/*
517dd7cddfSDavid du Colombier 	 * Set up the software configuration.
527dd7cddfSDavid du Colombier 	 * Use defaults for port, irq, mem and size
537dd7cddfSDavid du Colombier 	 * if not specified.
547dd7cddfSDavid du Colombier 	 * The manual says 16KB memory, the box
557dd7cddfSDavid du Colombier 	 * says 32KB. The manual seems to be correct.
567dd7cddfSDavid du Colombier 	 */
577dd7cddfSDavid du Colombier 	if(ether->port == 0)
587dd7cddfSDavid du Colombier 		ether->port = 0x300;
597dd7cddfSDavid du Colombier 	if(ether->irq == 0)
607dd7cddfSDavid du Colombier 		ether->irq = 9;
617dd7cddfSDavid du Colombier 	if(ether->mem == 0)
627dd7cddfSDavid du Colombier 		ether->mem = 0x4000;
637dd7cddfSDavid du Colombier 	if(ether->size == 0)
647dd7cddfSDavid du Colombier 		ether->size = 16*1024;
657dd7cddfSDavid du Colombier 	port = ether->port;
667dd7cddfSDavid du Colombier 
677dd7cddfSDavid du Colombier 	if(ioalloc(ether->port, 0x20, 0, "ec2t") < 0)
687dd7cddfSDavid du Colombier 		return -1;
697dd7cddfSDavid du Colombier 	slot = -1;
709a747e4fSDavid du Colombier 	for(ec2t = ec2tpcmcia; ec2t->name != nil; ec2t++){
719a747e4fSDavid du Colombier 		if((slot = pcmspecial(ec2t->name, ether)) >= 0)
727dd7cddfSDavid du Colombier 			break;
737dd7cddfSDavid du Colombier 	}
749a747e4fSDavid du Colombier 	if(ec2t->name == nil){
759a747e4fSDavid du Colombier 		ec2t = &tmpec2t;
769a747e4fSDavid du Colombier 		ec2t->name = nil;
779a747e4fSDavid du Colombier 		ec2t->iochecksum = 0;
78223a736eSDavid du Colombier 		for(i = 0; i < ether->nopt; i++){
799a747e4fSDavid du Colombier 			if(cistrncmp(ether->opt[i], "id=", 3) == 0){
809a747e4fSDavid du Colombier 				ec2t->name = &ether->opt[i][3];
819a747e4fSDavid du Colombier 				slot = pcmspecial(ec2t->name, ether);
829a747e4fSDavid du Colombier 			}
839a747e4fSDavid du Colombier 			else if(cistrncmp(ether->opt[i], "iochecksum", 10) == 0)
849a747e4fSDavid du Colombier 				ec2t->iochecksum = 1;
85223a736eSDavid du Colombier 		}
86223a736eSDavid du Colombier 	}
877dd7cddfSDavid du Colombier 	if(slot < 0){
887dd7cddfSDavid du Colombier 		iofree(port);
897dd7cddfSDavid du Colombier 		return -1;
907dd7cddfSDavid du Colombier 	}
917dd7cddfSDavid du Colombier 
927dd7cddfSDavid du Colombier 	ether->ctlr = malloc(sizeof(Dp8390));
937dd7cddfSDavid du Colombier 	ctlr = ether->ctlr;
94*aa72973aSDavid du Colombier 	if(ctlr == nil) {
95*aa72973aSDavid du Colombier 		iofree(port);
96*aa72973aSDavid du Colombier 		error(Enomem);
97*aa72973aSDavid du Colombier 	}
987dd7cddfSDavid du Colombier 	ctlr->width = 2;
997dd7cddfSDavid du Colombier 	ctlr->ram = 0;
1007dd7cddfSDavid du Colombier 
1017dd7cddfSDavid du Colombier 	ctlr->port = port;
1027dd7cddfSDavid du Colombier 	ctlr->data = port+Data;
1037dd7cddfSDavid du Colombier 
1047dd7cddfSDavid du Colombier 	ctlr->tstart = HOWMANY(ether->mem, Dp8390BufSz);
1057dd7cddfSDavid du Colombier 	ctlr->pstart = ctlr->tstart + HOWMANY(sizeof(Etherpkt), Dp8390BufSz);
1067dd7cddfSDavid du Colombier 	ctlr->pstop = ctlr->tstart + HOWMANY(ether->size, Dp8390BufSz);
1077dd7cddfSDavid du Colombier 
1087dd7cddfSDavid du Colombier 	ctlr->dummyrr = 0;
109223a736eSDavid du Colombier 	for(i = 0; i < ether->nopt; i++){
110223a736eSDavid du Colombier 		if(cistrcmp(ether->opt[i], "nodummyrr") == 0)
111223a736eSDavid du Colombier 			ctlr->dummyrr = 0;
112223a736eSDavid du Colombier 		else if(cistrncmp(ether->opt[i], "dummyrr=", 8) == 0)
113223a736eSDavid du Colombier 			ctlr->dummyrr = strtol(&ether->opt[i][8], nil, 0);
114223a736eSDavid du Colombier 	}
1157dd7cddfSDavid du Colombier 
1167dd7cddfSDavid du Colombier 	/*
1177dd7cddfSDavid du Colombier 	 * Reset the board. This is done by doing a read
1187dd7cddfSDavid du Colombier 	 * followed by a write to the Reset address.
1197dd7cddfSDavid du Colombier 	 */
1207dd7cddfSDavid du Colombier 	buf[0] = inb(port+Reset);
1217dd7cddfSDavid du Colombier 	delay(2);
1227dd7cddfSDavid du Colombier 	outb(port+Reset, buf[0]);
1237dd7cddfSDavid du Colombier 	delay(2);
1247dd7cddfSDavid du Colombier 
1257dd7cddfSDavid du Colombier 	/*
1267dd7cddfSDavid du Colombier 	 * Init the (possible) chip, then use the (possible)
1277dd7cddfSDavid du Colombier 	 * chip to read the (possible) PROM for ethernet address
1287dd7cddfSDavid du Colombier 	 * and a marker byte.
1297dd7cddfSDavid du Colombier 	 * Could just look at the DP8390 command register after
1307dd7cddfSDavid du Colombier 	 * initialisation has been tried, but that wouldn't be
1317dd7cddfSDavid du Colombier 	 * enough, there are other ethernet boards which could
1327dd7cddfSDavid du Colombier 	 * match.
1337dd7cddfSDavid du Colombier 	 */
1347dd7cddfSDavid du Colombier 	dp8390reset(ether);
1357dd7cddfSDavid du Colombier 	sum = 0;
1369a747e4fSDavid du Colombier 	if(ec2t->iochecksum){
1377dd7cddfSDavid du Colombier 		/*
1389a747e4fSDavid du Colombier 		 * These cards have the ethernet address in I/O space.
1397dd7cddfSDavid du Colombier 		 * There's a checksum over 8 bytes which sums to 0xFF.
1407dd7cddfSDavid du Colombier 		 */
1417dd7cddfSDavid du Colombier 		for(i = 0; i < 8; i++){
1427dd7cddfSDavid du Colombier 			x = inb(port+0x14+i);
1437dd7cddfSDavid du Colombier 			sum += x;
1447dd7cddfSDavid du Colombier 			buf[i] = (x<<8)|x;
1457dd7cddfSDavid du Colombier 		}
1467dd7cddfSDavid du Colombier 	}
147223a736eSDavid du Colombier 	else{
148223a736eSDavid du Colombier 		memset(buf, 0, sizeof(buf));
149223a736eSDavid du Colombier 		dp8390read(ctlr, buf, 0, sizeof(buf));
150223a736eSDavid du Colombier 		if((buf[0x0E] & 0xFF) == 0x57 && (buf[0x0F] & 0xFF) == 0x57)
151223a736eSDavid du Colombier 			sum = 0xFF;
152223a736eSDavid du Colombier 	}
1537dd7cddfSDavid du Colombier 	if(sum != 0xFF){
1547dd7cddfSDavid du Colombier 		pcmspecialclose(slot);
1557dd7cddfSDavid du Colombier 		iofree(ether->port);
1567dd7cddfSDavid du Colombier 		free(ether->ctlr);
1577dd7cddfSDavid du Colombier 		return -1;
1587dd7cddfSDavid du Colombier 	}
1597dd7cddfSDavid du Colombier 
1607dd7cddfSDavid du Colombier 	/*
1617dd7cddfSDavid du Colombier 	 * Stupid machine. Shorts were asked for,
1627dd7cddfSDavid du Colombier 	 * shorts were delivered, although the PROM is a byte array.
1637dd7cddfSDavid du Colombier 	 * Set the ethernet address.
1647dd7cddfSDavid du Colombier 	 */
1657dd7cddfSDavid du Colombier 	memset(ea, 0, Eaddrlen);
1667dd7cddfSDavid du Colombier 	if(memcmp(ea, ether->ea, Eaddrlen) == 0){
1677dd7cddfSDavid du Colombier 		for(i = 0; i < sizeof(ether->ea); i++)
1687dd7cddfSDavid du Colombier 			ether->ea[i] = buf[i];
1697dd7cddfSDavid du Colombier 	}
1707dd7cddfSDavid du Colombier 	dp8390setea(ether);
1717dd7cddfSDavid du Colombier 
1727dd7cddfSDavid du Colombier 	return 0;
1737dd7cddfSDavid du Colombier }
1747dd7cddfSDavid du Colombier 
1757dd7cddfSDavid du Colombier void
etherec2tlink(void)1767dd7cddfSDavid du Colombier etherec2tlink(void)
1777dd7cddfSDavid du Colombier {
1787dd7cddfSDavid du Colombier 	addethercard("EC2T", reset);
1797dd7cddfSDavid du Colombier }
180