-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
yaml --- r: 277996 b: refs/heads/master c: 0744dd0 h: refs/heads/master v: v3
- Loading branch information
Eric Dumazet
authored and
David S. Miller
committed
Nov 29, 2011
1 parent
38bb23d
commit 88d46f9
Showing
4 changed files
with
151 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,2 +1,2 @@ | ||
--- | ||
refs/heads/master: de0396f4003a24a57875b35f2996fdaa47bc1d0d | ||
refs/heads/master: 0744dd00c1b1be99a25b62b1b48df440e82e57e0 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
#ifndef _NET_FLOW_KEYS_H | ||
#define _NET_FLOW_KEYS_H | ||
|
||
struct flow_keys { | ||
__be32 src; | ||
__be32 dst; | ||
union { | ||
__be32 ports; | ||
__be16 port16[2]; | ||
}; | ||
u8 ip_proto; | ||
}; | ||
|
||
extern bool skb_flow_dissect(const struct sk_buff *skb, struct flow_keys *flow); | ||
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,134 @@ | ||
#include <linux/skbuff.h> | ||
#include <linux/ip.h> | ||
#include <linux/ipv6.h> | ||
#include <linux/if_vlan.h> | ||
#include <net/ip.h> | ||
#include <linux/if_tunnel.h> | ||
#include <linux/if_pppox.h> | ||
#include <linux/ppp_defs.h> | ||
#include <net/flow_keys.h> | ||
|
||
|
||
bool skb_flow_dissect(const struct sk_buff *skb, struct flow_keys *flow) | ||
{ | ||
int poff, nhoff = skb_network_offset(skb); | ||
u8 ip_proto; | ||
__be16 proto = skb->protocol; | ||
|
||
memset(flow, 0, sizeof(*flow)); | ||
|
||
again: | ||
switch (proto) { | ||
case __constant_htons(ETH_P_IP): { | ||
const struct iphdr *iph; | ||
struct iphdr _iph; | ||
ip: | ||
iph = skb_header_pointer(skb, nhoff, sizeof(_iph), &_iph); | ||
if (!iph) | ||
return false; | ||
|
||
if (ip_is_fragment(iph)) | ||
ip_proto = 0; | ||
else | ||
ip_proto = iph->protocol; | ||
flow->src = iph->saddr; | ||
flow->dst = iph->daddr; | ||
nhoff += iph->ihl * 4; | ||
break; | ||
} | ||
case __constant_htons(ETH_P_IPV6): { | ||
const struct ipv6hdr *iph; | ||
struct ipv6hdr _iph; | ||
ipv6: | ||
iph = skb_header_pointer(skb, nhoff, sizeof(_iph), &_iph); | ||
if (!iph) | ||
return false; | ||
|
||
ip_proto = iph->nexthdr; | ||
flow->src = iph->saddr.s6_addr32[3]; | ||
flow->dst = iph->daddr.s6_addr32[3]; | ||
nhoff += sizeof(struct ipv6hdr); | ||
break; | ||
} | ||
case __constant_htons(ETH_P_8021Q): { | ||
const struct vlan_hdr *vlan; | ||
struct vlan_hdr _vlan; | ||
|
||
vlan = skb_header_pointer(skb, nhoff, sizeof(_vlan), &_vlan); | ||
if (!vlan) | ||
return false; | ||
|
||
proto = vlan->h_vlan_encapsulated_proto; | ||
nhoff += sizeof(*vlan); | ||
goto again; | ||
} | ||
case __constant_htons(ETH_P_PPP_SES): { | ||
struct { | ||
struct pppoe_hdr hdr; | ||
__be16 proto; | ||
} *hdr, _hdr; | ||
hdr = skb_header_pointer(skb, nhoff, sizeof(_hdr), &_hdr); | ||
if (!hdr) | ||
return false; | ||
proto = hdr->proto; | ||
nhoff += PPPOE_SES_HLEN; | ||
switch (proto) { | ||
case __constant_htons(PPP_IP): | ||
goto ip; | ||
case __constant_htons(PPP_IPV6): | ||
goto ipv6; | ||
default: | ||
return false; | ||
} | ||
} | ||
default: | ||
return false; | ||
} | ||
|
||
switch (ip_proto) { | ||
case IPPROTO_GRE: { | ||
struct gre_hdr { | ||
__be16 flags; | ||
__be16 proto; | ||
} *hdr, _hdr; | ||
|
||
hdr = skb_header_pointer(skb, nhoff, sizeof(_hdr), &_hdr); | ||
if (!hdr) | ||
return false; | ||
/* | ||
* Only look inside GRE if version zero and no | ||
* routing | ||
*/ | ||
if (!(hdr->flags & (GRE_VERSION|GRE_ROUTING))) { | ||
proto = hdr->proto; | ||
nhoff += 4; | ||
if (hdr->flags & GRE_CSUM) | ||
nhoff += 4; | ||
if (hdr->flags & GRE_KEY) | ||
nhoff += 4; | ||
if (hdr->flags & GRE_SEQ) | ||
nhoff += 4; | ||
goto again; | ||
} | ||
break; | ||
} | ||
case IPPROTO_IPIP: | ||
goto again; | ||
default: | ||
break; | ||
} | ||
|
||
flow->ip_proto = ip_proto; | ||
poff = proto_ports_offset(ip_proto); | ||
if (poff >= 0) { | ||
__be32 *ports, _ports; | ||
|
||
nhoff += poff; | ||
ports = skb_header_pointer(skb, nhoff, sizeof(_ports), &_ports); | ||
if (ports) | ||
flow->ports = *ports; | ||
} | ||
|
||
return true; | ||
} | ||
EXPORT_SYMBOL(skb_flow_dissect); |