mirror of
https://github.com/Fishwaldo/Star64_linux.git
synced 2025-06-26 08:31:13 +00:00
net: sh_eth: add support for multicast filtering
Some controllers have TSU. It can filter multicast by hardware. This patch supports it. Signed-off-by: Yoshihiro Shimoda <yoshihiro.shimoda.uh@renesas.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
6ba88021c3
commit
6743fe6df4
2 changed files with 286 additions and 5 deletions
|
@ -1604,18 +1604,289 @@ static int sh_eth_do_ioctl(struct net_device *ndev, struct ifreq *rq,
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(SH_ETH_HAS_TSU)
|
#if defined(SH_ETH_HAS_TSU)
|
||||||
|
/* For TSU_POSTn. Please refer to the manual about this (strange) bitfields */
|
||||||
|
static void *sh_eth_tsu_get_post_reg_offset(struct sh_eth_private *mdp,
|
||||||
|
int entry)
|
||||||
|
{
|
||||||
|
return sh_eth_tsu_get_offset(mdp, TSU_POST1) + (entry / 8 * 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
static u32 sh_eth_tsu_get_post_mask(int entry)
|
||||||
|
{
|
||||||
|
return 0x0f << (28 - ((entry % 8) * 4));
|
||||||
|
}
|
||||||
|
|
||||||
|
static u32 sh_eth_tsu_get_post_bit(struct sh_eth_private *mdp, int entry)
|
||||||
|
{
|
||||||
|
return (0x08 >> (mdp->port << 1)) << (28 - ((entry % 8) * 4));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void sh_eth_tsu_enable_cam_entry_post(struct net_device *ndev,
|
||||||
|
int entry)
|
||||||
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
u32 tmp;
|
||||||
|
void *reg_offset;
|
||||||
|
|
||||||
|
reg_offset = sh_eth_tsu_get_post_reg_offset(mdp, entry);
|
||||||
|
tmp = ioread32(reg_offset);
|
||||||
|
iowrite32(tmp | sh_eth_tsu_get_post_bit(mdp, entry), reg_offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool sh_eth_tsu_disable_cam_entry_post(struct net_device *ndev,
|
||||||
|
int entry)
|
||||||
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
u32 post_mask, ref_mask, tmp;
|
||||||
|
void *reg_offset;
|
||||||
|
|
||||||
|
reg_offset = sh_eth_tsu_get_post_reg_offset(mdp, entry);
|
||||||
|
post_mask = sh_eth_tsu_get_post_mask(entry);
|
||||||
|
ref_mask = sh_eth_tsu_get_post_bit(mdp, entry) & ~post_mask;
|
||||||
|
|
||||||
|
tmp = ioread32(reg_offset);
|
||||||
|
iowrite32(tmp & ~post_mask, reg_offset);
|
||||||
|
|
||||||
|
/* If other port enables, the function returns "true" */
|
||||||
|
return tmp & ref_mask;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sh_eth_tsu_busy(struct net_device *ndev)
|
||||||
|
{
|
||||||
|
int timeout = SH_ETH_TSU_TIMEOUT_MS * 100;
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
|
||||||
|
while ((sh_eth_tsu_read(mdp, TSU_ADSBSY) & TSU_ADSBSY_0)) {
|
||||||
|
udelay(10);
|
||||||
|
timeout--;
|
||||||
|
if (timeout <= 0) {
|
||||||
|
dev_err(&ndev->dev, "%s: timeout\n", __func__);
|
||||||
|
return -ETIMEDOUT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sh_eth_tsu_write_entry(struct net_device *ndev, void *reg,
|
||||||
|
const u8 *addr)
|
||||||
|
{
|
||||||
|
u32 val;
|
||||||
|
|
||||||
|
val = addr[0] << 24 | addr[1] << 16 | addr[2] << 8 | addr[3];
|
||||||
|
iowrite32(val, reg);
|
||||||
|
if (sh_eth_tsu_busy(ndev) < 0)
|
||||||
|
return -EBUSY;
|
||||||
|
|
||||||
|
val = addr[4] << 8 | addr[5];
|
||||||
|
iowrite32(val, reg + 4);
|
||||||
|
if (sh_eth_tsu_busy(ndev) < 0)
|
||||||
|
return -EBUSY;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void sh_eth_tsu_read_entry(void *reg, u8 *addr)
|
||||||
|
{
|
||||||
|
u32 val;
|
||||||
|
|
||||||
|
val = ioread32(reg);
|
||||||
|
addr[0] = (val >> 24) & 0xff;
|
||||||
|
addr[1] = (val >> 16) & 0xff;
|
||||||
|
addr[2] = (val >> 8) & 0xff;
|
||||||
|
addr[3] = val & 0xff;
|
||||||
|
val = ioread32(reg + 4);
|
||||||
|
addr[4] = (val >> 8) & 0xff;
|
||||||
|
addr[5] = val & 0xff;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int sh_eth_tsu_find_entry(struct net_device *ndev, const u8 *addr)
|
||||||
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
void *reg_offset = sh_eth_tsu_get_offset(mdp, TSU_ADRH0);
|
||||||
|
int i;
|
||||||
|
u8 c_addr[ETH_ALEN];
|
||||||
|
|
||||||
|
for (i = 0; i < SH_ETH_TSU_CAM_ENTRIES; i++, reg_offset += 8) {
|
||||||
|
sh_eth_tsu_read_entry(reg_offset, c_addr);
|
||||||
|
if (memcmp(addr, c_addr, ETH_ALEN) == 0)
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -ENOENT;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sh_eth_tsu_find_empty(struct net_device *ndev)
|
||||||
|
{
|
||||||
|
u8 blank[ETH_ALEN];
|
||||||
|
int entry;
|
||||||
|
|
||||||
|
memset(blank, 0, sizeof(blank));
|
||||||
|
entry = sh_eth_tsu_find_entry(ndev, blank);
|
||||||
|
return (entry < 0) ? -ENOMEM : entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sh_eth_tsu_disable_cam_entry_table(struct net_device *ndev,
|
||||||
|
int entry)
|
||||||
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
void *reg_offset = sh_eth_tsu_get_offset(mdp, TSU_ADRH0);
|
||||||
|
int ret;
|
||||||
|
u8 blank[ETH_ALEN];
|
||||||
|
|
||||||
|
sh_eth_tsu_write(mdp, sh_eth_tsu_read(mdp, TSU_TEN) &
|
||||||
|
~(1 << (31 - entry)), TSU_TEN);
|
||||||
|
|
||||||
|
memset(blank, 0, sizeof(blank));
|
||||||
|
ret = sh_eth_tsu_write_entry(ndev, reg_offset + entry * 8, blank);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sh_eth_tsu_add_entry(struct net_device *ndev, const u8 *addr)
|
||||||
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
void *reg_offset = sh_eth_tsu_get_offset(mdp, TSU_ADRH0);
|
||||||
|
int i, ret;
|
||||||
|
|
||||||
|
if (!mdp->cd->tsu)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
i = sh_eth_tsu_find_entry(ndev, addr);
|
||||||
|
if (i < 0) {
|
||||||
|
/* No entry found, create one */
|
||||||
|
i = sh_eth_tsu_find_empty(ndev);
|
||||||
|
if (i < 0)
|
||||||
|
return -ENOMEM;
|
||||||
|
ret = sh_eth_tsu_write_entry(ndev, reg_offset + i * 8, addr);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/* Enable the entry */
|
||||||
|
sh_eth_tsu_write(mdp, sh_eth_tsu_read(mdp, TSU_TEN) |
|
||||||
|
(1 << (31 - i)), TSU_TEN);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Entry found or created, enable POST */
|
||||||
|
sh_eth_tsu_enable_cam_entry_post(ndev, i);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sh_eth_tsu_del_entry(struct net_device *ndev, const u8 *addr)
|
||||||
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
int i, ret;
|
||||||
|
|
||||||
|
if (!mdp->cd->tsu)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
i = sh_eth_tsu_find_entry(ndev, addr);
|
||||||
|
if (i) {
|
||||||
|
/* Entry found */
|
||||||
|
if (sh_eth_tsu_disable_cam_entry_post(ndev, i))
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
/* Disable the entry if both ports was disabled */
|
||||||
|
ret = sh_eth_tsu_disable_cam_entry_table(ndev, i);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
done:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int sh_eth_tsu_purge_all(struct net_device *ndev)
|
||||||
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
int i, ret;
|
||||||
|
|
||||||
|
if (unlikely(!mdp->cd->tsu))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
for (i = 0; i < SH_ETH_TSU_CAM_ENTRIES; i++) {
|
||||||
|
if (sh_eth_tsu_disable_cam_entry_post(ndev, i))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* Disable the entry if both ports was disabled */
|
||||||
|
ret = sh_eth_tsu_disable_cam_entry_table(ndev, i);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void sh_eth_tsu_purge_mcast(struct net_device *ndev)
|
||||||
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
u8 addr[ETH_ALEN];
|
||||||
|
void *reg_offset = sh_eth_tsu_get_offset(mdp, TSU_ADRH0);
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (unlikely(!mdp->cd->tsu))
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (i = 0; i < SH_ETH_TSU_CAM_ENTRIES; i++, reg_offset += 8) {
|
||||||
|
sh_eth_tsu_read_entry(reg_offset, addr);
|
||||||
|
if (is_multicast_ether_addr(addr))
|
||||||
|
sh_eth_tsu_del_entry(ndev, addr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Multicast reception directions set */
|
/* Multicast reception directions set */
|
||||||
static void sh_eth_set_multicast_list(struct net_device *ndev)
|
static void sh_eth_set_multicast_list(struct net_device *ndev)
|
||||||
{
|
{
|
||||||
|
struct sh_eth_private *mdp = netdev_priv(ndev);
|
||||||
|
u32 ecmr_bits;
|
||||||
|
int mcast_all = 0;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&mdp->lock, flags);
|
||||||
|
/*
|
||||||
|
* Initial condition is MCT = 1, PRM = 0.
|
||||||
|
* Depending on ndev->flags, set PRM or clear MCT
|
||||||
|
*/
|
||||||
|
ecmr_bits = (sh_eth_read(ndev, ECMR) & ~ECMR_PRM) | ECMR_MCT;
|
||||||
|
|
||||||
|
if (!(ndev->flags & IFF_MULTICAST)) {
|
||||||
|
sh_eth_tsu_purge_mcast(ndev);
|
||||||
|
mcast_all = 1;
|
||||||
|
}
|
||||||
|
if (ndev->flags & IFF_ALLMULTI) {
|
||||||
|
sh_eth_tsu_purge_mcast(ndev);
|
||||||
|
ecmr_bits &= ~ECMR_MCT;
|
||||||
|
mcast_all = 1;
|
||||||
|
}
|
||||||
|
|
||||||
if (ndev->flags & IFF_PROMISC) {
|
if (ndev->flags & IFF_PROMISC) {
|
||||||
/* Set promiscuous. */
|
sh_eth_tsu_purge_all(ndev);
|
||||||
sh_eth_write(ndev, (sh_eth_read(ndev, ECMR) & ~ECMR_MCT) |
|
ecmr_bits = (ecmr_bits & ~ECMR_MCT) | ECMR_PRM;
|
||||||
ECMR_PRM, ECMR);
|
} else if (mdp->cd->tsu) {
|
||||||
|
struct netdev_hw_addr *ha;
|
||||||
|
netdev_for_each_mc_addr(ha, ndev) {
|
||||||
|
if (mcast_all && is_multicast_ether_addr(ha->addr))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (sh_eth_tsu_add_entry(ndev, ha->addr) < 0) {
|
||||||
|
if (!mcast_all) {
|
||||||
|
sh_eth_tsu_purge_mcast(ndev);
|
||||||
|
ecmr_bits &= ~ECMR_MCT;
|
||||||
|
mcast_all = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
/* Normal, unicast/broadcast-only mode. */
|
/* Normal, unicast/broadcast-only mode. */
|
||||||
sh_eth_write(ndev, (sh_eth_read(ndev, ECMR) & ~ECMR_PRM) |
|
ecmr_bits = (ecmr_bits & ~ECMR_PRM) | ECMR_MCT;
|
||||||
ECMR_MCT, ECMR);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* update the ethernet mode */
|
||||||
|
sh_eth_write(ndev, ecmr_bits, ECMR);
|
||||||
|
|
||||||
|
spin_unlock_irqrestore(&mdp->lock, flags);
|
||||||
}
|
}
|
||||||
#endif /* SH_ETH_HAS_TSU */
|
#endif /* SH_ETH_HAS_TSU */
|
||||||
|
|
||||||
|
@ -1869,6 +2140,7 @@ static int sh_eth_drv_probe(struct platform_device *pdev)
|
||||||
}
|
}
|
||||||
mdp->tsu_addr = ioremap(rtsu->start,
|
mdp->tsu_addr = ioremap(rtsu->start,
|
||||||
resource_size(rtsu));
|
resource_size(rtsu));
|
||||||
|
mdp->port = devno % 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* initialize first or needed device */
|
/* initialize first or needed device */
|
||||||
|
|
|
@ -29,6 +29,8 @@
|
||||||
#define RX_RING_SIZE 64 /* Rx ring size */
|
#define RX_RING_SIZE 64 /* Rx ring size */
|
||||||
#define ETHERSMALL 60
|
#define ETHERSMALL 60
|
||||||
#define PKT_BUF_SZ 1538
|
#define PKT_BUF_SZ 1538
|
||||||
|
#define SH_ETH_TSU_TIMEOUT_MS 500
|
||||||
|
#define SH_ETH_TSU_CAM_ENTRIES 32
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
/* E-DMAC registers */
|
/* E-DMAC registers */
|
||||||
|
@ -778,6 +780,7 @@ struct sh_eth_private {
|
||||||
char post_rx; /* POST receive */
|
char post_rx; /* POST receive */
|
||||||
char post_fw; /* POST forward */
|
char post_fw; /* POST forward */
|
||||||
struct net_device_stats tsu_stats; /* TSU forward status */
|
struct net_device_stats tsu_stats; /* TSU forward status */
|
||||||
|
int port; /* for TSU */
|
||||||
|
|
||||||
unsigned no_ether_link:1;
|
unsigned no_ether_link:1;
|
||||||
unsigned ether_link_active_low:1;
|
unsigned ether_link_active_low:1;
|
||||||
|
@ -811,6 +814,12 @@ static inline unsigned long sh_eth_read(struct net_device *ndev,
|
||||||
return ioread32(mdp->addr + mdp->reg_offset[enum_index]);
|
return ioread32(mdp->addr + mdp->reg_offset[enum_index]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void *sh_eth_tsu_get_offset(struct sh_eth_private *mdp,
|
||||||
|
int enum_index)
|
||||||
|
{
|
||||||
|
return mdp->tsu_addr + mdp->reg_offset[enum_index];
|
||||||
|
}
|
||||||
|
|
||||||
static inline void sh_eth_tsu_write(struct sh_eth_private *mdp,
|
static inline void sh_eth_tsu_write(struct sh_eth_private *mdp,
|
||||||
unsigned long data, int enum_index)
|
unsigned long data, int enum_index)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue