]> bbs.cooldavid.org Git - net-next-2.6.git/blobdiff - drivers/net/sfc/siena.c
sfc: Enable IPv6 RSS using random key for Toeplitz hash
[net-next-2.6.git] / drivers / net / sfc / siena.c
index 1619fb5a64f5c0574b3dca25cdb49abd7d5f5571..7bf93faff3ab28b1770efb2ebc4094a1d9120da9 100644 (file)
@@ -12,6 +12,8 @@
 #include <linux/delay.h>
 #include <linux/pci.h>
 #include <linux/module.h>
+#include <linux/slab.h>
+#include <linux/random.h>
 #include "net_driver.h"
 #include "bitfield.h"
 #include "efx.h"
@@ -273,6 +275,9 @@ static int siena_probe_nic(struct efx_nic *efx)
                goto fail5;
        }
 
+       get_random_bytes(&nic_data->ipv6_rss_key,
+                        sizeof(nic_data->ipv6_rss_key));
+
        return 0;
 
 fail5:
@@ -292,6 +297,7 @@ fail1:
  */
 static int siena_init_nic(struct efx_nic *efx)
 {
+       struct siena_nic_data *nic_data = efx->nic_data;
        efx_oword_t temp;
        int rc;
 
@@ -318,6 +324,20 @@ static int siena_init_nic(struct efx_nic *efx)
        EFX_SET_OWORD_FIELD(temp, FRF_BZ_RX_INGR_EN, 1);
        efx_writeo(efx, &temp, FR_AZ_RX_CFG);
 
+       /* Enable IPv6 RSS */
+       BUILD_BUG_ON(sizeof(nic_data->ipv6_rss_key) !=
+                    2 * sizeof(temp) + FRF_CZ_RX_RSS_IPV6_TKEY_HI_WIDTH / 8 ||
+                    FRF_CZ_RX_RSS_IPV6_TKEY_HI_LBN != 0);
+       memcpy(&temp, nic_data->ipv6_rss_key, sizeof(temp));
+       efx_writeo(efx, &temp, FR_CZ_RX_RSS_IPV6_REG1);
+       memcpy(&temp, nic_data->ipv6_rss_key + sizeof(temp), sizeof(temp));
+       efx_writeo(efx, &temp, FR_CZ_RX_RSS_IPV6_REG2);
+       EFX_POPULATE_OWORD_2(temp, FRF_CZ_RX_RSS_IPV6_THASH_ENABLE, 1,
+                            FRF_CZ_RX_RSS_IPV6_IP_THASH_ENABLE, 1);
+       memcpy(&temp, nic_data->ipv6_rss_key + 2 * sizeof(temp),
+              FRF_CZ_RX_RSS_IPV6_TKEY_HI_WIDTH / 8);
+       efx_writeo(efx, &temp, FR_CZ_RX_RSS_IPV6_REG3);
+
        if (efx_nic_rx_xoff_thresh >= 0 || efx_nic_rx_xon_thresh >= 0)
                /* No MCDI operation has been defined to set thresholds */
                EFX_ERR(efx, "ignoring RX flow control thresholds\n");