ipt_connbytes.c 4.0 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12
/* Kernel module to match connection tracking byte counter.
 * GPL (C) 2002 Martin Devera (devik@cdi.cz).
 *
 * 2004-07-20 Harald Welte <laforge@netfilter.org>
 * 	- reimplemented to use per-connection accounting counters
 * 	- add functionality to match number of packets
 * 	- add functionality to match average packet size
 * 	- add support to match directions seperately
 *
 */
#include <linux/module.h>
#include <linux/skbuff.h>
13
#include <net/netfilter/nf_conntrack_compat.h>
14 15 16 17 18 19 20 21 22 23 24
#include <linux/netfilter_ipv4/ip_tables.h>
#include <linux/netfilter_ipv4/ipt_connbytes.h>

#include <asm/div64.h>
#include <asm/bitops.h>

MODULE_LICENSE("GPL");
MODULE_AUTHOR("Harald Welte <laforge@netfilter.org>");
MODULE_DESCRIPTION("iptables match for matching number of pkts/bytes per connection");

/* 64bit divisor, dividend and result. dynamic precision */
25
static u_int64_t div64_64(u_int64_t dividend, u_int64_t divisor)
26
{
27
	u_int32_t d = divisor;
28

29 30
	if (divisor > 0xffffffffULL) {
		unsigned int shift = fls(divisor >> 32);
31

32 33 34
		d = divisor >> shift;
		dividend >>= shift;
	}
35

36 37
	do_div(dividend, d);
	return dividend;
38 39 40 41 42 43 44 45 46 47 48 49
}

static int
match(const struct sk_buff *skb,
      const struct net_device *in,
      const struct net_device *out,
      const void *matchinfo,
      int offset,
      int *hotdrop)
{
	const struct ipt_connbytes_info *sinfo = matchinfo;
	u_int64_t what = 0;	/* initialize to make gcc happy */
50
	const struct ip_conntrack_counter *counters;
51

52
	if (!(counters = nf_ct_get_counters(skb)))
53 54 55
		return 0; /* no match */

	switch (sinfo->what) {
56
	case IPT_CONNBYTES_PKTS:
57 58
		switch (sinfo->direction) {
		case IPT_CONNBYTES_DIR_ORIGINAL:
59
			what = counters[IP_CT_DIR_ORIGINAL].packets;
60 61
			break;
		case IPT_CONNBYTES_DIR_REPLY:
62
			what = counters[IP_CT_DIR_REPLY].packets;
63 64
			break;
		case IPT_CONNBYTES_DIR_BOTH:
65 66
			what = counters[IP_CT_DIR_ORIGINAL].packets;
			what += counters[IP_CT_DIR_REPLY].packets;
67 68 69
			break;
		}
		break;
70
	case IPT_CONNBYTES_BYTES:
71 72
		switch (sinfo->direction) {
		case IPT_CONNBYTES_DIR_ORIGINAL:
73
			what = counters[IP_CT_DIR_ORIGINAL].bytes;
74 75
			break;
		case IPT_CONNBYTES_DIR_REPLY:
76
			what = counters[IP_CT_DIR_REPLY].bytes;
77 78
			break;
		case IPT_CONNBYTES_DIR_BOTH:
79 80
			what = counters[IP_CT_DIR_ORIGINAL].bytes;
			what += counters[IP_CT_DIR_REPLY].bytes;
81 82 83
			break;
		}
		break;
84
	case IPT_CONNBYTES_AVGPKT:
85 86
		switch (sinfo->direction) {
		case IPT_CONNBYTES_DIR_ORIGINAL:
87 88
			what = div64_64(counters[IP_CT_DIR_ORIGINAL].bytes,
					counters[IP_CT_DIR_ORIGINAL].packets);
89 90
			break;
		case IPT_CONNBYTES_DIR_REPLY:
91 92
			what = div64_64(counters[IP_CT_DIR_REPLY].bytes,
					counters[IP_CT_DIR_REPLY].packets);
93 94 95 96 97
			break;
		case IPT_CONNBYTES_DIR_BOTH:
			{
				u_int64_t bytes;
				u_int64_t pkts;
98 99 100 101
				bytes = counters[IP_CT_DIR_ORIGINAL].bytes +
					counters[IP_CT_DIR_REPLY].bytes;
				pkts = counters[IP_CT_DIR_ORIGINAL].packets+
					counters[IP_CT_DIR_REPLY].packets;
102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129

				/* FIXME_THEORETICAL: what to do if sum
				 * overflows ? */

				what = div64_64(bytes, pkts);
			}
			break;
		}
		break;
	}

	if (sinfo->count.to)
		return (what <= sinfo->count.to && what >= sinfo->count.from);
	else
		return (what >= sinfo->count.from);
}

static int check(const char *tablename,
		 const struct ipt_ip *ip,
		 void *matchinfo,
		 unsigned int matchsize,
		 unsigned int hook_mask)
{
	const struct ipt_connbytes_info *sinfo = matchinfo;

	if (matchsize != IPT_ALIGN(sizeof(struct ipt_connbytes_info)))
		return 0;

130 131 132
	if (sinfo->what != IPT_CONNBYTES_PKTS &&
	    sinfo->what != IPT_CONNBYTES_BYTES &&
	    sinfo->what != IPT_CONNBYTES_AVGPKT)
133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161
		return 0;

	if (sinfo->direction != IPT_CONNBYTES_DIR_ORIGINAL &&
	    sinfo->direction != IPT_CONNBYTES_DIR_REPLY &&
	    sinfo->direction != IPT_CONNBYTES_DIR_BOTH)
		return 0;

	return 1;
}

static struct ipt_match state_match = {
	.name		= "connbytes",
	.match		= &match,
	.checkentry	= &check,
	.me		= THIS_MODULE
};

static int __init init(void)
{
	return ipt_register_match(&state_match);
}

static void __exit fini(void)
{
	ipt_unregister_match(&state_match);
}

module_init(init);
module_exit(fini);