1 /* Kernel module to match connection tracking byte counter.
2 * GPL (C) 2002 Martin Devera (devik@cdi.cz).
4 * 2004-07-20 Harald Welte <laforge@netfilter.org>
5 * - reimplemented to use per-connection accounting counters
6 * - add functionality to match number of packets
7 * - add functionality to match average packet size
8 * - add support to match directions seperately
9 * 2005-10-16 Harald Welte <laforge@netfilter.org>
13 #include <linux/module.h>
14 #include <linux/skbuff.h>
15 #include <net/netfilter/nf_conntrack_compat.h>
16 #include <linux/netfilter/x_tables.h>
17 #include <linux/netfilter/xt_connbytes.h>
19 #include <asm/div64.h>
20 #include <asm/bitops.h>
22 MODULE_LICENSE("GPL");
23 MODULE_AUTHOR("Harald Welte <laforge@netfilter.org>");
24 MODULE_DESCRIPTION("iptables match for matching number of pkts/bytes per connection");
25 MODULE_ALIAS("ipt_connbytes");
27 /* 64bit divisor, dividend and result. dynamic precision */
28 static u_int64_t
div64_64(u_int64_t dividend
, u_int64_t divisor
)
30 u_int32_t d
= divisor
;
32 if (divisor
> 0xffffffffULL
) {
33 unsigned int shift
= fls(divisor
>> 32);
44 match(const struct sk_buff
*skb
,
45 const struct net_device
*in
,
46 const struct net_device
*out
,
47 const void *matchinfo
,
52 const struct xt_connbytes_info
*sinfo
= matchinfo
;
53 u_int64_t what
= 0; /* initialize to make gcc happy */
54 const struct ip_conntrack_counter
*counters
;
56 if (!(counters
= nf_ct_get_counters(skb
)))
57 return 0; /* no match */
59 switch (sinfo
->what
) {
60 case XT_CONNBYTES_PKTS
:
61 switch (sinfo
->direction
) {
62 case XT_CONNBYTES_DIR_ORIGINAL
:
63 what
= counters
[IP_CT_DIR_ORIGINAL
].packets
;
65 case XT_CONNBYTES_DIR_REPLY
:
66 what
= counters
[IP_CT_DIR_REPLY
].packets
;
68 case XT_CONNBYTES_DIR_BOTH
:
69 what
= counters
[IP_CT_DIR_ORIGINAL
].packets
;
70 what
+= counters
[IP_CT_DIR_REPLY
].packets
;
74 case XT_CONNBYTES_BYTES
:
75 switch (sinfo
->direction
) {
76 case XT_CONNBYTES_DIR_ORIGINAL
:
77 what
= counters
[IP_CT_DIR_ORIGINAL
].bytes
;
79 case XT_CONNBYTES_DIR_REPLY
:
80 what
= counters
[IP_CT_DIR_REPLY
].bytes
;
82 case XT_CONNBYTES_DIR_BOTH
:
83 what
= counters
[IP_CT_DIR_ORIGINAL
].bytes
;
84 what
+= counters
[IP_CT_DIR_REPLY
].bytes
;
88 case XT_CONNBYTES_AVGPKT
:
89 switch (sinfo
->direction
) {
90 case XT_CONNBYTES_DIR_ORIGINAL
:
91 what
= div64_64(counters
[IP_CT_DIR_ORIGINAL
].bytes
,
92 counters
[IP_CT_DIR_ORIGINAL
].packets
);
94 case XT_CONNBYTES_DIR_REPLY
:
95 what
= div64_64(counters
[IP_CT_DIR_REPLY
].bytes
,
96 counters
[IP_CT_DIR_REPLY
].packets
);
98 case XT_CONNBYTES_DIR_BOTH
:
102 bytes
= counters
[IP_CT_DIR_ORIGINAL
].bytes
+
103 counters
[IP_CT_DIR_REPLY
].bytes
;
104 pkts
= counters
[IP_CT_DIR_ORIGINAL
].packets
+
105 counters
[IP_CT_DIR_REPLY
].packets
;
107 /* FIXME_THEORETICAL: what to do if sum
110 what
= div64_64(bytes
, pkts
);
118 return (what
<= sinfo
->count
.to
&& what
>= sinfo
->count
.from
);
120 return (what
>= sinfo
->count
.from
);
123 static int check(const char *tablename
,
126 unsigned int matchsize
,
127 unsigned int hook_mask
)
129 const struct xt_connbytes_info
*sinfo
= matchinfo
;
131 if (matchsize
!= XT_ALIGN(sizeof(struct xt_connbytes_info
)))
134 if (sinfo
->what
!= XT_CONNBYTES_PKTS
&&
135 sinfo
->what
!= XT_CONNBYTES_BYTES
&&
136 sinfo
->what
!= XT_CONNBYTES_AVGPKT
)
139 if (sinfo
->direction
!= XT_CONNBYTES_DIR_ORIGINAL
&&
140 sinfo
->direction
!= XT_CONNBYTES_DIR_REPLY
&&
141 sinfo
->direction
!= XT_CONNBYTES_DIR_BOTH
)
147 static struct xt_match connbytes_match
= {
150 .checkentry
= &check
,
153 static struct xt_match connbytes6_match
= {
156 .checkentry
= &check
,
160 static int __init
init(void)
163 ret
= xt_register_match(AF_INET
, &connbytes_match
);
167 ret
= xt_register_match(AF_INET6
, &connbytes6_match
);
169 xt_unregister_match(AF_INET
, &connbytes_match
);
173 static void __exit
fini(void)
175 xt_unregister_match(AF_INET
, &connbytes_match
);
176 xt_unregister_match(AF_INET6
, &connbytes6_match
);