summaryrefslogtreecommitdiff
path: root/sys/dev/fdt/if_mvneta.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/fdt/if_mvneta.c')
-rw-r--r--sys/dev/fdt/if_mvneta.c100
1 files changed, 93 insertions, 7 deletions
diff --git a/sys/dev/fdt/if_mvneta.c b/sys/dev/fdt/if_mvneta.c
index 0e9d542c321..ee0920bbf35 100644
--- a/sys/dev/fdt/if_mvneta.c
+++ b/sys/dev/fdt/if_mvneta.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_mvneta.c,v 1.13 2020/07/10 13:26:36 patrick Exp $ */
+/* $OpenBSD: if_mvneta.c,v 1.14 2020/11/29 13:00:23 kettenis Exp $ */
/* $NetBSD: if_mvneta.c,v 1.41 2015/04/15 10:15:40 hsuenaga Exp $ */
/*
* Copyright (c) 2007, 2008, 2013 KIYOHARA Takashi
@@ -41,6 +41,7 @@
#include <sys/mbuf.h>
#include <machine/bus.h>
+#include <machine/cpufunc.h>
#include <machine/fdt.h>
#include <dev/ofw/openfirm.h>
@@ -159,6 +160,8 @@ struct mvneta_softc {
PHY_MODE_SGMII,
PHY_MODE_RGMII,
PHY_MODE_RGMII_ID,
+ PHY_MODE_1000BASEX,
+ PHY_MODE_2500BASEX,
} sc_phy_mode;
int sc_fixed_link;
int sc_inband_status;
@@ -166,6 +169,7 @@ struct mvneta_softc {
int sc_phyloc;
int sc_link;
int sc_sfp;
+ int sc_node;
};
@@ -268,6 +272,7 @@ mvneta_miibus_statchg(struct device *self)
void
mvneta_inband_statchg(struct mvneta_softc *sc)
{
+ uint64_t subtype = IFM_SUBTYPE(sc->sc_mii.mii_media_active);
uint32_t reg;
sc->sc_mii.mii_media_status = IFM_AVALID;
@@ -276,7 +281,11 @@ mvneta_inband_statchg(struct mvneta_softc *sc)
reg = MVNETA_READ(sc, MVNETA_PS0);
if (reg & MVNETA_PS0_LINKUP)
sc->sc_mii.mii_media_status |= IFM_ACTIVE;
- if (reg & MVNETA_PS0_GMIISPEED)
+ if (sc->sc_phy_mode == PHY_MODE_2500BASEX)
+ sc->sc_mii.mii_media_active |= subtype;
+ else if (sc->sc_phy_mode == PHY_MODE_1000BASEX)
+ sc->sc_mii.mii_media_active |= subtype;
+ else if (reg & MVNETA_PS0_GMIISPEED)
sc->sc_mii.mii_media_active |= IFM_1000_T;
else if (reg & MVNETA_PS0_MIISPEED)
sc->sc_mii.mii_media_active |= IFM_100_TX;
@@ -344,6 +353,64 @@ mvneta_wininit(struct mvneta_softc *sc)
MVNETA_WRITE(sc, MVNETA_BARE, en);
}
+#define COMPHY_SIP_POWER_ON 0x82000001
+#define COMPHY_SIP_POWER_OFF 0x82000002
+#define COMPHY_SPEED(x) ((x) << 2)
+#define COMPHY_SPEED_1_25G 0 /* SGMII 1G */
+#define COMPHY_SPEED_2_5G 1
+#define COMPHY_SPEED_3_125G 2 /* SGMII 2.5G */
+#define COMPHY_SPEED_5G 3
+#define COMPHY_SPEED_5_15625G 4 /* XFI 5G */
+#define COMPHY_SPEED_6G 5
+#define COMPHY_SPEED_10_3125G 6 /* XFI 10G */
+#define COMPHY_UNIT(x) ((x) << 8)
+#define COMPHY_MODE(x) ((x) << 12)
+#define COMPHY_MODE_SATA 1
+#define COMPHY_MODE_SGMII 2 /* SGMII 1G */
+#define COMPHY_MODE_HS_SGMII 3 /* SGMII 2.5G */
+#define COMPHY_MODE_USB3H 4
+#define COMPHY_MODE_USB3D 5
+#define COMPHY_MODE_PCIE 6
+#define COMPHY_MODE_RXAUI 7
+#define COMPHY_MODE_XFI 8
+#define COMPHY_MODE_SFI 9
+#define COMPHY_MODE_USB3 10
+
+void
+mvneta_comphy_init(struct mvneta_softc *sc)
+{
+ int node, phys[2], lane, unit;
+ uint32_t mode;
+
+ if (OF_getpropintarray(sc->sc_node, "phys", phys, sizeof(phys)) !=
+ sizeof(phys))
+ return;
+ node = OF_getnodebyphandle(phys[0]);
+ if (!node)
+ return;
+
+ lane = OF_getpropint(node, "reg", 0);
+ unit = phys[1];
+
+ switch (sc->sc_phy_mode) {
+ case PHY_MODE_1000BASEX:
+ case PHY_MODE_SGMII:
+ mode = COMPHY_MODE(COMPHY_MODE_SGMII) |
+ COMPHY_SPEED(COMPHY_SPEED_1_25G) |
+ COMPHY_UNIT(unit);
+ break;
+ case PHY_MODE_2500BASEX:
+ mode = COMPHY_MODE(COMPHY_MODE_HS_SGMII) |
+ COMPHY_SPEED(COMPHY_SPEED_3_125G) |
+ COMPHY_UNIT(unit);
+ break;
+ default:
+ return;
+ }
+
+ smc_call(COMPHY_SIP_POWER_ON, lane, mode, 0);
+}
+
int
mvneta_match(struct device *parent, void *cfdata, void *aux)
{
@@ -358,7 +425,7 @@ mvneta_attach(struct device *parent, struct device *self, void *aux)
{
struct mvneta_softc *sc = (struct mvneta_softc *) self;
struct fdt_attach_args *faa = aux;
- uint32_t ctl0, ctl2, panc;
+ uint32_t ctl0, ctl2, ctl4, panc;
struct ifnet *ifp;
int i, len, node;
char *phy_mode;
@@ -374,6 +441,7 @@ mvneta_attach(struct device *parent, struct device *self, void *aux)
return;
}
sc->sc_dmat = faa->fa_dmat;
+ sc->sc_node = faa->fa_node;
clock_enable(faa->fa_node, NULL);
@@ -395,6 +463,10 @@ mvneta_attach(struct device *parent, struct device *self, void *aux)
sc->sc_phy_mode = PHY_MODE_RGMII_ID;
else if (!strncmp(phy_mode, "rgmii", strlen("rgmii")))
sc->sc_phy_mode = PHY_MODE_RGMII;
+ else if (!strncmp(phy_mode, "1000base-x", strlen("1000base-x")))
+ sc->sc_phy_mode = PHY_MODE_1000BASEX;
+ else if (!strncmp(phy_mode, "2500base-x", strlen("2500base-x")))
+ sc->sc_phy_mode = PHY_MODE_2500BASEX;
else {
printf("%s: cannot use phy-mode %s\n", self->dv_xname,
phy_mode);
@@ -532,6 +604,7 @@ mvneta_attach(struct device *parent, struct device *self, void *aux)
/* Setup phy. */
ctl0 = MVNETA_READ(sc, MVNETA_PMACC0);
ctl2 = MVNETA_READ(sc, MVNETA_PMACC2);
+ ctl4 = MVNETA_READ(sc, MVNETA_PMACC4);
panc = MVNETA_READ(sc, MVNETA_PANC);
/* Force link down to change in-band settings. */
@@ -539,8 +612,11 @@ mvneta_attach(struct device *parent, struct device *self, void *aux)
panc |= MVNETA_PANC_FORCELINKFAIL;
MVNETA_WRITE(sc, MVNETA_PANC, panc);
+ mvneta_comphy_init(sc);
+
ctl0 &= ~MVNETA_PMACC0_PORTTYPE;
ctl2 &= ~(MVNETA_PMACC2_PORTMACRESET | MVNETA_PMACC2_INBANDAN);
+ ctl4 &= ~(MVNETA_PMACC4_SHORT_PREAMBLE);
panc &= ~(MVNETA_PANC_INBANDANEN | MVNETA_PANC_INBANDRESTARTAN |
MVNETA_PANC_SETMIISPEED | MVNETA_PANC_SETGMIISPEED |
MVNETA_PANC_ANSPEEDEN | MVNETA_PANC_SETFCEN |
@@ -559,6 +635,17 @@ mvneta_attach(struct device *parent, struct device *self, void *aux)
MVNETA_SERDESCFG_SGMII_PROTO);
ctl2 |= MVNETA_PMACC2_PCSEN;
break;
+ case PHY_MODE_1000BASEX:
+ MVNETA_WRITE(sc, MVNETA_SERDESCFG,
+ MVNETA_SERDESCFG_SGMII_PROTO);
+ ctl2 |= MVNETA_PMACC2_PCSEN;
+ break;
+ case PHY_MODE_2500BASEX:
+ MVNETA_WRITE(sc, MVNETA_SERDESCFG,
+ MVNETA_SERDESCFG_HSGMII_PROTO);
+ ctl2 |= MVNETA_PMACC2_PCSEN;
+ ctl4 |= MVNETA_PMACC4_SHORT_PREAMBLE;
+ break;
default:
break;
}
@@ -590,6 +677,7 @@ mvneta_attach(struct device *parent, struct device *self, void *aux)
MVNETA_WRITE(sc, MVNETA_PMACC0, ctl0);
MVNETA_WRITE(sc, MVNETA_PMACC2, ctl2);
+ MVNETA_WRITE(sc, MVNETA_PMACC4, ctl4);
MVNETA_WRITE(sc, MVNETA_PANC, panc);
/* Port reset */
@@ -661,10 +749,8 @@ mvneta_attach_deferred(struct device *self)
} else
ifmedia_set(&sc->sc_mii.mii_media, IFM_ETHER|IFM_AUTO);
} else {
- ifmedia_add(&sc->sc_mii.mii_media,
- IFM_ETHER|IFM_MANUAL, 0, NULL);
- ifmedia_set(&sc->sc_mii.mii_media,
- IFM_ETHER|IFM_MANUAL);
+ ifmedia_add(&sc->sc_mii.mii_media, IFM_ETHER|IFM_AUTO, 0, NULL);
+ ifmedia_set(&sc->sc_mii.mii_media, IFM_ETHER|IFM_AUTO);
if (sc->sc_inband_status) {
mvneta_inband_statchg(sc);