summaryrefslogtreecommitdiff
path: root/sys/dev/ic/hme.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/ic/hme.c')
-rw-r--r--sys/dev/ic/hme.c149
1 files changed, 102 insertions, 47 deletions
diff --git a/sys/dev/ic/hme.c b/sys/dev/ic/hme.c
index bf844c13f1a..3c58ffca01d 100644
--- a/sys/dev/ic/hme.c
+++ b/sys/dev/ic/hme.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: hme.c,v 1.25 2003/02/08 21:42:11 jason Exp $ */
+/* $OpenBSD: hme.c,v 1.26 2003/03/05 20:46:26 jason Exp $ */
/* $NetBSD: hme.c,v 1.21 2001/07/07 15:59:37 thorpej Exp $ */
/*-
@@ -248,12 +248,17 @@ hme_config(sc)
mii->mii_writereg = hme_mii_writereg;
mii->mii_statchg = hme_mii_statchg;
- ifmedia_init(&mii->mii_media, 0, hme_mediachange, hme_mediastatus);
+ ifmedia_init(&mii->mii_media, IFM_IMASK,
+ hme_mediachange, hme_mediastatus);
hme_mifinit(sc);
- mii_attach(&sc->sc_dev, mii, 0xffffffff,
- MII_PHY_ANY, MII_OFFSET_ANY, 0);
+ if (sc->sc_tcvr == -1)
+ mii_attach(&sc->sc_dev, mii, 0xffffffff, MII_PHY_ANY,
+ MII_OFFSET_ANY, 0);
+ else
+ mii_attach(&sc->sc_dev, mii, 0xffffffff, sc->sc_tcvr,
+ MII_OFFSET_ANY, 0);
child = LIST_FIRST(&mii->mii_phys);
if (child == NULL) {
@@ -371,6 +376,9 @@ hme_stop(sc)
timeout_del(&sc->sc_tick_ch);
mii_down(&sc->sc_mii);
+ /* Mask all interrupts */
+ bus_space_write_4(t, seb, HME_SEBI_IMASK, 0xffffffff);
+
/* Reset transmitter and receiver */
bus_space_write_4(t, seb, HME_SEBI_RESET,
(HME_SEB_RESET_ETX | HME_SEB_RESET_ERX));
@@ -475,7 +483,6 @@ hme_init(sc)
bus_space_handle_t etx = sc->sc_etx;
bus_space_handle_t erx = sc->sc_erx;
bus_space_handle_t mac = sc->sc_mac;
- bus_space_handle_t mif = sc->sc_mif;
u_int8_t *ea;
u_int32_t v;
@@ -591,12 +598,8 @@ hme_init(sc)
/* step 11. XIF Configuration */
v = bus_space_read_4(t, mac, HME_MACI_XIF);
v |= HME_MAC_XIF_OE;
- /* If an external transceiver is connected, enable its MII drivers */
- if ((bus_space_read_4(t, mif, HME_MIFI_CFG) & HME_MIF_CFG_MDI1) != 0)
- v |= HME_MAC_XIF_MIIENABLE;
bus_space_write_4(t, mac, HME_MACI_XIF, v);
-
/* step 12. RX_MAC Configuration Register */
v = bus_space_read_4(t, mac, HME_MACI_RXCFG);
v |= HME_MAC_RXCFG_ENABLE;
@@ -874,17 +877,31 @@ hme_mifinit(sc)
{
bus_space_tag_t t = sc->sc_bustag;
bus_space_handle_t mif = sc->sc_mif;
+ bus_space_handle_t mac = sc->sc_mac;
+ int phy;
u_int32_t v;
- /* Configure the MIF in frame mode */
v = bus_space_read_4(t, mif, HME_MIFI_CFG);
- v &= ~HME_MIF_CFG_BBMODE;
- bus_space_write_4(t, mif, HME_MIFI_CFG, v);
-
+ phy = HME_PHYAD_EXTERNAL;
if (v & HME_MIF_CFG_MDI1)
- sc->sc_tcvr = HME_PHYAD_EXTERNAL;
+ phy = sc->sc_tcvr = HME_PHYAD_EXTERNAL;
else if (v & HME_MIF_CFG_MDI0)
- sc->sc_tcvr = HME_PHYAD_INTERNAL;
+ phy = sc->sc_tcvr = HME_PHYAD_INTERNAL;
+ else
+ sc->sc_tcvr = -1;
+
+ /* Configure the MIF in frame mode, no poll, current phy select */
+ v = 0;
+ if (phy == HME_PHYAD_EXTERNAL)
+ v |= HME_MIF_CFG_PHY;
+ bus_space_write_4(t, mif, HME_MIFI_CFG, v);
+
+ /* If an external transceiver is selected, enable its MII drivers */
+ v = bus_space_read_4(t, mac, HME_MACI_XIF);
+ v &= ~HME_MAC_XIF_MIIENABLE;
+ if (phy == HME_PHYAD_EXTERNAL)
+ v |= HME_MAC_XIF_MIIENABLE;
+ bus_space_write_4(t, mac, HME_MACI_XIF, v);
}
/*
@@ -898,21 +915,28 @@ hme_mii_readreg(self, phy, reg)
struct hme_softc *sc = (struct hme_softc *)self;
bus_space_tag_t t = sc->sc_bustag;
bus_space_handle_t mif = sc->sc_mif;
+ bus_space_handle_t mac = sc->sc_mac;
+ u_int32_t v, xif_cfg, mifi_cfg;
int n;
- u_int32_t v;
- if (sc->sc_tcvr != phy)
+ if (phy != HME_PHYAD_EXTERNAL && phy != HME_PHYAD_INTERNAL)
return (0);
/* Select the desired PHY in the MIF configuration register */
- v = bus_space_read_4(t, mif, HME_MIFI_CFG);
- /* Clear PHY select bit */
+ v = mifi_cfg = bus_space_read_4(t, mif, HME_MIFI_CFG);
v &= ~HME_MIF_CFG_PHY;
if (phy == HME_PHYAD_EXTERNAL)
- /* Set PHY select bit to get at external device */
v |= HME_MIF_CFG_PHY;
bus_space_write_4(t, mif, HME_MIFI_CFG, v);
+ /* Enable MII drivers on external transceiver */
+ v = xif_cfg = bus_space_read_4(t, mac, HME_MACI_XIF);
+ if (phy == HME_PHYAD_EXTERNAL)
+ v |= HME_MAC_XIF_MIIENABLE;
+ else
+ v &= ~HME_MAC_XIF_MIIENABLE;
+ bus_space_write_4(t, mac, HME_MACI_XIF, v);
+
/* Construct the frame command */
v = (MII_COMMAND_START << HME_MIF_FO_ST_SHIFT) |
HME_MIF_FO_TAMSB |
@@ -924,12 +948,21 @@ hme_mii_readreg(self, phy, reg)
for (n = 0; n < 100; n++) {
DELAY(1);
v = bus_space_read_4(t, mif, HME_MIFI_FO);
- if (v & HME_MIF_FO_TALSB)
- return (v & HME_MIF_FO_DATA);
+ if (v & HME_MIF_FO_TALSB) {
+ v &= HME_MIF_FO_DATA;
+ goto out;
+ }
}
+ v = 0;
printf("%s: mii_read timeout\n", sc->sc_dev.dv_xname);
- return (0);
+
+out:
+ /* Restore MIFI_CFG register */
+ bus_space_write_4(t, mif, HME_MIFI_CFG, mifi_cfg);
+ /* Restore XIF register */
+ bus_space_write_4(t, mac, HME_MACI_XIF, xif_cfg);
+ return (v);
}
static void
@@ -940,21 +973,29 @@ hme_mii_writereg(self, phy, reg, val)
struct hme_softc *sc = (void *)self;
bus_space_tag_t t = sc->sc_bustag;
bus_space_handle_t mif = sc->sc_mif;
+ bus_space_handle_t mac = sc->sc_mac;
+ u_int32_t v, xif_cfg, mifi_cfg;
int n;
- u_int32_t v;
- if (sc->sc_tcvr != phy)
+ /* We can at most have two PHYs */
+ if (phy != HME_PHYAD_EXTERNAL && phy != HME_PHYAD_INTERNAL)
return;
/* Select the desired PHY in the MIF configuration register */
- v = bus_space_read_4(t, mif, HME_MIFI_CFG);
- /* Clear PHY select bit */
+ v = mifi_cfg = bus_space_read_4(t, mif, HME_MIFI_CFG);
v &= ~HME_MIF_CFG_PHY;
if (phy == HME_PHYAD_EXTERNAL)
- /* Set PHY select bit to get at external device */
v |= HME_MIF_CFG_PHY;
bus_space_write_4(t, mif, HME_MIFI_CFG, v);
+ /* Enable MII drivers on external transceiver */
+ v = xif_cfg = bus_space_read_4(t, mac, HME_MACI_XIF);
+ if (phy == HME_PHYAD_EXTERNAL)
+ v |= HME_MAC_XIF_MIIENABLE;
+ else
+ v &= ~HME_MAC_XIF_MIIENABLE;
+ bus_space_write_4(t, mac, HME_MACI_XIF, v);
+
/* Construct the frame command */
v = (MII_COMMAND_START << HME_MIF_FO_ST_SHIFT) |
HME_MIF_FO_TAMSB |
@@ -968,10 +1009,15 @@ hme_mii_writereg(self, phy, reg, val)
DELAY(1);
v = bus_space_read_4(t, mif, HME_MIFI_FO);
if (v & HME_MIF_FO_TALSB)
- return;
+ goto out;
}
printf("%s: mii_write timeout\n", sc->sc_dev.dv_xname);
+out:
+ /* Restore MIFI_CFG register */
+ bus_space_write_4(t, mif, HME_MIFI_CFG, mifi_cfg);
+ /* Restore XIF register */
+ bus_space_write_4(t, mac, HME_MACI_XIF, xif_cfg);
}
static void
@@ -979,26 +1025,18 @@ hme_mii_statchg(dev)
struct device *dev;
{
struct hme_softc *sc = (void *)dev;
- int instance = IFM_INST(sc->sc_mii.mii_media.ifm_cur->ifm_media);
- int phy = sc->sc_phys[instance];
bus_space_tag_t t = sc->sc_bustag;
- bus_space_handle_t mif = sc->sc_mif;
bus_space_handle_t mac = sc->sc_mac;
u_int32_t v;
#ifdef HMEDEBUG
if (sc->sc_debug)
- printf("hme_mii_statchg: status change: phy = %d\n", phy);
+ printf("hme_mii_statchg: status change\n", phy);
#endif
- /* Select the current PHY in the MIF configuration register */
- v = bus_space_read_4(t, mif, HME_MIFI_CFG);
- v &= ~HME_MIF_CFG_PHY;
- if (phy == HME_PHYAD_EXTERNAL)
- v |= HME_MIF_CFG_PHY;
- bus_space_write_4(t, mif, HME_MIFI_CFG, v);
-
/* Set the MAC Full Duplex bit appropriately */
+ /* Apparently the hme chip is SIMPLE if working in full duplex mode,
+ but not otherwise. */
v = bus_space_read_4(t, mac, HME_MACI_TXCFG);
if ((IFM_OPTIONS(sc->sc_mii.mii_media_active) & IFM_FDX) != 0) {
v |= HME_MAC_TXCFG_FULLDPLX;
@@ -1008,13 +1046,6 @@ hme_mii_statchg(dev)
sc->sc_arpcom.ac_if.if_flags &= ~IFF_SIMPLEX;
}
bus_space_write_4(t, mac, HME_MACI_TXCFG, v);
-
- /* If an external transceiver is selected, enable its MII drivers */
- v = bus_space_read_4(t, mac, HME_MACI_XIF);
- v &= ~HME_MAC_XIF_MIIENABLE;
- if (phy == HME_PHYAD_EXTERNAL)
- v |= HME_MAC_XIF_MIIENABLE;
- bus_space_write_4(t, mac, HME_MACI_XIF, v);
}
int
@@ -1022,10 +1053,34 @@ hme_mediachange(ifp)
struct ifnet *ifp;
{
struct hme_softc *sc = ifp->if_softc;
+ bus_space_tag_t t = sc->sc_bustag;
+ bus_space_handle_t mif = sc->sc_mif;
+ bus_space_handle_t mac = sc->sc_mac;
+ int instance = IFM_INST(sc->sc_mii.mii_media.ifm_cur->ifm_media);
+ int phy = sc->sc_phys[instance];
+ u_int32_t v;
+#ifdef HMEDEBUG
+ if (sc->sc_debug)
+ printf("hme_mediachange: phy = %d\n", phy);
+#endif
if (IFM_TYPE(sc->sc_media.ifm_media) != IFM_ETHER)
return (EINVAL);
+ /* Select the current PHY in the MIF configuration register */
+ v = bus_space_read_4(t, mif, HME_MIFI_CFG);
+ v &= ~HME_MIF_CFG_PHY;
+ if (phy == HME_PHYAD_EXTERNAL)
+ v |= HME_MIF_CFG_PHY;
+ bus_space_write_4(t, mif, HME_MIFI_CFG, v);
+
+ /* If an external transceiver is selected, enable its MII drivers */
+ v = bus_space_read_4(t, mac, HME_MACI_XIF);
+ v &= ~HME_MAC_XIF_MIIENABLE;
+ if (phy == HME_PHYAD_EXTERNAL)
+ v |= HME_MAC_XIF_MIIENABLE;
+ bus_space_write_4(t, mac, HME_MACI_XIF, v);
+
return (mii_mediachg(&sc->sc_mii));
}