return 0;
 }
 
-static void ixgb_set_speed_duplex(struct net_device *netdev)
+void ixgb_set_speed_duplex(struct net_device *netdev)
 {
        struct ixgb_adapter *adapter = netdev_priv(netdev);
        /* be optimistic about our link, since we were up before */
        return 0;
 }
 
-static u32
-ixgb_get_rx_csum(struct net_device *netdev)
-{
-       struct ixgb_adapter *adapter = netdev_priv(netdev);
-
-       return adapter->rx_csum;
-}
-
-static int
-ixgb_set_rx_csum(struct net_device *netdev, u32 data)
-{
-       struct ixgb_adapter *adapter = netdev_priv(netdev);
-
-       adapter->rx_csum = data;
-
-       if (netif_running(netdev)) {
-               ixgb_down(adapter, true);
-               ixgb_up(adapter);
-               ixgb_set_speed_duplex(netdev);
-       } else
-               ixgb_reset(adapter);
-       return 0;
-}
-
-static u32
-ixgb_get_tx_csum(struct net_device *netdev)
-{
-       return (netdev->features & NETIF_F_HW_CSUM) != 0;
-}
-
-static int
-ixgb_set_tx_csum(struct net_device *netdev, u32 data)
-{
-       if (data)
-               netdev->features |= NETIF_F_HW_CSUM;
-       else
-               netdev->features &= ~NETIF_F_HW_CSUM;
-
-       return 0;
-}
-
-static int
-ixgb_set_tso(struct net_device *netdev, u32 data)
-{
-       if (data)
-               netdev->features |= NETIF_F_TSO;
-       else
-               netdev->features &= ~NETIF_F_TSO;
-       return 0;
-}
-
 static u32
 ixgb_get_msglevel(struct net_device *netdev)
 {
        .set_ringparam = ixgb_set_ringparam,
        .get_pauseparam = ixgb_get_pauseparam,
        .set_pauseparam = ixgb_set_pauseparam,
-       .get_rx_csum = ixgb_get_rx_csum,
-       .set_rx_csum = ixgb_set_rx_csum,
-       .get_tx_csum = ixgb_get_tx_csum,
-       .set_tx_csum = ixgb_set_tx_csum,
-       .set_sg = ethtool_op_set_sg,
        .get_msglevel = ixgb_get_msglevel,
        .set_msglevel = ixgb_set_msglevel,
-       .set_tso = ixgb_set_tso,
        .get_strings = ixgb_get_strings,
        .set_phys_id = ixgb_set_phys_id,
        .get_sset_count = ixgb_get_sset_count,
 
        }
 }
 
+static int
+ixgb_set_features(struct net_device *netdev, u32 features)
+{
+       struct ixgb_adapter *adapter = netdev_priv(netdev);
+       u32 changed = features ^ netdev->features;
+
+       if (!(changed & NETIF_F_RXCSUM))
+               return 0;
+
+       adapter->rx_csum = !!(features & NETIF_F_RXCSUM);
+
+       if (netif_running(netdev)) {
+               ixgb_down(adapter, true);
+               ixgb_up(adapter);
+               ixgb_set_speed_duplex(netdev);
+       } else
+               ixgb_reset(adapter);
+
+       return 0;
+}
+
+
 static const struct net_device_ops ixgb_netdev_ops = {
        .ndo_open               = ixgb_open,
        .ndo_stop               = ixgb_close,
 #ifdef CONFIG_NET_POLL_CONTROLLER
        .ndo_poll_controller    = ixgb_netpoll,
 #endif
+       .ndo_set_features       = ixgb_set_features,
 };
 
 /**
        if (err)
                goto err_sw_init;
 
-       netdev->features = NETIF_F_SG |
-                          NETIF_F_HW_CSUM |
+       netdev->hw_features = NETIF_F_SG |
+                          NETIF_F_TSO |
+                          NETIF_F_HW_CSUM;
+       netdev->features = netdev->hw_features |
                           NETIF_F_HW_VLAN_TX |
                           NETIF_F_HW_VLAN_RX |
                           NETIF_F_HW_VLAN_FILTER;
-       netdev->features |= NETIF_F_TSO;
+       netdev->hw_features |= NETIF_F_RXCSUM;
 
        if (pci_using_dac) {
                netdev->features |= NETIF_F_HIGHDMA;