blob: c3757f389ba236043f0b6439190a8771a8b94660 [file] [log] [blame]
Bart De Schuymerff587202005-02-08 20:02:28 +00001/* ebt_arpreply
2 *
3 * Authors:
4 * Grzegorz Borowiak <grzes@gnu.univ.gda.pl>
5 * Bart De Schuymer <bdschuym@pandora.be>
6 *
7 * August, 2003
8 */
9
Bart De Schuymer5a824802003-08-14 19:24:25 +000010#include <stdio.h>
11#include <stdlib.h>
12#include <string.h>
Bart De Schuymer5a824802003-08-14 19:24:25 +000013#include <getopt.h>
14#include "../include/ebtables_u.h"
Bart De Schuymer57633b12010-12-18 16:15:49 +000015#include <netinet/ether.h>
Bart De Schuymer5a824802003-08-14 19:24:25 +000016#include <linux/netfilter_bridge/ebt_arpreply.h>
17
18static int mac_supplied;
19
20#define REPLY_MAC '1'
21#define REPLY_TARGET '2'
22static struct option opts[] =
23{
24 { "arpreply-mac" , required_argument, 0, REPLY_MAC },
25 { "arpreply-target" , required_argument, 0, REPLY_TARGET },
26 { 0 }
27};
28
29static void print_help()
30{
31 printf(
32 "arpreply target options:\n"
33 " --arpreply-mac address : source MAC of generated reply\n"
34 " --arpreply-target target : ACCEPT, DROP, RETURN or CONTINUE\n"
35 " (standard target is DROP)\n");
36}
37
38static void init(struct ebt_entry_target *target)
39{
40 struct ebt_arpreply_info *replyinfo =
41 (struct ebt_arpreply_info *)target->data;
42
43 replyinfo->target = EBT_DROP;
44 memset(replyinfo->mac, 0, ETH_ALEN);
45 mac_supplied = 0;
46}
47
48#define OPT_REPLY_MAC 0x01
49#define OPT_REPLY_TARGET 0x02
50static int parse(int c, char **argv, int argc,
51 const struct ebt_u_entry *entry, unsigned int *flags,
52 struct ebt_entry_target **target)
53{
54 struct ebt_arpreply_info *replyinfo =
55 (struct ebt_arpreply_info *)(*target)->data;
56 struct ether_addr *addr;
57
58 switch (c) {
59 case REPLY_MAC:
Bart De Schuymerff587202005-02-08 20:02:28 +000060 ebt_check_option2(flags, OPT_REPLY_MAC);
Bart De Schuymer5a824802003-08-14 19:24:25 +000061 if (!(addr = ether_aton(optarg)))
Bart De Schuymerff587202005-02-08 20:02:28 +000062 ebt_print_error2("Problem with specified --arpreply-mac mac");
Bart De Schuymer5a824802003-08-14 19:24:25 +000063 memcpy(replyinfo->mac, addr, ETH_ALEN);
64 mac_supplied = 1;
65 break;
66 case REPLY_TARGET:
Bart De Schuymerff587202005-02-08 20:02:28 +000067 ebt_check_option2(flags, OPT_REPLY_TARGET);
Bart De Schuymer5a824802003-08-14 19:24:25 +000068 if (FILL_TARGET(optarg, replyinfo->target))
Bart De Schuymerff587202005-02-08 20:02:28 +000069 ebt_print_error2("Illegal --arpreply-target target");
Bart De Schuymer5a824802003-08-14 19:24:25 +000070 break;
71
72 default:
73 return 0;
74 }
75 return 1;
76}
77
78static void final_check(const struct ebt_u_entry *entry,
79 const struct ebt_entry_target *target, const char *name,
80 unsigned int hookmask, unsigned int time)
81{
82 struct ebt_arpreply_info *replyinfo =
83 (struct ebt_arpreply_info *)target->data;
84
Bart De Schuymerff587202005-02-08 20:02:28 +000085 if (entry->ethproto != ETH_P_ARP || entry->invflags & EBT_IPROTO) {
86 ebt_print_error("For ARP replying the protocol must be specified as ARP");
87 } else if (time == 0 && mac_supplied == 0) {
Bart De Schuymer64182a32004-01-21 20:39:54 +000088 ebt_print_error("No arpreply mac supplied");
Bart De Schuymerff587202005-02-08 20:02:28 +000089 } else if (BASE_CHAIN && replyinfo->target == EBT_RETURN) {
90 ebt_print_error("--arpreply-target RETURN not allowed on base chain");
91 } else {
92 CLEAR_BASE_CHAIN_BIT;
93 if (strcmp(name, "nat") || hookmask & ~(1 << NF_BR_PRE_ROUTING))
94 ebt_print_error("arpreply only allowed in PREROUTING");
95 }
Bart De Schuymer5a824802003-08-14 19:24:25 +000096}
97
98static void print(const struct ebt_u_entry *entry,
99 const struct ebt_entry_target *target)
100{
101 struct ebt_arpreply_info *replyinfo =
102 (struct ebt_arpreply_info *)target->data;
103
104 printf("--arpreply-mac ");
Bart De Schuymer8339ff12004-01-14 20:05:27 +0000105 ebt_print_mac(replyinfo->mac);
Bart De Schuymer5a824802003-08-14 19:24:25 +0000106 if (replyinfo->target == EBT_DROP)
107 return;
108 printf(" --arpreply-target %s", TARGET_NAME(replyinfo->target));
109}
110
111static int compare(const struct ebt_entry_target *t1,
112 const struct ebt_entry_target *t2)
113{
114 struct ebt_arpreply_info *replyinfo1 =
115 (struct ebt_arpreply_info *)t1->data;
116 struct ebt_arpreply_info *replyinfo2 =
117 (struct ebt_arpreply_info *)t2->data;
118
119 return memcmp(replyinfo1->mac, replyinfo2->mac, ETH_ALEN) == 0
120 && replyinfo1->target == replyinfo2->target;
121}
122
123static struct ebt_u_target arpreply_target =
124{
Bart De Schuymerfbdebad2008-02-03 19:58:44 +0000125 .name = "arpreply",
Bart De Schuymer7cf1cca2003-08-30 16:20:19 +0000126 .size = sizeof(struct ebt_arpreply_info),
127 .help = print_help,
128 .init = init,
129 .parse = parse,
130 .final_check = final_check,
131 .print = print,
132 .compare = compare,
133 .extra_ops = opts,
Bart De Schuymer5a824802003-08-14 19:24:25 +0000134};
135
Bart De Schuymer64182a32004-01-21 20:39:54 +0000136void _init(void)
Bart De Schuymer5a824802003-08-14 19:24:25 +0000137{
Bart De Schuymer8339ff12004-01-14 20:05:27 +0000138 ebt_register_target(&arpreply_target);
Bart De Schuymer5a824802003-08-14 19:24:25 +0000139}