Commit c48ce926 authored by David S. Miller's avatar David S. Miller

[SPARC]: Update ethtool support in Sun net drivers.

parent 9696b971
/* $Id: sunbmac.c,v 1.30 2002/01/15 06:48:55 davem Exp $ /* $Id: sunbmac.c,v 1.30 2002/01/15 06:48:55 davem Exp $
* sunbmac.c: Driver for Sparc BigMAC 100baseT ethernet adapters. * sunbmac.c: Driver for Sparc BigMAC 100baseT ethernet adapters.
* *
* Copyright (C) 1997, 1998, 1999 David S. Miller (davem@redhat.com) * Copyright (C) 1997, 1998, 1999, 2003 David S. Miller (davem@redhat.com)
*/ */
#include <linux/module.h> #include <linux/module.h>
...@@ -18,6 +18,7 @@ ...@@ -18,6 +18,7 @@
#include <linux/init.h> #include <linux/init.h>
#include <linux/crc32.h> #include <linux/crc32.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/ethtool.h>
#include <linux/netdevice.h> #include <linux/netdevice.h>
#include <linux/etherdevice.h> #include <linux/etherdevice.h>
#include <linux/skbuff.h> #include <linux/skbuff.h>
...@@ -37,7 +38,7 @@ ...@@ -37,7 +38,7 @@
#include "sunbmac.h" #include "sunbmac.h"
static char version[] __initdata = static char version[] __initdata =
"sunbmac.c:v1.9 11/Sep/99 David S. Miller (davem@redhat.com)\n"; "sunbmac.c:v2.0 24/Nov/03 David S. Miller (davem@redhat.com)\n";
#undef DEBUG_PROBE #undef DEBUG_PROBE
#undef DEBUG_TX #undef DEBUG_TX
...@@ -1035,6 +1036,33 @@ static void bigmac_set_multicast(struct net_device *dev) ...@@ -1035,6 +1036,33 @@ static void bigmac_set_multicast(struct net_device *dev)
sbus_writel(tmp, bregs + BMAC_RXCFG); sbus_writel(tmp, bregs + BMAC_RXCFG);
} }
/* Ethtool support... */
static void bigmac_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info)
{
struct bigmac *bp = dev->priv;
strcpy(info->driver, "sunbmac");
strcpy(info->version, "2.0");
sprintf(info->bus_info, "SBUS:%d",
bp->qec_sdev->slot);
}
static u32 bigmac_get_link(struct net_device *dev)
{
struct bigmac *bp = dev->priv;
spin_lock_irq(&bp->lock);
bp->sw_bmsr = bigmac_tcvr_read(bp, bp->tregs, BIGMAC_BMSR);
spin_unlock_irq(&bp->lock);
return (bp->sw_bmsr & BMSR_LSTATUS);
}
static struct ethtool_ops bigmac_ethtool_ops = {
.get_drvinfo = bigmac_get_drvinfo,
.get_link = bigmac_get_link,
};
static int __init bigmac_ether_init(struct sbus_dev *qec_sdev) static int __init bigmac_ether_init(struct sbus_dev *qec_sdev)
{ {
struct net_device *dev; struct net_device *dev;
...@@ -1169,6 +1197,7 @@ static int __init bigmac_ether_init(struct sbus_dev *qec_sdev) ...@@ -1169,6 +1197,7 @@ static int __init bigmac_ether_init(struct sbus_dev *qec_sdev)
dev->open = &bigmac_open; dev->open = &bigmac_open;
dev->stop = &bigmac_close; dev->stop = &bigmac_close;
dev->hard_start_xmit = &bigmac_start_xmit; dev->hard_start_xmit = &bigmac_start_xmit;
dev->ethtool_ops = &bigmac_ethtool_ops;
/* Set links to BigMAC statistic and multi-cast loading code. */ /* Set links to BigMAC statistic and multi-cast loading code. */
dev->get_stats = &bigmac_get_stats; dev->get_stats = &bigmac_get_stats;
......
/* $Id: sungem.c,v 1.44.2.22 2002/03/13 01:18:12 davem Exp $ /* $Id: sungem.c,v 1.44.2.22 2002/03/13 01:18:12 davem Exp $
* sungem.c: Sun GEM ethernet driver. * sungem.c: Sun GEM ethernet driver.
* *
* Copyright (C) 2000, 2001, 2002 David S. Miller (davem@redhat.com) * Copyright (C) 2000, 2001, 2002, 2003 David S. Miller (davem@redhat.com)
* *
* Support for Apple GMAC and assorted PHYs by * Support for Apple GMAC and assorted PHYs by
* Benjamin Herrenscmidt (benh@kernel.crashing.org) * Benjamin Herrenscmidt (benh@kernel.crashing.org)
...@@ -70,8 +70,8 @@ ...@@ -70,8 +70,8 @@
SUPPORTED_1000baseT_Half | SUPPORTED_1000baseT_Full) SUPPORTED_1000baseT_Half | SUPPORTED_1000baseT_Full)
#define DRV_NAME "sungem" #define DRV_NAME "sungem"
#define DRV_VERSION "0.97" #define DRV_VERSION "0.98"
#define DRV_RELDATE "3/20/02" #define DRV_RELDATE "8/24/03"
#define DRV_AUTHOR "David S. Miller (davem@redhat.com)" #define DRV_AUTHOR "David S. Miller (davem@redhat.com)"
static char version[] __devinitdata = static char version[] __devinitdata =
...@@ -2317,177 +2317,134 @@ static void gem_set_multicast(struct net_device *dev) ...@@ -2317,177 +2317,134 @@ static void gem_set_multicast(struct net_device *dev)
spin_unlock_irq(&gp->lock); spin_unlock_irq(&gp->lock);
} }
/* Eventually add support for changing the advertisement static void gem_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info)
* on autoneg. {
*/ struct gem *gp = dev->priv;
static int gem_ethtool_ioctl(struct net_device *dev, void *ep_user)
strcpy(info->driver, DRV_NAME);
strcpy(info->version, DRV_VERSION);
strcpy(info->bus_info, pci_name(gp->pdev));
}
static int gem_get_settings(struct net_device *dev, struct ethtool_cmd *cmd)
{ {
struct gem *gp = dev->priv; struct gem *gp = dev->priv;
struct ethtool_cmd ecmd;
if (copy_from_user(&ecmd, ep_user, sizeof(ecmd)))
return -EFAULT;
switch(ecmd.cmd) {
case ETHTOOL_GDRVINFO: {
struct ethtool_drvinfo info = { .cmd = ETHTOOL_GDRVINFO };
strncpy(info.driver, DRV_NAME, ETHTOOL_BUSINFO_LEN); if (gp->phy_type == phy_mii_mdio0 ||
strncpy(info.version, DRV_VERSION, ETHTOOL_BUSINFO_LEN); gp->phy_type == phy_mii_mdio1) {
info.fw_version[0] = '\0'; if (gp->phy_mii.def)
strncpy(info.bus_info, pci_name(gp->pdev), ETHTOOL_BUSINFO_LEN); cmd->supported = gp->phy_mii.def->features;
info.regdump_len = 0; /*SUNGEM_NREGS;*/ else
cmd->supported = (SUPPORTED_10baseT_Half |
SUPPORTED_10baseT_Full);
if (copy_to_user(ep_user, &info, sizeof(info))) /* XXX hardcoded stuff for now */
return -EFAULT; cmd->port = PORT_MII;
cmd->transceiver = XCVR_EXTERNAL;
cmd->phy_address = 0; /* XXX fixed PHYAD */
return 0; /* Return current PHY settings */
} spin_lock_irq(&gp->lock);
cmd->autoneg = gp->want_autoneg;
case ETHTOOL_GSET: cmd->speed = gp->phy_mii.speed;
if (gp->phy_type == phy_mii_mdio0 || cmd->duplex = gp->phy_mii.duplex;
gp->phy_type == phy_mii_mdio1) { cmd->advertising = gp->phy_mii.advertising;
if (gp->phy_mii.def)
ecmd.supported = gp->phy_mii.def->features; /* If we started with a forced mode, we don't have a default
else * advertise set, we need to return something sensible so
ecmd.supported = SUPPORTED_10baseT_Half | SUPPORTED_10baseT_Full; * userland can re-enable autoneg properly.
*/
/* XXX hardcoded stuff for now */ if (cmd->advertising == 0)
ecmd.port = PORT_MII; cmd->advertising = cmd->supported;
ecmd.transceiver = XCVR_EXTERNAL; spin_unlock_irq(&gp->lock);
ecmd.phy_address = 0; /* XXX fixed PHYAD */ } else { // XXX PCS ?
cmd->supported =
/* Return current PHY settings */
spin_lock_irq(&gp->lock);
ecmd.autoneg = gp->want_autoneg;
ecmd.speed = gp->phy_mii.speed;
ecmd.duplex = gp->phy_mii.duplex;
ecmd.advertising = gp->phy_mii.advertising;
/* If we started with a forced mode, we don't have a default
* advertise set, we need to return something sensible so
* userland can re-enable autoneg properly */
if (ecmd.advertising == 0)
ecmd.advertising = ecmd.supported;
spin_unlock_irq(&gp->lock);
} else { // XXX PCS ?
ecmd.supported =
(SUPPORTED_10baseT_Half | SUPPORTED_10baseT_Full | (SUPPORTED_10baseT_Half | SUPPORTED_10baseT_Full |
SUPPORTED_100baseT_Half | SUPPORTED_100baseT_Full | SUPPORTED_100baseT_Half | SUPPORTED_100baseT_Full |
SUPPORTED_Autoneg); SUPPORTED_Autoneg);
ecmd.advertising = ecmd.supported; cmd->advertising = cmd->supported;
} cmd->speed = 0;
if (copy_to_user(ep_user, &ecmd, sizeof(ecmd))) cmd->duplex = cmd->port = cmd->phy_address =
return -EFAULT; cmd->transceiver = cmd->autoneg = 0;
return 0; }
cmd->maxtxpkt = cmd->maxrxpkt = 0;
case ETHTOOL_SSET:
/* Verify the settings we care about. */
if (ecmd.autoneg != AUTONEG_ENABLE &&
ecmd.autoneg != AUTONEG_DISABLE)
return -EINVAL;
if (ecmd.autoneg == AUTONEG_ENABLE &&
ecmd.advertising == 0)
return -EINVAL;
if (ecmd.autoneg == AUTONEG_DISABLE &&
((ecmd.speed != SPEED_1000 &&
ecmd.speed != SPEED_100 &&
ecmd.speed != SPEED_10) ||
(ecmd.duplex != DUPLEX_HALF &&
ecmd.duplex != DUPLEX_FULL)))
return -EINVAL;
/* Apply settings and restart link process. */
spin_lock_irq(&gp->lock);
gem_begin_auto_negotiation(gp, &ecmd);
spin_unlock_irq(&gp->lock);
return 0;
case ETHTOOL_NWAY_RST:
if (!gp->want_autoneg)
return -EINVAL;
/* Restart link process. */ return 0;
spin_lock_irq(&gp->lock); }
gem_begin_auto_negotiation(gp, NULL);
spin_unlock_irq(&gp->lock);
return 0; static int gem_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
{
struct gem *gp = dev->priv;
case ETHTOOL_GWOL: /* Verify the settings we care about. */
case ETHTOOL_SWOL: if (cmd->autoneg != AUTONEG_ENABLE &&
break; /* todo */ cmd->autoneg != AUTONEG_DISABLE)
return -EINVAL;
/* get link status */ if (cmd->autoneg == AUTONEG_ENABLE &&
case ETHTOOL_GLINK: { cmd->advertising == 0)
struct ethtool_value edata = { .cmd = ETHTOOL_GLINK }; return -EINVAL;
edata.data = (gp->lstate == link_up); if (cmd->autoneg == AUTONEG_DISABLE &&
if (copy_to_user(ep_user, &edata, sizeof(edata))) ((cmd->speed != SPEED_1000 &&
return -EFAULT; cmd->speed != SPEED_100 &&
return 0; cmd->speed != SPEED_10) ||
} (cmd->duplex != DUPLEX_HALF &&
cmd->duplex != DUPLEX_FULL)))
return -EINVAL;
/* Apply settings and restart link process. */
spin_lock_irq(&gp->lock);
gem_begin_auto_negotiation(gp, cmd);
spin_unlock_irq(&gp->lock);
/* get message-level */ return 0;
case ETHTOOL_GMSGLVL: { }
struct ethtool_value edata = { .cmd = ETHTOOL_GMSGLVL };
edata.data = gp->msg_enable; static int gem_nway_reset(struct net_device *dev)
if (copy_to_user(ep_user, &edata, sizeof(edata))) {
return -EFAULT; struct gem *gp = dev->priv;
return 0;
}
/* set message-level */ if (!gp->want_autoneg)
case ETHTOOL_SMSGLVL: { return -EINVAL;
struct ethtool_value edata;
if (copy_from_user(&edata, ep_user, sizeof(edata))) /* Restart link process. */
return -EFAULT; spin_lock_irq(&gp->lock);
gp->msg_enable = edata.data; gem_begin_auto_negotiation(gp, NULL);
return 0; spin_unlock_irq(&gp->lock);
}
#if 0 return 0;
case ETHTOOL_GREGS: { }
struct ethtool_regs regs;
u32 *regbuf;
int r = 0;
if (copy_from_user(&regs, useraddr, sizeof(regs))) static u32 gem_get_link(struct net_device *dev)
return -EFAULT; {
struct gem *gp = dev->priv;
if (regs.len > SUNGEM_NREGS) {
regs.len = SUNGEM_NREGS;
}
regs.version = 0;
if (copy_to_user(useraddr, &regs, sizeof(regs)))
return -EFAULT;
if (!gp->hw_running) return (gp->lstate == link_up);
return -ENODEV; }
useraddr += offsetof(struct ethtool_regs, data);
/* Use kmalloc to avoid bloating the stack */
regbuf = kmalloc(4 * SUNGEM_NREGS, GFP_KERNEL);
if (!regbuf)
return -ENOMEM;
spin_lock_irq(&np->lock);
gem_get_regs(gp, regbuf);
spin_unlock_irq(&np->lock);
if (copy_to_user(useraddr, regbuf, regs.len*sizeof(u32)))
r = -EFAULT;
kfree(regbuf);
return r;
}
#endif
};
return -EOPNOTSUPP; static u32 gem_get_msglevel(struct net_device *dev)
{
struct gem *gp = dev->priv;
return gp->msg_enable;
} }
static void gem_set_msglevel(struct net_device *dev, u32 value)
{
struct gem *gp = dev->priv;
gp->msg_enable = value;
}
static struct ethtool_ops gem_ethtool_ops = {
.get_drvinfo = gem_get_drvinfo,
.get_link = ethtool_op_get_link,
.get_settings = gem_get_settings,
.set_settings = gem_set_settings,
.nway_reset = gem_nway_reset,
.get_link = gem_get_link,
.get_msglevel = gem_get_msglevel,
.set_msglevel = gem_set_msglevel,
};
static int gem_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd) static int gem_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
{ {
...@@ -2501,10 +2458,6 @@ static int gem_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd) ...@@ -2501,10 +2458,6 @@ static int gem_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
down(&gp->pm_sem); down(&gp->pm_sem);
switch (cmd) { switch (cmd) {
case SIOCETHTOOL:
rc = gem_ethtool_ioctl(dev, ifr->ifr_data);
break;
case SIOCGMIIPHY: /* Get address of MII PHY in use. */ case SIOCGMIIPHY: /* Get address of MII PHY in use. */
data->phy_id = gp->mii_phy_addr; data->phy_id = gp->mii_phy_addr;
/* Fallthrough... */ /* Fallthrough... */
...@@ -2812,6 +2765,7 @@ static int __devinit gem_init_one(struct pci_dev *pdev, ...@@ -2812,6 +2765,7 @@ static int __devinit gem_init_one(struct pci_dev *pdev,
dev->get_stats = gem_get_stats; dev->get_stats = gem_get_stats;
dev->set_multicast_list = gem_set_multicast; dev->set_multicast_list = gem_set_multicast;
dev->do_ioctl = gem_ioctl; dev->do_ioctl = gem_ioctl;
dev->ethtool_ops = &gem_ethtool_ops;
dev->tx_timeout = gem_tx_timeout; dev->tx_timeout = gem_tx_timeout;
dev->watchdog_timeo = 5 * HZ; dev->watchdog_timeo = 5 * HZ;
dev->change_mtu = gem_change_mtu; dev->change_mtu = gem_change_mtu;
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
* auto carrier detecting ethernet driver. Also known as the * auto carrier detecting ethernet driver. Also known as the
* "Happy Meal Ethernet" found on SunSwift SBUS cards. * "Happy Meal Ethernet" found on SunSwift SBUS cards.
* *
* Copyright (C) 1996, 1998, 1999, 2002 David S. Miller (davem@redhat.com) * Copyright (C) 1996, 1998, 1999, 2002, 2003 David S. Miller (davem@redhat.com)
* *
* Changes : * Changes :
* 2000/11/11 Willy Tarreau <willy AT meta-x.org> * 2000/11/11 Willy Tarreau <willy AT meta-x.org>
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
*/ */
static char version[] = static char version[] =
"sunhme.c:v2.01 26/Mar/2002 David S. Miller (davem@redhat.com)\n"; "sunhme.c:v2.02 24/Aug/2003 David S. Miller (davem@redhat.com)\n";
#include <linux/module.h> #include <linux/module.h>
#include <linux/config.h> #include <linux/config.h>
...@@ -2426,85 +2426,109 @@ static void happy_meal_set_multicast(struct net_device *dev) ...@@ -2426,85 +2426,109 @@ static void happy_meal_set_multicast(struct net_device *dev)
} }
/* Ethtool support... */ /* Ethtool support... */
static int happy_meal_ioctl(struct net_device *dev, static int hme_get_settings(struct net_device *dev, struct ethtool_cmd *cmd)
struct ifreq *rq, int cmd)
{ {
struct happy_meal *hp = dev->priv; struct happy_meal *hp = dev->priv;
struct ethtool_cmd *ep_user = (struct ethtool_cmd *) rq->ifr_data;
struct ethtool_cmd ecmd;
if (cmd != SIOCETHTOOL)
return -EOPNOTSUPP;
if (copy_from_user(&ecmd, ep_user, sizeof(ecmd)))
return -EFAULT;
if (ecmd.cmd == ETHTOOL_GSET) {
ecmd.supported =
(SUPPORTED_10baseT_Half | SUPPORTED_10baseT_Full |
SUPPORTED_100baseT_Half | SUPPORTED_100baseT_Full |
SUPPORTED_Autoneg | SUPPORTED_TP | SUPPORTED_MII);
/* XXX hardcoded stuff for now */
ecmd.port = PORT_TP; /* XXX no MII support */
ecmd.transceiver = XCVR_INTERNAL; /* XXX no external xcvr support */
ecmd.phy_address = 0; /* XXX fixed PHYAD */
/* Record PHY settings. */
spin_lock_irq(&hp->happy_lock);
hp->sw_bmcr = happy_meal_tcvr_read(hp, hp->tcvregs, MII_BMCR);
hp->sw_lpa = happy_meal_tcvr_read(hp, hp->tcvregs, MII_LPA);
spin_unlock_irq(&hp->happy_lock);
if (hp->sw_bmcr & BMCR_ANENABLE) { cmd->supported =
ecmd.autoneg = AUTONEG_ENABLE; (SUPPORTED_10baseT_Half | SUPPORTED_10baseT_Full |
ecmd.speed = SUPPORTED_100baseT_Half | SUPPORTED_100baseT_Full |
(hp->sw_lpa & (LPA_100HALF | LPA_100FULL)) ? SUPPORTED_Autoneg | SUPPORTED_TP | SUPPORTED_MII);
SPEED_100 : SPEED_10;
if (ecmd.speed == SPEED_100) /* XXX hardcoded stuff for now */
ecmd.duplex = cmd->port = PORT_TP; /* XXX no MII support */
(hp->sw_lpa & (LPA_100FULL)) ? cmd->transceiver = XCVR_INTERNAL; /* XXX no external xcvr support */
DUPLEX_FULL : DUPLEX_HALF; cmd->phy_address = 0; /* XXX fixed PHYAD */
else
ecmd.duplex = /* Record PHY settings. */
(hp->sw_lpa & (LPA_10FULL)) ? spin_lock_irq(&hp->happy_lock);
DUPLEX_FULL : DUPLEX_HALF; hp->sw_bmcr = happy_meal_tcvr_read(hp, hp->tcvregs, MII_BMCR);
} else { hp->sw_lpa = happy_meal_tcvr_read(hp, hp->tcvregs, MII_LPA);
ecmd.autoneg = AUTONEG_DISABLE; spin_unlock_irq(&hp->happy_lock);
ecmd.speed =
(hp->sw_bmcr & BMCR_SPEED100) ? if (hp->sw_bmcr & BMCR_ANENABLE) {
SPEED_100 : SPEED_10; cmd->autoneg = AUTONEG_ENABLE;
ecmd.duplex = cmd->speed =
(hp->sw_bmcr & BMCR_FULLDPLX) ? (hp->sw_lpa & (LPA_100HALF | LPA_100FULL)) ?
SPEED_100 : SPEED_10;
if (cmd->speed == SPEED_100)
cmd->duplex =
(hp->sw_lpa & (LPA_100FULL)) ?
DUPLEX_FULL : DUPLEX_HALF; DUPLEX_FULL : DUPLEX_HALF;
} else
if (copy_to_user(ep_user, &ecmd, sizeof(ecmd))) cmd->duplex =
return -EFAULT; (hp->sw_lpa & (LPA_10FULL)) ?
return 0; DUPLEX_FULL : DUPLEX_HALF;
} else if (ecmd.cmd == ETHTOOL_SSET) { } else {
/* Verify the settings we care about. */ cmd->autoneg = AUTONEG_DISABLE;
if (ecmd.autoneg != AUTONEG_ENABLE && cmd->speed =
ecmd.autoneg != AUTONEG_DISABLE) (hp->sw_bmcr & BMCR_SPEED100) ?
return -EINVAL; SPEED_100 : SPEED_10;
if (ecmd.autoneg == AUTONEG_DISABLE && cmd->duplex =
((ecmd.speed != SPEED_100 && (hp->sw_bmcr & BMCR_FULLDPLX) ?
ecmd.speed != SPEED_10) || DUPLEX_FULL : DUPLEX_HALF;
(ecmd.duplex != DUPLEX_HALF && }
ecmd.duplex != DUPLEX_FULL))) return 0;
return -EINVAL; }
/* Ok, do it to it. */
spin_lock_irq(&hp->happy_lock);
del_timer(&hp->happy_timer);
happy_meal_begin_auto_negotiation(hp,
hp->tcvregs,
&ecmd);
spin_unlock_irq(&hp->happy_lock);
return 0; static int hme_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
} else {
return -EOPNOTSUPP; struct happy_meal *hp = dev->priv;
/* Verify the settings we care about. */
if (cmd->autoneg != AUTONEG_ENABLE &&
cmd->autoneg != AUTONEG_DISABLE)
return -EINVAL;
if (cmd->autoneg == AUTONEG_DISABLE &&
((cmd->speed != SPEED_100 &&
cmd->speed != SPEED_10) ||
(cmd->duplex != DUPLEX_HALF &&
cmd->duplex != DUPLEX_FULL)))
return -EINVAL;
/* Ok, do it to it. */
spin_lock_irq(&hp->happy_lock);
del_timer(&hp->happy_timer);
happy_meal_begin_auto_negotiation(hp, hp->tcvregs, cmd);
spin_unlock_irq(&hp->happy_lock);
return 0;
}
static void hme_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info)
{
struct happy_meal *hp = dev->priv;
strcpy(info->driver, "sunhme");
strcpy(info->version, "2.02");
if (hp->happy_flags & HFLAG_PCI) {
struct pci_dev *pdev = hp->happy_dev;
strcpy(info->bus_info, pci_name(pdev));
} else {
struct sbus_dev *sdev = hp->happy_dev;
sprintf(info->bus_info, "SBUS:%d",
sdev->slot);
}
} }
static u32 hme_get_link(struct net_device *dev)
{
struct happy_meal *hp = dev->priv;
spin_lock_irq(&hp->happy_lock);
hp->sw_bmcr = happy_meal_tcvr_read(hp, hp->tcvregs, MII_BMCR);
spin_unlock_irq(&hp->happy_lock);
return (hp->sw_bmsr & BMSR_LSTATUS);
}
static struct ethtool_ops hme_ethtool_ops = {
.get_settings = hme_get_settings,
.set_settings = hme_set_settings,
.get_drvinfo = hme_get_drvinfo,
.get_link = hme_get_link,
};
static int hme_version_printed; static int hme_version_printed;
#ifdef CONFIG_SBUS #ifdef CONFIG_SBUS
...@@ -2797,7 +2821,7 @@ static int __init happy_meal_sbus_init(struct sbus_dev *sdev, int is_qfe) ...@@ -2797,7 +2821,7 @@ static int __init happy_meal_sbus_init(struct sbus_dev *sdev, int is_qfe)
dev->set_multicast_list = &happy_meal_set_multicast; dev->set_multicast_list = &happy_meal_set_multicast;
dev->tx_timeout = &happy_meal_tx_timeout; dev->tx_timeout = &happy_meal_tx_timeout;
dev->watchdog_timeo = 5*HZ; dev->watchdog_timeo = 5*HZ;
dev->do_ioctl = &happy_meal_ioctl; dev->ethtool_ops = &hme_ethtool_ops;
/* Happy Meal can do it all... except VLAN. */ /* Happy Meal can do it all... except VLAN. */
dev->features |= NETIF_F_SG | NETIF_F_HW_CSUM | NETIF_F_VLAN_CHALLENGED; dev->features |= NETIF_F_SG | NETIF_F_HW_CSUM | NETIF_F_VLAN_CHALLENGED;
...@@ -3141,7 +3165,7 @@ static int __init happy_meal_pci_init(struct pci_dev *pdev) ...@@ -3141,7 +3165,7 @@ static int __init happy_meal_pci_init(struct pci_dev *pdev)
dev->set_multicast_list = &happy_meal_set_multicast; dev->set_multicast_list = &happy_meal_set_multicast;
dev->tx_timeout = &happy_meal_tx_timeout; dev->tx_timeout = &happy_meal_tx_timeout;
dev->watchdog_timeo = 5*HZ; dev->watchdog_timeo = 5*HZ;
dev->do_ioctl = &happy_meal_ioctl; dev->ethtool_ops = &hme_ethtool_ops;
dev->irq = pdev->irq; dev->irq = pdev->irq;
dev->dma = 0; dev->dma = 0;
......
...@@ -70,7 +70,7 @@ ...@@ -70,7 +70,7 @@
#undef DEBUG_DRIVER #undef DEBUG_DRIVER
static char version[] = static char version[] =
"sunlance.c:v2.01 08/Nov/01 Miguel de Icaza (miguel@nuclecu.unam.mx)\n"; "sunlance.c:v2.02 24/Aug/03 Miguel de Icaza (miguel@nuclecu.unam.mx)\n";
static char lancestr[] = "LANCE"; static char lancestr[] = "LANCE";
...@@ -93,6 +93,7 @@ static char lancestr[] = "LANCE"; ...@@ -93,6 +93,7 @@ static char lancestr[] = "LANCE";
#include <linux/netdevice.h> #include <linux/netdevice.h>
#include <linux/etherdevice.h> #include <linux/etherdevice.h>
#include <linux/skbuff.h> #include <linux/skbuff.h>
#include <linux/ethtool.h>
#include <asm/system.h> #include <asm/system.h>
#include <asm/bitops.h> #include <asm/bitops.h>
...@@ -1287,6 +1288,30 @@ static void lance_free_hwresources(struct lance_private *lp) ...@@ -1287,6 +1288,30 @@ static void lance_free_hwresources(struct lance_private *lp)
} }
} }
/* Ethtool support... */
static void sparc_lance_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info)
{
struct lance_private *lp = dev->priv;
strcpy(info->driver, "sunlance");
strcpy(info->version, "2.02");
sprintf(info->bus_info, "SBUS:%d",
lp->sdev->slot);
}
static u32 sparc_lance_get_link(struct net_device *dev)
{
/* We really do not keep track of this, but this
* is better than not reporting anything at all.
*/
return 1;
}
static struct ethtool_ops sparc_lance_ethtool_ops = {
.get_drvinfo = sparc_lance_get_drvinfo,
.get_link = sparc_lance_get_link,
};
static int __init sparc_lance_init(struct net_device *dev, static int __init sparc_lance_init(struct net_device *dev,
struct sbus_dev *sdev, struct sbus_dev *sdev,
struct sbus_dma *ledma, struct sbus_dma *ledma,
...@@ -1456,6 +1481,7 @@ static int __init sparc_lance_init(struct net_device *dev, ...@@ -1456,6 +1481,7 @@ static int __init sparc_lance_init(struct net_device *dev,
dev->watchdog_timeo = 5*HZ; dev->watchdog_timeo = 5*HZ;
dev->get_stats = &lance_get_stats; dev->get_stats = &lance_get_stats;
dev->set_multicast_list = &lance_set_multicast; dev->set_multicast_list = &lance_set_multicast;
dev->ethtool_ops = &sparc_lance_ethtool_ops;
dev->irq = sdev->irqs[0]; dev->irq = sdev->irqs[0];
......
...@@ -4,11 +4,11 @@ ...@@ -4,11 +4,11 @@
* controller out there can be most efficiently programmed * controller out there can be most efficiently programmed
* if you make it look like a LANCE. * if you make it look like a LANCE.
* *
* Copyright (C) 1996, 1999 David S. Miller (davem@redhat.com) * Copyright (C) 1996, 1999, 2003 David S. Miller (davem@redhat.com)
*/ */
static char version[] = static char version[] =
"sunqe.c:v2.9 9/11/99 David S. Miller (davem@redhat.com)\n"; "sunqe.c:v3.0 8/24/03 David S. Miller (davem@redhat.com)\n";
#include <linux/module.h> #include <linux/module.h>
#include <linux/kernel.h> #include <linux/kernel.h>
...@@ -26,6 +26,7 @@ static char version[] = ...@@ -26,6 +26,7 @@ static char version[] =
#include <linux/netdevice.h> #include <linux/netdevice.h>
#include <linux/etherdevice.h> #include <linux/etherdevice.h>
#include <linux/skbuff.h> #include <linux/skbuff.h>
#include <linux/ethtool.h>
#include <asm/system.h> #include <asm/system.h>
#include <asm/bitops.h> #include <asm/bitops.h>
...@@ -684,6 +685,35 @@ static void qe_set_multicast(struct net_device *dev) ...@@ -684,6 +685,35 @@ static void qe_set_multicast(struct net_device *dev)
netif_wake_queue(dev); netif_wake_queue(dev);
} }
/* Ethtool support... */
static void qe_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info)
{
struct sunqe *qep = dev->priv;
strcpy(info->driver, "sunqe");
strcpy(info->version, "3.0");
sprintf(info->bus_info, "SBUS:%d",
qep->qe_sdev->slot);
}
static u32 qe_get_link(struct net_device *dev)
{
struct sunqe *qep = dev->priv;
unsigned long mregs = qep->mregs;
u8 phyconfig;
spin_lock_irq(&qep->lock);
phyconfig = sbus_readb(mregs + MREGS_PHYCONFIG);
spin_unlock_irq(&qep->lock);
return (phyconfig & MREGS_PHYCONFIG_LSTAT);
}
static struct ethtool_ops qe_ethtool_ops = {
.get_drvinfo = qe_get_drvinfo,
.get_link = qe_get_link,
};
/* This is only called once at boot time for each card probed. */ /* This is only called once at boot time for each card probed. */
static inline void qec_init_once(struct sunqec *qecp, struct sbus_dev *qsdev) static inline void qec_init_once(struct sunqec *qecp, struct sbus_dev *qsdev)
{ {
...@@ -850,6 +880,7 @@ static int __init qec_ether_init(struct net_device *dev, struct sbus_dev *sdev) ...@@ -850,6 +880,7 @@ static int __init qec_ether_init(struct net_device *dev, struct sbus_dev *sdev)
qe_devs[i]->watchdog_timeo = 5*HZ; qe_devs[i]->watchdog_timeo = 5*HZ;
qe_devs[i]->irq = sdev->irqs[0]; qe_devs[i]->irq = sdev->irqs[0];
qe_devs[i]->dma = 0; qe_devs[i]->dma = 0;
qe_devs[i]->ethtool_ops = &qe_ethtool_ops;
} }
/* QEC receives interrupts from each QE, then it sends the actual /* QEC receives interrupts from each QE, then it sends the actual
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment