hyperv: Add handling of IP header with option field in netvsc_set_hash()
In case that the IP header has optional field at the end, this patch will get the port numbers after that field, and compute the hash. The general parser skb_flow_dissect() is used here. Signed-off-by: Haiyang Zhang <haiyangz@microsoft.com> Reviewed-by: K. Y. Srinivasan <kys@microsoft.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
f47de068f6
commit
f88e67149f
@ -162,7 +162,7 @@ union sub_key {
|
|||||||
* data: network byte order
|
* data: network byte order
|
||||||
* return: host byte order
|
* return: host byte order
|
||||||
*/
|
*/
|
||||||
static u32 comp_hash(u8 *key, int klen, u8 *data, int dlen)
|
static u32 comp_hash(u8 *key, int klen, void *data, int dlen)
|
||||||
{
|
{
|
||||||
union sub_key subk;
|
union sub_key subk;
|
||||||
int k_next = 4;
|
int k_next = 4;
|
||||||
@ -176,7 +176,7 @@ static u32 comp_hash(u8 *key, int klen, u8 *data, int dlen)
|
|||||||
for (i = 0; i < dlen; i++) {
|
for (i = 0; i < dlen; i++) {
|
||||||
subk.kb = key[k_next];
|
subk.kb = key[k_next];
|
||||||
k_next = (k_next + 1) % klen;
|
k_next = (k_next + 1) % klen;
|
||||||
dt = data[i];
|
dt = ((u8 *)data)[i];
|
||||||
for (j = 0; j < 8; j++) {
|
for (j = 0; j < 8; j++) {
|
||||||
if (dt & 0x80)
|
if (dt & 0x80)
|
||||||
ret ^= subk.ka;
|
ret ^= subk.ka;
|
||||||
@ -190,26 +190,20 @@ static u32 comp_hash(u8 *key, int klen, u8 *data, int dlen)
|
|||||||
|
|
||||||
static bool netvsc_set_hash(u32 *hash, struct sk_buff *skb)
|
static bool netvsc_set_hash(u32 *hash, struct sk_buff *skb)
|
||||||
{
|
{
|
||||||
struct iphdr *iphdr;
|
struct flow_keys flow;
|
||||||
int data_len;
|
int data_len;
|
||||||
bool ret = false;
|
|
||||||
|
|
||||||
if (eth_hdr(skb)->h_proto != htons(ETH_P_IP))
|
if (!skb_flow_dissect(skb, &flow) || flow.n_proto != htons(ETH_P_IP))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
iphdr = ip_hdr(skb);
|
if (flow.ip_proto == IPPROTO_TCP)
|
||||||
|
data_len = 12;
|
||||||
|
else
|
||||||
|
data_len = 8;
|
||||||
|
|
||||||
if (iphdr->version == 4) {
|
*hash = comp_hash(netvsc_hash_key, HASH_KEYLEN, &flow, data_len);
|
||||||
if (iphdr->protocol == IPPROTO_TCP)
|
|
||||||
data_len = 12;
|
|
||||||
else
|
|
||||||
data_len = 8;
|
|
||||||
*hash = comp_hash(netvsc_hash_key, HASH_KEYLEN,
|
|
||||||
(u8 *)&iphdr->saddr, data_len);
|
|
||||||
ret = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static u16 netvsc_select_queue(struct net_device *ndev, struct sk_buff *skb,
|
static u16 netvsc_select_queue(struct net_device *ndev, struct sk_buff *skb,
|
||||||
|
Loading…
Reference in New Issue
Block a user