diff options
Diffstat (limited to 'net/sched/cls_flow.c')
| -rw-r--r-- | net/sched/cls_flow.c | 17 |
1 files changed, 15 insertions, 2 deletions
diff --git a/net/sched/cls_flow.c b/net/sched/cls_flow.c index 8d7698621f0a..971b867e0484 100644 --- a/net/sched/cls_flow.c +++ b/net/sched/cls_flow.c | |||
| @@ -19,6 +19,7 @@ | |||
| 19 | #include <linux/in.h> | 19 | #include <linux/in.h> |
| 20 | #include <linux/ip.h> | 20 | #include <linux/ip.h> |
| 21 | #include <linux/ipv6.h> | 21 | #include <linux/ipv6.h> |
| 22 | #include <linux/if_vlan.h> | ||
| 22 | 23 | ||
| 23 | #include <net/pkt_cls.h> | 24 | #include <net/pkt_cls.h> |
| 24 | #include <net/ip.h> | 25 | #include <net/ip.h> |
| @@ -270,6 +271,15 @@ static u32 flow_get_skgid(const struct sk_buff *skb) | |||
| 270 | return 0; | 271 | return 0; |
| 271 | } | 272 | } |
| 272 | 273 | ||
| 274 | static u32 flow_get_vlan_tag(const struct sk_buff *skb) | ||
| 275 | { | ||
| 276 | u16 uninitialized_var(tag); | ||
| 277 | |||
| 278 | if (vlan_get_tag(skb, &tag) < 0) | ||
| 279 | return 0; | ||
| 280 | return tag & VLAN_VID_MASK; | ||
| 281 | } | ||
| 282 | |||
| 273 | static u32 flow_key_get(const struct sk_buff *skb, int key) | 283 | static u32 flow_key_get(const struct sk_buff *skb, int key) |
| 274 | { | 284 | { |
| 275 | switch (key) { | 285 | switch (key) { |
| @@ -305,6 +315,8 @@ static u32 flow_key_get(const struct sk_buff *skb, int key) | |||
| 305 | return flow_get_skuid(skb); | 315 | return flow_get_skuid(skb); |
| 306 | case FLOW_KEY_SKGID: | 316 | case FLOW_KEY_SKGID: |
| 307 | return flow_get_skgid(skb); | 317 | return flow_get_skgid(skb); |
| 318 | case FLOW_KEY_VLAN_TAG: | ||
| 319 | return flow_get_vlan_tag(skb); | ||
| 308 | default: | 320 | default: |
| 309 | WARN_ON(1); | 321 | WARN_ON(1); |
| 310 | return 0; | 322 | return 0; |
| @@ -402,12 +414,13 @@ static int flow_change(struct tcf_proto *tp, unsigned long base, | |||
| 402 | 414 | ||
| 403 | if (tb[TCA_FLOW_KEYS]) { | 415 | if (tb[TCA_FLOW_KEYS]) { |
| 404 | keymask = nla_get_u32(tb[TCA_FLOW_KEYS]); | 416 | keymask = nla_get_u32(tb[TCA_FLOW_KEYS]); |
| 405 | if (fls(keymask) - 1 > FLOW_KEY_MAX) | ||
| 406 | return -EOPNOTSUPP; | ||
| 407 | 417 | ||
| 408 | nkeys = hweight32(keymask); | 418 | nkeys = hweight32(keymask); |
| 409 | if (nkeys == 0) | 419 | if (nkeys == 0) |
| 410 | return -EINVAL; | 420 | return -EINVAL; |
| 421 | |||
| 422 | if (fls(keymask) - 1 > FLOW_KEY_MAX) | ||
| 423 | return -EOPNOTSUPP; | ||
| 411 | } | 424 | } |
| 412 | 425 | ||
| 413 | err = tcf_exts_validate(tp, tb, tca[TCA_RATE], &e, &flow_ext_map); | 426 | err = tcf_exts_validate(tp, tb, tca[TCA_RATE], &e, &flow_ext_map); |
