diff options
Diffstat (limited to 'drivers/net/typhoon.c')
-rw-r--r-- | drivers/net/typhoon.c | 174 |
1 files changed, 55 insertions, 119 deletions
diff --git a/drivers/net/typhoon.c b/drivers/net/typhoon.c index 2e50077ff450..3de4283344e9 100644 --- a/drivers/net/typhoon.c +++ b/drivers/net/typhoon.c | |||
@@ -24,10 +24,6 @@ | |||
24 | 3XP Processor. It has been tested on x86 and sparc64. | 24 | 3XP Processor. It has been tested on x86 and sparc64. |
25 | 25 | ||
26 | KNOWN ISSUES: | 26 | KNOWN ISSUES: |
27 | *) The current firmware always strips the VLAN tag off, even if | ||
28 | we tell it not to. You should filter VLANs at the switch | ||
29 | as a workaround (good practice in any event) until we can | ||
30 | get this fixed. | ||
31 | *) Cannot DMA Rx packets to a 2 byte aligned address. Also firmware | 27 | *) Cannot DMA Rx packets to a 2 byte aligned address. Also firmware |
32 | issue. Hopefully 3Com will fix it. | 28 | issue. Hopefully 3Com will fix it. |
33 | *) Waiting for a command response takes 8ms due to non-preemptable | 29 | *) Waiting for a command response takes 8ms due to non-preemptable |
@@ -127,12 +123,11 @@ static const int multicast_filter_limit = 32; | |||
127 | #include <linux/in6.h> | 123 | #include <linux/in6.h> |
128 | #include <linux/dma-mapping.h> | 124 | #include <linux/dma-mapping.h> |
129 | #include <linux/firmware.h> | 125 | #include <linux/firmware.h> |
130 | #include <generated/utsrelease.h> | ||
131 | 126 | ||
132 | #include "typhoon.h" | 127 | #include "typhoon.h" |
133 | 128 | ||
134 | MODULE_AUTHOR("David Dillow <dave@thedillows.org>"); | 129 | MODULE_AUTHOR("David Dillow <dave@thedillows.org>"); |
135 | MODULE_VERSION(UTS_RELEASE); | 130 | MODULE_VERSION("1.0"); |
136 | MODULE_LICENSE("GPL"); | 131 | MODULE_LICENSE("GPL"); |
137 | MODULE_FIRMWARE(FIRMWARE_NAME); | 132 | MODULE_FIRMWARE(FIRMWARE_NAME); |
138 | MODULE_DESCRIPTION("3Com Typhoon Family (3C990, 3CR990, and variants)"); | 133 | MODULE_DESCRIPTION("3Com Typhoon Family (3C990, 3CR990, and variants)"); |
@@ -280,8 +275,6 @@ struct typhoon { | |||
280 | struct pci_dev * pdev; | 275 | struct pci_dev * pdev; |
281 | struct net_device * dev; | 276 | struct net_device * dev; |
282 | struct napi_struct napi; | 277 | struct napi_struct napi; |
283 | spinlock_t state_lock; | ||
284 | struct vlan_group * vlgrp; | ||
285 | struct basic_ring rxHiRing; | 278 | struct basic_ring rxHiRing; |
286 | struct basic_ring rxBuffRing; | 279 | struct basic_ring rxBuffRing; |
287 | struct rxbuff_ent rxbuffers[RXENT_ENTRIES]; | 280 | struct rxbuff_ent rxbuffers[RXENT_ENTRIES]; |
@@ -541,7 +534,7 @@ cleanup: | |||
541 | 534 | ||
542 | indexes->respCleared = cpu_to_le32(cleared); | 535 | indexes->respCleared = cpu_to_le32(cleared); |
543 | wmb(); | 536 | wmb(); |
544 | return (resp_save == NULL); | 537 | return resp_save == NULL; |
545 | } | 538 | } |
546 | 539 | ||
547 | static inline int | 540 | static inline int |
@@ -695,44 +688,6 @@ out: | |||
695 | return err; | 688 | return err; |
696 | } | 689 | } |
697 | 690 | ||
698 | static void | ||
699 | typhoon_vlan_rx_register(struct net_device *dev, struct vlan_group *grp) | ||
700 | { | ||
701 | struct typhoon *tp = netdev_priv(dev); | ||
702 | struct cmd_desc xp_cmd; | ||
703 | int err; | ||
704 | |||
705 | spin_lock_bh(&tp->state_lock); | ||
706 | if(!tp->vlgrp != !grp) { | ||
707 | /* We've either been turned on for the first time, or we've | ||
708 | * been turned off. Update the 3XP. | ||
709 | */ | ||
710 | if(grp) | ||
711 | tp->offload |= TYPHOON_OFFLOAD_VLAN; | ||
712 | else | ||
713 | tp->offload &= ~TYPHOON_OFFLOAD_VLAN; | ||
714 | |||
715 | /* If the interface is up, the runtime is running -- and we | ||
716 | * must be up for the vlan core to call us. | ||
717 | * | ||
718 | * Do the command outside of the spin lock, as it is slow. | ||
719 | */ | ||
720 | INIT_COMMAND_WITH_RESPONSE(&xp_cmd, | ||
721 | TYPHOON_CMD_SET_OFFLOAD_TASKS); | ||
722 | xp_cmd.parm2 = tp->offload; | ||
723 | xp_cmd.parm3 = tp->offload; | ||
724 | spin_unlock_bh(&tp->state_lock); | ||
725 | err = typhoon_issue_command(tp, 1, &xp_cmd, 0, NULL); | ||
726 | if(err < 0) | ||
727 | netdev_err(tp->dev, "vlan offload error %d\n", -err); | ||
728 | spin_lock_bh(&tp->state_lock); | ||
729 | } | ||
730 | |||
731 | /* now make the change visible */ | ||
732 | tp->vlgrp = grp; | ||
733 | spin_unlock_bh(&tp->state_lock); | ||
734 | } | ||
735 | |||
736 | static inline void | 691 | static inline void |
737 | typhoon_tso_fill(struct sk_buff *skb, struct transmit_ring *txRing, | 692 | typhoon_tso_fill(struct sk_buff *skb, struct transmit_ring *txRing, |
738 | u32 ring_dma) | 693 | u32 ring_dma) |
@@ -818,7 +773,7 @@ typhoon_start_tx(struct sk_buff *skb, struct net_device *dev) | |||
818 | first_txd->processFlags |= | 773 | first_txd->processFlags |= |
819 | TYPHOON_TX_PF_INSERT_VLAN | TYPHOON_TX_PF_VLAN_PRIORITY; | 774 | TYPHOON_TX_PF_INSERT_VLAN | TYPHOON_TX_PF_VLAN_PRIORITY; |
820 | first_txd->processFlags |= | 775 | first_txd->processFlags |= |
821 | cpu_to_le32(ntohs(vlan_tx_tag_get(skb)) << | 776 | cpu_to_le32(htons(vlan_tx_tag_get(skb)) << |
822 | TYPHOON_TX_PF_VLAN_TAG_SHIFT); | 777 | TYPHOON_TX_PF_VLAN_TAG_SHIFT); |
823 | } | 778 | } |
824 | 779 | ||
@@ -891,7 +846,7 @@ typhoon_start_tx(struct sk_buff *skb, struct net_device *dev) | |||
891 | if(typhoon_num_free_tx(txRing) < (numDesc + 2)) { | 846 | if(typhoon_num_free_tx(txRing) < (numDesc + 2)) { |
892 | netif_stop_queue(dev); | 847 | netif_stop_queue(dev); |
893 | 848 | ||
894 | /* A Tx complete IRQ could have gotten inbetween, making | 849 | /* A Tx complete IRQ could have gotten between, making |
895 | * the ring free again. Only need to recheck here, since | 850 | * the ring free again. Only need to recheck here, since |
896 | * Tx is serialized. | 851 | * Tx is serialized. |
897 | */ | 852 | */ |
@@ -936,7 +891,7 @@ typhoon_set_rx_mode(struct net_device *dev) | |||
936 | filter |= TYPHOON_RX_FILTER_MCAST_HASH; | 891 | filter |= TYPHOON_RX_FILTER_MCAST_HASH; |
937 | } | 892 | } |
938 | 893 | ||
939 | INIT_COMMAND_NO_RESPONSE(&xp_cmd, TYPHOON_CMD_SET_RX_FILTER); | 894 | INIT_COMMAND_WITH_RESPONSE(&xp_cmd, TYPHOON_CMD_SET_RX_FILTER); |
940 | xp_cmd.parm1 = filter; | 895 | xp_cmd.parm1 = filter; |
941 | typhoon_issue_command(tp, 1, &xp_cmd, 0, NULL); | 896 | typhoon_issue_command(tp, 1, &xp_cmd, 0, NULL); |
942 | } | 897 | } |
@@ -962,36 +917,34 @@ typhoon_do_get_stats(struct typhoon *tp) | |||
962 | * The extra status reported would be a good candidate for | 917 | * The extra status reported would be a good candidate for |
963 | * ethtool_ops->get_{strings,stats}() | 918 | * ethtool_ops->get_{strings,stats}() |
964 | */ | 919 | */ |
965 | stats->tx_packets = le32_to_cpu(s->txPackets); | 920 | stats->tx_packets = le32_to_cpu(s->txPackets) + |
966 | stats->tx_bytes = le64_to_cpu(s->txBytes); | 921 | saved->tx_packets; |
967 | stats->tx_errors = le32_to_cpu(s->txCarrierLost); | 922 | stats->tx_bytes = le64_to_cpu(s->txBytes) + |
968 | stats->tx_carrier_errors = le32_to_cpu(s->txCarrierLost); | 923 | saved->tx_bytes; |
969 | stats->collisions = le32_to_cpu(s->txMultipleCollisions); | 924 | stats->tx_errors = le32_to_cpu(s->txCarrierLost) + |
970 | stats->rx_packets = le32_to_cpu(s->rxPacketsGood); | 925 | saved->tx_errors; |
971 | stats->rx_bytes = le64_to_cpu(s->rxBytesGood); | 926 | stats->tx_carrier_errors = le32_to_cpu(s->txCarrierLost) + |
972 | stats->rx_fifo_errors = le32_to_cpu(s->rxFifoOverruns); | 927 | saved->tx_carrier_errors; |
928 | stats->collisions = le32_to_cpu(s->txMultipleCollisions) + | ||
929 | saved->collisions; | ||
930 | stats->rx_packets = le32_to_cpu(s->rxPacketsGood) + | ||
931 | saved->rx_packets; | ||
932 | stats->rx_bytes = le64_to_cpu(s->rxBytesGood) + | ||
933 | saved->rx_bytes; | ||
934 | stats->rx_fifo_errors = le32_to_cpu(s->rxFifoOverruns) + | ||
935 | saved->rx_fifo_errors; | ||
973 | stats->rx_errors = le32_to_cpu(s->rxFifoOverruns) + | 936 | stats->rx_errors = le32_to_cpu(s->rxFifoOverruns) + |
974 | le32_to_cpu(s->BadSSD) + le32_to_cpu(s->rxCrcErrors); | 937 | le32_to_cpu(s->BadSSD) + le32_to_cpu(s->rxCrcErrors) + |
975 | stats->rx_crc_errors = le32_to_cpu(s->rxCrcErrors); | 938 | saved->rx_errors; |
976 | stats->rx_length_errors = le32_to_cpu(s->rxOversized); | 939 | stats->rx_crc_errors = le32_to_cpu(s->rxCrcErrors) + |
940 | saved->rx_crc_errors; | ||
941 | stats->rx_length_errors = le32_to_cpu(s->rxOversized) + | ||
942 | saved->rx_length_errors; | ||
977 | tp->speed = (s->linkStatus & TYPHOON_LINK_100MBPS) ? | 943 | tp->speed = (s->linkStatus & TYPHOON_LINK_100MBPS) ? |
978 | SPEED_100 : SPEED_10; | 944 | SPEED_100 : SPEED_10; |
979 | tp->duplex = (s->linkStatus & TYPHOON_LINK_FULL_DUPLEX) ? | 945 | tp->duplex = (s->linkStatus & TYPHOON_LINK_FULL_DUPLEX) ? |
980 | DUPLEX_FULL : DUPLEX_HALF; | 946 | DUPLEX_FULL : DUPLEX_HALF; |
981 | 947 | ||
982 | /* add in the saved statistics | ||
983 | */ | ||
984 | stats->tx_packets += saved->tx_packets; | ||
985 | stats->tx_bytes += saved->tx_bytes; | ||
986 | stats->tx_errors += saved->tx_errors; | ||
987 | stats->collisions += saved->collisions; | ||
988 | stats->rx_packets += saved->rx_packets; | ||
989 | stats->rx_bytes += saved->rx_bytes; | ||
990 | stats->rx_fifo_errors += saved->rx_fifo_errors; | ||
991 | stats->rx_errors += saved->rx_errors; | ||
992 | stats->rx_crc_errors += saved->rx_crc_errors; | ||
993 | stats->rx_length_errors += saved->rx_length_errors; | ||
994 | |||
995 | return 0; | 948 | return 0; |
996 | } | 949 | } |
997 | 950 | ||
@@ -1050,7 +1003,6 @@ typhoon_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info) | |||
1050 | } | 1003 | } |
1051 | 1004 | ||
1052 | strcpy(info->driver, KBUILD_MODNAME); | 1005 | strcpy(info->driver, KBUILD_MODNAME); |
1053 | strcpy(info->version, UTS_RELEASE); | ||
1054 | strcpy(info->bus_info, pci_name(pci_dev)); | 1006 | strcpy(info->bus_info, pci_name(pci_dev)); |
1055 | } | 1007 | } |
1056 | 1008 | ||
@@ -1098,7 +1050,7 @@ typhoon_get_settings(struct net_device *dev, struct ethtool_cmd *cmd) | |||
1098 | 1050 | ||
1099 | /* need to get stats to make these link speed/duplex valid */ | 1051 | /* need to get stats to make these link speed/duplex valid */ |
1100 | typhoon_do_get_stats(tp); | 1052 | typhoon_do_get_stats(tp); |
1101 | cmd->speed = tp->speed; | 1053 | ethtool_cmd_speed_set(cmd, tp->speed); |
1102 | cmd->duplex = tp->duplex; | 1054 | cmd->duplex = tp->duplex; |
1103 | cmd->phy_address = 0; | 1055 | cmd->phy_address = 0; |
1104 | cmd->transceiver = XCVR_INTERNAL; | 1056 | cmd->transceiver = XCVR_INTERNAL; |
@@ -1116,25 +1068,26 @@ static int | |||
1116 | typhoon_set_settings(struct net_device *dev, struct ethtool_cmd *cmd) | 1068 | typhoon_set_settings(struct net_device *dev, struct ethtool_cmd *cmd) |
1117 | { | 1069 | { |
1118 | struct typhoon *tp = netdev_priv(dev); | 1070 | struct typhoon *tp = netdev_priv(dev); |
1071 | u32 speed = ethtool_cmd_speed(cmd); | ||
1119 | struct cmd_desc xp_cmd; | 1072 | struct cmd_desc xp_cmd; |
1120 | __le16 xcvr; | 1073 | __le16 xcvr; |
1121 | int err; | 1074 | int err; |
1122 | 1075 | ||
1123 | err = -EINVAL; | 1076 | err = -EINVAL; |
1124 | if(cmd->autoneg == AUTONEG_ENABLE) { | 1077 | if (cmd->autoneg == AUTONEG_ENABLE) { |
1125 | xcvr = TYPHOON_XCVR_AUTONEG; | 1078 | xcvr = TYPHOON_XCVR_AUTONEG; |
1126 | } else { | 1079 | } else { |
1127 | if(cmd->duplex == DUPLEX_HALF) { | 1080 | if (cmd->duplex == DUPLEX_HALF) { |
1128 | if(cmd->speed == SPEED_10) | 1081 | if (speed == SPEED_10) |
1129 | xcvr = TYPHOON_XCVR_10HALF; | 1082 | xcvr = TYPHOON_XCVR_10HALF; |
1130 | else if(cmd->speed == SPEED_100) | 1083 | else if (speed == SPEED_100) |
1131 | xcvr = TYPHOON_XCVR_100HALF; | 1084 | xcvr = TYPHOON_XCVR_100HALF; |
1132 | else | 1085 | else |
1133 | goto out; | 1086 | goto out; |
1134 | } else if(cmd->duplex == DUPLEX_FULL) { | 1087 | } else if (cmd->duplex == DUPLEX_FULL) { |
1135 | if(cmd->speed == SPEED_10) | 1088 | if (speed == SPEED_10) |
1136 | xcvr = TYPHOON_XCVR_10FULL; | 1089 | xcvr = TYPHOON_XCVR_10FULL; |
1137 | else if(cmd->speed == SPEED_100) | 1090 | else if (speed == SPEED_100) |
1138 | xcvr = TYPHOON_XCVR_100FULL; | 1091 | xcvr = TYPHOON_XCVR_100FULL; |
1139 | else | 1092 | else |
1140 | goto out; | 1093 | goto out; |
@@ -1153,7 +1106,7 @@ typhoon_set_settings(struct net_device *dev, struct ethtool_cmd *cmd) | |||
1153 | tp->speed = 0xff; /* invalid */ | 1106 | tp->speed = 0xff; /* invalid */ |
1154 | tp->duplex = 0xff; /* invalid */ | 1107 | tp->duplex = 0xff; /* invalid */ |
1155 | } else { | 1108 | } else { |
1156 | tp->speed = cmd->speed; | 1109 | tp->speed = speed; |
1157 | tp->duplex = cmd->duplex; | 1110 | tp->duplex = cmd->duplex; |
1158 | } | 1111 | } |
1159 | 1112 | ||
@@ -1192,14 +1145,6 @@ typhoon_set_wol(struct net_device *dev, struct ethtool_wolinfo *wol) | |||
1192 | return 0; | 1145 | return 0; |
1193 | } | 1146 | } |
1194 | 1147 | ||
1195 | static u32 | ||
1196 | typhoon_get_rx_csum(struct net_device *dev) | ||
1197 | { | ||
1198 | /* For now, we don't allow turning off RX checksums. | ||
1199 | */ | ||
1200 | return 1; | ||
1201 | } | ||
1202 | |||
1203 | static void | 1148 | static void |
1204 | typhoon_get_ringparam(struct net_device *dev, struct ethtool_ringparam *ering) | 1149 | typhoon_get_ringparam(struct net_device *dev, struct ethtool_ringparam *ering) |
1205 | { | 1150 | { |
@@ -1221,10 +1166,6 @@ static const struct ethtool_ops typhoon_ethtool_ops = { | |||
1221 | .get_wol = typhoon_get_wol, | 1166 | .get_wol = typhoon_get_wol, |
1222 | .set_wol = typhoon_set_wol, | 1167 | .set_wol = typhoon_set_wol, |
1223 | .get_link = ethtool_op_get_link, | 1168 | .get_link = ethtool_op_get_link, |
1224 | .get_rx_csum = typhoon_get_rx_csum, | ||
1225 | .set_tx_csum = ethtool_op_set_tx_csum, | ||
1226 | .set_sg = ethtool_op_set_sg, | ||
1227 | .set_tso = ethtool_op_set_tso, | ||
1228 | .get_ringparam = typhoon_get_ringparam, | 1169 | .get_ringparam = typhoon_get_ringparam, |
1229 | }; | 1170 | }; |
1230 | 1171 | ||
@@ -1311,9 +1252,9 @@ typhoon_init_interface(struct typhoon *tp) | |||
1311 | 1252 | ||
1312 | tp->offload = TYPHOON_OFFLOAD_IP_CHKSUM | TYPHOON_OFFLOAD_TCP_CHKSUM; | 1253 | tp->offload = TYPHOON_OFFLOAD_IP_CHKSUM | TYPHOON_OFFLOAD_TCP_CHKSUM; |
1313 | tp->offload |= TYPHOON_OFFLOAD_UDP_CHKSUM | TSO_OFFLOAD_ON; | 1254 | tp->offload |= TYPHOON_OFFLOAD_UDP_CHKSUM | TSO_OFFLOAD_ON; |
1255 | tp->offload |= TYPHOON_OFFLOAD_VLAN; | ||
1314 | 1256 | ||
1315 | spin_lock_init(&tp->command_lock); | 1257 | spin_lock_init(&tp->command_lock); |
1316 | spin_lock_init(&tp->state_lock); | ||
1317 | 1258 | ||
1318 | /* Force the writes to the shared memory area out before continuing. */ | 1259 | /* Force the writes to the shared memory area out before continuing. */ |
1319 | wmb(); | 1260 | wmb(); |
@@ -1330,7 +1271,7 @@ typhoon_init_rings(struct typhoon *tp) | |||
1330 | tp->rxHiRing.lastWrite = 0; | 1271 | tp->rxHiRing.lastWrite = 0; |
1331 | tp->rxBuffRing.lastWrite = 0; | 1272 | tp->rxBuffRing.lastWrite = 0; |
1332 | tp->cmdRing.lastWrite = 0; | 1273 | tp->cmdRing.lastWrite = 0; |
1333 | tp->cmdRing.lastWrite = 0; | 1274 | tp->respRing.lastWrite = 0; |
1334 | 1275 | ||
1335 | tp->txLoRing.lastRead = 0; | 1276 | tp->txLoRing.lastRead = 0; |
1336 | tp->txHiRing.lastRead = 0; | 1277 | tp->txHiRing.lastRead = 0; |
@@ -1762,15 +1703,12 @@ typhoon_rx(struct typhoon *tp, struct basic_ring *rxRing, volatile __le32 * read | |||
1762 | (TYPHOON_RX_IP_CHK_GOOD | TYPHOON_RX_UDP_CHK_GOOD)) { | 1703 | (TYPHOON_RX_IP_CHK_GOOD | TYPHOON_RX_UDP_CHK_GOOD)) { |
1763 | new_skb->ip_summed = CHECKSUM_UNNECESSARY; | 1704 | new_skb->ip_summed = CHECKSUM_UNNECESSARY; |
1764 | } else | 1705 | } else |
1765 | new_skb->ip_summed = CHECKSUM_NONE; | 1706 | skb_checksum_none_assert(new_skb); |
1766 | 1707 | ||
1767 | spin_lock(&tp->state_lock); | 1708 | if (rx->rxStatus & TYPHOON_RX_VLAN) |
1768 | if(tp->vlgrp != NULL && rx->rxStatus & TYPHOON_RX_VLAN) | 1709 | __vlan_hwaccel_put_tag(new_skb, |
1769 | vlan_hwaccel_receive_skb(new_skb, tp->vlgrp, | 1710 | ntohl(rx->vlanTag) & 0xffff); |
1770 | ntohl(rx->vlanTag) & 0xffff); | 1711 | netif_receive_skb(new_skb); |
1771 | else | ||
1772 | netif_receive_skb(new_skb); | ||
1773 | spin_unlock(&tp->state_lock); | ||
1774 | 1712 | ||
1775 | received++; | 1713 | received++; |
1776 | budget--; | 1714 | budget--; |
@@ -1991,11 +1929,9 @@ typhoon_start_runtime(struct typhoon *tp) | |||
1991 | goto error_out; | 1929 | goto error_out; |
1992 | 1930 | ||
1993 | INIT_COMMAND_NO_RESPONSE(&xp_cmd, TYPHOON_CMD_SET_OFFLOAD_TASKS); | 1931 | INIT_COMMAND_NO_RESPONSE(&xp_cmd, TYPHOON_CMD_SET_OFFLOAD_TASKS); |
1994 | spin_lock_bh(&tp->state_lock); | ||
1995 | xp_cmd.parm2 = tp->offload; | 1932 | xp_cmd.parm2 = tp->offload; |
1996 | xp_cmd.parm3 = tp->offload; | 1933 | xp_cmd.parm3 = tp->offload; |
1997 | err = typhoon_issue_command(tp, 1, &xp_cmd, 0, NULL); | 1934 | err = typhoon_issue_command(tp, 1, &xp_cmd, 0, NULL); |
1998 | spin_unlock_bh(&tp->state_lock); | ||
1999 | if(err < 0) | 1935 | if(err < 0) |
2000 | goto error_out; | 1936 | goto error_out; |
2001 | 1937 | ||
@@ -2233,13 +2169,9 @@ typhoon_suspend(struct pci_dev *pdev, pm_message_t state) | |||
2233 | if(!netif_running(dev)) | 2169 | if(!netif_running(dev)) |
2234 | return 0; | 2170 | return 0; |
2235 | 2171 | ||
2236 | spin_lock_bh(&tp->state_lock); | 2172 | /* TYPHOON_OFFLOAD_VLAN is always on now, so this doesn't work */ |
2237 | if(tp->vlgrp && tp->wol_events & TYPHOON_WAKE_MAGIC_PKT) { | 2173 | if(tp->wol_events & TYPHOON_WAKE_MAGIC_PKT) |
2238 | spin_unlock_bh(&tp->state_lock); | 2174 | netdev_warn(dev, "cannot do WAKE_MAGIC with VLAN offloading\n"); |
2239 | netdev_err(dev, "cannot do WAKE_MAGIC with VLANS\n"); | ||
2240 | return -EBUSY; | ||
2241 | } | ||
2242 | spin_unlock_bh(&tp->state_lock); | ||
2243 | 2175 | ||
2244 | netif_device_detach(dev); | 2176 | netif_device_detach(dev); |
2245 | 2177 | ||
@@ -2340,7 +2272,6 @@ static const struct net_device_ops typhoon_netdev_ops = { | |||
2340 | .ndo_validate_addr = eth_validate_addr, | 2272 | .ndo_validate_addr = eth_validate_addr, |
2341 | .ndo_set_mac_address = typhoon_set_mac_address, | 2273 | .ndo_set_mac_address = typhoon_set_mac_address, |
2342 | .ndo_change_mtu = eth_change_mtu, | 2274 | .ndo_change_mtu = eth_change_mtu, |
2343 | .ndo_vlan_rx_register = typhoon_vlan_rx_register, | ||
2344 | }; | 2275 | }; |
2345 | 2276 | ||
2346 | static int __devinit | 2277 | static int __devinit |
@@ -2524,10 +2455,15 @@ typhoon_init_one(struct pci_dev *pdev, const struct pci_device_id *ent) | |||
2524 | 2455 | ||
2525 | /* We can handle scatter gather, up to 16 entries, and | 2456 | /* We can handle scatter gather, up to 16 entries, and |
2526 | * we can do IP checksumming (only version 4, doh...) | 2457 | * we can do IP checksumming (only version 4, doh...) |
2458 | * | ||
2459 | * There's no way to turn off the RX VLAN offloading and stripping | ||
2460 | * on the current 3XP firmware -- it does not respect the offload | ||
2461 | * settings -- so we only allow the user to toggle the TX processing. | ||
2527 | */ | 2462 | */ |
2528 | dev->features |= NETIF_F_SG | NETIF_F_IP_CSUM; | 2463 | dev->hw_features = NETIF_F_SG | NETIF_F_IP_CSUM | NETIF_F_TSO | |
2529 | dev->features |= NETIF_F_HW_VLAN_TX | NETIF_F_HW_VLAN_RX; | 2464 | NETIF_F_HW_VLAN_TX; |
2530 | dev->features |= NETIF_F_TSO; | 2465 | dev->features = dev->hw_features | |
2466 | NETIF_F_HW_VLAN_RX | NETIF_F_RXCSUM; | ||
2531 | 2467 | ||
2532 | if(register_netdev(dev) < 0) { | 2468 | if(register_netdev(dev) < 0) { |
2533 | err_msg = "unable to register netdev"; | 2469 | err_msg = "unable to register netdev"; |