blob: bde43a5d5a238ade06cc633e84187af27e90b71c [file] [log] [blame]
/*
* Atheros AR71xx built-in ethernet mac driver
*
* Copyright (c) 2013 The Linux Foundation. All rights reserved.
* Copyright (C) 2008-2010 Gabor Juhos <juhosg@openwrt.org>
* Copyright (C) 2008 Imre Kaloz <kaloz@openwrt.org>
*
* Based on Atheros' AG7100 driver
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation.
*/
#include "ag71xx.h"
#define AR934X_PHY_SPECIFIC_STATUS_REG 0x11
#define AR934X_PORT_STATUS_SPEED_MASK 0xc000
#define AR934X_PORT_STATUS_SPEED_10 (0 << 14)
#define AR934X_PORT_STATUS_SPEED_100 (1 << 14)
#define AR934X_PORT_STATUS_DUPLEX (1 << 13)
static int ag71xx_ethtool_get_settings_switchport(struct net_device *dev,
struct ethtool_cmd *cmd)
{
struct ag71xx *ag = netdev_priv(dev);
struct platform_device *pdev = ag->pdev;
struct net_device *netdev = ag->dev;
uint16_t status;
status = ag->mii_bus->read(ag->mii_bus, netdev->dev_id,
AR934X_PHY_SPECIFIC_STATUS_REG);
if (status == 0xffff)
return -ENODEV;
switch (status & AR934X_PORT_STATUS_SPEED_MASK) {
case AR934X_PORT_STATUS_SPEED_10:
ethtool_cmd_speed_set(cmd, 10);
break;
case AR934X_PORT_STATUS_SPEED_100:
ethtool_cmd_speed_set(cmd, 100);
break;
default:
ethtool_cmd_speed_set(cmd, 0);
break;
}
cmd->duplex = !!(status & AR934X_PORT_STATUS_DUPLEX);
cmd->port = PORT_MII;
cmd->transceiver = XCVR_EXTERNAL;
cmd->autoneg = 1;
return 0;
}
static int ag71xx_ethtool_get_settings(struct net_device *dev,
struct ethtool_cmd *cmd)
{
struct ag71xx *ag = netdev_priv(dev);
struct phy_device *phydev = ag->phy_dev;
if (!phydev)
return ag71xx_ethtool_get_settings_switchport(dev, cmd);
return phy_ethtool_gset(phydev, cmd);
}
static int ag71xx_ethtool_set_settings(struct net_device *dev,
struct ethtool_cmd *cmd)
{
struct ag71xx *ag = netdev_priv(dev);
struct phy_device *phydev = ag->phy_dev;
if (!phydev)
return -ENODEV;
return phy_ethtool_sset(phydev, cmd);
}
static void ag71xx_ethtool_get_drvinfo(struct net_device *dev,
struct ethtool_drvinfo *info)
{
struct ag71xx *ag = netdev_priv(dev);
strcpy(info->driver, ag->pdev->dev.driver->name);
strcpy(info->version, AG71XX_DRV_VERSION);
strcpy(info->bus_info, dev_name(&ag->pdev->dev));
}
static u32 ag71xx_ethtool_get_msglevel(struct net_device *dev)
{
struct ag71xx *ag = netdev_priv(dev);
return ag->msg_enable;
}
static void ag71xx_ethtool_set_msglevel(struct net_device *dev, u32 msg_level)
{
struct ag71xx *ag = netdev_priv(dev);
ag->msg_enable = msg_level;
}
static void ag71xx_ethtool_get_ringparam(struct net_device *dev,
struct ethtool_ringparam *er)
{
struct ag71xx *ag = netdev_priv(dev);
er->tx_max_pending = AG71XX_TX_RING_SIZE_MAX;
er->rx_max_pending = AG71XX_RX_RING_SIZE_MAX;
er->rx_mini_max_pending = 0;
er->rx_jumbo_max_pending = 0;
er->tx_pending = ag->tx_ring.size;
er->rx_pending = ag->rx_ring.size;
er->rx_mini_pending = 0;
er->rx_jumbo_pending = 0;
}
/*
* Return the next largest power of 2.
*/
static int ag71xx_next_power_of_2(unsigned int i)
{
i--;
i = (i >> 1) | i;
i = (i >> 2) | i;
i = (i >> 4) | i;
i = (i >> 8) | i;
i = (i >> 16) | i;
i++;
return i;
}
static int ag71xx_ethtool_set_ringparam(struct net_device *dev,
struct ethtool_ringparam *er)
{
struct ag71xx *ag = netdev_priv(dev);
unsigned tx_size;
unsigned rx_size;
int err;
if (er->rx_mini_pending != 0||
er->rx_jumbo_pending != 0 ||
er->rx_pending == 0 ||
er->tx_pending == 0)
return -EINVAL;
tx_size = er->tx_pending < AG71XX_TX_RING_SIZE_MAX ?
er->tx_pending : AG71XX_TX_RING_SIZE_MAX;
tx_size = ag71xx_next_power_of_2(tx_size);
rx_size = er->rx_pending < AG71XX_RX_RING_SIZE_MAX ?
er->rx_pending : AG71XX_RX_RING_SIZE_MAX;
rx_size = ag71xx_next_power_of_2(rx_size);
if (netif_running(dev)) {
err = dev->netdev_ops->ndo_stop(dev);
if (err)
return err;
}
ag->tx_ring.size = tx_size;
ag->tx_ring.mask = tx_size - 1;
ag->rx_ring.size = rx_size;
ag->rx_ring.mask = rx_size - 1;
if (netif_running(dev))
err = dev->netdev_ops->ndo_open(dev);
return err;
}
static int ag71xx_ethtool_get_regs_len(struct net_device *netdev)
{
#define AG71XX_REGS_LEN 23
return AG71XX_REGS_LEN * sizeof(u32);
}
static void ag71xx_ethtool_get_regs(struct net_device *netdev,
struct ethtool_regs *regs, void *p)
{
struct ag71xx *ag = netdev_priv(netdev);
u32 *regs_buff = p;
memset(p, 0, AG71XX_REGS_LEN * sizeof(u32));
regs_buff[0] = ag71xx_rr(ag, AG71XX_REG_MAC_CFG1);
regs_buff[1] = ag71xx_rr(ag, AG71XX_REG_MAC_CFG2);
regs_buff[2] = ag71xx_rr(ag, AG71XX_REG_MAC_IPG);
regs_buff[3] = ag71xx_rr(ag, AG71XX_REG_MAC_HDX);
regs_buff[4] = ag71xx_rr(ag, AG71XX_REG_MAC_MFL);
regs_buff[5] = 0xFFFFFFFF;
regs_buff[6] = 0xFFFFFFFF;
regs_buff[7] = 0xFFFFFFFF;
regs_buff[8] = ag71xx_rr(ag, AG71XX_REG_MII_CFG);
regs_buff[9] = 0xFFFFFFFF;
regs_buff[10] = 0xFFFFFFFF;
regs_buff[11] = 0xFFFFFFFF;
regs_buff[12] = 0xFFFFFFFF;
regs_buff[13] = 0xFFFFFFFF;
regs_buff[14] = ag71xx_rr(ag, AG71XX_REG_MAC_IFCTL);
regs_buff[15] = ag71xx_rr(ag, AG71XX_REG_MAC_ADDR1);
regs_buff[16] = ag71xx_rr(ag, AG71XX_REG_MAC_ADDR2);
regs_buff[17] = ag71xx_rr(ag, AG71XX_REG_FIFO_CFG0);
regs_buff[18] = ag71xx_rr(ag, AG71XX_REG_FIFO_CFG1);
regs_buff[19] = ag71xx_rr(ag, AG71XX_REG_FIFO_CFG2);
regs_buff[20] = ag71xx_rr(ag, AG71XX_REG_FIFO_CFG3);
regs_buff[21] = ag71xx_rr(ag, AG71XX_REG_FIFO_CFG4);
regs_buff[22] = ag71xx_rr(ag, AG71XX_REG_FIFO_CFG5);
}
struct ethtool_ops ag71xx_ethtool_ops = {
.set_settings = ag71xx_ethtool_set_settings,
.get_settings = ag71xx_ethtool_get_settings,
.get_drvinfo = ag71xx_ethtool_get_drvinfo,
.get_regs_len = ag71xx_ethtool_get_regs_len,
.get_regs = ag71xx_ethtool_get_regs,
.get_msglevel = ag71xx_ethtool_get_msglevel,
.set_msglevel = ag71xx_ethtool_set_msglevel,
.get_ringparam = ag71xx_ethtool_get_ringparam,
.set_ringparam = ag71xx_ethtool_set_ringparam,
.get_link = ethtool_op_get_link,
};