stmmac: fix the rx csum feature
For new GMACs it is possible to turn-on/off the COE. In the current driver, when disabled the Rx-checksum via ethtool, the tool reported that csum was disabled but the HW continued to set the IPC. Indeed this is because the fix_features allows this. So the patch fixes this problem by adding the set_features. Signed-off-by: Giuseppe Cavallaro <peppe.cavallaro@st.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
8a3cf39b72
commit
d2afb5bdff
|
@ -445,6 +445,7 @@ struct mac_device_info {
|
||||||
int multicast_filter_bins;
|
int multicast_filter_bins;
|
||||||
int unicast_filter_entries;
|
int unicast_filter_entries;
|
||||||
int mcast_bits_log2;
|
int mcast_bits_log2;
|
||||||
|
unsigned int rx_csum;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mac_device_info *dwmac1000_setup(void __iomem *ioaddr, int mcbins,
|
struct mac_device_info *dwmac1000_setup(void __iomem *ioaddr, int mcbins,
|
||||||
|
|
|
@ -58,7 +58,11 @@ static int dwmac1000_rx_ipc_enable(struct mac_device_info *hw)
|
||||||
void __iomem *ioaddr = hw->pcsr;
|
void __iomem *ioaddr = hw->pcsr;
|
||||||
u32 value = readl(ioaddr + GMAC_CONTROL);
|
u32 value = readl(ioaddr + GMAC_CONTROL);
|
||||||
|
|
||||||
value |= GMAC_CONTROL_IPC;
|
if (hw->rx_csum)
|
||||||
|
value |= GMAC_CONTROL_IPC;
|
||||||
|
else
|
||||||
|
value &= ~GMAC_CONTROL_IPC;
|
||||||
|
|
||||||
writel(value, ioaddr + GMAC_CONTROL);
|
writel(value, ioaddr + GMAC_CONTROL);
|
||||||
|
|
||||||
value = readl(ioaddr + GMAC_CONTROL);
|
value = readl(ioaddr + GMAC_CONTROL);
|
||||||
|
|
|
@ -1673,6 +1673,7 @@ static int stmmac_hw_setup(struct net_device *dev)
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
pr_warn(" RX IPC Checksum Offload disabled\n");
|
pr_warn(" RX IPC Checksum Offload disabled\n");
|
||||||
priv->plat->rx_coe = STMMAC_RX_COE_NONE;
|
priv->plat->rx_coe = STMMAC_RX_COE_NONE;
|
||||||
|
priv->hw->rx_csum = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Enable the MAC Rx/Tx */
|
/* Enable the MAC Rx/Tx */
|
||||||
|
@ -2111,7 +2112,7 @@ static int stmmac_rx(struct stmmac_priv *priv, int limit)
|
||||||
unsigned int entry = priv->cur_rx % rxsize;
|
unsigned int entry = priv->cur_rx % rxsize;
|
||||||
unsigned int next_entry;
|
unsigned int next_entry;
|
||||||
unsigned int count = 0;
|
unsigned int count = 0;
|
||||||
int coe = priv->plat->rx_coe;
|
int coe = priv->hw->rx_csum;
|
||||||
|
|
||||||
if (netif_msg_rx_status(priv)) {
|
if (netif_msg_rx_status(priv)) {
|
||||||
pr_debug("%s: descriptor ring:\n", __func__);
|
pr_debug("%s: descriptor ring:\n", __func__);
|
||||||
|
@ -2334,6 +2335,7 @@ static netdev_features_t stmmac_fix_features(struct net_device *dev,
|
||||||
features &= ~NETIF_F_RXCSUM;
|
features &= ~NETIF_F_RXCSUM;
|
||||||
else if (priv->plat->rx_coe == STMMAC_RX_COE_TYPE1)
|
else if (priv->plat->rx_coe == STMMAC_RX_COE_TYPE1)
|
||||||
features &= ~NETIF_F_IPV6_CSUM;
|
features &= ~NETIF_F_IPV6_CSUM;
|
||||||
|
|
||||||
if (!priv->plat->tx_coe)
|
if (!priv->plat->tx_coe)
|
||||||
features &= ~NETIF_F_ALL_CSUM;
|
features &= ~NETIF_F_ALL_CSUM;
|
||||||
|
|
||||||
|
@ -2348,6 +2350,24 @@ static netdev_features_t stmmac_fix_features(struct net_device *dev,
|
||||||
return features;
|
return features;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int stmmac_set_features(struct net_device *netdev,
|
||||||
|
netdev_features_t features)
|
||||||
|
{
|
||||||
|
struct stmmac_priv *priv = netdev_priv(netdev);
|
||||||
|
|
||||||
|
/* Keep the COE Type in case of csum is supporting */
|
||||||
|
if (features & NETIF_F_RXCSUM)
|
||||||
|
priv->hw->rx_csum = priv->plat->rx_coe;
|
||||||
|
else
|
||||||
|
priv->hw->rx_csum = 0;
|
||||||
|
/* No check needed because rx_coe has been set before and it will be
|
||||||
|
* fixed in case of issue.
|
||||||
|
*/
|
||||||
|
priv->hw->mac->rx_ipc(priv->hw);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* stmmac_interrupt - main ISR
|
* stmmac_interrupt - main ISR
|
||||||
* @irq: interrupt number.
|
* @irq: interrupt number.
|
||||||
|
@ -2628,6 +2648,7 @@ static const struct net_device_ops stmmac_netdev_ops = {
|
||||||
.ndo_stop = stmmac_release,
|
.ndo_stop = stmmac_release,
|
||||||
.ndo_change_mtu = stmmac_change_mtu,
|
.ndo_change_mtu = stmmac_change_mtu,
|
||||||
.ndo_fix_features = stmmac_fix_features,
|
.ndo_fix_features = stmmac_fix_features,
|
||||||
|
.ndo_set_features = stmmac_set_features,
|
||||||
.ndo_set_rx_mode = stmmac_set_rx_mode,
|
.ndo_set_rx_mode = stmmac_set_rx_mode,
|
||||||
.ndo_tx_timeout = stmmac_tx_timeout,
|
.ndo_tx_timeout = stmmac_tx_timeout,
|
||||||
.ndo_do_ioctl = stmmac_ioctl,
|
.ndo_do_ioctl = stmmac_ioctl,
|
||||||
|
@ -2704,9 +2725,11 @@ static int stmmac_hw_init(struct stmmac_priv *priv)
|
||||||
/* To use alternate (extended) or normal descriptor structures */
|
/* To use alternate (extended) or normal descriptor structures */
|
||||||
stmmac_selec_desc_mode(priv);
|
stmmac_selec_desc_mode(priv);
|
||||||
|
|
||||||
if (priv->plat->rx_coe)
|
if (priv->plat->rx_coe) {
|
||||||
|
priv->hw->rx_csum = priv->plat->rx_coe;
|
||||||
pr_info(" RX Checksum Offload Engine supported (type %d)\n",
|
pr_info(" RX Checksum Offload Engine supported (type %d)\n",
|
||||||
priv->plat->rx_coe);
|
priv->plat->rx_coe);
|
||||||
|
}
|
||||||
if (priv->plat->tx_coe)
|
if (priv->plat->tx_coe)
|
||||||
pr_info(" TX Checksum insertion supported\n");
|
pr_info(" TX Checksum insertion supported\n");
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue