NetBSD/sys/dev/usb/if_mos.c
2020-01-29 06:17:07 +00:00

814 lines
20 KiB
C

/* $NetBSD: if_mos.c,v 1.4 2020/01/29 06:26:32 thorpej Exp $ */
/* $OpenBSD: if_mos.c,v 1.40 2019/07/07 06:40:10 kevlo Exp $ */
/*
* Copyright (c) 2008 Johann Christian Rode <jcrode@gmx.net>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
/*
* Copyright (c) 2005, 2006, 2007 Jonathan Gray <jsg@openbsd.org>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
/*
* Copyright (c) 1997, 1998, 1999, 2000-2003
* Bill Paul <wpaul@windriver.com>. 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.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by Bill Paul.
* 4. Neither the name of the author nor the names of any co-contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY Bill Paul 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 Bill Paul OR THE VOICES IN HIS HEAD
* 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.
*/
/*
* Moschip MCS7730/MCS7830/MCS7832 USB to Ethernet controller
* The datasheet is available at the following URL:
* http://www.moschip.com/data/products/MCS7830/Data%20Sheet_7830.pdf
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: if_mos.c,v 1.4 2020/01/29 06:26:32 thorpej Exp $");
#include <sys/param.h>
#include <dev/usb/usbnet.h>
#include <dev/usb/if_mosreg.h>
#define MOS_PAUSE_REWRITES 3
#define MOS_TIMEOUT 1000
#define MOS_RX_LIST_CNT 1
#define MOS_TX_LIST_CNT 1
/* Maximum size of a fast ethernet frame plus one byte for the status */
#define MOS_BUFSZ (ETHER_MAX_LEN+1)
/*
* USB endpoints.
*/
#define MOS_ENDPT_RX 0
#define MOS_ENDPT_TX 1
#define MOS_ENDPT_INTR 2
#define MOS_ENDPT_MAX 3
/*
* USB vendor requests.
*/
#define MOS_UR_READREG 0x0e
#define MOS_UR_WRITEREG 0x0d
#define MOS_CONFIG_NO 1
#define MOS_IFACE_IDX 0
struct mos_type {
struct usb_devno mos_dev;
u_int16_t mos_flags;
#define MCS7730 0x0001 /* MCS7730 */
#define MCS7830 0x0002 /* MCS7830 */
#define MCS7832 0x0004 /* MCS7832 */
};
#define MOS_INC(x, y) (x) = (x + 1) % y
#ifdef MOS_DEBUG
#define DPRINTF(x) do { if (mosdebug) printf x; } while (0)
#define DPRINTFN(n,x) do { if (mosdebug >= (n)) printf x; } while (0)
int mosdebug = 0;
#else
#define DPRINTF(x) __nothing
#define DPRINTFN(n,x) __nothing
#endif
/*
* Various supported device vendors/products.
*/
static const struct mos_type mos_devs[] = {
{ { USB_VENDOR_MOSCHIP, USB_PRODUCT_MOSCHIP_MCS7730 }, MCS7730 },
{ { USB_VENDOR_MOSCHIP, USB_PRODUCT_MOSCHIP_MCS7830 }, MCS7830 },
{ { USB_VENDOR_MOSCHIP, USB_PRODUCT_MOSCHIP_MCS7832 }, MCS7832 },
{ { USB_VENDOR_SITECOMEU, USB_PRODUCT_SITECOMEU_LN030 }, MCS7830 },
};
#define mos_lookup(v, p) ((const struct mos_type *)usb_lookup(mos_devs, v, p))
static int mos_match(device_t, cfdata_t, void *);
static void mos_attach(device_t, device_t, void *);
CFATTACH_DECL_NEW(mos, sizeof(struct usbnet),
mos_match, mos_attach, usbnet_detach, usbnet_activate);
static void mos_rx_loop(struct usbnet *, struct usbnet_chain *, uint32_t);
static unsigned mos_tx_prepare(struct usbnet *, struct mbuf *,
struct usbnet_chain *);
static int mos_ioctl(struct ifnet *, u_long, void *);
static int mos_init(struct ifnet *);
static void mos_chip_init(struct usbnet *);
static void mos_stop(struct ifnet *ifp, int disable);
static int mos_mii_read_reg(struct usbnet *, int, int, uint16_t *);
static int mos_mii_write_reg(struct usbnet *, int, int, uint16_t);
static void mos_mii_statchg(struct ifnet *);
static void mos_reset(struct usbnet *);
static int mos_reg_read_1(struct usbnet *, int);
static int mos_reg_read_2(struct usbnet *, int);
static int mos_reg_write_1(struct usbnet *, int, int);
static int mos_reg_write_2(struct usbnet *, int, int);
static int mos_readmac(struct usbnet *);
static int mos_writemac(struct usbnet *);
static int mos_write_mcast(struct usbnet *, uint8_t *);
static const struct usbnet_ops mos_ops = {
.uno_stop = mos_stop,
.uno_ioctl = mos_ioctl,
.uno_read_reg = mos_mii_read_reg,
.uno_write_reg = mos_mii_write_reg,
.uno_statchg = mos_mii_statchg,
.uno_tx_prepare = mos_tx_prepare,
.uno_rx_loop = mos_rx_loop,
.uno_init = mos_init,
};
static int
mos_reg_read_1(struct usbnet *un, int reg)
{
usb_device_request_t req;
usbd_status err;
uByte val = 0;
if (usbnet_isdying(un))
return 0;
req.bmRequestType = UT_READ_VENDOR_DEVICE;
req.bRequest = MOS_UR_READREG;
USETW(req.wValue, 0);
USETW(req.wIndex, reg);
USETW(req.wLength, 1);
err = usbd_do_request(un->un_udev, &req, &val);
if (err) {
aprint_error_dev(un->un_dev, "read reg %x\n", reg);
return 0;
}
return val;
}
static int
mos_reg_read_2(struct usbnet *un, int reg)
{
usb_device_request_t req;
usbd_status err;
uWord val;
if (usbnet_isdying(un))
return 0;
USETW(val,0);
req.bmRequestType = UT_READ_VENDOR_DEVICE;
req.bRequest = MOS_UR_READREG;
USETW(req.wValue, 0);
USETW(req.wIndex, reg);
USETW(req.wLength, 2);
err = usbd_do_request(un->un_udev, &req, &val);
if (err) {
aprint_error_dev(un->un_dev, "read reg2 %x\n", reg);
return 0;
}
return UGETW(val);
}
static int
mos_reg_write_1(struct usbnet *un, int reg, int aval)
{
usb_device_request_t req;
usbd_status err;
uByte val;
if (usbnet_isdying(un))
return 0;
val = aval;
req.bmRequestType = UT_WRITE_VENDOR_DEVICE;
req.bRequest = MOS_UR_WRITEREG;
USETW(req.wValue, 0);
USETW(req.wIndex, reg);
USETW(req.wLength, 1);
err = usbd_do_request(un->un_udev, &req, &val);
if (err)
aprint_error_dev(un->un_dev, "write reg %x <- %x\n",
reg, aval);
return 0;
}
static int
mos_reg_write_2(struct usbnet *un, int reg, int aval)
{
usb_device_request_t req;
usbd_status err;
uWord val;
USETW(val, aval);
if (usbnet_isdying(un))
return EIO;
req.bmRequestType = UT_WRITE_VENDOR_DEVICE;
req.bRequest = MOS_UR_WRITEREG;
USETW(req.wValue, 0);
USETW(req.wIndex, reg);
USETW(req.wLength, 2);
err = usbd_do_request(un->un_udev, &req, &val);
if (err)
aprint_error_dev(un->un_dev, "write reg2 %x <- %x\n",
reg, aval);
return 0;
}
static int
mos_readmac(struct usbnet *un)
{
usb_device_request_t req;
usbd_status err;
if (usbnet_isdying(un))
return 0;
req.bmRequestType = UT_READ_VENDOR_DEVICE;
req.bRequest = MOS_UR_READREG;
USETW(req.wValue, 0);
USETW(req.wIndex, MOS_MAC);
USETW(req.wLength, ETHER_ADDR_LEN);
err = usbd_do_request(un->un_udev, &req, un->un_eaddr);
if (err)
aprint_error_dev(un->un_dev, "%s: failed", __func__);
return err;
}
static int
mos_writemac(struct usbnet *un)
{
usb_device_request_t req;
usbd_status err;
if (usbnet_isdying(un))
return 0;
req.bmRequestType = UT_WRITE_VENDOR_DEVICE;
req.bRequest = MOS_UR_WRITEREG;
USETW(req.wValue, 0);
USETW(req.wIndex, MOS_MAC);
USETW(req.wLength, ETHER_ADDR_LEN);
err = usbd_do_request(un->un_udev, &req, un->un_eaddr);
if (err)
aprint_error_dev(un->un_dev, "%s: failed", __func__);
return 0;
}
static int
mos_write_mcast(struct usbnet *un, uint8_t *hashtbl)
{
usb_device_request_t req;
usbd_status err;
if (usbnet_isdying(un))
return EIO;
req.bmRequestType = UT_WRITE_VENDOR_DEVICE;
req.bRequest = MOS_UR_WRITEREG;
USETW(req.wValue, 0);
USETW(req.wIndex, MOS_MCAST_TABLE);
USETW(req.wLength, 8);
err = usbd_do_request(un->un_udev, &req, hashtbl);
if (err) {
aprint_error_dev(un->un_dev, "%s: failed", __func__);
return(-1);
}
return 0;
}
static int
mos_mii_read_reg(struct usbnet *un, int phy, int reg, uint16_t *val)
{
int i, res;
mos_reg_write_2(un, MOS_PHY_DATA, 0);
mos_reg_write_1(un, MOS_PHY_CTL, (phy & MOS_PHYCTL_PHYADDR) |
MOS_PHYCTL_READ);
mos_reg_write_1(un, MOS_PHY_STS, (reg & MOS_PHYSTS_PHYREG) |
MOS_PHYSTS_PENDING);
for (i = 0; i < MOS_TIMEOUT; i++) {
if (mos_reg_read_1(un, MOS_PHY_STS) & MOS_PHYSTS_READY)
break;
}
if (i == MOS_TIMEOUT) {
aprint_error_dev(un->un_dev, "read PHY failed\n");
return EIO;
}
res = mos_reg_read_2(un, MOS_PHY_DATA);
*val = res;
DPRINTFN(10,("%s: %s: phy %d reg %d val %u\n",
device_xname(un->un_dev), __func__, phy, reg, res));
return 0;
}
static int
mos_mii_write_reg(struct usbnet *un, int phy, int reg, uint16_t val)
{
int i;
DPRINTFN(10,("%s: %s: phy %d reg %d val %u\n",
device_xname(un->un_dev), __func__, phy, reg, val));
mos_reg_write_2(un, MOS_PHY_DATA, val);
mos_reg_write_1(un, MOS_PHY_CTL, (phy & MOS_PHYCTL_PHYADDR) |
MOS_PHYCTL_WRITE);
mos_reg_write_1(un, MOS_PHY_STS, (reg & MOS_PHYSTS_PHYREG) |
MOS_PHYSTS_PENDING);
for (i = 0; i < MOS_TIMEOUT; i++) {
if (mos_reg_read_1(un, MOS_PHY_STS) & MOS_PHYSTS_READY)
break;
}
if (i == MOS_TIMEOUT) {
aprint_error_dev(un->un_dev, "write PHY failed\n");
return EIO;
}
return 0;
}
void
mos_mii_statchg(struct ifnet *ifp)
{
struct usbnet * const un = ifp->if_softc;
struct mii_data * const mii = usbnet_mii(un);
int val, err;
if (usbnet_isdying(un))
return;
DPRINTFN(10,("%s: %s: enter\n", device_xname(un->un_dev), __func__));
usbnet_lock_mii(un);
/* disable RX, TX prior to changing FDX, SPEEDSEL */
val = mos_reg_read_1(un, MOS_CTL);
val &= ~(MOS_CTL_TX_ENB | MOS_CTL_RX_ENB);
mos_reg_write_1(un, MOS_CTL, val);
/* reset register which counts dropped frames */
mos_reg_write_1(un, MOS_FRAME_DROP_CNT, 0);
if ((mii->mii_media_active & IFM_GMASK) == IFM_FDX)
val |= MOS_CTL_FDX_ENB;
else
val &= ~(MOS_CTL_FDX_ENB);
if ((mii->mii_media_status & (IFM_ACTIVE | IFM_AVALID)) ==
(IFM_ACTIVE | IFM_AVALID)) {
switch (IFM_SUBTYPE(mii->mii_media_active)) {
case IFM_100_TX:
val |= MOS_CTL_SPEEDSEL;
break;
case IFM_10_T:
val &= ~(MOS_CTL_SPEEDSEL);
break;
}
usbnet_set_link(un, true);
}
/* re-enable TX, RX */
val |= (MOS_CTL_TX_ENB | MOS_CTL_RX_ENB);
err = mos_reg_write_1(un, MOS_CTL, val);
usbnet_unlock_mii(un);
if (err)
aprint_error_dev(un->un_dev, "media change failed\n");
}
static void
mos_setiff_locked(struct usbnet *un)
{
struct ifnet *ifp = usbnet_ifp(un);
struct ethercom *ec = usbnet_ec(un);
struct ether_multi *enm;
struct ether_multistep step;
u_int32_t h = 0;
u_int8_t rxmode, hashtbl[8] = { 0, 0, 0, 0, 0, 0, 0, 0 };
if (usbnet_isdying(un))
return;
rxmode = mos_reg_read_1(un, MOS_CTL);
rxmode &= ~(MOS_CTL_ALLMULTI | MOS_CTL_RX_PROMISC);
ETHER_LOCK(ec);
if (ifp->if_flags & IFF_PROMISC) {
allmulti:
ec->ec_flags |= ETHER_F_ALLMULTI;
rxmode |= MOS_CTL_ALLMULTI;
if (ifp->if_flags & IFF_PROMISC)
rxmode |= MOS_CTL_RX_PROMISC;
} else {
/* now program new ones */
ec->ec_flags &= ~ETHER_F_ALLMULTI;
ETHER_FIRST_MULTI(step, ec, enm);
while (enm != NULL) {
if (memcmp(enm->enm_addrlo, enm->enm_addrhi,
ETHER_ADDR_LEN)) {
memset(hashtbl, 0, sizeof(hashtbl));
goto allmulti;
}
h = ether_crc32_be(enm->enm_addrlo,
ETHER_ADDR_LEN) >> 26;
hashtbl[h / 8] |= 1 << (h % 8);
ETHER_NEXT_MULTI(step, enm);
}
}
ETHER_UNLOCK(ec);
/*
* The datasheet claims broadcast frames were always accepted
* regardless of filter settings. But the hardware seems to
* filter broadcast frames, so pass them explicitly.
*/
h = ether_crc32_be(etherbroadcastaddr, ETHER_ADDR_LEN) >> 26;
hashtbl[h / 8] |= 1 << (h % 8);
mos_write_mcast(un, hashtbl);
mos_reg_write_1(un, MOS_CTL, rxmode);
}
static void
mos_setiff(struct usbnet *un)
{
usbnet_lock_mii(un);
mos_setiff_locked(un);
usbnet_unlock_mii(un);
}
static void
mos_reset(struct usbnet *un)
{
u_int8_t ctl;
if (usbnet_isdying(un))
return;
ctl = mos_reg_read_1(un, MOS_CTL);
ctl &= ~(MOS_CTL_RX_PROMISC | MOS_CTL_ALLMULTI | MOS_CTL_TX_ENB |
MOS_CTL_RX_ENB);
/* Disable RX, TX, promiscuous and allmulticast mode */
mos_reg_write_1(un, MOS_CTL, ctl);
/* Reset frame drop counter register to zero */
mos_reg_write_1(un, MOS_FRAME_DROP_CNT, 0);
/* Wait a little while for the chip to get its brains in order. */
DELAY(1000);
}
void
mos_chip_init(struct usbnet *un)
{
int i;
/*
* Rev.C devices have a pause threshold register which needs to be set
* at startup.
*/
if (mos_reg_read_1(un, MOS_PAUSE_TRHD) != -1) {
for (i = 0; i < MOS_PAUSE_REWRITES; i++)
mos_reg_write_1(un, MOS_PAUSE_TRHD, 0);
}
}
/*
* Probe for a MCS7x30 chip.
*/
static int
mos_match(device_t parent, cfdata_t match, void *aux)
{
struct usb_attach_arg *uaa = aux;
return (mos_lookup(uaa->uaa_vendor, uaa->uaa_product) != NULL ?
UMATCH_VENDOR_PRODUCT : UMATCH_NONE);
}
/*
* Attach the interface.
*/
static void
mos_attach(device_t parent, device_t self, void *aux)
{
USBNET_MII_DECL_DEFAULT(unm);
struct usbnet * un = device_private(self);
struct usb_attach_arg *uaa = aux;
struct usbd_device *dev = uaa->uaa_device;
usbd_status err;
usb_interface_descriptor_t *id;
usb_endpoint_descriptor_t *ed;
char *devinfop;
int i;
aprint_naive("\n");
aprint_normal("\n");
devinfop = usbd_devinfo_alloc(dev, 0);
aprint_normal_dev(self, "%s\n", devinfop);
usbd_devinfo_free(devinfop);
un->un_dev = self;
un->un_udev = dev;
un->un_sc = un;
un->un_ops = &mos_ops;
un->un_rx_xfer_flags = USBD_SHORT_XFER_OK;
un->un_tx_xfer_flags = USBD_FORCE_SHORT_XFER;
un->un_rx_list_cnt = MOS_RX_LIST_CNT;
un->un_tx_list_cnt = MOS_TX_LIST_CNT;
un->un_rx_bufsz = un->un_tx_bufsz = MOS_BUFSZ;
err = usbd_set_config_no(dev, MOS_CONFIG_NO, 1);
if (err) {
aprint_error_dev(self, "failed to set configuration"
", err=%s\n", usbd_errstr(err));
return;
}
err = usbd_device2interface_handle(dev, MOS_IFACE_IDX, &un->un_iface);
if (err) {
aprint_error_dev(self, "failed getting interface handle"
", err=%s\n", usbd_errstr(err));
return;
}
un->un_flags = mos_lookup(uaa->uaa_vendor, uaa->uaa_product)->mos_flags;
id = usbd_get_interface_descriptor(un->un_iface);
/* Find endpoints. */
for (i = 0; i < id->bNumEndpoints; i++) {
ed = usbd_interface2endpoint_descriptor(un->un_iface, i);
if (!ed) {
aprint_error_dev(self, "couldn't get ep %d\n", i);
return;
}
if (UE_GET_DIR(ed->bEndpointAddress) == UE_DIR_IN &&
UE_GET_XFERTYPE(ed->bmAttributes) == UE_BULK) {
un->un_ed[USBNET_ENDPT_RX] = ed->bEndpointAddress;
} else if (UE_GET_DIR(ed->bEndpointAddress) == UE_DIR_OUT &&
UE_GET_XFERTYPE(ed->bmAttributes) == UE_BULK) {
un->un_ed[USBNET_ENDPT_TX] = ed->bEndpointAddress;
} else if (UE_GET_DIR(ed->bEndpointAddress) == UE_DIR_IN &&
UE_GET_XFERTYPE(ed->bmAttributes) == UE_INTERRUPT) {
un->un_ed[USBNET_ENDPT_INTR] = ed->bEndpointAddress;
}
}
if (un->un_flags & MCS7730)
aprint_normal_dev(self, "MCS7730\n");
else if (un->un_flags & MCS7830)
aprint_normal_dev(self, "MCS7830\n");
else if (un->un_flags & MCS7832)
aprint_normal_dev(self, "MCS7832\n");
/* Set these up now for register access. */
usbnet_attach(un, "mosdet");
mos_chip_init(un);
/*
* Read MAC address, inform the world.
*/
err = mos_readmac(un);
if (err) {
aprint_error_dev(self, "couldn't read MAC address\n");
return;
}
struct ifnet *ifp = usbnet_ifp(un);
ifp->if_capabilities = ETHERCAP_VLAN_MTU;
usbnet_attach_ifp(un, IFF_SIMPLEX | IFF_BROADCAST | IFF_MULTICAST,
0, &unm);
}
/*
* A frame has been uploaded: pass the resulting mbuf chain up to
* the higher level protocols.
*/
void
mos_rx_loop(struct usbnet * un, struct usbnet_chain *c, uint32_t total_len)
{
struct ifnet *ifp = usbnet_ifp(un);
uint8_t *buf = c->unc_buf;
u_int8_t rxstat;
u_int16_t pktlen = 0;
DPRINTFN(5,("%s: %s: enter len %u\n",
device_xname(un->un_dev), __func__, total_len));
if (total_len <= 1)
return;
/* evaluate status byte at the end */
pktlen = total_len - 1;
if (pktlen > un->un_rx_bufsz) {
if_statinc(ifp, if_ierrors);
return;
}
rxstat = buf[pktlen] & MOS_RXSTS_MASK;
if (rxstat != MOS_RXSTS_VALID) {
DPRINTF(("%s: erroneous frame received: ",
device_xname(un->un_dev)));
if (rxstat & MOS_RXSTS_SHORT_FRAME)
DPRINTF(("frame size less than 64 bytes\n"));
if (rxstat & MOS_RXSTS_LARGE_FRAME)
DPRINTF(("frame size larger than 1532 bytes\n"));
if (rxstat & MOS_RXSTS_CRC_ERROR)
DPRINTF(("CRC error\n"));
if (rxstat & MOS_RXSTS_ALIGN_ERROR)
DPRINTF(("alignment error\n"));
if_statinc(ifp, if_ierrors);
return;
}
if (pktlen < sizeof(struct ether_header) ) {
if_statinc(ifp, if_ierrors);
return;
}
usbnet_enqueue(un, c->unc_buf, pktlen, 0, 0, 0);
}
static unsigned
mos_tx_prepare(struct usbnet *un, struct mbuf *m, struct usbnet_chain *c)
{
int length;
usbnet_isowned_tx(un);
if ((unsigned)m->m_pkthdr.len > un->un_tx_bufsz)
return 0;
m_copydata(m, 0, m->m_pkthdr.len, c->unc_buf);
length = m->m_pkthdr.len;
DPRINTFN(5,("%s: %s: len %u\n",
device_xname(un->un_dev), __func__, length));
return length;
}
static int
mos_init_locked(struct ifnet *ifp)
{
struct usbnet * const un = ifp->if_softc;
u_int8_t rxmode;
unsigned char ipgs[2];
if (usbnet_isdying(un))
return EIO;
/* Cancel pending I/O */
usbnet_stop(un, ifp, 1);
usbnet_lock_mii_un_locked(un);
/* Reset the ethernet interface. */
mos_reset(un);
/* Write MAC address. */
mos_writemac(un);
/* Read and set transmitter IPG values */
ipgs[0] = mos_reg_read_1(un, MOS_IPG0);
ipgs[1] = mos_reg_read_1(un, MOS_IPG1);
mos_reg_write_1(un, MOS_IPG0, ipgs[0]);
mos_reg_write_1(un, MOS_IPG1, ipgs[1]);
/* Program promiscuous mode and multicast filters. */
mos_setiff_locked(un);
/* Enable receiver and transmitter, bridge controls speed/duplex mode */
rxmode = mos_reg_read_1(un, MOS_CTL);
rxmode |= MOS_CTL_RX_ENB | MOS_CTL_TX_ENB | MOS_CTL_BS_ENB;
rxmode &= ~(MOS_CTL_SLEEP);
mos_reg_write_1(un, MOS_CTL, rxmode);
usbnet_unlock_mii_un_locked(un);
return usbnet_init_rx_tx(un);
}
static int
mos_init(struct ifnet *ifp)
{
struct usbnet * const un = ifp->if_softc;
usbnet_lock(un);
int ret = mos_init_locked(ifp);
usbnet_unlock(un);
return ret;
}
static int
mos_ioctl(struct ifnet *ifp, u_long cmd, void *data)
{
struct usbnet * const un = ifp->if_softc;
switch (cmd) {
case SIOCADDMULTI:
case SIOCDELMULTI:
mos_setiff(un);
break;
default:
break;
}
return 0;
}
void
mos_stop(struct ifnet *ifp, int disable)
{
struct usbnet * const un = ifp->if_softc;
usbnet_lock_mii_un_locked(un);
mos_reset(un);
usbnet_unlock_mii_un_locked(un);
}