5 #include <linux/netfilter/xt_recent.h>
21 F_RCHECK = 1 << O_RCHECK,
22 F_UPDATE = 1 << O_UPDATE,
23 F_REMOVE = 1 << O_REMOVE,
24 F_SECONDS = 1 << O_SECONDS,
25 F_ANY_OP = F_SET | F_RCHECK | F_UPDATE | F_REMOVE,
28 #define s struct xt_recent_mtinfo
29 static const struct xt_option_entry recent_opts_v0[] = {
30 {.name = "set", .id = O_SET, .type = XTTYPE_NONE,
31 .excl = F_ANY_OP, .flags = XTOPT_INVERT},
32 {.name = "rcheck", .id = O_RCHECK, .type = XTTYPE_NONE,
33 .excl = F_ANY_OP, .flags = XTOPT_INVERT},
34 {.name = "update", .id = O_UPDATE, .type = XTTYPE_NONE,
35 .excl = F_ANY_OP, .flags = XTOPT_INVERT},
36 {.name = "remove", .id = O_REMOVE, .type = XTTYPE_NONE,
37 .excl = F_ANY_OP, .flags = XTOPT_INVERT},
38 {.name = "seconds", .id = O_SECONDS, .type = XTTYPE_UINT32,
39 .flags = XTOPT_PUT, XTOPT_POINTER(s, seconds), .min = 1},
40 {.name = "reap", .id = O_REAP, .type = XTTYPE_NONE,
42 {.name = "hitcount", .id = O_HITCOUNT, .type = XTTYPE_UINT32,
43 .flags = XTOPT_PUT, XTOPT_POINTER(s, hit_count)},
44 {.name = "rttl", .id = O_RTTL, .type = XTTYPE_NONE,
45 .excl = F_SET | F_REMOVE},
46 {.name = "name", .id = O_NAME, .type = XTTYPE_STRING,
47 .flags = XTOPT_PUT, XTOPT_POINTER(s, name)},
48 {.name = "rsource", .id = O_RSOURCE, .type = XTTYPE_NONE},
49 {.name = "rdest", .id = O_RDEST, .type = XTTYPE_NONE},
54 #define s struct xt_recent_mtinfo_v1
55 static const struct xt_option_entry recent_opts_v1[] = {
56 {.name = "set", .id = O_SET, .type = XTTYPE_NONE,
57 .excl = F_ANY_OP, .flags = XTOPT_INVERT},
58 {.name = "rcheck", .id = O_RCHECK, .type = XTTYPE_NONE,
59 .excl = F_ANY_OP, .flags = XTOPT_INVERT},
60 {.name = "update", .id = O_UPDATE, .type = XTTYPE_NONE,
61 .excl = F_ANY_OP, .flags = XTOPT_INVERT},
62 {.name = "remove", .id = O_REMOVE, .type = XTTYPE_NONE,
63 .excl = F_ANY_OP, .flags = XTOPT_INVERT},
64 {.name = "seconds", .id = O_SECONDS, .type = XTTYPE_UINT32,
65 .flags = XTOPT_PUT, XTOPT_POINTER(s, seconds), .min = 1},
66 {.name = "reap", .id = O_REAP, .type = XTTYPE_NONE,
68 {.name = "hitcount", .id = O_HITCOUNT, .type = XTTYPE_UINT32,
69 .flags = XTOPT_PUT, XTOPT_POINTER(s, hit_count)},
70 {.name = "rttl", .id = O_RTTL, .type = XTTYPE_NONE,
71 .excl = F_SET | F_REMOVE},
72 {.name = "name", .id = O_NAME, .type = XTTYPE_STRING,
73 .flags = XTOPT_PUT, XTOPT_POINTER(s, name)},
74 {.name = "rsource", .id = O_RSOURCE, .type = XTTYPE_NONE},
75 {.name = "rdest", .id = O_RDEST, .type = XTTYPE_NONE},
76 {.name = "mask", .id = O_MASK, .type = XTTYPE_HOST,
77 .flags = XTOPT_PUT, XTOPT_POINTER(s, mask)},
82 static void recent_help(void)
85 "recent match options:\n"
86 "[!] --set Add source address to list, always matches.\n"
87 "[!] --rcheck Match if source address in list.\n"
88 "[!] --update Match if source address in list, also update last-seen time.\n"
89 "[!] --remove Match if source address in list, also removes that address from list.\n"
90 " --seconds seconds For check and update commands above.\n"
91 " Specifies that the match will only occur if source address last seen within\n"
92 " the last 'seconds' seconds.\n"
93 " --reap Purge entries older then 'seconds'.\n"
94 " Can only be used in conjunction with the seconds option.\n"
95 " --hitcount hits For check and update commands above.\n"
96 " Specifies that the match will only occur if source address seen hits times.\n"
97 " May be used in conjunction with the seconds option.\n"
98 " --rttl For check and update commands above.\n"
99 " Specifies that the match will only occur if the source address and the TTL\n"
100 " match between this packet and the one which was set.\n"
101 " Useful if you have problems with people spoofing their source address in order\n"
102 " to DoS you via this module.\n"
103 " --name name Name of the recent list to be used. DEFAULT used if none given.\n"
104 " --rsource Match/Save the source address of each packet in the recent list table (default).\n"
105 " --rdest Match/Save the destination address of each packet in the recent list table.\n"
106 " --mask netmask Netmask that will be applied to this recent list.\n"
107 "xt_recent by: Stephen Frost <sfrost@snowman.net>.\n");
115 static void recent_init(struct xt_entry_match *match, unsigned int rev)
117 struct xt_recent_mtinfo *info = (struct xt_recent_mtinfo *)match->data;
118 struct xt_recent_mtinfo_v1 *info_v1 =
119 (struct xt_recent_mtinfo_v1 *)match->data;
121 strncpy(info->name,"DEFAULT", XT_RECENT_NAME_LEN);
122 /* even though XT_RECENT_NAME_LEN is currently defined as 200,
123 * better be safe, than sorry */
124 info->name[XT_RECENT_NAME_LEN-1] = '\0';
125 info->side = XT_RECENT_SOURCE;
126 if (rev == XT_RECENT_REV_1)
127 memset(&info_v1->mask, 0xFF, sizeof(info_v1->mask));
130 static void recent_parse(struct xt_option_call *cb)
132 struct xt_recent_mtinfo *info = cb->data;
134 xtables_option_parse(cb);
135 switch (cb->entry->id) {
137 info->check_set |= XT_RECENT_SET;
142 info->check_set |= XT_RECENT_CHECK;
147 info->check_set |= XT_RECENT_UPDATE;
152 info->check_set |= XT_RECENT_REMOVE;
157 info->check_set |= XT_RECENT_TTL;
160 info->side = XT_RECENT_SOURCE;
163 info->side = XT_RECENT_DEST;
166 info->check_set |= XT_RECENT_REAP;
171 static void recent_check(struct xt_fcheck_call *cb)
173 if (!(cb->xflags & F_ANY_OP))
174 xtables_error(PARAMETER_PROBLEM,
175 "recent: you must specify one of `--set', `--rcheck' "
176 "`--update' or `--remove'");
179 static void recent_print(const void *ip, const struct xt_entry_match *match,
182 const struct xt_recent_mtinfo_v1 *info = (const void *)match->data;
188 if (info->check_set & XT_RECENT_SET)
190 if (info->check_set & XT_RECENT_CHECK)
192 if (info->check_set & XT_RECENT_UPDATE)
194 if (info->check_set & XT_RECENT_REMOVE)
196 if(info->seconds) printf(" seconds: %d", info->seconds);
197 if (info->check_set & XT_RECENT_REAP)
199 if(info->hit_count) printf(" hit_count: %d", info->hit_count);
200 if (info->check_set & XT_RECENT_TTL)
201 printf(" TTL-Match");
202 printf(" name: %s", info->name);
203 if (info->side == XT_RECENT_SOURCE)
204 printf(" side: source");
205 if (info->side == XT_RECENT_DEST)
206 printf(" side: dest");
211 xtables_ipaddr_to_numeric(&info->mask.in));
215 xtables_ip6addr_to_numeric(&info->mask.in6));
220 static void recent_save(const void *ip, const struct xt_entry_match *match,
223 const struct xt_recent_mtinfo_v1 *info = (const void *)match->data;
228 if (info->check_set & XT_RECENT_SET)
230 if (info->check_set & XT_RECENT_CHECK)
232 if (info->check_set & XT_RECENT_UPDATE)
234 if (info->check_set & XT_RECENT_REMOVE)
236 if(info->seconds) printf(" --seconds %d", info->seconds);
237 if (info->check_set & XT_RECENT_REAP)
239 if(info->hit_count) printf(" --hitcount %d", info->hit_count);
240 if (info->check_set & XT_RECENT_TTL)
242 printf(" --name %s",info->name);
247 xtables_ipaddr_to_numeric(&info->mask.in));
251 xtables_ip6addr_to_numeric(&info->mask.in6));
255 if (info->side == XT_RECENT_SOURCE)
256 printf(" --rsource");
257 if (info->side == XT_RECENT_DEST)
261 static void recent_init_v0(struct xt_entry_match *match)
263 recent_init(match, XT_RECENT_REV_0);
266 static void recent_init_v1(struct xt_entry_match *match)
268 recent_init(match, XT_RECENT_REV_1);
271 static void recent_save_v0(const void *ip, const struct xt_entry_match *match)
273 recent_save(ip, match, NFPROTO_UNSPEC);
276 static void recent_save_v4(const void *ip, const struct xt_entry_match *match)
278 recent_save(ip, match, NFPROTO_IPV4);
281 static void recent_save_v6(const void *ip, const struct xt_entry_match *match)
283 recent_save(ip, match, NFPROTO_IPV6);
286 static void recent_print_v0(const void *ip, const struct xt_entry_match *match,
289 recent_print(ip, match, NFPROTO_UNSPEC);
292 static void recent_print_v4(const void *ip, const struct xt_entry_match *match,
295 recent_print(ip, match, NFPROTO_IPV4);
298 static void recent_print_v6(const void *ip, const struct xt_entry_match *match,
301 recent_print(ip, match, NFPROTO_IPV6);
304 static struct xtables_match recent_mt_reg[] = {
307 .version = XTABLES_VERSION,
309 .family = NFPROTO_UNSPEC,
310 .size = XT_ALIGN(sizeof(struct xt_recent_mtinfo)),
311 .userspacesize = XT_ALIGN(sizeof(struct xt_recent_mtinfo)),
313 .init = recent_init_v0,
314 .x6_parse = recent_parse,
315 .x6_fcheck = recent_check,
316 .print = recent_print_v0,
317 .save = recent_save_v0,
318 .x6_options = recent_opts_v0,
322 .version = XTABLES_VERSION,
324 .family = NFPROTO_IPV4,
325 .size = XT_ALIGN(sizeof(struct xt_recent_mtinfo_v1)),
326 .userspacesize = XT_ALIGN(sizeof(struct xt_recent_mtinfo_v1)),
328 .init = recent_init_v1,
329 .x6_parse = recent_parse,
330 .x6_fcheck = recent_check,
331 .print = recent_print_v4,
332 .save = recent_save_v4,
333 .x6_options = recent_opts_v1,
337 .version = XTABLES_VERSION,
339 .family = NFPROTO_IPV6,
340 .size = XT_ALIGN(sizeof(struct xt_recent_mtinfo_v1)),
341 .userspacesize = XT_ALIGN(sizeof(struct xt_recent_mtinfo_v1)),
343 .init = recent_init_v1,
344 .x6_parse = recent_parse,
345 .x6_fcheck = recent_check,
346 .print = recent_print_v6,
347 .save = recent_save_v6,
348 .x6_options = recent_opts_v1,
354 xtables_register_matches(recent_mt_reg, ARRAY_SIZE(recent_mt_reg));