blob: 5901a5ef894bfb6134c8ad151ed5ab3de73b3f72 [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. */
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +000070static void REJECT_help(void)
Marc Bouchere6869a82000-03-20 06:03:29 +000071{
72 printf(
73"REJECT options:\n"
74"--reject-with type drop input packet and send back\n"
Harald Welte7d774512005-04-15 09:39:55 +000075" a reply packet according to type:\n");
Marc Bouchere6869a82000-03-20 06:03:29 +000076
77 print_reject_types();
Harald Welte5a52c512003-05-24 11:44:18 +000078
79 printf("(*) See man page or read the INCOMPATIBILITES file for compatibility issues.\n");
Marc Bouchere6869a82000-03-20 06:03:29 +000080}
81
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +000082static const struct option REJECT_opts[] = {
Patrick McHardy500f4832007-09-08 15:59:04 +000083 { "reject-with", 1, NULL, '1' },
Max Kellermann9ee386a2008-01-29 13:48:05 +000084 { .name = NULL }
Marc Bouchere6869a82000-03-20 06:03:29 +000085};
86
87/* Allocate and initialize the target. */
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +000088static void REJECT_init(struct xt_entry_target *t)
Marc Bouchere6869a82000-03-20 06:03:29 +000089{
90 struct ipt_reject_info *reject = (struct ipt_reject_info *)t->data;
Rusty Russell7e53bf92000-03-20 07:03:28 +000091
Marc Bouchere6869a82000-03-20 06:03:29 +000092 /* default */
93 reject->with = IPT_ICMP_PORT_UNREACHABLE;
Rusty Russell7e53bf92000-03-20 07:03:28 +000094
Marc Bouchere6869a82000-03-20 06:03:29 +000095}
96
97/* Function which parses command options; returns true if it
98 ate an option */
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +000099static int REJECT_parse(int c, char **argv, int invert, unsigned int *flags,
100 const void *entry, struct xt_entry_target **target)
Rusty Russell7e53bf92000-03-20 07:03:28 +0000101{
Marc Bouchere6869a82000-03-20 06:03:29 +0000102 struct ipt_reject_info *reject = (struct ipt_reject_info *)(*target)->data;
103 unsigned int limit = sizeof(reject_table)/sizeof(struct reject_names);
104 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000105
Marc Bouchere6869a82000-03-20 06:03:29 +0000106 switch(c) {
107 case '1':
Harald Welteb77f1da2002-03-14 11:35:58 +0000108 if (check_inverse(optarg, &invert, NULL, 0))
Rusty Russell7e53bf92000-03-20 07:03:28 +0000109 exit_error(PARAMETER_PROBLEM,
Marc Bouchere6869a82000-03-20 06:03:29 +0000110 "Unexpected `!' after --reject-with");
111 for (i = 0; i < limit; i++) {
112 if ((strncasecmp(reject_table[i].name, optarg, strlen(optarg)) == 0)
113 || (strncasecmp(reject_table[i].alias, optarg, strlen(optarg)) == 0)) {
114 reject->with = reject_table[i].with;
115 return 1;
116 }
117 }
Rusty Russellbd8382b2000-12-18 05:09:52 +0000118 /* This due to be dropped late in 2.4 pre-release cycle --RR */
119 if (strncasecmp("echo-reply", optarg, strlen(optarg)) == 0
120 || strncasecmp("echoreply", optarg, strlen(optarg)) == 0)
121 fprintf(stderr, "--reject-with echo-reply no longer"
122 " supported\n");
Marc Bouchere6869a82000-03-20 06:03:29 +0000123 exit_error(PARAMETER_PROBLEM, "unknown reject type `%s'",optarg);
124 default:
125 /* Fall through */
Marc Boucher6e9bfc72002-01-19 12:48:05 +0000126 break;
Marc Bouchere6869a82000-03-20 06:03:29 +0000127 }
128 return 0;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000129}
Marc Bouchere6869a82000-03-20 06:03:29 +0000130
Marc Bouchere6869a82000-03-20 06:03:29 +0000131/* Prints out ipt_reject_info. */
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +0000132static void REJECT_print(const void *ip, const struct xt_entry_target *target,
133 int numeric)
Marc Bouchere6869a82000-03-20 06:03:29 +0000134{
Rusty Russell7e53bf92000-03-20 07:03:28 +0000135 const struct ipt_reject_info *reject
Marc Bouchere6869a82000-03-20 06:03:29 +0000136 = (const struct ipt_reject_info *)target->data;
137 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000138
Marc Bouchere6869a82000-03-20 06:03:29 +0000139 for (i = 0; i < sizeof(reject_table)/sizeof(struct reject_names); i++) {
140 if (reject_table[i].with == reject->with)
141 break;
142 }
143 printf("reject-with %s ", reject_table[i].name);
144}
145
146/* Saves ipt_reject in parsable form to stdout. */
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +0000147static void REJECT_save(const void *ip, const struct xt_entry_target *target)
Marc Bouchere6869a82000-03-20 06:03:29 +0000148{
Rusty Russell7e53bf92000-03-20 07:03:28 +0000149 const struct ipt_reject_info *reject
Marc Bouchere6869a82000-03-20 06:03:29 +0000150 = (const struct ipt_reject_info *)target->data;
Harald Welte97b3fde2001-05-12 05:22:18 +0000151 unsigned int i;
Rusty Russell7e53bf92000-03-20 07:03:28 +0000152
Harald Welte97b3fde2001-05-12 05:22:18 +0000153 for (i = 0; i < sizeof(reject_table)/sizeof(struct reject_names); i++)
154 if (reject_table[i].with == reject->with)
155 break;
156
157 printf("--reject-with %s ", reject_table[i].name);
Marc Bouchere6869a82000-03-20 06:03:29 +0000158}
159
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +0000160static struct iptables_target reject_target = {
Pablo Neira8caee8b2004-12-28 13:11:59 +0000161 .name = "REJECT",
162 .version = IPTABLES_VERSION,
163 .size = IPT_ALIGN(sizeof(struct ipt_reject_info)),
164 .userspacesize = IPT_ALIGN(sizeof(struct ipt_reject_info)),
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +0000165 .help = REJECT_help,
166 .init = REJECT_init,
167 .parse = REJECT_parse,
168 .print = REJECT_print,
169 .save = REJECT_save,
170 .extra_opts = REJECT_opts,
Marc Bouchere6869a82000-03-20 06:03:29 +0000171};
172
173void _init(void)
174{
Jan Engelhardt1d5b63d2007-10-04 16:29:00 +0000175 register_target(&reject_target);
Marc Bouchere6869a82000-03-20 06:03:29 +0000176}