blob: 90ec163f347b77b8f37c5a5c21a22df81b597f0e [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"},
Harald Welte11b85912005-11-22 08:54:28 +000047 {"tcp-reset", "tcp-rst",
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
Patrick McHardy500f4832007-09-08 15:59:04 +000054print_reject_types(void)
Marc Bouchere6869a82000-03-20 06:03:29 +000055{
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"
Harald Welte7d774512005-04-15 09:39:55 +000076" a reply packet according to type:\n");
Marc Bouchere6869a82000-03-20 06:03:29 +000077
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
Jan Engelhardt661f1122007-07-30 14:46:51 +000083static const struct option opts[] = {
Patrick McHardy500f4832007-09-08 15:59:04 +000084 { "reject-with", 1, NULL, '1' },
85 { }
Marc Bouchere6869a82000-03-20 06:03:29 +000086};
87
88/* Allocate and initialize the target. */
89static void
Peter Rileyea146a92007-09-02 13:09:07 +000090init(struct xt_entry_target *t)
Marc Bouchere6869a82000-03-20 06:03:29 +000091{
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}
98
99/* Function which parses command options; returns true if it
100 ate an option */
101static int
102parse(int c, char **argv, int invert, unsigned int *flags,
Yasuyuki KOZAKAIc0a9ab92007-07-24 06:02:05 +0000103 const void *entry,
Yasuyuki KOZAKAI193df8e2007-07-24 05:57:28 +0000104 struct xt_entry_target **target)
Rusty Russell7e53bf92000-03-20 07:03:28 +0000105{
Marc Bouchere6869a82000-03-20 06:03:29 +0000106 struct ipt_reject_info *reject = (struct ipt_reject_info *)(*target)->data;
107 unsigned int limit = sizeof(reject_table)/sizeof(struct reject_names);
108 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000109
Marc Bouchere6869a82000-03-20 06:03:29 +0000110 switch(c) {
111 case '1':
Harald Welteb77f1da2002-03-14 11:35:58 +0000112 if (check_inverse(optarg, &invert, NULL, 0))
Rusty Russell7e53bf92000-03-20 07:03:28 +0000113 exit_error(PARAMETER_PROBLEM,
Marc Bouchere6869a82000-03-20 06:03:29 +0000114 "Unexpected `!' after --reject-with");
115 for (i = 0; i < limit; i++) {
116 if ((strncasecmp(reject_table[i].name, optarg, strlen(optarg)) == 0)
117 || (strncasecmp(reject_table[i].alias, optarg, strlen(optarg)) == 0)) {
118 reject->with = reject_table[i].with;
119 return 1;
120 }
121 }
Rusty Russellbd8382b2000-12-18 05:09:52 +0000122 /* This due to be dropped late in 2.4 pre-release cycle --RR */
123 if (strncasecmp("echo-reply", optarg, strlen(optarg)) == 0
124 || strncasecmp("echoreply", optarg, strlen(optarg)) == 0)
125 fprintf(stderr, "--reject-with echo-reply no longer"
126 " supported\n");
Marc Bouchere6869a82000-03-20 06:03:29 +0000127 exit_error(PARAMETER_PROBLEM, "unknown reject type `%s'",optarg);
128 default:
129 /* Fall through */
Marc Boucher6e9bfc72002-01-19 12:48:05 +0000130 break;
Marc Bouchere6869a82000-03-20 06:03:29 +0000131 }
132 return 0;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000133}
Marc Bouchere6869a82000-03-20 06:03:29 +0000134
135/* Final check; nothing. */
136static void final_check(unsigned int flags)
137{
138}
139
140/* Prints out ipt_reject_info. */
Rusty Russell7e53bf92000-03-20 07:03:28 +0000141static void
Yasuyuki KOZAKAIc0a9ab92007-07-24 06:02:05 +0000142print(const void *ip,
Yasuyuki KOZAKAI193df8e2007-07-24 05:57:28 +0000143 const struct xt_entry_target *target,
Marc Bouchere6869a82000-03-20 06:03:29 +0000144 int numeric)
145{
Rusty Russell7e53bf92000-03-20 07:03:28 +0000146 const struct ipt_reject_info *reject
Marc Bouchere6869a82000-03-20 06:03:29 +0000147 = (const struct ipt_reject_info *)target->data;
148 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000149
Marc Bouchere6869a82000-03-20 06:03:29 +0000150 for (i = 0; i < sizeof(reject_table)/sizeof(struct reject_names); i++) {
151 if (reject_table[i].with == reject->with)
152 break;
153 }
154 printf("reject-with %s ", reject_table[i].name);
155}
156
157/* Saves ipt_reject in parsable form to stdout. */
Yasuyuki KOZAKAIc0a9ab92007-07-24 06:02:05 +0000158static void save(const void *ip, const struct xt_entry_target *target)
Marc Bouchere6869a82000-03-20 06:03:29 +0000159{
Rusty Russell7e53bf92000-03-20 07:03:28 +0000160 const struct ipt_reject_info *reject
Marc Bouchere6869a82000-03-20 06:03:29 +0000161 = (const struct ipt_reject_info *)target->data;
Harald Welte97b3fde2001-05-12 05:22:18 +0000162 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000163
Harald Welte97b3fde2001-05-12 05:22:18 +0000164 for (i = 0; i < sizeof(reject_table)/sizeof(struct reject_names); i++)
165 if (reject_table[i].with == reject->with)
166 break;
167
168 printf("--reject-with %s ", reject_table[i].name);
Marc Bouchere6869a82000-03-20 06:03:29 +0000169}
170
Harald Welte7d774512005-04-15 09:39:55 +0000171static struct iptables_target reject = {
Pablo Neira8caee8b2004-12-28 13:11:59 +0000172 .name = "REJECT",
173 .version = IPTABLES_VERSION,
174 .size = IPT_ALIGN(sizeof(struct ipt_reject_info)),
175 .userspacesize = IPT_ALIGN(sizeof(struct ipt_reject_info)),
176 .help = &help,
177 .init = &init,
178 .parse = &parse,
179 .final_check = &final_check,
180 .print = &print,
181 .save = &save,
182 .extra_opts = opts
Marc Bouchere6869a82000-03-20 06:03:29 +0000183};
184
185void _init(void)
186{
Harald Welte7d774512005-04-15 09:39:55 +0000187 register_target(&reject);
Marc Bouchere6869a82000-03-20 06:03:29 +0000188}