Commit 03a49a14 authored by David S. Miller's avatar David S. Miller

Merge branch 'stmmac-hash-table'

Biao Huang says:

====================
stmmac: fix out-of-boundary issue and add taller hash table support

Fix mac address out-of-boundary issue in net-next tree.
and resend the patch which was discussed in
https://lore.kernel.org/patchwork/patch/1082117
but with no further progress.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 6ec3d4d2 b8ef7020
...@@ -326,6 +326,7 @@ struct dma_features { ...@@ -326,6 +326,7 @@ struct dma_features {
/* 802.3az - Energy-Efficient Ethernet (EEE) */ /* 802.3az - Energy-Efficient Ethernet (EEE) */
unsigned int eee; unsigned int eee;
unsigned int av; unsigned int av;
unsigned int hash_tb_sz;
unsigned int tsoen; unsigned int tsoen;
/* TX and RX csum */ /* TX and RX csum */
unsigned int tx_coe; unsigned int tx_coe;
...@@ -424,9 +425,9 @@ struct mac_device_info { ...@@ -424,9 +425,9 @@ struct mac_device_info {
struct mii_regs mii; /* MII register Addresses */ struct mii_regs mii; /* MII register Addresses */
struct mac_link link; struct mac_link link;
void __iomem *pcsr; /* vpointer to device CSRs */ void __iomem *pcsr; /* vpointer to device CSRs */
int multicast_filter_bins; unsigned int multicast_filter_bins;
int unicast_filter_entries; unsigned int unicast_filter_entries;
int mcast_bits_log2; unsigned int mcast_bits_log2;
unsigned int rx_csum; unsigned int rx_csum;
unsigned int pcs; unsigned int pcs;
unsigned int pmt; unsigned int pmt;
......
...@@ -15,8 +15,7 @@ ...@@ -15,8 +15,7 @@
/* MAC registers */ /* MAC registers */
#define GMAC_CONFIG 0x00000000 #define GMAC_CONFIG 0x00000000
#define GMAC_PACKET_FILTER 0x00000008 #define GMAC_PACKET_FILTER 0x00000008
#define GMAC_HASH_TAB_0_31 0x00000010 #define GMAC_HASH_TAB(x) (0x10 + (x) * 4)
#define GMAC_HASH_TAB_32_63 0x00000014
#define GMAC_RX_FLOW_CTRL 0x00000090 #define GMAC_RX_FLOW_CTRL 0x00000090
#define GMAC_QX_TX_FLOW_CTRL(x) (0x70 + x * 4) #define GMAC_QX_TX_FLOW_CTRL(x) (0x70 + x * 4)
#define GMAC_TXQ_PRTY_MAP0 0x98 #define GMAC_TXQ_PRTY_MAP0 0x98
...@@ -181,6 +180,7 @@ enum power_event { ...@@ -181,6 +180,7 @@ enum power_event {
#define GMAC_HW_FEAT_MIISEL BIT(0) #define GMAC_HW_FEAT_MIISEL BIT(0)
/* MAC HW features1 bitmap */ /* MAC HW features1 bitmap */
#define GMAC_HW_HASH_TB_SZ GENMASK(25, 24)
#define GMAC_HW_FEAT_AVSEL BIT(20) #define GMAC_HW_FEAT_AVSEL BIT(20)
#define GMAC_HW_TSOEN BIT(18) #define GMAC_HW_TSOEN BIT(18)
#define GMAC_HW_TXFIFOSIZE GENMASK(10, 6) #define GMAC_HW_TXFIFOSIZE GENMASK(10, 6)
......
...@@ -400,41 +400,50 @@ static void dwmac4_set_filter(struct mac_device_info *hw, ...@@ -400,41 +400,50 @@ static void dwmac4_set_filter(struct mac_device_info *hw,
struct net_device *dev) struct net_device *dev)
{ {
void __iomem *ioaddr = (void __iomem *)dev->base_addr; void __iomem *ioaddr = (void __iomem *)dev->base_addr;
unsigned int value = 0; int numhashregs = (hw->multicast_filter_bins >> 5);
int mcbitslog2 = hw->mcast_bits_log2;
unsigned int value;
int i;
value = readl(ioaddr + GMAC_PACKET_FILTER);
value &= ~GMAC_PACKET_FILTER_HMC;
value &= ~GMAC_PACKET_FILTER_HPF;
value &= ~GMAC_PACKET_FILTER_PCF;
value &= ~GMAC_PACKET_FILTER_PM;
value &= ~GMAC_PACKET_FILTER_PR;
if (dev->flags & IFF_PROMISC) { if (dev->flags & IFF_PROMISC) {
value = GMAC_PACKET_FILTER_PR | GMAC_PACKET_FILTER_PCF; value = GMAC_PACKET_FILTER_PR | GMAC_PACKET_FILTER_PCF;
} else if ((dev->flags & IFF_ALLMULTI) || } else if ((dev->flags & IFF_ALLMULTI) ||
(netdev_mc_count(dev) > HASH_TABLE_SIZE)) { (netdev_mc_count(dev) > hw->multicast_filter_bins)) {
/* Pass all multi */ /* Pass all multi */
value = GMAC_PACKET_FILTER_PM; value |= GMAC_PACKET_FILTER_PM;
/* Set the 64 bits of the HASH tab. To be updated if taller /* Set all the bits of the HASH tab */
* hash table is used for (i = 0; i < numhashregs; i++)
*/ writel(0xffffffff, ioaddr + GMAC_HASH_TAB(i));
writel(0xffffffff, ioaddr + GMAC_HASH_TAB_0_31);
writel(0xffffffff, ioaddr + GMAC_HASH_TAB_32_63);
} else if (!netdev_mc_empty(dev)) { } else if (!netdev_mc_empty(dev)) {
u32 mc_filter[2];
struct netdev_hw_addr *ha; struct netdev_hw_addr *ha;
u32 mc_filter[8];
/* Hash filter for multicast */ /* Hash filter for multicast */
value = GMAC_PACKET_FILTER_HMC; value |= GMAC_PACKET_FILTER_HMC;
memset(mc_filter, 0, sizeof(mc_filter)); memset(mc_filter, 0, sizeof(mc_filter));
netdev_for_each_mc_addr(ha, dev) { netdev_for_each_mc_addr(ha, dev) {
/* The upper 6 bits of the calculated CRC are used to /* The upper n bits of the calculated CRC are used to
* index the content of the Hash Table Reg 0 and 1. * index the contents of the hash table. The number of
* bits used depends on the hardware configuration
* selected at core configuration time.
*/ */
int bit_nr = int bit_nr = bitrev32(~crc32_le(~0, ha->addr,
(bitrev32(~crc32_le(~0, ha->addr, 6)) >> 26); ETH_ALEN)) >> (32 - mcbitslog2);
/* The most significant bit determines the register /* The most significant bit determines the register to
* to use while the other 5 bits determines the bit * use (H/L) while the other 5 bits determine the bit
* within the selected register * within the register.
*/ */
mc_filter[bit_nr >> 5] |= (1 << (bit_nr & 0x1F)); mc_filter[bit_nr >> 5] |= (1 << (bit_nr & 0x1f));
} }
writel(mc_filter[0], ioaddr + GMAC_HASH_TAB_0_31); for (i = 0; i < numhashregs; i++)
writel(mc_filter[1], ioaddr + GMAC_HASH_TAB_32_63); writel(mc_filter[i], ioaddr + GMAC_HASH_TAB(i));
} }
value |= GMAC_PACKET_FILTER_HPF; value |= GMAC_PACKET_FILTER_HPF;
...@@ -454,7 +463,7 @@ static void dwmac4_set_filter(struct mac_device_info *hw, ...@@ -454,7 +463,7 @@ static void dwmac4_set_filter(struct mac_device_info *hw,
reg++; reg++;
} }
while (reg <= GMAC_MAX_PERFECT_ADDRESSES) { while (reg < GMAC_MAX_PERFECT_ADDRESSES) {
writel(0, ioaddr + GMAC_ADDR_HIGH(reg)); writel(0, ioaddr + GMAC_ADDR_HIGH(reg));
writel(0, ioaddr + GMAC_ADDR_LOW(reg)); writel(0, ioaddr + GMAC_ADDR_LOW(reg));
reg++; reg++;
......
...@@ -351,6 +351,7 @@ static void dwmac4_get_hw_feature(void __iomem *ioaddr, ...@@ -351,6 +351,7 @@ static void dwmac4_get_hw_feature(void __iomem *ioaddr,
/* MAC HW feature1 */ /* MAC HW feature1 */
hw_cap = readl(ioaddr + GMAC_HW_FEATURE1); hw_cap = readl(ioaddr + GMAC_HW_FEATURE1);
dma_cap->hash_tb_sz = (hw_cap & GMAC_HW_HASH_TB_SZ) >> 24;
dma_cap->av = (hw_cap & GMAC_HW_FEAT_AVSEL) >> 20; dma_cap->av = (hw_cap & GMAC_HW_FEAT_AVSEL) >> 20;
dma_cap->tsoen = (hw_cap & GMAC_HW_TSOEN) >> 18; dma_cap->tsoen = (hw_cap & GMAC_HW_TSOEN) >> 18;
/* RX and TX FIFO sizes are encoded as log2(n / 128). Undo that by /* RX and TX FIFO sizes are encoded as log2(n / 128). Undo that by
......
...@@ -4107,6 +4107,12 @@ static int stmmac_hw_init(struct stmmac_priv *priv) ...@@ -4107,6 +4107,12 @@ static int stmmac_hw_init(struct stmmac_priv *priv)
priv->plat->enh_desc = priv->dma_cap.enh_desc; priv->plat->enh_desc = priv->dma_cap.enh_desc;
priv->plat->pmt = priv->dma_cap.pmt_remote_wake_up; priv->plat->pmt = priv->dma_cap.pmt_remote_wake_up;
priv->hw->pmt = priv->plat->pmt; priv->hw->pmt = priv->plat->pmt;
if (priv->dma_cap.hash_tb_sz) {
priv->hw->multicast_filter_bins =
(BIT(priv->dma_cap.hash_tb_sz) << 5);
priv->hw->mcast_bits_log2 =
ilog2(priv->hw->multicast_filter_bins);
}
/* TXCOE doesn't work in thresh DMA mode */ /* TXCOE doesn't work in thresh DMA mode */
if (priv->plat->force_thresh_dma_mode) if (priv->plat->force_thresh_dma_mode)
......
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