diff options
Diffstat (limited to 'net/sched/cls_fw.c')
-rw-r--r-- | net/sched/cls_fw.c | 31 |
1 files changed, 27 insertions, 4 deletions
diff --git a/net/sched/cls_fw.c b/net/sched/cls_fw.c index fdfc83af3d1f..29d8b9a4d162 100644 --- a/net/sched/cls_fw.c +++ b/net/sched/cls_fw.c | |||
@@ -46,9 +46,11 @@ | |||
46 | #include <net/act_api.h> | 46 | #include <net/act_api.h> |
47 | #include <net/pkt_cls.h> | 47 | #include <net/pkt_cls.h> |
48 | 48 | ||
49 | #define HTSIZE (PAGE_SIZE/sizeof(struct fw_filter *)) | ||
50 | |||
49 | struct fw_head | 51 | struct fw_head |
50 | { | 52 | { |
51 | struct fw_filter *ht[256]; | 53 | struct fw_filter *ht[HTSIZE]; |
52 | }; | 54 | }; |
53 | 55 | ||
54 | struct fw_filter | 56 | struct fw_filter |
@@ -69,7 +71,28 @@ static struct tcf_ext_map fw_ext_map = { | |||
69 | 71 | ||
70 | static __inline__ int fw_hash(u32 handle) | 72 | static __inline__ int fw_hash(u32 handle) |
71 | { | 73 | { |
72 | return handle&0xFF; | 74 | if (HTSIZE == 4096) |
75 | return ((handle >> 24) & 0xFFF) ^ | ||
76 | ((handle >> 12) & 0xFFF) ^ | ||
77 | (handle & 0xFFF); | ||
78 | else if (HTSIZE == 2048) | ||
79 | return ((handle >> 22) & 0x7FF) ^ | ||
80 | ((handle >> 11) & 0x7FF) ^ | ||
81 | (handle & 0x7FF); | ||
82 | else if (HTSIZE == 1024) | ||
83 | return ((handle >> 20) & 0x3FF) ^ | ||
84 | ((handle >> 10) & 0x3FF) ^ | ||
85 | (handle & 0x3FF); | ||
86 | else if (HTSIZE == 512) | ||
87 | return (handle >> 27) ^ | ||
88 | ((handle >> 18) & 0x1FF) ^ | ||
89 | ((handle >> 9) & 0x1FF) ^ | ||
90 | (handle & 0x1FF); | ||
91 | else if (HTSIZE == 256) { | ||
92 | u8 *t = (u8 *) &handle; | ||
93 | return t[0] ^ t[1] ^ t[2] ^ t[3]; | ||
94 | } else | ||
95 | return handle & (HTSIZE - 1); | ||
73 | } | 96 | } |
74 | 97 | ||
75 | static int fw_classify(struct sk_buff *skb, struct tcf_proto *tp, | 98 | static int fw_classify(struct sk_buff *skb, struct tcf_proto *tp, |
@@ -152,7 +175,7 @@ static void fw_destroy(struct tcf_proto *tp) | |||
152 | if (head == NULL) | 175 | if (head == NULL) |
153 | return; | 176 | return; |
154 | 177 | ||
155 | for (h=0; h<256; h++) { | 178 | for (h=0; h<HTSIZE; h++) { |
156 | while ((f=head->ht[h]) != NULL) { | 179 | while ((f=head->ht[h]) != NULL) { |
157 | head->ht[h] = f->next; | 180 | head->ht[h] = f->next; |
158 | fw_delete_filter(tp, f); | 181 | fw_delete_filter(tp, f); |
@@ -291,7 +314,7 @@ static void fw_walk(struct tcf_proto *tp, struct tcf_walker *arg) | |||
291 | if (arg->stop) | 314 | if (arg->stop) |
292 | return; | 315 | return; |
293 | 316 | ||
294 | for (h = 0; h < 256; h++) { | 317 | for (h = 0; h < HTSIZE; h++) { |
295 | struct fw_filter *f; | 318 | struct fw_filter *f; |
296 | 319 | ||
297 | for (f = head->ht[h]; f; f = f->next) { | 320 | for (f = head->ht[h]; f; f = f->next) { |