1 #include <linux/module.h>
2 #include <linux/skbuff.h>
5 #include <linux/sctp.h>
7 #include <linux/netfilter/x_tables.h>
8 #include <linux/netfilter/xt_sctp.h>
9 #include <linux/netfilter_ipv4/ip_tables.h>
10 #include <linux/netfilter_ipv6/ip6_tables.h>
12 MODULE_LICENSE("GPL");
13 MODULE_AUTHOR("Kiran Kumar Immidi");
14 MODULE_DESCRIPTION("Match for SCTP protocol packets");
15 MODULE_ALIAS("ipt_sctp");
18 #define duprintf(format, args...) printk(format , ## args)
20 #define duprintf(format, args...)
23 #define SCCHECK(cond, option, flag, invflag) (!((flag) & (option)) \
24 || (!!((invflag) & (option)) ^ (cond)))
27 match_flags(const struct xt_sctp_flag_info
*flag_info
,
34 for (i
= 0; i
< flag_count
; i
++)
35 if (flag_info
[i
].chunktype
== chunktype
)
36 return (chunkflags
& flag_info
[i
].flag_mask
) == flag_info
[i
].flag
;
42 match_packet(const struct sk_buff
*skb
,
44 const u_int32_t
*chunkmap
,
46 const struct xt_sctp_flag_info
*flag_info
,
50 u_int32_t chunkmapcopy
[256 / sizeof (u_int32_t
)];
51 sctp_chunkhdr_t _sch
, *sch
;
57 if (chunk_match_type
== SCTP_CHUNK_MATCH_ALL
)
58 SCTP_CHUNKMAP_COPY(chunkmapcopy
, chunkmap
);
61 sch
= skb_header_pointer(skb
, offset
, sizeof(_sch
), &_sch
);
62 if (sch
== NULL
|| sch
->length
== 0) {
63 duprintf("Dropping invalid SCTP packet.\n");
68 duprintf("Chunk num: %d\toffset: %d\ttype: %d\tlength: %d\tflags: %x\n",
69 ++i
, offset
, sch
->type
, htons(sch
->length
), sch
->flags
);
71 offset
+= (ntohs(sch
->length
) + 3) & ~3;
73 duprintf("skb->len: %d\toffset: %d\n", skb
->len
, offset
);
75 if (SCTP_CHUNKMAP_IS_SET(chunkmap
, sch
->type
)) {
76 switch (chunk_match_type
) {
77 case SCTP_CHUNK_MATCH_ANY
:
78 if (match_flags(flag_info
, flag_count
,
79 sch
->type
, sch
->flags
)) {
84 case SCTP_CHUNK_MATCH_ALL
:
85 if (match_flags(flag_info
, flag_count
,
86 sch
->type
, sch
->flags
))
87 SCTP_CHUNKMAP_CLEAR(chunkmapcopy
, sch
->type
);
90 case SCTP_CHUNK_MATCH_ONLY
:
91 if (!match_flags(flag_info
, flag_count
,
92 sch
->type
, sch
->flags
))
97 switch (chunk_match_type
) {
98 case SCTP_CHUNK_MATCH_ONLY
:
102 } while (offset
< skb
->len
);
104 switch (chunk_match_type
) {
105 case SCTP_CHUNK_MATCH_ALL
:
106 return SCTP_CHUNKMAP_IS_CLEAR(chunkmap
);
107 case SCTP_CHUNK_MATCH_ANY
:
109 case SCTP_CHUNK_MATCH_ONLY
:
113 /* This will never be reached, but required to stop compiler whine */
118 match(const struct sk_buff
*skb
,
119 const struct net_device
*in
,
120 const struct net_device
*out
,
121 const struct xt_match
*match
,
122 const void *matchinfo
,
124 unsigned int protoff
,
127 const struct xt_sctp_info
*info
= matchinfo
;
128 sctp_sctphdr_t _sh
, *sh
;
131 duprintf("Dropping non-first fragment.. FIXME\n");
135 sh
= skb_header_pointer(skb
, protoff
, sizeof(_sh
), &_sh
);
137 duprintf("Dropping evil TCP offset=0 tinygram.\n");
141 duprintf("spt: %d\tdpt: %d\n", ntohs(sh
->source
), ntohs(sh
->dest
));
143 return SCCHECK(ntohs(sh
->source
) >= info
->spts
[0]
144 && ntohs(sh
->source
) <= info
->spts
[1],
145 XT_SCTP_SRC_PORTS
, info
->flags
, info
->invflags
)
146 && SCCHECK(ntohs(sh
->dest
) >= info
->dpts
[0]
147 && ntohs(sh
->dest
) <= info
->dpts
[1],
148 XT_SCTP_DEST_PORTS
, info
->flags
, info
->invflags
)
149 && SCCHECK(match_packet(skb
, protoff
+ sizeof (sctp_sctphdr_t
),
150 info
->chunkmap
, info
->chunk_match_type
,
151 info
->flag_info
, info
->flag_count
,
153 XT_SCTP_CHUNK_TYPES
, info
->flags
, info
->invflags
);
157 checkentry(const char *tablename
,
159 const struct xt_match
*match
,
161 unsigned int hook_mask
)
163 const struct xt_sctp_info
*info
= matchinfo
;
165 return !(info
->flags
& ~XT_SCTP_VALID_FLAGS
)
166 && !(info
->invflags
& ~XT_SCTP_VALID_FLAGS
)
167 && !(info
->invflags
& ~info
->flags
)
168 && ((!(info
->flags
& XT_SCTP_CHUNK_TYPES
)) ||
169 (info
->chunk_match_type
&
170 (SCTP_CHUNK_MATCH_ALL
171 | SCTP_CHUNK_MATCH_ANY
172 | SCTP_CHUNK_MATCH_ONLY
)));
175 static struct xt_match xt_sctp_match
[] __read_mostly
= {
179 .checkentry
= checkentry
,
181 .matchsize
= sizeof(struct xt_sctp_info
),
182 .proto
= IPPROTO_SCTP
,
188 .checkentry
= checkentry
,
190 .matchsize
= sizeof(struct xt_sctp_info
),
191 .proto
= IPPROTO_SCTP
,
196 static int __init
xt_sctp_init(void)
198 return xt_register_matches(xt_sctp_match
, ARRAY_SIZE(xt_sctp_match
));
201 static void __exit
xt_sctp_fini(void)
203 xt_unregister_matches(xt_sctp_match
, ARRAY_SIZE(xt_sctp_match
));
206 module_init(xt_sctp_init
);
207 module_exit(xt_sctp_fini
);