blob: 8170edd806b88b98c8f09e10c67b5e28d66b8928 [file] [log] [blame]
Marc Bouchere6869a82000-03-20 06:03:29 +00001/* Shared library add-on to iptables to add customized REJECT support.
2 *
3 * (C) 2000 Jozsef Kadlecsik <kadlec@blackhole.kfki.hu>
4 */
5#include <stdio.h>
6#include <string.h>
7#include <stdlib.h>
8#include <getopt.h>
9#include <iptables.h>
10#include <linux/netfilter_ipv4/ip_tables.h>
11#include <linux/netfilter_ipv4/ipt_REJECT.h>
Harald Welte5a52c512003-05-24 11:44:18 +000012#include <linux/version.h>
13
14/* If we are compiling against a kernel that does not support
15 * IPT_ICMP_ADMIN_PROHIBITED, we are emulating it.
16 * The result will be a plain DROP of the packet instead of
17 * reject. -- Maciej Soltysiak <solt@dns.toxicfilms.tv>
18 */
19#ifndef IPT_ICMP_ADMIN_PROHIBITED
20#define IPT_ICMP_ADMIN_PROHIBITED IPT_TCP_RESET + 1
21#endif
Marc Bouchere6869a82000-03-20 06:03:29 +000022
23struct reject_names {
24 const char *name;
25 const char *alias;
26 enum ipt_reject_with with;
27 const char *desc;
28};
29
30static const struct reject_names reject_table[] = {
Rusty Russell7e53bf92000-03-20 07:03:28 +000031 {"icmp-net-unreachable", "net-unreach",
Marc Bouchere6869a82000-03-20 06:03:29 +000032 IPT_ICMP_NET_UNREACHABLE, "ICMP network unreachable"},
33 {"icmp-host-unreachable", "host-unreach",
34 IPT_ICMP_HOST_UNREACHABLE, "ICMP host unreachable"},
Marc Bouchere6869a82000-03-20 06:03:29 +000035 {"icmp-proto-unreachable", "proto-unreach",
36 IPT_ICMP_PROT_UNREACHABLE, "ICMP protocol unreachable"},
Harald Welte30d920a2001-06-16 19:02:25 +000037 {"icmp-port-unreachable", "port-unreach",
38 IPT_ICMP_PORT_UNREACHABLE, "ICMP port unreachable (default)"},
Rusty Russellbd8382b2000-12-18 05:09:52 +000039#if 0
Marc Bouchere6869a82000-03-20 06:03:29 +000040 {"echo-reply", "echoreply",
Rusty Russellf7e72d52000-06-20 19:07:29 +000041 IPT_ICMP_ECHOREPLY, "for ICMP echo only: faked ICMP echo reply"},
Rusty Russellbd8382b2000-12-18 05:09:52 +000042#endif
Rusty Russellf7e72d52000-06-20 19:07:29 +000043 {"icmp-net-prohibited", "net-prohib",
44 IPT_ICMP_NET_PROHIBITED, "ICMP network prohibited"},
45 {"icmp-host-prohibited", "host-prohib",
46 IPT_ICMP_HOST_PROHIBITED, "ICMP host prohibited"},
Rusty Russell52e440e2000-07-12 07:30:55 +000047 {"tcp-reset", "tcp-reset",
Harald Welte5a52c512003-05-24 11:44:18 +000048 IPT_TCP_RESET, "TCP RST packet"},
49 {"icmp-admin-prohibited", "admin-prohib",
50 IPT_ICMP_ADMIN_PROHIBITED, "ICMP administratively prohibited (*)"}
Marc Bouchere6869a82000-03-20 06:03:29 +000051};
52
Rusty Russell7e53bf92000-03-20 07:03:28 +000053static void
Marc Bouchere6869a82000-03-20 06:03:29 +000054print_reject_types()
55{
56 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +000057
Marc Bouchere6869a82000-03-20 06:03:29 +000058 printf("Valid reject types:\n");
59
60 for (i = 0; i < sizeof(reject_table)/sizeof(struct reject_names); i++) {
61 printf(" %-25s\t%s\n", reject_table[i].name, reject_table[i].desc);
62 printf(" %-25s\talias\n", reject_table[i].alias);
63 }
64 printf("\n");
65}
66
67/* Saves the union ipt_targinfo in parsable form to stdout. */
68
69/* Function which prints out usage message. */
Rusty Russell7e53bf92000-03-20 07:03:28 +000070static void
Marc Bouchere6869a82000-03-20 06:03:29 +000071help(void)
72{
73 printf(
74"REJECT options:\n"
75"--reject-with type drop input packet and send back\n"
76" a reply packet according to type:\n");
77
78 print_reject_types();
Harald Welte5a52c512003-05-24 11:44:18 +000079
80 printf("(*) See man page or read the INCOMPATIBILITES file for compatibility issues.\n");
Marc Bouchere6869a82000-03-20 06:03:29 +000081}
82
83static struct option opts[] = {
84 { "reject-with", 1, 0, '1' },
85 { 0 }
86};
87
88/* Allocate and initialize the target. */
89static void
90init(struct ipt_entry_target *t, unsigned int *nfcache)
91{
92 struct ipt_reject_info *reject = (struct ipt_reject_info *)t->data;
Rusty Russell7e53bf92000-03-20 07:03:28 +000093
Marc Bouchere6869a82000-03-20 06:03:29 +000094 /* default */
95 reject->with = IPT_ICMP_PORT_UNREACHABLE;
Rusty Russell7e53bf92000-03-20 07:03:28 +000096
Marc Bouchere6869a82000-03-20 06:03:29 +000097 /* Can't cache this */
98 *nfcache |= NFC_UNKNOWN;
99}
100
101/* Function which parses command options; returns true if it
102 ate an option */
103static int
104parse(int c, char **argv, int invert, unsigned int *flags,
105 const struct ipt_entry *entry,
106 struct ipt_entry_target **target)
Rusty Russell7e53bf92000-03-20 07:03:28 +0000107{
Marc Bouchere6869a82000-03-20 06:03:29 +0000108 struct ipt_reject_info *reject = (struct ipt_reject_info *)(*target)->data;
109 unsigned int limit = sizeof(reject_table)/sizeof(struct reject_names);
110 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000111
Marc Bouchere6869a82000-03-20 06:03:29 +0000112 switch(c) {
113 case '1':
Harald Welteb77f1da2002-03-14 11:35:58 +0000114 if (check_inverse(optarg, &invert, NULL, 0))
Rusty Russell7e53bf92000-03-20 07:03:28 +0000115 exit_error(PARAMETER_PROBLEM,
Marc Bouchere6869a82000-03-20 06:03:29 +0000116 "Unexpected `!' after --reject-with");
117 for (i = 0; i < limit; i++) {
118 if ((strncasecmp(reject_table[i].name, optarg, strlen(optarg)) == 0)
119 || (strncasecmp(reject_table[i].alias, optarg, strlen(optarg)) == 0)) {
120 reject->with = reject_table[i].with;
121 return 1;
122 }
123 }
Rusty Russellbd8382b2000-12-18 05:09:52 +0000124 /* This due to be dropped late in 2.4 pre-release cycle --RR */
125 if (strncasecmp("echo-reply", optarg, strlen(optarg)) == 0
126 || strncasecmp("echoreply", optarg, strlen(optarg)) == 0)
127 fprintf(stderr, "--reject-with echo-reply no longer"
128 " supported\n");
Marc Bouchere6869a82000-03-20 06:03:29 +0000129 exit_error(PARAMETER_PROBLEM, "unknown reject type `%s'",optarg);
130 default:
131 /* Fall through */
Marc Boucher6e9bfc72002-01-19 12:48:05 +0000132 break;
Marc Bouchere6869a82000-03-20 06:03:29 +0000133 }
134 return 0;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000135}
Marc Bouchere6869a82000-03-20 06:03:29 +0000136
137/* Final check; nothing. */
138static void final_check(unsigned int flags)
139{
140}
141
142/* Prints out ipt_reject_info. */
Rusty Russell7e53bf92000-03-20 07:03:28 +0000143static void
Marc Bouchere6869a82000-03-20 06:03:29 +0000144print(const struct ipt_ip *ip,
Rusty Russell7e53bf92000-03-20 07:03:28 +0000145 const struct ipt_entry_target *target,
Marc Bouchere6869a82000-03-20 06:03:29 +0000146 int numeric)
147{
Rusty Russell7e53bf92000-03-20 07:03:28 +0000148 const struct ipt_reject_info *reject
Marc Bouchere6869a82000-03-20 06:03:29 +0000149 = (const struct ipt_reject_info *)target->data;
150 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000151
Marc Bouchere6869a82000-03-20 06:03:29 +0000152 for (i = 0; i < sizeof(reject_table)/sizeof(struct reject_names); i++) {
153 if (reject_table[i].with == reject->with)
154 break;
155 }
156 printf("reject-with %s ", reject_table[i].name);
157}
158
159/* Saves ipt_reject in parsable form to stdout. */
160static void save(const struct ipt_ip *ip, const struct ipt_entry_target *target)
161{
Rusty Russell7e53bf92000-03-20 07:03:28 +0000162 const struct ipt_reject_info *reject
Marc Bouchere6869a82000-03-20 06:03:29 +0000163 = (const struct ipt_reject_info *)target->data;
Harald Welte97b3fde2001-05-12 05:22:18 +0000164 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000165
Harald Welte97b3fde2001-05-12 05:22:18 +0000166 for (i = 0; i < sizeof(reject_table)/sizeof(struct reject_names); i++)
167 if (reject_table[i].with == reject->with)
168 break;
169
170 printf("--reject-with %s ", reject_table[i].name);
Marc Bouchere6869a82000-03-20 06:03:29 +0000171}
172
Harald Welte3efb6ea2001-08-06 18:50:21 +0000173static
Marc Bouchere6869a82000-03-20 06:03:29 +0000174struct iptables_target reject
175= { NULL,
176 "REJECT",
Harald Welte80fe35d2002-05-29 13:08:15 +0000177 IPTABLES_VERSION,
Rusty Russell73f72f52000-07-03 10:17:57 +0000178 IPT_ALIGN(sizeof(struct ipt_reject_info)),
179 IPT_ALIGN(sizeof(struct ipt_reject_info)),
Marc Bouchere6869a82000-03-20 06:03:29 +0000180 &help,
181 &init,
182 &parse,
183 &final_check,
184 &print,
185 &save,
186 opts
187};
188
189void _init(void)
190{
191 register_target(&reject);
192}