Initial commit
Some checks failed
Build Kernel / Build all affected Kernels (push) Has been cancelled
Build all core packages / Build all core packages for selected target (push) Has been cancelled
Build and Push prebuilt tools container / Build and Push all prebuilt containers (push) Has been cancelled
Build Toolchains / Build Toolchains for each target (push) Has been cancelled
Build host tools / Build host tools for linux and macos based systems (push) Has been cancelled
Coverity scan build / Coverity x86/64 build (push) Has been cancelled
Some checks failed
Build Kernel / Build all affected Kernels (push) Has been cancelled
Build all core packages / Build all core packages for selected target (push) Has been cancelled
Build and Push prebuilt tools container / Build and Push all prebuilt containers (push) Has been cancelled
Build Toolchains / Build Toolchains for each target (push) Has been cancelled
Build host tools / Build host tools for linux and macos based systems (push) Has been cancelled
Coverity scan build / Coverity x86/64 build (push) Has been cancelled
This commit is contained in:
234
target/linux/ramips/files/drivers/net/ethernet/ralink/ethtool.c
Normal file
234
target/linux/ramips/files/drivers/net/ethernet/ralink/ethtool.c
Normal file
@@ -0,0 +1,234 @@
|
||||
/* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; version 2 of the License
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* Copyright (C) 2009-2015 John Crispin <blogic@openwrt.org>
|
||||
* Copyright (C) 2009-2015 Felix Fietkau <nbd@nbd.name>
|
||||
* Copyright (C) 2013-2015 Michael Lee <igvtee@gmail.com>
|
||||
*/
|
||||
|
||||
#include "mtk_eth_soc.h"
|
||||
|
||||
static const char fe_gdma_str[][ETH_GSTRING_LEN] = {
|
||||
#define _FE(x...) # x,
|
||||
FE_STAT_REG_DECLARE
|
||||
#undef _FE
|
||||
};
|
||||
|
||||
static int fe_get_link_ksettings(struct net_device *ndev,
|
||||
struct ethtool_link_ksettings *cmd)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(ndev);
|
||||
|
||||
if (!priv->phy_dev)
|
||||
return -ENODEV;
|
||||
|
||||
if (priv->phy_flags == FE_PHY_FLAG_ATTACH) {
|
||||
if (phy_read_status(priv->phy_dev))
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
phy_ethtool_ksettings_get(ndev->phydev, cmd);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int fe_set_link_ksettings(struct net_device *ndev,
|
||||
const struct ethtool_link_ksettings *cmd)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(ndev);
|
||||
|
||||
if (!priv->phy_dev)
|
||||
goto out_sset;
|
||||
|
||||
if (cmd->base.phy_address != priv->phy_dev->mdio.addr) {
|
||||
if (priv->phy->phy_node[cmd->base.phy_address]) {
|
||||
priv->phy_dev = priv->phy->phy[cmd->base.phy_address];
|
||||
priv->phy_flags = FE_PHY_FLAG_PORT;
|
||||
} else if (priv->mii_bus && mdiobus_get_phy(priv->mii_bus, cmd->base.phy_address)) {
|
||||
priv->phy_dev = mdiobus_get_phy(priv->mii_bus, cmd->base.phy_address);
|
||||
priv->phy_flags = FE_PHY_FLAG_ATTACH;
|
||||
} else {
|
||||
goto out_sset;
|
||||
}
|
||||
}
|
||||
|
||||
return phy_ethtool_ksettings_set(ndev->phydev, cmd);
|
||||
|
||||
out_sset:
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
static void fe_get_drvinfo(struct net_device *dev,
|
||||
struct ethtool_drvinfo *info)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(dev);
|
||||
struct fe_soc_data *soc = priv->soc;
|
||||
|
||||
strlcpy(info->driver, priv->dev->driver->name, sizeof(info->driver));
|
||||
strlcpy(info->version, MTK_FE_DRV_VERSION, sizeof(info->version));
|
||||
strlcpy(info->bus_info, dev_name(priv->dev), sizeof(info->bus_info));
|
||||
|
||||
if (soc->reg_table[FE_REG_FE_COUNTER_BASE])
|
||||
info->n_stats = ARRAY_SIZE(fe_gdma_str);
|
||||
}
|
||||
|
||||
static u32 fe_get_msglevel(struct net_device *dev)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(dev);
|
||||
|
||||
return priv->msg_enable;
|
||||
}
|
||||
|
||||
static void fe_set_msglevel(struct net_device *dev, u32 value)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(dev);
|
||||
|
||||
priv->msg_enable = value;
|
||||
}
|
||||
|
||||
static int fe_nway_reset(struct net_device *dev)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(dev);
|
||||
|
||||
if (!priv->phy_dev)
|
||||
goto out_nway_reset;
|
||||
|
||||
return genphy_restart_aneg(priv->phy_dev);
|
||||
|
||||
out_nway_reset:
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
|
||||
static u32 fe_get_link(struct net_device *dev)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(dev);
|
||||
int err;
|
||||
|
||||
if (!priv->phy_dev)
|
||||
goto out_get_link;
|
||||
|
||||
if (priv->phy_flags == FE_PHY_FLAG_ATTACH) {
|
||||
err = genphy_update_link(priv->phy_dev);
|
||||
if (err)
|
||||
goto out_get_link;
|
||||
}
|
||||
|
||||
return priv->phy_dev->link;
|
||||
|
||||
out_get_link:
|
||||
return ethtool_op_get_link(dev);
|
||||
}
|
||||
|
||||
static int fe_set_ringparam(struct net_device *dev,
|
||||
struct ethtool_ringparam *ring,
|
||||
struct kernel_ethtool_ringparam *kernel_rp,
|
||||
struct netlink_ext_ack *extack)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(dev);
|
||||
|
||||
if ((ring->tx_pending < 2) ||
|
||||
(ring->rx_pending < 2) ||
|
||||
(ring->rx_pending > MAX_DMA_DESC) ||
|
||||
(ring->tx_pending > MAX_DMA_DESC))
|
||||
return -EINVAL;
|
||||
|
||||
dev->netdev_ops->ndo_stop(dev);
|
||||
|
||||
priv->tx_ring.tx_ring_size = BIT(fls(ring->tx_pending) - 1);
|
||||
priv->rx_ring.rx_ring_size = BIT(fls(ring->rx_pending) - 1);
|
||||
|
||||
dev->netdev_ops->ndo_open(dev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void fe_get_ringparam(struct net_device *dev,
|
||||
struct ethtool_ringparam *ring,
|
||||
struct kernel_ethtool_ringparam *kernel_rp,
|
||||
struct netlink_ext_ack *extack)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(dev);
|
||||
|
||||
ring->rx_max_pending = MAX_DMA_DESC;
|
||||
ring->tx_max_pending = MAX_DMA_DESC;
|
||||
ring->rx_pending = priv->rx_ring.rx_ring_size;
|
||||
ring->tx_pending = priv->tx_ring.tx_ring_size;
|
||||
}
|
||||
|
||||
static void fe_get_strings(struct net_device *dev, u32 stringset, u8 *data)
|
||||
{
|
||||
switch (stringset) {
|
||||
case ETH_SS_STATS:
|
||||
ethtool_puts(&data, *fe_gdma_str);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static int fe_get_sset_count(struct net_device *dev, int sset)
|
||||
{
|
||||
switch (sset) {
|
||||
case ETH_SS_STATS:
|
||||
return ARRAY_SIZE(fe_gdma_str);
|
||||
default:
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
}
|
||||
|
||||
static void fe_get_ethtool_stats(struct net_device *dev,
|
||||
struct ethtool_stats *stats, u64 *data)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(dev);
|
||||
struct fe_hw_stats *hwstats = priv->hw_stats;
|
||||
u64 *data_src, *data_dst;
|
||||
unsigned int start;
|
||||
int i;
|
||||
|
||||
if (netif_running(dev) && netif_device_present(dev)) {
|
||||
if (spin_trylock(&hwstats->stats_lock)) {
|
||||
fe_stats_update(priv);
|
||||
spin_unlock(&hwstats->stats_lock);
|
||||
}
|
||||
}
|
||||
|
||||
do {
|
||||
data_src = &hwstats->tx_bytes;
|
||||
data_dst = data;
|
||||
start = u64_stats_fetch_begin(&hwstats->syncp);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(fe_gdma_str); i++)
|
||||
*data_dst++ = *data_src++;
|
||||
|
||||
} while (u64_stats_fetch_retry(&hwstats->syncp, start));
|
||||
}
|
||||
|
||||
static struct ethtool_ops fe_ethtool_ops = {
|
||||
.get_link_ksettings = fe_get_link_ksettings,
|
||||
.set_link_ksettings = fe_set_link_ksettings,
|
||||
.get_drvinfo = fe_get_drvinfo,
|
||||
.get_msglevel = fe_get_msglevel,
|
||||
.set_msglevel = fe_set_msglevel,
|
||||
.nway_reset = fe_nway_reset,
|
||||
.get_link = fe_get_link,
|
||||
.set_ringparam = fe_set_ringparam,
|
||||
.get_ringparam = fe_get_ringparam,
|
||||
};
|
||||
|
||||
void fe_set_ethtool_ops(struct net_device *netdev)
|
||||
{
|
||||
struct fe_priv *priv = netdev_priv(netdev);
|
||||
struct fe_soc_data *soc = priv->soc;
|
||||
|
||||
if (soc->reg_table[FE_REG_FE_COUNTER_BASE]) {
|
||||
fe_ethtool_ops.get_strings = fe_get_strings;
|
||||
fe_ethtool_ops.get_sset_count = fe_get_sset_count;
|
||||
fe_ethtool_ops.get_ethtool_stats = fe_get_ethtool_stats;
|
||||
}
|
||||
|
||||
netdev->ethtool_ops = &fe_ethtool_ops;
|
||||
}
|
||||
Reference in New Issue
Block a user