a321935999
than VLAN groups. Some chips (eg this rtl8366rb) has a VLAN group per port - you first define a set of VLANs in a vlan group, then you assign a VLAN group to a port. Other chips (eg the AR8xxx switch chips) have a VLAN ID array per port - there's no group per se, just a list of vlans that can be configured. So for now, the switch API will use the latter and rely on drivers doing the heavy lifting if one wishes to use the VLAN group method. Maybe later on both can be supported. PR: kern/177878 PR: kern/177873 Submitted by: Luiz Otavio O Souza <loos.br@gmail.com> Reviewed by: ray
770 lines
21 KiB
C
770 lines
21 KiB
C
/*-
|
|
* Copyright (c) 2011-2012 Stefan Bethke.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*
|
|
* $FreeBSD$
|
|
*/
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/bus.h>
|
|
#include <sys/errno.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/module.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/sockio.h>
|
|
#include <sys/sysctl.h>
|
|
#include <sys/systm.h>
|
|
|
|
#include <net/if.h>
|
|
#include <net/if_arp.h>
|
|
#include <net/ethernet.h>
|
|
#include <net/if_dl.h>
|
|
#include <net/if_media.h>
|
|
#include <net/if_types.h>
|
|
|
|
#include <machine/bus.h>
|
|
#include <dev/iicbus/iic.h>
|
|
#include <dev/iicbus/iiconf.h>
|
|
#include <dev/iicbus/iicbus.h>
|
|
#include <dev/mii/mii.h>
|
|
#include <dev/mii/miivar.h>
|
|
|
|
#include <dev/etherswitch/etherswitch.h>
|
|
#include <dev/etherswitch/rtl8366/rtl8366rbvar.h>
|
|
|
|
#include "iicbus_if.h"
|
|
#include "miibus_if.h"
|
|
#include "etherswitch_if.h"
|
|
|
|
|
|
struct rtl8366rb_softc {
|
|
struct mtx sc_mtx; /* serialize access to softc */
|
|
int smi_acquired; /* serialize access to SMI/I2C bus */
|
|
struct mtx callout_mtx; /* serialize callout */
|
|
device_t dev;
|
|
int vid[RTL8366RB_NUM_VLANS];
|
|
char *ifname[RTL8366RB_NUM_PHYS];
|
|
device_t miibus[RTL8366RB_NUM_PHYS];
|
|
struct ifnet *ifp[RTL8366RB_NUM_PHYS];
|
|
struct callout callout_tick;
|
|
};
|
|
|
|
static etherswitch_info_t etherswitch_info = {
|
|
.es_nports = RTL8366RB_NUM_PORTS,
|
|
.es_nvlangroups = RTL8366RB_NUM_VLANS,
|
|
.es_name = "Realtek RTL8366RB"
|
|
};
|
|
|
|
#define RTL_LOCK(_sc) mtx_lock(&(_sc)->sc_mtx)
|
|
#define RTL_UNLOCK(_sc) mtx_unlock(&(_sc)->sc_mtx)
|
|
#define RTL_LOCK_ASSERT(_sc, _what) mtx_assert(&(_s)c->sc_mtx, (_what))
|
|
#define RTL_TRYLOCK(_sc) mtx_trylock(&(_sc)->sc_mtx)
|
|
|
|
#define RTL_WAITOK 0
|
|
#define RTL_NOWAIT 1
|
|
|
|
#define RTL_SMI_ACQUIRED 1
|
|
#define RTL_SMI_ACQUIRED_ASSERT(_sc) \
|
|
KASSERT((_sc)->smi_acquired == RTL_SMI_ACQUIRED, ("smi must be acquired @%s", __FUNCTION__))
|
|
|
|
#if defined(DEBUG)
|
|
#define DPRINTF(dev, args...) device_printf(dev, args)
|
|
#define DEVERR(dev, err, fmt, args...) do { \
|
|
if (err != 0) device_printf(dev, fmt, err, args); \
|
|
} while (0)
|
|
#define DEBUG_INCRVAR(var) do { \
|
|
var++; \
|
|
} while (0)
|
|
|
|
static int callout_blocked = 0;
|
|
static int iic_select_retries = 0;
|
|
static int phy_access_retries = 0;
|
|
static SYSCTL_NODE(_debug, OID_AUTO, rtl8366rb, CTLFLAG_RD, 0, "rtl8366rb");
|
|
SYSCTL_INT(_debug_rtl8366rb, OID_AUTO, callout_blocked, CTLFLAG_RW, &callout_blocked, 0,
|
|
"number of times the callout couldn't acquire the bus");
|
|
SYSCTL_INT(_debug_rtl8366rb, OID_AUTO, iic_select_retries, CTLFLAG_RW, &iic_select_retries, 0,
|
|
"number of times the I2C bus selection had to be retried");
|
|
SYSCTL_INT(_debug_rtl8366rb, OID_AUTO, phy_access_retries, CTLFLAG_RW, &phy_access_retries, 0,
|
|
"number of times PHY register access had to be retried");
|
|
#else
|
|
#define DPRINTF(dev, args...)
|
|
#define DEVERR(dev, err, fmt, args...)
|
|
#define DEBUG_INCRVAR(var)
|
|
#endif
|
|
|
|
static int smi_probe(device_t dev);
|
|
static int smi_read(device_t dev, uint16_t addr, uint16_t *data, int sleep);
|
|
static int smi_write(device_t dev, uint16_t addr, uint16_t data, int sleep);
|
|
static int smi_rmw(device_t dev, uint16_t addr, uint16_t mask, uint16_t data, int sleep);
|
|
static void rtl8366rb_tick(void *arg);
|
|
static int rtl8366rb_ifmedia_upd(struct ifnet *);
|
|
static void rtl8366rb_ifmedia_sts(struct ifnet *, struct ifmediareq *);
|
|
|
|
static void
|
|
rtl8366rb_identify(driver_t *driver, device_t parent)
|
|
{
|
|
device_t child;
|
|
struct iicbus_ivar *devi;
|
|
|
|
if (device_find_child(parent, "rtl8366rb", -1) == NULL) {
|
|
child = BUS_ADD_CHILD(parent, 0, "rtl8366rb", -1);
|
|
devi = IICBUS_IVAR(child);
|
|
devi->addr = RTL8366RB_IIC_ADDR;
|
|
}
|
|
}
|
|
|
|
static int
|
|
rtl8366rb_probe(device_t dev)
|
|
{
|
|
if (smi_probe(dev) != 0)
|
|
return (ENXIO);
|
|
device_set_desc(dev, "RTL8366RB Ethernet Switch Controller");
|
|
return (BUS_PROBE_DEFAULT);
|
|
}
|
|
|
|
static void
|
|
rtl8366rb_init(device_t dev)
|
|
{
|
|
/* Initialisation for TL-WR1043ND */
|
|
smi_rmw(dev, RTL8366RB_RCR,
|
|
RTL8366RB_RCR_HARD_RESET,
|
|
RTL8366RB_RCR_HARD_RESET, RTL_WAITOK);
|
|
DELAY(100000);
|
|
/* Enable 16 VLAN mode */
|
|
smi_rmw(dev, RTL8366RB_SGCR,
|
|
RTL8366RB_SGCR_EN_VLAN | RTL8366RB_SGCR_EN_VLAN_4KTB,
|
|
RTL8366RB_SGCR_EN_VLAN, RTL_WAITOK);
|
|
/* remove port 0 form VLAN 0 */
|
|
smi_rmw(dev, RTL8366RB_VMCR(RTL8366RB_VMCR_MU_REG, 0),
|
|
(1 << 0), 0, RTL_WAITOK);
|
|
/* add port 0 untagged and port 5 tagged to VLAN 1 */
|
|
smi_rmw(dev, RTL8366RB_VMCR(RTL8366RB_VMCR_MU_REG, 1),
|
|
((1 << 5 | 1 << 0) << RTL8366RB_VMCR_MU_MEMBER_SHIFT)
|
|
| ((1 << 5 | 1 << 0) << RTL8366RB_VMCR_MU_UNTAG_SHIFT),
|
|
((1 << 5 | 1 << 0) << RTL8366RB_VMCR_MU_MEMBER_SHIFT
|
|
| ((1 << 0) << RTL8366RB_VMCR_MU_UNTAG_SHIFT)),
|
|
RTL_WAITOK);
|
|
/* set PVLAN 1 for port 0 */
|
|
smi_rmw(dev, RTL8366RB_PVCR_REG(0),
|
|
RTL8366RB_PVCR_VAL(0, RTL8366RB_PVCR_PORT_MASK),
|
|
RTL8366RB_PVCR_VAL(0, 1), RTL_WAITOK);
|
|
}
|
|
|
|
static int
|
|
rtl8366rb_attach(device_t dev)
|
|
{
|
|
uint16_t rev = 0;
|
|
struct rtl8366rb_softc *sc;
|
|
char name[IFNAMSIZ];
|
|
int err = 0;
|
|
int i;
|
|
|
|
sc = device_get_softc(dev);
|
|
bzero(sc, sizeof(*sc));
|
|
sc->dev = dev;
|
|
mtx_init(&sc->sc_mtx, "rtl8366rb", NULL, MTX_DEF);
|
|
sc->smi_acquired = 0;
|
|
mtx_init(&sc->callout_mtx, "rtl8366rbcallout", NULL, MTX_DEF);
|
|
|
|
rtl8366rb_init(dev);
|
|
smi_read(dev, RTL8366RB_CVCR, &rev, RTL_WAITOK);
|
|
device_printf(dev, "rev. %d\n", rev & 0x000f);
|
|
|
|
/* attach miibus and phys */
|
|
/* PHYs need an interface, so we generate a dummy one */
|
|
for (i = 0; i < RTL8366RB_NUM_PHYS; i++) {
|
|
sc->ifp[i] = if_alloc(IFT_ETHER);
|
|
sc->ifp[i]->if_softc = sc;
|
|
sc->ifp[i]->if_flags |= IFF_UP | IFF_BROADCAST | IFF_DRV_RUNNING
|
|
| IFF_SIMPLEX;
|
|
snprintf(name, IFNAMSIZ, "%sport", device_get_nameunit(dev));
|
|
sc->ifname[i] = malloc(strlen(name)+1, M_DEVBUF, M_WAITOK);
|
|
bcopy(name, sc->ifname[i], strlen(name)+1);
|
|
if_initname(sc->ifp[i], sc->ifname[i], i);
|
|
err = mii_attach(dev, &sc->miibus[i], sc->ifp[i], rtl8366rb_ifmedia_upd, \
|
|
rtl8366rb_ifmedia_sts, BMSR_DEFCAPMASK, \
|
|
i, MII_OFFSET_ANY, 0);
|
|
if (err != 0) {
|
|
device_printf(dev, "attaching PHY %d failed\n", i);
|
|
return (err);
|
|
}
|
|
}
|
|
|
|
bus_generic_probe(dev);
|
|
bus_enumerate_hinted_children(dev);
|
|
err = bus_generic_attach(dev);
|
|
if (err != 0)
|
|
return (err);
|
|
|
|
callout_init_mtx(&sc->callout_tick, &sc->callout_mtx, 0);
|
|
rtl8366rb_tick(sc);
|
|
|
|
return (err);
|
|
}
|
|
|
|
static int
|
|
rtl8366rb_detach(device_t dev)
|
|
{
|
|
struct rtl8366rb_softc *sc = device_get_softc(dev);
|
|
int i;
|
|
|
|
for (i=0; i < RTL8366RB_NUM_PHYS; i++) {
|
|
if (sc->miibus[i])
|
|
device_delete_child(dev, sc->miibus[i]);
|
|
if (sc->ifp[i] != NULL)
|
|
if_free(sc->ifp[i]);
|
|
free(sc->ifname[i], M_DEVBUF);
|
|
}
|
|
bus_generic_detach(dev);
|
|
callout_drain(&sc->callout_tick);
|
|
mtx_destroy(&sc->callout_mtx);
|
|
mtx_destroy(&sc->sc_mtx);
|
|
|
|
return (0);
|
|
}
|
|
|
|
static void
|
|
rtl8366rb_update_ifmedia(int portstatus, u_int *media_status, u_int *media_active)
|
|
{
|
|
*media_active = IFM_ETHER;
|
|
*media_status = IFM_AVALID;
|
|
if ((portstatus & RTL8366RB_PLSR_LINK) != 0)
|
|
*media_status |= IFM_ACTIVE;
|
|
else {
|
|
*media_active |= IFM_NONE;
|
|
return;
|
|
}
|
|
switch (portstatus & RTL8366RB_PLSR_SPEED_MASK) {
|
|
case RTL8366RB_PLSR_SPEED_10:
|
|
*media_active |= IFM_10_T;
|
|
break;
|
|
case RTL8366RB_PLSR_SPEED_100:
|
|
*media_active |= IFM_100_TX;
|
|
break;
|
|
case RTL8366RB_PLSR_SPEED_1000:
|
|
*media_active |= IFM_1000_T;
|
|
break;
|
|
}
|
|
if ((portstatus & RTL8366RB_PLSR_FULLDUPLEX) == 0)
|
|
*media_active |= IFM_FDX;
|
|
else
|
|
*media_active |= IFM_HDX;
|
|
if ((portstatus & RTL8366RB_PLSR_TXPAUSE) != 0)
|
|
*media_active |= IFM_ETH_TXPAUSE;
|
|
if ((portstatus & RTL8366RB_PLSR_RXPAUSE) != 0)
|
|
*media_active |= IFM_ETH_RXPAUSE;
|
|
}
|
|
|
|
static void
|
|
rtl833rb_miipollstat(struct rtl8366rb_softc *sc)
|
|
{
|
|
int i;
|
|
struct mii_data *mii;
|
|
struct mii_softc *miisc;
|
|
uint16_t value;
|
|
int portstatus;
|
|
|
|
for (i = 0; i < RTL8366RB_NUM_PHYS; i++) {
|
|
mii = device_get_softc(sc->miibus[i]);
|
|
if ((i % 2) == 0) {
|
|
if (smi_read(sc->dev, RTL8366RB_PLSR_BASE + i/2, &value, RTL_NOWAIT) != 0) {
|
|
DEBUG_INCRVAR(callout_blocked);
|
|
return;
|
|
}
|
|
portstatus = value & 0xff;
|
|
} else {
|
|
portstatus = (value >> 8) & 0xff;
|
|
}
|
|
rtl8366rb_update_ifmedia(portstatus, &mii->mii_media_status, &mii->mii_media_active);
|
|
LIST_FOREACH(miisc, &mii->mii_phys, mii_list) {
|
|
if (IFM_INST(mii->mii_media.ifm_cur->ifm_media) != miisc->mii_inst)
|
|
continue;
|
|
mii_phy_update(miisc, MII_POLLSTAT);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
rtl8366rb_tick(void *arg)
|
|
{
|
|
struct rtl8366rb_softc *sc = arg;
|
|
|
|
rtl833rb_miipollstat(sc);
|
|
callout_reset(&sc->callout_tick, hz, rtl8366rb_tick, sc);
|
|
}
|
|
|
|
static int
|
|
smi_probe(device_t dev)
|
|
{
|
|
device_t iicbus, iicha;
|
|
int err, i;
|
|
uint16_t chipid;
|
|
char bytes[2];
|
|
int xferd;
|
|
|
|
bytes[0] = RTL8366RB_CIR & 0xff;
|
|
bytes[1] = (RTL8366RB_CIR >> 8) & 0xff;
|
|
iicbus = device_get_parent(dev);
|
|
iicha = device_get_parent(iicbus);
|
|
iicbus_reset(iicbus, IIC_FASTEST, RTL8366RB_IIC_ADDR, NULL);
|
|
for (i=3; i--; ) {
|
|
IICBUS_STOP(iicha);
|
|
/*
|
|
* we go directly to the host adapter because iicbus.c
|
|
* only issues a stop on a bus that was successfully started.
|
|
*/
|
|
}
|
|
err = iicbus_request_bus(iicbus, dev, IIC_WAIT);
|
|
if (err != 0)
|
|
goto out;
|
|
err = iicbus_start(iicbus, RTL8366RB_IIC_ADDR | RTL_IICBUS_READ, RTL_IICBUS_TIMEOUT);
|
|
if (err != 0)
|
|
goto out;
|
|
err = iicbus_write(iicbus, bytes, 2, &xferd, RTL_IICBUS_TIMEOUT);
|
|
if (err != 0)
|
|
goto out;
|
|
err = iicbus_read(iicbus, bytes, 2, &xferd, IIC_LAST_READ, 0);
|
|
if (err != 0)
|
|
goto out;
|
|
chipid = ((bytes[1] & 0xff) << 8) | (bytes[0] & 0xff);
|
|
DPRINTF(dev, "chip id 0x%04x\n", chipid);
|
|
if (chipid != RTL8366RB_CIR_ID8366RB)
|
|
err = ENXIO;
|
|
out:
|
|
iicbus_stop(iicbus);
|
|
iicbus_release_bus(iicbus, dev);
|
|
return (err == 0 ? 0 : ENXIO);
|
|
}
|
|
|
|
static int
|
|
smi_acquire(struct rtl8366rb_softc *sc, int sleep)
|
|
{
|
|
int r = 0;
|
|
if (sleep == RTL_WAITOK)
|
|
RTL_LOCK(sc);
|
|
else
|
|
if (RTL_TRYLOCK(sc) == 0)
|
|
return (EWOULDBLOCK);
|
|
if (sc->smi_acquired == RTL_SMI_ACQUIRED)
|
|
r = EBUSY;
|
|
else {
|
|
r = iicbus_request_bus(device_get_parent(sc->dev), sc->dev, \
|
|
sleep == RTL_WAITOK ? IIC_WAIT : IIC_DONTWAIT);
|
|
if (r == 0)
|
|
sc->smi_acquired = RTL_SMI_ACQUIRED;
|
|
}
|
|
RTL_UNLOCK(sc);
|
|
return (r);
|
|
}
|
|
|
|
static int
|
|
smi_release(struct rtl8366rb_softc *sc, int sleep)
|
|
{
|
|
if (sleep == RTL_WAITOK)
|
|
RTL_LOCK(sc);
|
|
else
|
|
if (RTL_TRYLOCK(sc) == 0)
|
|
return (EWOULDBLOCK);
|
|
RTL_SMI_ACQUIRED_ASSERT(sc);
|
|
iicbus_release_bus(device_get_parent(sc->dev), sc->dev);
|
|
sc->smi_acquired = 0;
|
|
RTL_UNLOCK(sc);
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
smi_select(device_t dev, int op, int sleep)
|
|
{
|
|
int err, i;
|
|
device_t iicbus = device_get_parent(dev);
|
|
struct iicbus_ivar *devi = IICBUS_IVAR(dev);
|
|
int slave = devi->addr;
|
|
|
|
RTL_SMI_ACQUIRED_ASSERT((struct rtl8366rb_softc *)device_get_softc(dev));
|
|
/*
|
|
* The chip does not use clock stretching when it is busy,
|
|
* instead ignoring the command. Retry a few times.
|
|
*/
|
|
for (i = RTL_IICBUS_RETRIES; i--; ) {
|
|
err = iicbus_start(iicbus, slave | op, RTL_IICBUS_TIMEOUT);
|
|
if (err != IIC_ENOACK)
|
|
break;
|
|
if (sleep == RTL_WAITOK) {
|
|
DEBUG_INCRVAR(iic_select_retries);
|
|
pause("smi_select", RTL_IICBUS_RETRY_SLEEP);
|
|
} else
|
|
break;
|
|
}
|
|
return (err);
|
|
}
|
|
|
|
static int
|
|
smi_read_locked(struct rtl8366rb_softc *sc, uint16_t addr, uint16_t *data, int sleep)
|
|
{
|
|
int err;
|
|
device_t iicbus = device_get_parent(sc->dev);
|
|
char bytes[2];
|
|
int xferd;
|
|
|
|
RTL_SMI_ACQUIRED_ASSERT(sc);
|
|
bytes[0] = addr & 0xff;
|
|
bytes[1] = (addr >> 8) & 0xff;
|
|
err = smi_select(sc->dev, RTL_IICBUS_READ, sleep);
|
|
if (err != 0)
|
|
goto out;
|
|
err = iicbus_write(iicbus, bytes, 2, &xferd, RTL_IICBUS_TIMEOUT);
|
|
if (err != 0)
|
|
goto out;
|
|
err = iicbus_read(iicbus, bytes, 2, &xferd, IIC_LAST_READ, 0);
|
|
if (err != 0)
|
|
goto out;
|
|
*data = ((bytes[1] & 0xff) << 8) | (bytes[0] & 0xff);
|
|
|
|
out:
|
|
iicbus_stop(iicbus);
|
|
return (err);
|
|
}
|
|
|
|
static int
|
|
smi_write_locked(struct rtl8366rb_softc *sc, uint16_t addr, uint16_t data, int sleep)
|
|
{
|
|
int err;
|
|
device_t iicbus = device_get_parent(sc->dev);
|
|
char bytes[4];
|
|
int xferd;
|
|
|
|
RTL_SMI_ACQUIRED_ASSERT(sc);
|
|
bytes[0] = addr & 0xff;
|
|
bytes[1] = (addr >> 8) & 0xff;
|
|
bytes[2] = data & 0xff;
|
|
bytes[3] = (data >> 8) & 0xff;
|
|
|
|
err = smi_select(sc->dev, RTL_IICBUS_WRITE, sleep);
|
|
if (err == 0)
|
|
err = iicbus_write(iicbus, bytes, 4, &xferd, RTL_IICBUS_TIMEOUT);
|
|
iicbus_stop(iicbus);
|
|
|
|
return (err);
|
|
}
|
|
|
|
static int
|
|
smi_read(device_t dev, uint16_t addr, uint16_t *data, int sleep)
|
|
{
|
|
struct rtl8366rb_softc *sc = device_get_softc(dev);
|
|
int err;
|
|
|
|
err = smi_acquire(sc, sleep);
|
|
if (err != 0)
|
|
return (EBUSY);
|
|
err = smi_read_locked(sc, addr, data, sleep);
|
|
smi_release(sc, sleep);
|
|
DEVERR(dev, err, "smi_read()=%d: addr=%04x\n", addr);
|
|
return (err == 0 ? 0 : EIO);
|
|
}
|
|
|
|
static int
|
|
smi_write(device_t dev, uint16_t addr, uint16_t data, int sleep)
|
|
{
|
|
struct rtl8366rb_softc *sc = device_get_softc(dev);
|
|
int err;
|
|
|
|
err = smi_acquire(sc, sleep);
|
|
if (err != 0)
|
|
return (EBUSY);
|
|
err = smi_write_locked(sc, addr, data, sleep);
|
|
smi_release(sc, sleep);
|
|
DEVERR(dev, err, "smi_write()=%d: addr=%04x\n", addr);
|
|
return (err == 0 ? 0 : EIO);
|
|
}
|
|
|
|
static int
|
|
smi_rmw(device_t dev, uint16_t addr, uint16_t mask, uint16_t data, int sleep)
|
|
{
|
|
struct rtl8366rb_softc *sc = device_get_softc(dev);
|
|
int err;
|
|
uint16_t oldv, newv;
|
|
|
|
err = smi_acquire(sc, sleep);
|
|
if (err != 0)
|
|
return (EBUSY);
|
|
if (err == 0) {
|
|
err = smi_read_locked(sc, addr, &oldv, sleep);
|
|
if (err == 0) {
|
|
newv = oldv & ~mask;
|
|
newv |= data & mask;
|
|
if (newv != oldv)
|
|
err = smi_write_locked(sc, addr, newv, sleep);
|
|
}
|
|
}
|
|
smi_release(sc, sleep);
|
|
DEVERR(dev, err, "smi_rmw()=%d: addr=%04x\n", addr);
|
|
return (err == 0 ? 0 : EIO);
|
|
}
|
|
|
|
static etherswitch_info_t *
|
|
rtl_getinfo(device_t dev)
|
|
{
|
|
return (ðerswitch_info);
|
|
}
|
|
|
|
static int
|
|
rtl_readreg(device_t dev, int reg)
|
|
{
|
|
uint16_t data = 0;
|
|
|
|
smi_read(dev, reg, &data, RTL_WAITOK);
|
|
return (data);
|
|
}
|
|
|
|
static int
|
|
rtl_writereg(device_t dev, int reg, int value)
|
|
{
|
|
return (smi_write(dev, reg, value, RTL_WAITOK));
|
|
}
|
|
|
|
static int
|
|
rtl_getport(device_t dev, etherswitch_port_t *p)
|
|
{
|
|
struct rtl8366rb_softc *sc;
|
|
struct ifmedia *ifm;
|
|
struct mii_data *mii;
|
|
struct ifmediareq *ifmr = &p->es_ifmr;
|
|
uint16_t v;
|
|
int err, vlangroup;
|
|
|
|
if (p->es_port < 0 || p->es_port >= RTL8366RB_NUM_PORTS)
|
|
return (ENXIO);
|
|
sc = device_get_softc(dev);
|
|
vlangroup = RTL8366RB_PVCR_GET(p->es_port,
|
|
rtl_readreg(dev, RTL8366RB_PVCR_REG(p->es_port)));
|
|
p->es_pvid = sc->vid[vlangroup];
|
|
|
|
if (p->es_port < RTL8366RB_NUM_PHYS) {
|
|
mii = device_get_softc(sc->miibus[p->es_port]);
|
|
ifm = &mii->mii_media;
|
|
err = ifmedia_ioctl(sc->ifp[p->es_port], &p->es_ifr, ifm, SIOCGIFMEDIA);
|
|
if (err)
|
|
return (err);
|
|
} else {
|
|
/* fill in fixed values for CPU port */
|
|
ifmr->ifm_count = 0;
|
|
smi_read(dev, RTL8366RB_PLSR_BASE + (RTL8366RB_NUM_PHYS)/2, &v, RTL_WAITOK);
|
|
v = v >> (8 * ((RTL8366RB_NUM_PHYS) % 2));
|
|
rtl8366rb_update_ifmedia(v, &ifmr->ifm_status, &ifmr->ifm_active);
|
|
ifmr->ifm_current = ifmr->ifm_active;
|
|
ifmr->ifm_mask = 0;
|
|
ifmr->ifm_status = IFM_ACTIVE | IFM_AVALID;
|
|
}
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
rtl_setport(device_t dev, etherswitch_port_t *p)
|
|
{
|
|
int i, err, vlangroup;
|
|
struct rtl8366rb_softc *sc;
|
|
struct ifmedia *ifm;
|
|
struct mii_data *mii;
|
|
|
|
if (p->es_port < 0 || p->es_port >= RTL8366RB_NUM_PHYS)
|
|
return (ENXIO);
|
|
sc = device_get_softc(dev);
|
|
vlangroup = -1;
|
|
for (i = 0; i < RTL8366RB_NUM_VLANS; i++) {
|
|
if (sc->vid[i] == p->es_pvid) {
|
|
vlangroup = i;
|
|
break;
|
|
}
|
|
}
|
|
if (vlangroup == -1)
|
|
return (ENXIO);
|
|
err = smi_rmw(dev, RTL8366RB_PVCR_REG(p->es_port),
|
|
RTL8366RB_PVCR_VAL(p->es_port, RTL8366RB_PVCR_PORT_MASK),
|
|
RTL8366RB_PVCR_VAL(p->es_port, vlangroup), RTL_WAITOK);
|
|
if (err)
|
|
return (err);
|
|
mii = device_get_softc(sc->miibus[p->es_port]);
|
|
ifm = &mii->mii_media;
|
|
err = ifmedia_ioctl(sc->ifp[p->es_port], &p->es_ifr, ifm, SIOCSIFMEDIA);
|
|
return (err);
|
|
}
|
|
|
|
static int
|
|
rtl_getvgroup(device_t dev, etherswitch_vlangroup_t *vg)
|
|
{
|
|
uint16_t vmcr[3];
|
|
int i;
|
|
|
|
for (i=0; i<3; i++)
|
|
vmcr[i] = rtl_readreg(dev, RTL8366RB_VMCR(i, vg->es_vlangroup));
|
|
|
|
vg->es_vid = RTL8366RB_VMCR_VID(vmcr);
|
|
vg->es_member_ports = RTL8366RB_VMCR_MEMBER(vmcr);
|
|
vg->es_untagged_ports = RTL8366RB_VMCR_UNTAG(vmcr);
|
|
vg->es_fid = RTL8366RB_VMCR_FID(vmcr);
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
rtl_setvgroup(device_t dev, etherswitch_vlangroup_t *vg)
|
|
{
|
|
struct rtl8366rb_softc *sc;
|
|
int g = vg->es_vlangroup;
|
|
|
|
sc = device_get_softc(dev);
|
|
sc->vid[g] = vg->es_vid;
|
|
rtl_writereg(dev, RTL8366RB_VMCR(RTL8366RB_VMCR_DOT1Q_REG, g),
|
|
(vg->es_vid << RTL8366RB_VMCR_DOT1Q_VID_SHIFT) & RTL8366RB_VMCR_DOT1Q_VID_MASK);
|
|
rtl_writereg(dev, RTL8366RB_VMCR(RTL8366RB_VMCR_MU_REG, g),
|
|
((vg->es_member_ports << RTL8366RB_VMCR_MU_MEMBER_SHIFT) & RTL8366RB_VMCR_MU_MEMBER_MASK) |
|
|
((vg->es_untagged_ports << RTL8366RB_VMCR_MU_UNTAG_SHIFT) & RTL8366RB_VMCR_MU_UNTAG_MASK));
|
|
rtl_writereg(dev, RTL8366RB_VMCR(RTL8366RB_VMCR_FID_REG, g),
|
|
vg->es_fid);
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
rtl_readphy(device_t dev, int phy, int reg)
|
|
{
|
|
struct rtl8366rb_softc *sc = device_get_softc(dev);
|
|
uint16_t data = 0;
|
|
int err, i, sleep;
|
|
|
|
if (phy < 0 || phy >= RTL8366RB_NUM_PHYS)
|
|
return (ENXIO);
|
|
if (reg < 0 || reg >= RTL8366RB_NUM_PHY_REG)
|
|
return (ENXIO);
|
|
sleep = RTL_WAITOK;
|
|
err = smi_acquire(sc, sleep);
|
|
if (err != 0)
|
|
return (EBUSY);
|
|
for (i = RTL_IICBUS_RETRIES; i--; ) {
|
|
err = smi_write_locked(sc, RTL8366RB_PACR, RTL8366RB_PACR_READ, sleep);
|
|
if (err == 0)
|
|
err = smi_write_locked(sc, RTL8366RB_PHYREG(phy, 0, reg), 0, sleep);
|
|
if (err == 0) {
|
|
err = smi_read_locked(sc, RTL8366RB_PADR, &data, sleep);
|
|
break;
|
|
}
|
|
DEBUG_INCRVAR(phy_access_retries);
|
|
DPRINTF(dev, "rtl_readphy(): chip not responsive, retrying %d more times\n", i);
|
|
pause("rtl_readphy", RTL_IICBUS_RETRY_SLEEP);
|
|
}
|
|
smi_release(sc, sleep);
|
|
DEVERR(dev, err, "rtl_readphy()=%d: phy=%d.%02x\n", phy, reg);
|
|
return (data);
|
|
}
|
|
|
|
static int
|
|
rtl_writephy(device_t dev, int phy, int reg, int data)
|
|
{
|
|
struct rtl8366rb_softc *sc = device_get_softc(dev);
|
|
int err, i, sleep;
|
|
|
|
if (phy < 0 || phy >= RTL8366RB_NUM_PHYS)
|
|
return (ENXIO);
|
|
if (reg < 0 || reg >= RTL8366RB_NUM_PHY_REG)
|
|
return (ENXIO);
|
|
sleep = RTL_WAITOK;
|
|
err = smi_acquire(sc, sleep);
|
|
if (err != 0)
|
|
return (EBUSY);
|
|
for (i = RTL_IICBUS_RETRIES; i--; ) {
|
|
err = smi_write_locked(sc, RTL8366RB_PACR, RTL8366RB_PACR_WRITE, sleep);
|
|
if (err == 0)
|
|
err = smi_write_locked(sc, RTL8366RB_PHYREG(phy, 0, reg), data, sleep);
|
|
if (err == 0) {
|
|
break;
|
|
}
|
|
DEBUG_INCRVAR(phy_access_retries);
|
|
DPRINTF(dev, "rtl_writephy(): chip not responsive, retrying %d more tiems\n", i);
|
|
pause("rtl_writephy", RTL_IICBUS_RETRY_SLEEP);
|
|
}
|
|
smi_release(sc, sleep);
|
|
DEVERR(dev, err, "rtl_writephy()=%d: phy=%d.%02x\n", phy, reg);
|
|
return (err == 0 ? 0 : EIO);
|
|
}
|
|
|
|
static int
|
|
rtl8366rb_ifmedia_upd(struct ifnet *ifp)
|
|
{
|
|
struct rtl8366rb_softc *sc = ifp->if_softc;
|
|
struct mii_data *mii = device_get_softc(sc->miibus[ifp->if_dunit]);
|
|
|
|
mii_mediachg(mii);
|
|
return (0);
|
|
}
|
|
|
|
static void
|
|
rtl8366rb_ifmedia_sts(struct ifnet *ifp, struct ifmediareq *ifmr)
|
|
{
|
|
struct rtl8366rb_softc *sc = ifp->if_softc;
|
|
struct mii_data *mii = device_get_softc(sc->miibus[ifp->if_dunit]);
|
|
|
|
mii_pollstat(mii);
|
|
ifmr->ifm_active = mii->mii_media_active;
|
|
ifmr->ifm_status = mii->mii_media_status;
|
|
}
|
|
|
|
|
|
static device_method_t rtl8366rb_methods[] = {
|
|
/* Device interface */
|
|
DEVMETHOD(device_identify, rtl8366rb_identify),
|
|
DEVMETHOD(device_probe, rtl8366rb_probe),
|
|
DEVMETHOD(device_attach, rtl8366rb_attach),
|
|
DEVMETHOD(device_detach, rtl8366rb_detach),
|
|
|
|
/* bus interface */
|
|
DEVMETHOD(bus_add_child, device_add_child_ordered),
|
|
|
|
/* MII interface */
|
|
DEVMETHOD(miibus_readreg, rtl_readphy),
|
|
DEVMETHOD(miibus_writereg, rtl_writephy),
|
|
|
|
/* etherswitch interface */
|
|
DEVMETHOD(etherswitch_getinfo, rtl_getinfo),
|
|
DEVMETHOD(etherswitch_readreg, rtl_readreg),
|
|
DEVMETHOD(etherswitch_writereg, rtl_writereg),
|
|
DEVMETHOD(etherswitch_readphyreg, rtl_readphy),
|
|
DEVMETHOD(etherswitch_writephyreg, rtl_writephy),
|
|
DEVMETHOD(etherswitch_getport, rtl_getport),
|
|
DEVMETHOD(etherswitch_setport, rtl_setport),
|
|
DEVMETHOD(etherswitch_getvgroup, rtl_getvgroup),
|
|
DEVMETHOD(etherswitch_setvgroup, rtl_setvgroup),
|
|
|
|
DEVMETHOD_END
|
|
};
|
|
|
|
DEFINE_CLASS_0(rtl8366rb, rtl8366rb_driver, rtl8366rb_methods,
|
|
sizeof(struct rtl8366rb_softc));
|
|
static devclass_t rtl8366rb_devclass;
|
|
|
|
DRIVER_MODULE(rtl8366rb, iicbus, rtl8366rb_driver, rtl8366rb_devclass, 0, 0);
|
|
DRIVER_MODULE(miibus, rtl8366rb, miibus_driver, miibus_devclass, 0, 0);
|
|
DRIVER_MODULE(etherswitch, rtl8366rb, etherswitch_driver, etherswitch_devclass, 0, 0);
|
|
MODULE_VERSION(rtl8366rb, 1);
|
|
MODULE_DEPEND(rtl8366rb, iicbus, 1, 1, 1); /* XXX which versions? */
|
|
MODULE_DEPEND(rtl8366rb, miibus, 1, 1, 1); /* XXX which versions? */
|
|
MODULE_DEPEND(rtl8366rb, etherswitch, 1, 1, 1); /* XXX which versions? */
|