/* $NetBSD: 3c509.c,v 1.10 2008/12/14 18:46:33 christos Exp $ */
/* stripped down from freebsd:sys/i386/netboot/3c509.c */
/**************************************************************************
NETBOOT - BOOTP/TFTP Bootstrap Program
Author: Martin Renters.
Date: Mar 22 1995
This code is based heavily on David Greenman's if_ed.c driver and
Andres Vega Garcia's if_ep.c driver.
Copyright (C) 1993-1994, David Greenman, Martin Renters.
Copyright (C) 1993-1995, Andres Vega Garcia.
Copyright (C) 1995, Serge Babkin.
This software may be used, modified, copied, distributed, and sold, in
both source and binary form provided that the above copyright and these
terms are retained. Under no circumstances are the authors responsible for
the proper functioning of this software, nor do the authors assume any
responsibility for damages incurred with its use.
3c509 support added by Serge Babkin (babkin@hq.icb.chel.su)
3c509.c,v 1.2 1995/05/30 07:58:52 rgrimes Exp
***************************************************************************/
#include <sys/types.h>
#include <machine/pio.h>
#include <lib/libsa/stand.h>
#include <lib/libkern/libkern.h>
#include <libi386.h>
#ifdef _STANDALONE
#include <bootinfo.h>
#endif
#include "etherdrv.h"
#include "3c509.h"
unsigned ether_medium;
unsigned short eth_base;
extern void epreset(void);
extern int ep_get_e(int);
static int send_ID_sequence(int);
static int get_eeprom_data(int, int);
u_char eth_myaddr[6];
static struct mtabentry {
int address_cfg; /* configured connector */
int config_bit; /* connector present */
char *name;
} mediatab[] = { /* indexed by media type - etherdrv.h */
{3, IS_BNC, "BNC"},
{0, IS_UTP, "UTP"},
{1, IS_AUI, "AUI"},
};
#ifdef _STANDALONE
static struct btinfo_netif bi_netif;
#endif
#ifndef _STANDALONE
extern int mapio(void);
#endif
/**************************************************************************
ETH_PROBE - Look for an adapter
***************************************************************************/
int
EtherInit(unsigned char *myadr)
{
/* common variables */
int i;
/* variables for 3C509 */
int data, j, id_port = EP_ID_PORT;
u_short k;
/* int ep_current_tag = EP_LAST_TAG + 1; */
u_short *p;
struct mtabentry *m;
#ifndef _STANDALONE
if (mapio()) {
printf("no IO access\n");
return 0;
}
#endif
/*********************************************************
Search for 3Com 509 card
***********************************************************/
/*
ep_current_tag--;
*/
/* Look for the ISA boards. Init and leave them actived */
/* search for the first card, ignore all others */
outb(id_port, 0xc0); /* Global reset */
delay(1000);
/*
for (i = 0; i < EP_MAX_BOARDS; i++) {
*/
outb(id_port, 0);
outb(id_port, 0);
send_ID_sequence(id_port);
data = get_eeprom_data(id_port, EEPROM_MFG_ID);
if (data != MFG_ID)
return 0;
/* resolve contention using the Ethernet address */
for (j = 0; j < 3; j++)
data = get_eeprom_data(id_port, j);
eth_base =
(get_eeprom_data(id_port, EEPROM_ADDR_CFG) & 0x1f) * 0x10 + 0x200;
outb(id_port, EP_LAST_TAG); /* tags board */
outb(id_port, ACTIVATE_ADAPTER_TO_CONFIG);
/*
ep_current_tag--;
break;
}
if (i == EP_MAX_BOARDS)
return 0;
*/
/*
* The iobase was found and MFG_ID was 0x6d50. PROD_ID should be
* 0x9[0-f]50
*/
GO_WINDOW(0);
k = (u_int)ep_get_e(EEPROM_PROD_ID);
if ((k & 0xf0ff) != (PROD_ID & 0xf0ff))
return 0;
printf("3C5x9 board on ISA at 0x%x - ", eth_base);
/* test for presence of connectors */
i = inw(IS_BASE + EP_W0_CONFIG_CTRL);
j = inw(IS_BASE + EP_W0_ADDRESS_CFG) >> 14;
for (ether_medium = 0, m = mediatab;
ether_medium < sizeof(mediatab) / sizeof(mediatab[0]);
ether_medium++, m++) {
if (j == m->address_cfg) {
if (!(i & m->config_bit)) {
printf("%s not present\n", m->name);
return 0;
}
printf("using %s\n", m->name);
goto ok;
}
}
printf("unknown connector\n");
return 0;
ok:
/*
* Read the station address from the eeprom
*/
p = (u_short *) eth_myaddr;
for (i = 0; i < 3; i++) {
u_short help;
GO_WINDOW(0);
help = ep_get_e(i);
p[i] = ((help & 0xff) << 8) | ((help & 0xff00) >> 8);
GO_WINDOW(2);
outw(BASE + EP_W2_ADDR_0 + (i * 2), help);
}
for (i = 0; i < 6; i++)
myadr[i] = eth_myaddr[i];
epreset();
#ifdef _STANDALONE
strncpy(bi_netif.ifname, "ep", sizeof(bi_netif.ifname));
bi_netif.bus = BI_BUS_ISA;
bi_netif.addr.iobase = eth_base;
BI_ADD(&bi_netif, BTINFO_NETIF, sizeof(bi_netif));
#endif
return 1;
}
static int
send_ID_sequence(int port)
{
int cx, al;
for (al = 0xff, cx = 0; cx < 255; cx++) {
outb(port, al);
al <<= 1;
if (al & 0x100)
al ^= 0xcf;
}
return 1;
}
/*
* We get eeprom data from the id_port given an offset into the eeprom.
* Basically; after the ID_sequence is sent to all of the cards; they enter
* the ID_CMD state where they will accept command requests. 0x80-0xbf loads
* the eeprom data. We then read the port 16 times and with every read; the
* cards check for contention (ie: if one card writes a 0 bit and another
* writes a 1 bit then the host sees a 0. At the end of the cycle; each card
* compares the data on the bus; if there is a difference then that card goes
* into ID_WAIT state again). In the meantime; one bit of data is returned in
* the AX register which is conveniently returned to us by inb(). Hence; we
* read 16 times getting one bit of data with each read.
*/
static int
get_eeprom_data(int id_port, int offset)
{
int i, data = 0;
outb(id_port, 0x80 + offset);
delay(1000);
for (i = 0; i < 16; i++)
data = (data << 1) | (inw(id_port) & 1);
return data;
}