diff options
author | Eric Dumazet <dada1@cosmosbay.com> | 2009-02-19 05:17:17 -0500 |
---|---|---|
committer | Patrick McHardy <kaber@trash.net> | 2009-02-19 05:17:17 -0500 |
commit | eacc17fb64f03b6c268aaf6cea320100d19d8af5 (patch) | |
tree | 52124f837c720697ee3999ad29a09113a76e02c6 | |
parent | 4323362e49bd10b8ff3fe5cf183fdd52662ff4a3 (diff) |
netfilter: xt_physdev: unfold two loops in physdev_mt()
xt_physdev netfilter module can use an ifname_compare() helper
so that two loops are unfolded.
Signed-off-by: Eric Dumazet <dada1@cosmosbay.com>
Signed-off-by: Patrick McHardy <kaber@trash.net>
-rw-r--r-- | net/netfilter/xt_physdev.c | 32 |
1 files changed, 21 insertions, 11 deletions
diff --git a/net/netfilter/xt_physdev.c b/net/netfilter/xt_physdev.c index 4b13ef7ce145..44a234ef4439 100644 --- a/net/netfilter/xt_physdev.c +++ b/net/netfilter/xt_physdev.c | |||
@@ -20,10 +20,27 @@ MODULE_DESCRIPTION("Xtables: Bridge physical device match"); | |||
20 | MODULE_ALIAS("ipt_physdev"); | 20 | MODULE_ALIAS("ipt_physdev"); |
21 | MODULE_ALIAS("ip6t_physdev"); | 21 | MODULE_ALIAS("ip6t_physdev"); |
22 | 22 | ||
23 | static unsigned long ifname_compare(const char *_a, const char *_b, const char *_mask) | ||
24 | { | ||
25 | const unsigned long *a = (const unsigned long *)_a; | ||
26 | const unsigned long *b = (const unsigned long *)_b; | ||
27 | const unsigned long *mask = (const unsigned long *)_mask; | ||
28 | unsigned long ret; | ||
29 | |||
30 | ret = (a[0] ^ b[0]) & mask[0]; | ||
31 | if (IFNAMSIZ > sizeof(unsigned long)) | ||
32 | ret |= (a[1] ^ b[1]) & mask[1]; | ||
33 | if (IFNAMSIZ > 2 * sizeof(unsigned long)) | ||
34 | ret |= (a[2] ^ b[2]) & mask[2]; | ||
35 | if (IFNAMSIZ > 3 * sizeof(unsigned long)) | ||
36 | ret |= (a[3] ^ b[3]) & mask[3]; | ||
37 | BUILD_BUG_ON(IFNAMSIZ > 4 * sizeof(unsigned long)); | ||
38 | return ret; | ||
39 | } | ||
40 | |||
23 | static bool | 41 | static bool |
24 | physdev_mt(const struct sk_buff *skb, const struct xt_match_param *par) | 42 | physdev_mt(const struct sk_buff *skb, const struct xt_match_param *par) |
25 | { | 43 | { |
26 | int i; | ||
27 | static const char nulldevname[IFNAMSIZ] __attribute__((aligned(sizeof(long)))); | 44 | static const char nulldevname[IFNAMSIZ] __attribute__((aligned(sizeof(long)))); |
28 | const struct xt_physdev_info *info = par->matchinfo; | 45 | const struct xt_physdev_info *info = par->matchinfo; |
29 | unsigned long ret; | 46 | unsigned long ret; |
@@ -68,11 +85,7 @@ physdev_mt(const struct sk_buff *skb, const struct xt_match_param *par) | |||
68 | if (!(info->bitmask & XT_PHYSDEV_OP_IN)) | 85 | if (!(info->bitmask & XT_PHYSDEV_OP_IN)) |
69 | goto match_outdev; | 86 | goto match_outdev; |
70 | indev = nf_bridge->physindev ? nf_bridge->physindev->name : nulldevname; | 87 | indev = nf_bridge->physindev ? nf_bridge->physindev->name : nulldevname; |
71 | for (i = 0, ret = 0; i < IFNAMSIZ/sizeof(unsigned long); i++) { | 88 | ret = ifname_compare(indev, info->physindev, info->in_mask); |
72 | ret |= (((const unsigned long *)indev)[i] | ||
73 | ^ ((const unsigned long *)info->physindev)[i]) | ||
74 | & ((const unsigned long *)info->in_mask)[i]; | ||
75 | } | ||
76 | 89 | ||
77 | if (!ret ^ !(info->invert & XT_PHYSDEV_OP_IN)) | 90 | if (!ret ^ !(info->invert & XT_PHYSDEV_OP_IN)) |
78 | return false; | 91 | return false; |
@@ -82,11 +95,8 @@ match_outdev: | |||
82 | return true; | 95 | return true; |
83 | outdev = nf_bridge->physoutdev ? | 96 | outdev = nf_bridge->physoutdev ? |
84 | nf_bridge->physoutdev->name : nulldevname; | 97 | nf_bridge->physoutdev->name : nulldevname; |
85 | for (i = 0, ret = 0; i < IFNAMSIZ/sizeof(unsigned long); i++) { | 98 | ret = ifname_compare(outdev, info->physoutdev, info->out_mask); |
86 | ret |= (((const unsigned long *)outdev)[i] | 99 | |
87 | ^ ((const unsigned long *)info->physoutdev)[i]) | ||
88 | & ((const unsigned long *)info->out_mask)[i]; | ||
89 | } | ||
90 | return (!!ret ^ !(info->invert & XT_PHYSDEV_OP_OUT)); | 100 | return (!!ret ^ !(info->invert & XT_PHYSDEV_OP_OUT)); |
91 | } | 101 | } |
92 | 102 | ||