blob: 16e7b080428760c69559135a49bd3cfe76cbad81 [file] [log] [blame]
Harald Welte2e4e6a12006-01-12 13:30:04 -08001/* (C) 1999-2001 Michal Ludvig <michal@logix.cz>
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 as
5 * published by the Free Software Foundation.
6 */
7
8#include <linux/module.h>
9#include <linux/skbuff.h>
10#include <linux/if_ether.h>
11#include <linux/if_packet.h>
Phil Oester28658c82006-07-24 22:54:14 -070012#include <linux/in.h>
13#include <linux/ip.h>
Harald Welte2e4e6a12006-01-12 13:30:04 -080014
15#include <linux/netfilter/xt_pkttype.h>
16#include <linux/netfilter/x_tables.h>
17
18MODULE_LICENSE("GPL");
19MODULE_AUTHOR("Michal Ludvig <michal@logix.cz>");
20MODULE_DESCRIPTION("IP tables match to match on linklayer packet type");
21MODULE_ALIAS("ipt_pkttype");
22MODULE_ALIAS("ip6t_pkttype");
23
24static int match(const struct sk_buff *skb,
25 const struct net_device *in,
26 const struct net_device *out,
Patrick McHardyc4986732006-03-20 18:02:56 -080027 const struct xt_match *match,
Harald Welte2e4e6a12006-01-12 13:30:04 -080028 const void *matchinfo,
29 int offset,
30 unsigned int protoff,
31 int *hotdrop)
32{
Phil Oester28658c82006-07-24 22:54:14 -070033 u_int8_t type;
Harald Welte2e4e6a12006-01-12 13:30:04 -080034 const struct xt_pkttype_info *info = matchinfo;
35
Phil Oester28658c82006-07-24 22:54:14 -070036 if (skb->pkt_type == PACKET_LOOPBACK)
37 type = (MULTICAST(skb->nh.iph->daddr)
38 ? PACKET_MULTICAST
39 : PACKET_BROADCAST);
40 else
41 type = skb->pkt_type;
42
43 return (type == info->pkttype) ^ info->invert;
Harald Welte2e4e6a12006-01-12 13:30:04 -080044}
45
Patrick McHardy4470bbc2006-08-22 00:34:04 -070046static struct xt_match xt_pkttype_match[] = {
47 {
48 .name = "pkttype",
49 .family = AF_INET,
50 .match = match,
51 .matchsize = sizeof(struct xt_pkttype_info),
52 .me = THIS_MODULE,
53 },
54 {
55 .name = "pkttype",
56 .family = AF_INET6,
57 .match = match,
58 .matchsize = sizeof(struct xt_pkttype_info),
59 .me = THIS_MODULE,
60 },
Patrick McHardy5d04bff2006-03-20 18:01:58 -080061};
Harald Welte2e4e6a12006-01-12 13:30:04 -080062
Andrew Morton65b4b4e2006-03-28 16:37:06 -080063static int __init xt_pkttype_init(void)
Harald Welte2e4e6a12006-01-12 13:30:04 -080064{
Patrick McHardy4470bbc2006-08-22 00:34:04 -070065 return xt_register_matches(xt_pkttype_match,
66 ARRAY_SIZE(xt_pkttype_match));
Harald Welte2e4e6a12006-01-12 13:30:04 -080067}
68
Andrew Morton65b4b4e2006-03-28 16:37:06 -080069static void __exit xt_pkttype_fini(void)
Harald Welte2e4e6a12006-01-12 13:30:04 -080070{
Patrick McHardy4470bbc2006-08-22 00:34:04 -070071 xt_unregister_matches(xt_pkttype_match, ARRAY_SIZE(xt_pkttype_match));
Harald Welte2e4e6a12006-01-12 13:30:04 -080072}
73
Andrew Morton65b4b4e2006-03-28 16:37:06 -080074module_init(xt_pkttype_init);
75module_exit(xt_pkttype_fini);