MFp4: Omnibus ed changes
o Attach AX88x90's MII bus to system, and require its presence. o Reorg the mii code a little, and move more of it into pccard attachment. o Eliminate ed_pccard_{read,write}_attrmem in favor of a more appropriate function in the pccard layer. o Update comments to reflect knowledge gained. o Update how re recognize a NE-2000 ROM. I found a couple of different datasheets that define the structure of the PROM data, so the code's old heuristics have been removed, and comments updated to reflect the structure. o Eliminate work around for EC2T. It is no longer needed, and was wrong headed since the EC2T has a Winbound 82C926C in it, not a AX88x90. o Add copyright to if_ed_pccard.c, since I believe I've re-written more than 3/4 of it. # With these changes, all of my 20-odd ed based cards work, except for the # NetGear FA-410, and I'm pretty sure that's a MII/PHY problem.
This commit is contained in:
parent
ff1de32e3b
commit
d21d33847c
@ -1635,43 +1635,6 @@ ed_pio_write_mbufs(struct ed_softc *sc, struct mbuf *m, bus_size_t dst)
|
||||
}
|
||||
|
||||
#ifndef ED_NO_MIIBUS
|
||||
/*
|
||||
* MII bus support routines.
|
||||
*/
|
||||
int
|
||||
ed_miibus_readreg(device_t dev, int phy, int reg)
|
||||
{
|
||||
struct ed_softc *sc;
|
||||
int failed, val;
|
||||
|
||||
sc = device_get_softc(dev);
|
||||
(*sc->mii_writebits)(sc, 0xffffffff, 32);
|
||||
(*sc->mii_writebits)(sc, ED_MII_STARTDELIM, ED_MII_STARTDELIM_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_READOP, ED_MII_OP_BITS);
|
||||
(*sc->mii_writebits)(sc, phy, ED_MII_PHY_BITS);
|
||||
(*sc->mii_writebits)(sc, reg, ED_MII_REG_BITS);
|
||||
failed = (*sc->mii_readbits)(sc, ED_MII_ACK_BITS);
|
||||
val = (*sc->mii_readbits)(sc, ED_MII_DATA_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_IDLE, ED_MII_IDLE_BITS);
|
||||
return (failed ? 0 : val);
|
||||
}
|
||||
|
||||
void
|
||||
ed_miibus_writereg(device_t dev, int phy, int reg, int data)
|
||||
{
|
||||
struct ed_softc *sc;
|
||||
|
||||
sc = device_get_softc(dev);
|
||||
(*sc->mii_writebits)(sc, 0xffffffff, 32);
|
||||
(*sc->mii_writebits)(sc, ED_MII_STARTDELIM, ED_MII_STARTDELIM_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_WRITEOP, ED_MII_OP_BITS);
|
||||
(*sc->mii_writebits)(sc, phy, ED_MII_PHY_BITS);
|
||||
(*sc->mii_writebits)(sc, reg, ED_MII_REG_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_TURNAROUND, ED_MII_TURNAROUND_BITS);
|
||||
(*sc->mii_writebits)(sc, data, ED_MII_DATA_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_IDLE, ED_MII_IDLE_BITS);
|
||||
}
|
||||
|
||||
int
|
||||
ed_ifmedia_upd(struct ifnet *ifp)
|
||||
{
|
||||
|
@ -1,4 +1,5 @@
|
||||
/*-
|
||||
* Copyright (c) 2005, M. Warner Losh
|
||||
* Copyright (c) 1995, David Greenman
|
||||
* All rights reserved.
|
||||
*
|
||||
@ -51,6 +52,7 @@
|
||||
#include <dev/ed/if_edreg.h>
|
||||
#include <dev/ed/if_edvar.h>
|
||||
#include <dev/pccard/pccardvar.h>
|
||||
#include <dev/pccard/pccardreg.h>
|
||||
#include <dev/pccard/pccard_cis.h>
|
||||
#ifndef ED_NO_MIIBUS
|
||||
#include <dev/mii/mii.h>
|
||||
@ -79,7 +81,9 @@ MODULE_DEPEND(ed, ether, 1, 1, 1);
|
||||
*
|
||||
* This applies only to the older, NE-2000 compatbile cards. The newer
|
||||
* cards based on the AX88x90 or DL100XX chipsets have a specific place
|
||||
* to look for MAC information.
|
||||
* to look for MAC information. And only to those NE-2000 compatible cards
|
||||
* that don't the NE-2000 compatible thing of placing the PROM contents
|
||||
* starting at location 0 of memory.
|
||||
*/
|
||||
#define ED_DEFAULT_MAC_OFFSET 0xff0
|
||||
|
||||
@ -186,15 +190,18 @@ static const struct ed_product {
|
||||
static int ed_pccard_probe(device_t);
|
||||
static int ed_pccard_attach(device_t);
|
||||
|
||||
static int ed_pccard_memread(device_t dev, off_t offset, u_char *byte);
|
||||
static int ed_pccard_memwrite(device_t dev, off_t offset, u_char byte);
|
||||
|
||||
static int ed_pccard_dl100xx(device_t dev, const struct ed_product *);
|
||||
#ifndef ED_NO_MIIBUS
|
||||
static void ed_pccard_dl10xx_mii_reset(struct ed_softc *sc);
|
||||
static u_int ed_pccard_dl10xx_mii_readbits(struct ed_softc *sc, int nbits);
|
||||
static void ed_pccard_dl10xx_mii_writebits(struct ed_softc *sc, u_int val,
|
||||
static void ed_pccard_dl100xx_mii_reset(struct ed_softc *sc);
|
||||
static u_int ed_pccard_dl100xx_mii_readbits(struct ed_softc *sc, int nbits);
|
||||
static void ed_pccard_dl100xx_mii_writebits(struct ed_softc *sc, u_int val,
|
||||
int nbits);
|
||||
|
||||
static void ed_pccard_ax88x90_mii_reset(struct ed_softc *sc);
|
||||
static u_int ed_pccard_ax88x90_mii_readbits(struct ed_softc *sc, int nbits);
|
||||
static void ed_pccard_ax88x90_mii_writebits(struct ed_softc *sc, u_int val,
|
||||
int nbits);
|
||||
static int ed_miibus_readreg(device_t dev, int phy, int reg);
|
||||
#endif
|
||||
|
||||
static int ed_pccard_ax88x90(device_t dev, const struct ed_product *);
|
||||
@ -232,25 +239,50 @@ static int
|
||||
ed_pccard_rom_mac(device_t dev, uint8_t *enaddr)
|
||||
{
|
||||
struct ed_softc *sc = device_get_softc(dev);
|
||||
uint8_t romdata[16];
|
||||
uint8_t romdata[32];
|
||||
int i;
|
||||
|
||||
/*
|
||||
* Read in the rom data at location 0. We should see one of
|
||||
* two patterns. Either you'll see odd locations 0xff and
|
||||
* even locations data, or you'll see odd and even locations
|
||||
* mirror each other. In addition, the last two even locations
|
||||
* should be 0. If they aren't 0, then we'll assume that
|
||||
* there's no valid ROM data on this card and try another method
|
||||
* to recover the MAC address. Since there are no NE-1000
|
||||
* based PC Card devices, we'll assume we're 16-bit.
|
||||
* Read in the rom data at location 0. Since there are no
|
||||
* NE-1000 based PC Card devices, we'll assume we're 16-bit.
|
||||
*
|
||||
* In researching what format this takes, I've found that the
|
||||
* following appears to be true for multiple cards based on
|
||||
* observation as well as datasheet digging.
|
||||
*
|
||||
* Data is stored in some ROM and is copied out 8 bits at a time
|
||||
* into 16-bit wide locations. This means that the odd locations
|
||||
* of the ROM are not used (and can be either 0 or ff).
|
||||
*
|
||||
* The contents appears to be as follows:
|
||||
* PROM RAM
|
||||
* Offset Offset What
|
||||
* 0 0 ENETADDR 0
|
||||
* 1 2 ENETADDR 1
|
||||
* 2 4 ENETADDR 2
|
||||
* 3 6 ENETADDR 3
|
||||
* 4 8 ENETADDR 4
|
||||
* 5 10 ENETADDR 5
|
||||
* 6-13 12-26 Reserved (varies by manufacturer)
|
||||
* 14 28 0x57
|
||||
* 15 30 0x57
|
||||
*
|
||||
* Some manufacturers have another image of enetaddr from
|
||||
* PROM offset 0x10 to 0x15 with 0x42 in 0x1e and 0x1f, but
|
||||
* this doesn't appear to be universally documented in the
|
||||
* datasheets. Some manufactuers have a card type, card config
|
||||
* checksums, etc encoded into PROM offset 6-13, but deciphering it
|
||||
* requires more knowledge about the exact underlying chipset than
|
||||
* we possess (and maybe can possess).
|
||||
*/
|
||||
ed_pio_readmem(sc, 0, romdata, 16);
|
||||
if (romdata[12] != 0 || romdata[14] != 0)
|
||||
return 0;
|
||||
ed_pio_readmem(sc, 0, romdata, 32);
|
||||
if (bootverbose)
|
||||
printf("ROM DATA: %32D\n", romdata, " ");
|
||||
if (romdata[28] != 0x57 || romdata[30] != 0x57)
|
||||
return (0);
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i++)
|
||||
enaddr[i] = romdata[i * 2];
|
||||
return 1;
|
||||
return (1);
|
||||
}
|
||||
|
||||
static int
|
||||
@ -311,7 +343,7 @@ ed_pccard_attach(device_t dev)
|
||||
error = ENXIO;
|
||||
if (error != 0)
|
||||
error = ed_pccard_dl100xx(dev, pp);
|
||||
if (error != 0 && pp->flags & NE2000DVF_AX88X90)
|
||||
if (error != 0)
|
||||
error = ed_pccard_ax88x90(dev, pp);
|
||||
if (error != 0)
|
||||
error = ed_probe_Novell_generic(dev, device_get_flags(dev));
|
||||
@ -349,17 +381,17 @@ ed_pccard_attach(device_t dev)
|
||||
}
|
||||
if (sum == 0 && pp->flags & NE2000DVF_ENADDR) {
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i++) {
|
||||
ed_pccard_memread(dev, pp->enoff + i * 2,
|
||||
pccard_attr_read_1(dev, pp->enoff + i * 2,
|
||||
enaddr + i);
|
||||
sum |= enaddr[i];
|
||||
}
|
||||
if (bootverbose)
|
||||
device_printf(dev, "Hint MAC %6D\n", enaddr,
|
||||
":");
|
||||
device_printf(dev, "Hint %x MAC %6D\n",
|
||||
pp->enoff, enaddr, ":");
|
||||
}
|
||||
if (sum == 0) {
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i++) {
|
||||
ed_pccard_memread(dev, ED_DEFAULT_MAC_OFFSET +
|
||||
pccard_attr_read_1(dev, ED_DEFAULT_MAC_OFFSET +
|
||||
i * 2, enaddr + i);
|
||||
sum |= enaddr[i];
|
||||
}
|
||||
@ -382,11 +414,17 @@ ed_pccard_attach(device_t dev)
|
||||
if (sc->chip_type == ED_CHIP_TYPE_DL10019 ||
|
||||
sc->chip_type == ED_CHIP_TYPE_DL10022) {
|
||||
/* Probe for an MII bus, but ignore errors. */
|
||||
ed_pccard_dl10xx_mii_reset(sc);
|
||||
sc->mii_readbits = ed_pccard_dl10xx_mii_readbits;
|
||||
sc->mii_writebits = ed_pccard_dl10xx_mii_writebits;
|
||||
ed_pccard_dl100xx_mii_reset(sc);
|
||||
mii_phy_probe(dev, &sc->miibus, ed_ifmedia_upd,
|
||||
ed_ifmedia_sts);
|
||||
} else if (sc->chip_type == ED_CHIP_TYPE_AX88190) {
|
||||
ed_pccard_ax88x90_mii_reset(sc);
|
||||
if ((error = mii_phy_probe(dev, &sc->miibus, ed_ifmedia_upd,
|
||||
ed_ifmedia_sts)) != 0) {
|
||||
device_printf(dev, "Missing mii!\n");
|
||||
goto bad;
|
||||
}
|
||||
|
||||
}
|
||||
#endif
|
||||
if (sc->modem_rid != -1)
|
||||
@ -397,52 +435,6 @@ ed_pccard_attach(device_t dev)
|
||||
return (error);
|
||||
}
|
||||
|
||||
static int
|
||||
ed_pccard_memwrite(device_t dev, off_t offset, u_char byte)
|
||||
{
|
||||
int cis_rid;
|
||||
struct resource *cis;
|
||||
|
||||
cis_rid = 0;
|
||||
cis = bus_alloc_resource(dev, SYS_RES_MEMORY, &cis_rid, 0, ~0,
|
||||
4 << 10, RF_ACTIVE | RF_SHAREABLE);
|
||||
if (cis == NULL)
|
||||
return (ENXIO);
|
||||
CARD_SET_RES_FLAGS(device_get_parent(dev), dev, SYS_RES_MEMORY,
|
||||
cis_rid, PCCARD_A_MEM_ATTR);
|
||||
|
||||
bus_space_write_1(rman_get_bustag(cis), rman_get_bushandle(cis),
|
||||
offset, byte);
|
||||
|
||||
bus_deactivate_resource(dev, SYS_RES_MEMORY, cis_rid, cis);
|
||||
bus_release_resource(dev, SYS_RES_MEMORY, cis_rid, cis);
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
||||
static int
|
||||
ed_pccard_memread(device_t dev, off_t offset, u_char *byte)
|
||||
{
|
||||
int cis_rid;
|
||||
struct resource *cis;
|
||||
|
||||
cis_rid = 0;
|
||||
cis = bus_alloc_resource(dev, SYS_RES_MEMORY, &cis_rid, 0, ~0,
|
||||
4 << 10, RF_ACTIVE | RF_SHAREABLE);
|
||||
if (cis == NULL)
|
||||
return (ENXIO);
|
||||
CARD_SET_RES_FLAGS(device_get_parent(dev), dev, SYS_RES_MEMORY,
|
||||
cis_rid, PCCARD_A_MEM_ATTR);
|
||||
|
||||
*byte = bus_space_read_1(rman_get_bustag(cis), rman_get_bushandle(cis),
|
||||
offset);
|
||||
|
||||
bus_deactivate_resource(dev, SYS_RES_MEMORY, cis_rid, cis);
|
||||
bus_release_resource(dev, SYS_RES_MEMORY, cis_rid, cis);
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
||||
/*
|
||||
* Probe the Ethernet MAC addrees for PCMCIA Linksys EtherFast 10/100
|
||||
* and compatible cards (DL10019C Ethernet controller).
|
||||
@ -459,9 +451,13 @@ ed_pccard_dl100xx(device_t dev, const struct ed_product *pp)
|
||||
uint8_t id;
|
||||
int i, error;
|
||||
|
||||
if (!pp->flags & NE2000DVF_DL100XX)
|
||||
if (!(pp->flags & NE2000DVF_DL100XX))
|
||||
return (ENXIO);
|
||||
if (bootverbose)
|
||||
device_printf(dev, "Trying DL100xx probing\n");
|
||||
error = ed_probe_Novell_generic(dev, device_get_flags(dev));
|
||||
if (bootverbose && error)
|
||||
device_printf(dev, "Novell generic probe failed: %d\n", error);
|
||||
if (error != 0)
|
||||
return (error);
|
||||
|
||||
@ -474,8 +470,13 @@ ed_pccard_dl100xx(device_t dev, const struct ed_product *pp)
|
||||
*/
|
||||
for (sum = 0, i = 0x04; i < 0x0c; i++)
|
||||
sum += ed_asic_inb(sc, i);
|
||||
if (sum != 0xff)
|
||||
if (sum != 0xff) {
|
||||
if (bootverbose)
|
||||
device_printf(dev, "Bad checksum %#x\n", sum);
|
||||
return (ENXIO); /* invalid DL10019C */
|
||||
}
|
||||
if (bootverbose)
|
||||
device_printf(dev, "CIR is %d\n", ed_asic_inb(sc, 0xa));
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i++)
|
||||
sc->enaddr[i] = ed_asic_inb(sc, 0x04 + i);
|
||||
ed_nic_outb(sc, ED_P0_DCR, ED_DCR_WTS | ED_DCR_FT1 | ED_DCR_LS);
|
||||
@ -486,83 +487,85 @@ ed_pccard_dl100xx(device_t dev, const struct ed_product *pp)
|
||||
sc->chip_type = (id & 0x90) == 0x90 ?
|
||||
ED_CHIP_TYPE_DL10022 : ED_CHIP_TYPE_DL10019;
|
||||
sc->type_str = ((id & 0x90) == 0x90) ? "DL10022" : "DL10019";
|
||||
sc->mii_readbits = ed_pccard_dl100xx_mii_readbits;
|
||||
sc->mii_writebits = ed_pccard_dl100xx_mii_writebits;
|
||||
return (0);
|
||||
}
|
||||
|
||||
#ifndef ED_NO_MIIBUS
|
||||
/* MII bit-twiddling routines for cards using Dlink chipset */
|
||||
#define DL10XX_MIISET(sc, x) ed_asic_outb(sc, ED_DL10XX_MIIBUS, \
|
||||
ed_asic_inb(sc, ED_DL10XX_MIIBUS) | (x))
|
||||
#define DL10XX_MIICLR(sc, x) ed_asic_outb(sc, ED_DL10XX_MIIBUS, \
|
||||
ed_asic_inb(sc, ED_DL10XX_MIIBUS) & ~(x))
|
||||
#define DL100XX_MIISET(sc, x) ed_asic_outb(sc, ED_DL100XX_MIIBUS, \
|
||||
ed_asic_inb(sc, ED_DL100XX_MIIBUS) | (x))
|
||||
#define DL100XX_MIICLR(sc, x) ed_asic_outb(sc, ED_DL100XX_MIIBUS, \
|
||||
ed_asic_inb(sc, ED_DL100XX_MIIBUS) & ~(x))
|
||||
|
||||
static void
|
||||
ed_pccard_dl10xx_mii_reset(struct ed_softc *sc)
|
||||
ed_pccard_dl100xx_mii_reset(struct ed_softc *sc)
|
||||
{
|
||||
if (sc->chip_type != ED_CHIP_TYPE_DL10022)
|
||||
return;
|
||||
|
||||
ed_asic_outb(sc, ED_DL10XX_MIIBUS, ED_DL10XX_MII_RESET2);
|
||||
ed_asic_outb(sc, ED_DL100XX_MIIBUS, ED_DL100XX_MII_RESET2);
|
||||
DELAY(10);
|
||||
ed_asic_outb(sc, ED_DL10XX_MIIBUS,
|
||||
ED_DL10XX_MII_RESET2 | ED_DL10XX_MII_RESET1);
|
||||
ed_asic_outb(sc, ED_DL100XX_MIIBUS,
|
||||
ED_DL100XX_MII_RESET2 | ED_DL100XX_MII_RESET1);
|
||||
DELAY(10);
|
||||
ed_asic_outb(sc, ED_DL10XX_MIIBUS, ED_DL10XX_MII_RESET2);
|
||||
ed_asic_outb(sc, ED_DL100XX_MIIBUS, ED_DL100XX_MII_RESET2);
|
||||
DELAY(10);
|
||||
ed_asic_outb(sc, ED_DL10XX_MIIBUS,
|
||||
ED_DL10XX_MII_RESET2 | ED_DL10XX_MII_RESET1);
|
||||
ed_asic_outb(sc, ED_DL100XX_MIIBUS,
|
||||
ED_DL100XX_MII_RESET2 | ED_DL100XX_MII_RESET1);
|
||||
DELAY(10);
|
||||
ed_asic_outb(sc, ED_DL10XX_MIIBUS, 0);
|
||||
ed_asic_outb(sc, ED_DL100XX_MIIBUS, 0);
|
||||
}
|
||||
|
||||
static void
|
||||
ed_pccard_dl10xx_mii_writebits(struct ed_softc *sc, u_int val, int nbits)
|
||||
ed_pccard_dl100xx_mii_writebits(struct ed_softc *sc, u_int val, int nbits)
|
||||
{
|
||||
int i;
|
||||
|
||||
if (sc->chip_type == ED_CHIP_TYPE_DL10022)
|
||||
DL10XX_MIISET(sc, ED_DL10XX_MII_DIROUT_22);
|
||||
DL100XX_MIISET(sc, ED_DL100XX_MII_DIROUT_22);
|
||||
else
|
||||
DL10XX_MIISET(sc, ED_DL10XX_MII_DIROUT_19);
|
||||
DL100XX_MIISET(sc, ED_DL100XX_MII_DIROUT_19);
|
||||
|
||||
for (i = nbits - 1; i >= 0; i--) {
|
||||
if ((val >> i) & 1)
|
||||
DL10XX_MIISET(sc, ED_DL10XX_MII_DATAOUT);
|
||||
DL100XX_MIISET(sc, ED_DL100XX_MII_DATAOUT);
|
||||
else
|
||||
DL10XX_MIICLR(sc, ED_DL10XX_MII_DATAOUT);
|
||||
DL100XX_MIICLR(sc, ED_DL100XX_MII_DATAOUT);
|
||||
DELAY(10);
|
||||
DL10XX_MIISET(sc, ED_DL10XX_MII_CLK);
|
||||
DL100XX_MIISET(sc, ED_DL100XX_MII_CLK);
|
||||
DELAY(10);
|
||||
DL10XX_MIICLR(sc, ED_DL10XX_MII_CLK);
|
||||
DL100XX_MIICLR(sc, ED_DL100XX_MII_CLK);
|
||||
DELAY(10);
|
||||
}
|
||||
}
|
||||
|
||||
static u_int
|
||||
ed_pccard_dl10xx_mii_readbits(struct ed_softc *sc, int nbits)
|
||||
ed_pccard_dl100xx_mii_readbits(struct ed_softc *sc, int nbits)
|
||||
{
|
||||
int i;
|
||||
u_int val = 0;
|
||||
|
||||
if (sc->chip_type == ED_CHIP_TYPE_DL10022)
|
||||
DL10XX_MIICLR(sc, ED_DL10XX_MII_DIROUT_22);
|
||||
DL100XX_MIICLR(sc, ED_DL100XX_MII_DIROUT_22);
|
||||
else
|
||||
DL10XX_MIICLR(sc, ED_DL10XX_MII_DIROUT_19);
|
||||
DL100XX_MIICLR(sc, ED_DL100XX_MII_DIROUT_19);
|
||||
|
||||
for (i = nbits - 1; i >= 0; i--) {
|
||||
DL10XX_MIISET(sc, ED_DL10XX_MII_CLK);
|
||||
DL100XX_MIISET(sc, ED_DL100XX_MII_CLK);
|
||||
DELAY(10);
|
||||
val <<= 1;
|
||||
if (ed_asic_inb(sc, ED_DL10XX_MIIBUS) & ED_DL10XX_MII_DATATIN)
|
||||
if (ed_asic_inb(sc, ED_DL100XX_MIIBUS) & ED_DL100XX_MII_DATATIN)
|
||||
val++;
|
||||
DL10XX_MIICLR(sc, ED_DL10XX_MII_CLK);
|
||||
DL100XX_MIICLR(sc, ED_DL100XX_MII_CLK);
|
||||
DELAY(10);
|
||||
}
|
||||
return val;
|
||||
}
|
||||
#endif
|
||||
|
||||
static void
|
||||
static int
|
||||
ed_pccard_ax88x90_geteprom(struct ed_softc *sc)
|
||||
{
|
||||
int prom[16],i;
|
||||
@ -598,33 +601,13 @@ ed_pccard_ax88x90_geteprom(struct ed_softc *sc)
|
||||
ed_nic_outb(sc, pg_seq[i].offset, pg_seq[i].value);
|
||||
|
||||
/* Get Data */
|
||||
for (i = 0; i < 16; i++)
|
||||
for (i = 0; i < ETHER_ADDR_LEN / 2; i++)
|
||||
prom[i] = ed_asic_inw(sc, 0);
|
||||
|
||||
/*
|
||||
* Work around a bug I've seen on Linksys EC2T cards. On
|
||||
* these cards, the node address is contained in the low order
|
||||
* bytes of the prom, with the upper byte being 0. On other
|
||||
* cards, the bytes are packed two per word. I'm unsure why
|
||||
* this is the case, and why no other open source OS has a
|
||||
* similar workaround. The Linksys EC2T card is still extremely
|
||||
* popular on E-Bay, fetching way more than any other 10Mbps
|
||||
* only card. I might be able to test to see if prom[7] and
|
||||
* prom[15] == 0x5757, since that appears to be a reliable
|
||||
* test. On the EC2T cards, I get 0x0057 in prom[14,15] instead.
|
||||
*/
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i++)
|
||||
if (prom[i] & 0xff00)
|
||||
break;
|
||||
if (i == ETHER_ADDR_LEN) {
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i++)
|
||||
sc->enaddr[i] = prom[i] & 0xff;
|
||||
} else {
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i += 2) {
|
||||
sc->enaddr[i] = prom[i / 2] & 0xff;
|
||||
sc->enaddr[i + 1] = (prom[i / 2] >> 8) & 0xff;
|
||||
}
|
||||
for (i = 0; i < ETHER_ADDR_LEN; i += 2) {
|
||||
sc->enaddr[i] = prom[i / 2] & 0xff;
|
||||
sc->enaddr[i + 1] = (prom[i / 2] >> 8) & 0xff;
|
||||
}
|
||||
return (0);
|
||||
}
|
||||
|
||||
/*
|
||||
@ -633,30 +616,68 @@ ed_pccard_ax88x90_geteprom(struct ed_softc *sc)
|
||||
static int
|
||||
ed_pccard_ax88x90(device_t dev, const struct ed_product *pp)
|
||||
{
|
||||
int error;
|
||||
int iobase;
|
||||
int error, iobase, i, id;
|
||||
char *ts;
|
||||
struct ed_softc *sc = device_get_softc(dev);
|
||||
|
||||
if (!pp->flags & NE2000DVF_AX88X90)
|
||||
if (!(pp->flags & NE2000DVF_AX88X90))
|
||||
return (ENXIO);
|
||||
|
||||
/* XXX
|
||||
* Set Attribute Memory IOBASE Register. Is this a deficiency in
|
||||
* the PC Card layer, or an ax88190 specific issue? The card
|
||||
* definitely doesn't work without it.
|
||||
if (bootverbose)
|
||||
device_printf(dev, "Checking AX88x90\n");
|
||||
|
||||
/*
|
||||
* Set the IOBASE Register. The AX88x90 cards are potentially
|
||||
* multifunction cards, and thus requires a slight workaround.
|
||||
* We write the address the card is at.
|
||||
*/
|
||||
iobase = rman_get_start(sc->port_res);
|
||||
ed_pccard_memwrite(dev, ED_AX88190_IOBASE0, iobase & 0xff);
|
||||
ed_pccard_memwrite(dev, ED_AX88190_IOBASE1, (iobase >> 8) & 0xff);
|
||||
pccard_ccr_write_1(dev, PCCARD_CCR_IOBASE0, iobase & 0xff);
|
||||
pccard_ccr_write_1(dev, PCCARD_CCR_IOBASE1, (iobase >> 8) & 0xff);
|
||||
|
||||
#ifdef ED_NO_MIIBUS
|
||||
return (ENXIO);
|
||||
#else
|
||||
/*
|
||||
* Check to see if we have a MII PHY ID at any of the first 17
|
||||
* locations. All AX88x90 devices have MII and a PHY, so we use
|
||||
* this to weed out chips that would otherwise make it through
|
||||
* the tests we have after this point.
|
||||
*/
|
||||
sc->mii_readbits = ed_pccard_ax88x90_mii_readbits;
|
||||
sc->mii_writebits = ed_pccard_ax88x90_mii_writebits;
|
||||
for (i = 0; i < 17; i++) {
|
||||
id = ed_miibus_readreg(dev, i, MII_PHYIDR1);
|
||||
if (id != 0 && id != 0xffff)
|
||||
break;
|
||||
}
|
||||
if (i == 17) {
|
||||
sc->mii_readbits = 0;
|
||||
sc->mii_writebits = 0;
|
||||
return (ENXIO);
|
||||
}
|
||||
|
||||
|
||||
ts = "AX88190";
|
||||
if (ed_asic_inb(sc, ED_ASIX_TEST) != 0) {
|
||||
ed_pccard_memwrite(dev, ED_AX88790_CSR, ED_AX88790_CSR_PWRDWN);
|
||||
/*
|
||||
* AX88790 (and I think AX88190A) chips need to be
|
||||
* powered down. There's an erratum that says we should
|
||||
* power down the PHY for 2.5s, but this seems to power
|
||||
* down the whole card. I'm unsure why this was done, but
|
||||
* appears to be required for proper operation.
|
||||
*/
|
||||
pccard_ccr_write_1(dev, PCCARD_CCR_STATUS,
|
||||
PCCARD_CCR_STATUS_PWRDWN);
|
||||
ts = "AX88790";
|
||||
}
|
||||
sc->chip_type = ED_CHIP_TYPE_AX88190;
|
||||
ed_pccard_ax88x90_geteprom(sc);
|
||||
error = ed_pccard_ax88x90_geteprom(sc);
|
||||
if (error)
|
||||
return (error);
|
||||
error = ed_probe_Novell_generic(dev, device_get_flags(dev));
|
||||
if (bootverbose)
|
||||
device_printf(dev, "probe novel returns %d\n", error);
|
||||
if (error == 0) {
|
||||
sc->vendor = ED_VENDOR_NOVELL;
|
||||
sc->type = ED_TYPE_NE2000;
|
||||
@ -664,8 +685,116 @@ ed_pccard_ax88x90(device_t dev, const struct ed_product *pp)
|
||||
sc->type_str = ts;
|
||||
}
|
||||
return (error);
|
||||
#endif
|
||||
}
|
||||
|
||||
#ifndef ED_NO_MIIBUS
|
||||
/* MII bit-twiddling routines for cards using Dlink chipset */
|
||||
#define AX88X90_MIISET(sc, x) ed_asic_outb(sc, ED_AX88X90_MIIBUS, \
|
||||
ed_asic_inb(sc, ED_AX88X90_MIIBUS) | (x))
|
||||
#define AX88X90_MIICLR(sc, x) ed_asic_outb(sc, ED_AX88X90_MIIBUS, \
|
||||
ed_asic_inb(sc, ED_AX88X90_MIIBUS) & ~(x))
|
||||
|
||||
static void
|
||||
ed_pccard_ax88x90_mii_reset(struct ed_softc *sc)
|
||||
{
|
||||
/* Do nothing! */
|
||||
}
|
||||
|
||||
static void
|
||||
ed_pccard_ax88x90_mii_writebits(struct ed_softc *sc, u_int val, int nbits)
|
||||
{
|
||||
int i;
|
||||
|
||||
AX88X90_MIICLR(sc, ED_AX88X90_MII_DIROUT);
|
||||
for (i = nbits - 1; i >= 0; i--) {
|
||||
if ((val >> i) & 1)
|
||||
AX88X90_MIISET(sc, ED_AX88X90_MII_DATAOUT);
|
||||
else
|
||||
AX88X90_MIICLR(sc, ED_AX88X90_MII_DATAOUT);
|
||||
DELAY(10);
|
||||
AX88X90_MIISET(sc, ED_AX88X90_MII_CLK);
|
||||
DELAY(10);
|
||||
AX88X90_MIICLR(sc, ED_AX88X90_MII_CLK);
|
||||
DELAY(10);
|
||||
}
|
||||
}
|
||||
|
||||
static u_int
|
||||
ed_pccard_ax88x90_mii_readbits(struct ed_softc *sc, int nbits)
|
||||
{
|
||||
int i;
|
||||
u_int val = 0;
|
||||
|
||||
AX88X90_MIISET(sc, ED_AX88X90_MII_DIROUT);
|
||||
for (i = nbits - 1; i >= 0; i--) {
|
||||
AX88X90_MIISET(sc, ED_AX88X90_MII_CLK);
|
||||
DELAY(10);
|
||||
val <<= 1;
|
||||
if (ed_asic_inb(sc, ED_AX88X90_MIIBUS) & ED_AX88X90_MII_DATATIN)
|
||||
val++;
|
||||
AX88X90_MIICLR(sc, ED_AX88X90_MII_CLK);
|
||||
DELAY(10);
|
||||
}
|
||||
return val;
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifndef ED_NO_MIIBUS
|
||||
/*
|
||||
* MII bus support routines.
|
||||
*/
|
||||
static int
|
||||
ed_miibus_readreg(device_t dev, int phy, int reg)
|
||||
{
|
||||
struct ed_softc *sc;
|
||||
int failed, val;
|
||||
|
||||
/*
|
||||
* The AX88790 seem to have phy 0..f external, and 0x10 internal.
|
||||
* but they also seem to have a bogus one that shows up at phy
|
||||
* 0x11 through 0x1f.
|
||||
*/
|
||||
if (phy >= 0x11)
|
||||
return (0);
|
||||
|
||||
sc = device_get_softc(dev);
|
||||
(*sc->mii_writebits)(sc, 0xffffffff, 32);
|
||||
(*sc->mii_writebits)(sc, ED_MII_STARTDELIM, ED_MII_STARTDELIM_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_READOP, ED_MII_OP_BITS);
|
||||
(*sc->mii_writebits)(sc, phy, ED_MII_PHY_BITS);
|
||||
(*sc->mii_writebits)(sc, reg, ED_MII_REG_BITS);
|
||||
failed = (*sc->mii_readbits)(sc, ED_MII_ACK_BITS);
|
||||
val = (*sc->mii_readbits)(sc, ED_MII_DATA_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_IDLE, ED_MII_IDLE_BITS);
|
||||
return (failed ? 0 : val);
|
||||
}
|
||||
|
||||
static void
|
||||
ed_miibus_writereg(device_t dev, int phy, int reg, int data)
|
||||
{
|
||||
struct ed_softc *sc;
|
||||
|
||||
/*
|
||||
* The AX88790 seem to have phy 0..f external, and 0x10 internal.
|
||||
* but they also seem to have a bogus one that shows up at phy
|
||||
* 0x11 through 0x1f.
|
||||
*/
|
||||
if (phy >= 0x11)
|
||||
return;
|
||||
|
||||
sc = device_get_softc(dev);
|
||||
(*sc->mii_writebits)(sc, 0xffffffff, 32);
|
||||
(*sc->mii_writebits)(sc, ED_MII_STARTDELIM, ED_MII_STARTDELIM_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_WRITEOP, ED_MII_OP_BITS);
|
||||
(*sc->mii_writebits)(sc, phy, ED_MII_PHY_BITS);
|
||||
(*sc->mii_writebits)(sc, reg, ED_MII_REG_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_TURNAROUND, ED_MII_TURNAROUND_BITS);
|
||||
(*sc->mii_writebits)(sc, data, ED_MII_DATA_BITS);
|
||||
(*sc->mii_writebits)(sc, ED_MII_IDLE, ED_MII_IDLE_BITS);
|
||||
}
|
||||
#endif
|
||||
|
||||
static device_method_t ed_pccard_methods[] = {
|
||||
/* Device interface */
|
||||
DEVMETHOD(device_probe, ed_pccard_probe),
|
||||
|
@ -1072,25 +1072,14 @@ struct ed_ring {
|
||||
#define ED_CHIP_TYPE_DL10019 0x03
|
||||
#define ED_CHIP_TYPE_DL10022 0x04
|
||||
|
||||
/*
|
||||
* AX88190 configuration status register.
|
||||
*/
|
||||
#define ED_AX88790_CSR 0x3c2
|
||||
#define ED_AX88790_CSR_PWRDWN 0x04
|
||||
/*
|
||||
* AX88190 IOBASE registers, I'm pretty sure these don't need to be written
|
||||
* to to make the card work by ed.
|
||||
*/
|
||||
#define ED_AX88190_IOBASE0 0x3ca
|
||||
#define ED_AX88190_IOBASE1 0x3cc
|
||||
|
||||
/*
|
||||
* Test for AX88790 vs 88190 cards.
|
||||
*/
|
||||
#define ED_ASIX_TEST 0x05
|
||||
|
||||
/*
|
||||
* MII bus definitions.
|
||||
* MII bus definitions. These are common to both DL100xx and AX88x90
|
||||
* MII definitions, most likely because they are standards based.
|
||||
*/
|
||||
#define ED_MII_STARTDELIM 0x01
|
||||
#define ED_MII_WRITEOP 0x01
|
||||
@ -1108,13 +1097,21 @@ struct ed_ring {
|
||||
#define ED_MII_IDLE_BITS 1
|
||||
|
||||
/* Dlink chipset used on some Netgear and Dlink PCMCIA cards */
|
||||
#define ED_DL10XX_MIIBUS 0x0c /* MII bus register on ASIC */
|
||||
#define ED_DL100XX_MIIBUS 0x0c /* MII bus register on ASIC */
|
||||
|
||||
#define ED_DL10XX_MII_RESET1 0x04
|
||||
#define ED_DL10XX_MII_RESET2 0x08
|
||||
#define ED_DL100XX_MII_RESET1 0x04
|
||||
#define ED_DL100XX_MII_RESET2 0x08
|
||||
|
||||
#define ED_DL10XX_MII_DATATIN 0x10
|
||||
#define ED_DL10XX_MII_DIROUT_22 0x20
|
||||
#define ED_DL10XX_MII_DIROUT_19 0x10
|
||||
#define ED_DL10XX_MII_DATAOUT 0x40
|
||||
#define ED_DL10XX_MII_CLK 0x80
|
||||
#define ED_DL100XX_MII_DATATIN 0x10
|
||||
#define ED_DL100XX_MII_DIROUT_22 0x20
|
||||
#define ED_DL100XX_MII_DIROUT_19 0x10
|
||||
#define ED_DL100XX_MII_DATAOUT 0x40
|
||||
#define ED_DL100XX_MII_CLK 0x80
|
||||
|
||||
/* AX88x90 based miibus defines */
|
||||
#define ED_AX88X90_MIIBUS 0x04 /* MII bus register on ASIC */
|
||||
|
||||
#define ED_AX88X90_MII_DATAOUT 0x08
|
||||
#define ED_AX88X90_MII_DATATIN 0x04
|
||||
#define ED_AX88X90_MII_DIROUT 0x02
|
||||
#define ED_AX88X90_MII_CLK 0x01
|
||||
|
@ -213,8 +213,6 @@ void ed_shmem_readmem8(struct ed_softc *, bus_size_t, uint8_t *, uint16_t);
|
||||
void ed_pio_readmem(struct ed_softc *, bus_size_t, uint8_t *, uint16_t);
|
||||
void ed_pio_writemem(struct ed_softc *, uint8_t *, uint16_t, uint16_t);
|
||||
#ifndef ED_NO_MIIBUS
|
||||
int ed_miibus_readreg(device_t, int, int);
|
||||
void ed_miibus_writereg(device_t, int, int, int);
|
||||
int ed_ifmedia_upd(struct ifnet *);
|
||||
void ed_ifmedia_sts(struct ifnet *, struct ifmediareq *);
|
||||
void ed_child_detached(device_t, device_t);
|
||||
|
Loading…
Reference in New Issue
Block a user