diff options
Diffstat (limited to 'drivers/net/skge.c')
-rw-r--r-- | drivers/net/skge.c | 47 |
1 files changed, 34 insertions, 13 deletions
diff --git a/drivers/net/skge.c b/drivers/net/skge.c index c3d2e0a2c4e6..eea75a401b0c 100644 --- a/drivers/net/skge.c +++ b/drivers/net/skge.c | |||
@@ -77,13 +77,13 @@ static const struct pci_device_id skge_id_table[] = { | |||
77 | { PCI_DEVICE(PCI_VENDOR_ID_3COM, PCI_DEVICE_ID_3COM_3C940B) }, | 77 | { PCI_DEVICE(PCI_VENDOR_ID_3COM, PCI_DEVICE_ID_3COM_3C940B) }, |
78 | { PCI_DEVICE(PCI_VENDOR_ID_SYSKONNECT, PCI_DEVICE_ID_SYSKONNECT_GE) }, | 78 | { PCI_DEVICE(PCI_VENDOR_ID_SYSKONNECT, PCI_DEVICE_ID_SYSKONNECT_GE) }, |
79 | { PCI_DEVICE(PCI_VENDOR_ID_SYSKONNECT, PCI_DEVICE_ID_SYSKONNECT_YU) }, | 79 | { PCI_DEVICE(PCI_VENDOR_ID_SYSKONNECT, PCI_DEVICE_ID_SYSKONNECT_YU) }, |
80 | { PCI_DEVICE(PCI_VENDOR_ID_DLINK, PCI_DEVICE_ID_DLINK_DGE510T), }, | 80 | { PCI_DEVICE(PCI_VENDOR_ID_DLINK, PCI_DEVICE_ID_DLINK_DGE510T) }, |
81 | { PCI_DEVICE(PCI_VENDOR_ID_DLINK, 0x4b01) }, /* DGE-530T */ | 81 | { PCI_DEVICE(PCI_VENDOR_ID_DLINK, 0x4b01) }, /* DGE-530T */ |
82 | { PCI_DEVICE(PCI_VENDOR_ID_MARVELL, 0x4320) }, | 82 | { PCI_DEVICE(PCI_VENDOR_ID_MARVELL, 0x4320) }, |
83 | { PCI_DEVICE(PCI_VENDOR_ID_MARVELL, 0x5005) }, /* Belkin */ | 83 | { PCI_DEVICE(PCI_VENDOR_ID_MARVELL, 0x5005) }, /* Belkin */ |
84 | { PCI_DEVICE(PCI_VENDOR_ID_CNET, PCI_DEVICE_ID_CNET_GIGACARD) }, | 84 | { PCI_DEVICE(PCI_VENDOR_ID_CNET, PCI_DEVICE_ID_CNET_GIGACARD) }, |
85 | { PCI_DEVICE(PCI_VENDOR_ID_LINKSYS, PCI_DEVICE_ID_LINKSYS_EG1064) }, | 85 | { PCI_DEVICE(PCI_VENDOR_ID_LINKSYS, PCI_DEVICE_ID_LINKSYS_EG1064) }, |
86 | { PCI_VENDOR_ID_LINKSYS, 0x1032, PCI_ANY_ID, 0x0015, }, | 86 | { PCI_VENDOR_ID_LINKSYS, 0x1032, PCI_ANY_ID, 0x0015 }, |
87 | { 0 } | 87 | { 0 } |
88 | }; | 88 | }; |
89 | MODULE_DEVICE_TABLE(pci, skge_id_table); | 89 | MODULE_DEVICE_TABLE(pci, skge_id_table); |
@@ -2767,6 +2767,17 @@ static int skge_change_mtu(struct net_device *dev, int new_mtu) | |||
2767 | return err; | 2767 | return err; |
2768 | } | 2768 | } |
2769 | 2769 | ||
2770 | static const u8 pause_mc_addr[ETH_ALEN] = { 0x1, 0x80, 0xc2, 0x0, 0x0, 0x1 }; | ||
2771 | |||
2772 | static void genesis_add_filter(u8 filter[8], const u8 *addr) | ||
2773 | { | ||
2774 | u32 crc, bit; | ||
2775 | |||
2776 | crc = ether_crc_le(ETH_ALEN, addr); | ||
2777 | bit = ~crc & 0x3f; | ||
2778 | filter[bit/8] |= 1 << (bit%8); | ||
2779 | } | ||
2780 | |||
2770 | static void genesis_set_multicast(struct net_device *dev) | 2781 | static void genesis_set_multicast(struct net_device *dev) |
2771 | { | 2782 | { |
2772 | struct skge_port *skge = netdev_priv(dev); | 2783 | struct skge_port *skge = netdev_priv(dev); |
@@ -2788,24 +2799,33 @@ static void genesis_set_multicast(struct net_device *dev) | |||
2788 | memset(filter, 0xff, sizeof(filter)); | 2799 | memset(filter, 0xff, sizeof(filter)); |
2789 | else { | 2800 | else { |
2790 | memset(filter, 0, sizeof(filter)); | 2801 | memset(filter, 0, sizeof(filter)); |
2791 | for (i = 0; list && i < count; i++, list = list->next) { | 2802 | |
2792 | u32 crc, bit; | 2803 | if (skge->flow_status == FLOW_STAT_REM_SEND |
2793 | crc = ether_crc_le(ETH_ALEN, list->dmi_addr); | 2804 | || skge->flow_status == FLOW_STAT_SYMMETRIC) |
2794 | bit = ~crc & 0x3f; | 2805 | genesis_add_filter(filter, pause_mc_addr); |
2795 | filter[bit/8] |= 1 << (bit%8); | 2806 | |
2796 | } | 2807 | for (i = 0; list && i < count; i++, list = list->next) |
2808 | genesis_add_filter(filter, list->dmi_addr); | ||
2797 | } | 2809 | } |
2798 | 2810 | ||
2799 | xm_write32(hw, port, XM_MODE, mode); | 2811 | xm_write32(hw, port, XM_MODE, mode); |
2800 | xm_outhash(hw, port, XM_HSM, filter); | 2812 | xm_outhash(hw, port, XM_HSM, filter); |
2801 | } | 2813 | } |
2802 | 2814 | ||
2815 | static void yukon_add_filter(u8 filter[8], const u8 *addr) | ||
2816 | { | ||
2817 | u32 bit = ether_crc(ETH_ALEN, addr) & 0x3f; | ||
2818 | filter[bit/8] |= 1 << (bit%8); | ||
2819 | } | ||
2820 | |||
2803 | static void yukon_set_multicast(struct net_device *dev) | 2821 | static void yukon_set_multicast(struct net_device *dev) |
2804 | { | 2822 | { |
2805 | struct skge_port *skge = netdev_priv(dev); | 2823 | struct skge_port *skge = netdev_priv(dev); |
2806 | struct skge_hw *hw = skge->hw; | 2824 | struct skge_hw *hw = skge->hw; |
2807 | int port = skge->port; | 2825 | int port = skge->port; |
2808 | struct dev_mc_list *list = dev->mc_list; | 2826 | struct dev_mc_list *list = dev->mc_list; |
2827 | int rx_pause = (skge->flow_status == FLOW_STAT_REM_SEND | ||
2828 | || skge->flow_status == FLOW_STAT_SYMMETRIC); | ||
2809 | u16 reg; | 2829 | u16 reg; |
2810 | u8 filter[8]; | 2830 | u8 filter[8]; |
2811 | 2831 | ||
@@ -2818,16 +2838,17 @@ static void yukon_set_multicast(struct net_device *dev) | |||
2818 | reg &= ~(GM_RXCR_UCF_ENA | GM_RXCR_MCF_ENA); | 2838 | reg &= ~(GM_RXCR_UCF_ENA | GM_RXCR_MCF_ENA); |
2819 | else if (dev->flags & IFF_ALLMULTI) /* all multicast */ | 2839 | else if (dev->flags & IFF_ALLMULTI) /* all multicast */ |
2820 | memset(filter, 0xff, sizeof(filter)); | 2840 | memset(filter, 0xff, sizeof(filter)); |
2821 | else if (dev->mc_count == 0) /* no multicast */ | 2841 | else if (dev->mc_count == 0 && !rx_pause)/* no multicast */ |
2822 | reg &= ~GM_RXCR_MCF_ENA; | 2842 | reg &= ~GM_RXCR_MCF_ENA; |
2823 | else { | 2843 | else { |
2824 | int i; | 2844 | int i; |
2825 | reg |= GM_RXCR_MCF_ENA; | 2845 | reg |= GM_RXCR_MCF_ENA; |
2826 | 2846 | ||
2827 | for (i = 0; list && i < dev->mc_count; i++, list = list->next) { | 2847 | if (rx_pause) |
2828 | u32 bit = ether_crc(ETH_ALEN, list->dmi_addr) & 0x3f; | 2848 | yukon_add_filter(filter, pause_mc_addr); |
2829 | filter[bit/8] |= 1 << (bit%8); | 2849 | |
2830 | } | 2850 | for (i = 0; list && i < dev->mc_count; i++, list = list->next) |
2851 | yukon_add_filter(filter, list->dmi_addr); | ||
2831 | } | 2852 | } |
2832 | 2853 | ||
2833 | 2854 | ||