
jengelh at computergmbh
Sep 15, 2007, 9:07 AM
Post #1 of 9
(6046 views)
Permalink
|
For 2.6.24. Please comment, thanks! === ipt_time from POM-ng augmented by: * xtables * ipv6 support * day-of-month support (for example "match on the 15th of each month") * timezone (it was there before though) - it does *what the user expects* (which is good). Signed-off-by: Jan Engelhardt <jengelh [at] computergmbh> --- include/linux/netfilter/xt_time.h | 13 ++ net/netfilter/Kconfig | 14 ++ net/netfilter/Makefile | 1 net/netfilter/xt_time.c | 208 ++++++++++++++++++++++++++++++++++++++ 4 files changed, 236 insertions(+) Index: linux-2.6.23/include/linux/netfilter/xt_time.h =================================================================== --- /dev/null +++ linux-2.6.23/include/linux/netfilter/xt_time.h @@ -0,0 +1,13 @@ +#ifndef _XT_TIME_H +#define _XT_TIME_H 1 + +struct xt_time_info { + time_t date_start; + time_t date_stop; + u_int32_t monthdays_match; + u_int16_t time_start; + u_int16_t time_stop; + u_int8_t weekdays_match; +}; + +#endif /* _XT_TIME_H */ Index: linux-2.6.23/net/netfilter/Kconfig =================================================================== --- linux-2.6.23.orig/net/netfilter/Kconfig +++ linux-2.6.23/net/netfilter/Kconfig @@ -665,6 +665,20 @@ config NETFILTER_XT_MATCH_TCPMSS To compile it as a module, choose M here. If unsure, say N. +config NETFILTER_XT_MATCH_TIME + tristate '"time" match support' + depends on NETFILTER_XTABLES + ---help--- + This option adds a "time" match, which allows you to match based on + the packet arrival time (at the machine which netfilter is running) + on) or departure time/date (for locally generated packets). + + If you say Y here, try `iptables -m time --help` for + more information. + + If you want to compile it as a module, say M here. + If unsure, say N. + config NETFILTER_XT_MATCH_U32 tristate '"u32" match support' depends on NETFILTER_XTABLES Index: linux-2.6.23/net/netfilter/Makefile =================================================================== --- linux-2.6.23.orig/net/netfilter/Makefile +++ linux-2.6.23/net/netfilter/Makefile @@ -73,6 +73,7 @@ obj-$(CONFIG_NETFILTER_XT_MATCH_SCTP) += obj-$(CONFIG_NETFILTER_XT_MATCH_STATE) += xt_state.o obj-$(CONFIG_NETFILTER_XT_MATCH_STATISTIC) += xt_statistic.o obj-$(CONFIG_NETFILTER_XT_MATCH_STRING) += xt_string.o +obj-$(CONFIG_NETFILTER_XT_MATCH_TIME) += xt_time.o obj-$(CONFIG_NETFILTER_XT_MATCH_TCPMSS) += xt_tcpmss.o obj-$(CONFIG_NETFILTER_XT_MATCH_PHYSDEV) += xt_physdev.o obj-$(CONFIG_NETFILTER_XT_MATCH_U32) += xt_u32.o Index: linux-2.6.23/net/netfilter/xt_time.c =================================================================== --- /dev/null +++ linux-2.6.23/net/netfilter/xt_time.c @@ -0,0 +1,208 @@ +/* + * xt_time + * Copyright © Jan Engelhardt <jengelh [at] computergmbh>, 2007 + * + * based on ipt_time by Fabrice MARIE <fabrice [at] netfilter> + * This is a module which is used for time matching + * It is using some modified code from dietlibc (localtime() function) + * that you can find at http://www.fefe.de/dietlibc/ + * This file is distributed under the terms of the GNU General Public + * License (GPL). Copies of the GPL can be obtained from gnu.org/gpl. + */ +#include <linux/ktime.h> +#include <linux/module.h> +#include <linux/skbuff.h> +#include <linux/netfilter/x_tables.h> +#include "xt_time.h" + +struct xtm { + unsigned int month; /* (1-12) */ + unsigned int monthday; /* (1-31) */ + unsigned int weekday; /* (1-7) */ + unsigned int hour; /* (0-23) */ + unsigned int minute; /* (0-59) */ +}; + +extern struct timezone sys_tz; /* ouch */ + +static const u_int16_t days_since_year[] = { + 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, +}; + +static const u_int16_t days_since_leapyear[] = { + 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, +}; + +/* + * Since time progresses forward, it is best to organize this array in reverse, + * to minimize lookup time. + */ +enum { + DSE_FIRST = 2039, +}; +static const u_int16_t days_since_epoch[] = { + /* 2039 - 2030 */ + 25202, 24837, 24472, 24106, 23741, 23376, 23011, 22645, 22280, 21915, + /* 2029 - 2020 */ + 21550, 21184, 20819, 20454, 20089, 19723, 19358, 18993, 18628, 18262, + /* 2019 - 2010 */ + 17897, 17532, 17167, 16801, 16436, 16071, 15706, 15340, 14975, 14610, + /* 2009 - 2000 */ + 14245, 13879, 13514, 13149, 12784, 12418, 12053, 11688, 11323, 10957, + /* 1999 - 1990 */ + 10592, 10227, 9862, 9496, 9131, 8766, 8401, 8035, 7670, 7305, + /* 1989 - 1980 */ + 6940, 6574, 6209, 5844, 5479, 5113, 4748, 4383, 4018, 3652, + /* 1979 - 1970 */ + 3287, 2922, 2557, 2191, 1826, 1461, 1096, 730, 365, 0, +}; + +static inline bool is_leap(unsigned int y) +{ + return y % 4 == 0 && (y % 100 != 0 || y % 400 == 0); +} + +static void localtime(struct xtm *r, time_t time) +{ + unsigned int year, i, w; + + time -= 60 * sys_tz.tz_minuteswest; + + /* 1d=86400s. Daytime. */ + w = time % 86400; + w /= 60; + r->minute = w % 60; + r->hour = w / 60; + + /* Days since 1970-01-01 */ + w = time / 86400; + r->weekday = (4 + w) % 7; + + /* + * Consider x=21551. Loop will abort on dse[i] <= x, which is with + * dse[i] = 21550 (=> 2029-01-02). + */ + for (i = 0, year = DSE_FIRST; days_since_epoch[i] > w; + ++i, --year) + /* just loop */; + + w -= days_since_epoch[i]; + /* r->yearday = w; */ + + if (is_leap(year)) { + for (i = ARRAY_SIZE(days_since_leapyear) - 1; + i > 0 && days_since_year[i] > w; --i) + /* just loop */; + } else { + for (i = ARRAY_SIZE(days_since_year) - 1; + i > 0 && days_since_year[i] > w; --i) + /* just loop */; + } + + r->month = i + 1; + r->monthday = w - days_since_year[i] + 1; + printk(KERN_ERR "m=%u md=%u\n", r->month, r->monthday); + return; +} + +static bool xt_time_match(const struct sk_buff *skb, + const struct net_device *in, + const struct net_device *out, + const struct xt_match *match, const void *matchinfo, + int offset, unsigned int protoff, bool *hotdrop) +{ + const struct xt_time_info *info = matchinfo; + unsigned int packet_time; + struct xtm current_time; + s64 stamp; + + if (skb->tstamp.tv64 == 0) + __net_timestamp((struct sk_buff *)skb); + + stamp = skb->tstamp.tv64; + do_div(stamp, NSEC_PER_SEC); + + /* + * xt_time will match when _all_ of the following hold: + * - 'now' is in the global time range date_start..date_end + * - 'now' is in the monthday mask + * - 'now' is in the weekday mask + * - 'now' is in the daytime range time_start..time_end + * (and by default, libxt_time will set these so as to match) + */ + + if (stamp < info->date_start || stamp > info->date_stop) + return false; + + localtime(¤t_time, stamp); + + if (!(info->monthdays_match & (1 << current_time.monthday))) + return false; + if (!(info->weekdays_match & (1 << current_time.weekday))) + return false; + + packet_time = current_time.hour * 60 + current_time.minute; + if (info->time_start < info->time_stop) { + if (packet_time < info->time_start || + packet_time > info->time_stop) + return false; + } else { + if (packet_time < info->time_start && + packet_time > info->time_stop) + return false; + } + + return true; +} + +static bool xt_time_check(const char *tablename, const void *ip, + const struct xt_match *match, void *matchinfo, + unsigned int hook_mask) +{ + struct xt_time_info *info = matchinfo; + + /* xt_time's granularity is a minute, hence 24*60 = 1day */ + if (info->time_start > 24 * 60 || info->time_stop >= 24 * 60) { + printk(KERN_WARNING "ipt_time: invalid argument - start or stop time greater than 23:59h\n"); + return false; + } + + return true; +} + +static struct xt_match xt_time_reg[] __read_mostly = { + { + .name = "time", + .family = AF_INET, + .match = xt_time_match, + .matchsize = sizeof(struct xt_time_info), + .checkentry = xt_time_check, + .me = THIS_MODULE, + }, + { + .name = "time", + .family = AF_INET6, + .match = xt_time_match, + .matchsize = sizeof(struct xt_time_info), + .checkentry = xt_time_check, + .me = THIS_MODULE, + }, +}; + +static int __init xt_time_init(void) +{ + return xt_register_matches(xt_time_reg, ARRAY_SIZE(xt_time_reg)); +} + +static void __exit xt_time_exit(void) +{ + xt_unregister_matches(xt_time_reg, ARRAY_SIZE(xt_time_reg)); +} + +module_init(xt_time_init); +module_exit(xt_time_exit); +MODULE_AUTHOR("Jan Engelhardt <jengelh [at] computergmbh>"); +MODULE_DESCRIPTION("netfilter time match"); +MODULE_LICENSE("GPL"); +MODULE_ALIAS("ipt_time"); +MODULE_ALIAS("ip6t_time");
|