blob: 97722d6cad942d67b968e8e812e1dada6f035f01 [file] [log] [blame]
Jan Engelhardt32b8e612010-07-23 21:16:14 +02001#include <stdbool.h>
Jan Engelhardtd441ad62011-05-06 17:45:12 +02002#include <stdint.h>
Patrick McHardy524bb802005-11-19 09:00:03 +00003#include <stdio.h>
Patrick McHardy524bb802005-11-19 09:00:03 +00004#include <string.h>
Patrick McHardy524bb802005-11-19 09:00:03 +00005#include <netdb.h>
Jan Engelhardt5d9678a2008-11-20 10:15:35 +01006#include <xtables.h>
Jan Engelhardtcd300542009-05-31 22:43:12 +02007#include <linux/netfilter/xt_policy.h>
Patrick McHardy524bb802005-11-19 09:00:03 +00008
Jan Engelhardtd441ad62011-05-06 17:45:12 +02009enum {
10 O_DIRECTION = 0,
11 O_POLICY,
12 O_STRICT,
13 O_REQID,
14 O_SPI,
15 O_PROTO,
16 O_MODE,
17 O_TUNNELSRC,
18 O_TUNNELDST,
Jan Engelhardt15392932011-05-12 12:46:40 +020019 O_NEXT,
20 F_STRICT = 1 << O_STRICT,
Jan Engelhardtd441ad62011-05-06 17:45:12 +020021};
Patrick McHardy524bb802005-11-19 09:00:03 +000022
Jan Engelhardt59d16402007-10-04 16:28:39 +000023static void policy_help(void)
Patrick McHardy524bb802005-11-19 09:00:03 +000024{
25 printf(
Jan Engelhardt8b7c64d2008-04-15 11:48:25 +020026"policy match options:\n"
Patrick McHardy524bb802005-11-19 09:00:03 +000027" --dir in|out match policy applied during decapsulation/\n"
28" policy to be applied during encapsulation\n"
29" --pol none|ipsec match policy\n"
30" --strict match entire policy instead of single element\n"
31" at any position\n"
Jan Engelhardt15392932011-05-12 12:46:40 +020032"These options may be used repeatedly, to describe policy elements:\n"
Patrick McHardy524bb802005-11-19 09:00:03 +000033"[!] --reqid reqid match reqid\n"
34"[!] --spi spi match SPI\n"
35"[!] --proto proto match protocol (ah/esp/ipcomp)\n"
36"[!] --mode mode match mode (transport/tunnel)\n"
37"[!] --tunnel-src addr/mask match tunnel source\n"
38"[!] --tunnel-dst addr/mask match tunnel destination\n"
Jan Engelhardt8b7c64d2008-04-15 11:48:25 +020039" --next begin next element in policy\n");
Patrick McHardy524bb802005-11-19 09:00:03 +000040}
41
Jan Engelhardtd441ad62011-05-06 17:45:12 +020042static const struct xt_option_entry policy_opts[] = {
43 {.name = "dir", .id = O_DIRECTION, .type = XTTYPE_STRING,
44 .flags = XTOPT_INVERT},
45 {.name = "pol", .id = O_POLICY, .type = XTTYPE_STRING},
46 {.name = "strict", .id = O_STRICT, .type = XTTYPE_NONE},
Jan Engelhardt15392932011-05-12 12:46:40 +020047 {.name = "reqid", .id = O_REQID, .type = XTTYPE_UINT32,
48 .flags = XTOPT_MULTI | XTOPT_INVERT},
49 {.name = "spi", .id = O_SPI, .type = XTTYPE_UINT32,
50 .flags = XTOPT_MULTI | XTOPT_INVERT},
51 {.name = "tunnel-src", .id = O_TUNNELSRC, .type = XTTYPE_HOSTMASK,
52 .flags = XTOPT_MULTI | XTOPT_INVERT},
53 {.name = "tunnel-dst", .id = O_TUNNELDST, .type = XTTYPE_HOSTMASK,
54 .flags = XTOPT_MULTI | XTOPT_INVERT},
Jan Engelhardtcdc8e0b2011-05-12 13:59:38 +020055 {.name = "proto", .id = O_PROTO, .type = XTTYPE_PROTOCOL,
Jan Engelhardt15392932011-05-12 12:46:40 +020056 .flags = XTOPT_MULTI | XTOPT_INVERT},
57 {.name = "mode", .id = O_MODE, .type = XTTYPE_STRING,
58 .flags = XTOPT_MULTI | XTOPT_INVERT},
59 {.name = "next", .id = O_NEXT, .type = XTTYPE_NONE,
60 .flags = XTOPT_MULTI, .also = F_STRICT},
Jan Engelhardtd441ad62011-05-06 17:45:12 +020061 XTOPT_TABLEEND,
Patrick McHardy524bb802005-11-19 09:00:03 +000062};
63
Jan Engelhardtd441ad62011-05-06 17:45:12 +020064static int parse_direction(const char *s)
Patrick McHardy524bb802005-11-19 09:00:03 +000065{
66 if (strcmp(s, "in") == 0)
Jan Engelhardtcd300542009-05-31 22:43:12 +020067 return XT_POLICY_MATCH_IN;
Patrick McHardy524bb802005-11-19 09:00:03 +000068 if (strcmp(s, "out") == 0)
Jan Engelhardtcd300542009-05-31 22:43:12 +020069 return XT_POLICY_MATCH_OUT;
Jan Engelhardt1829ed42009-02-21 03:29:44 +010070 xtables_error(PARAMETER_PROBLEM, "policy_match: invalid dir \"%s\"", s);
Patrick McHardy524bb802005-11-19 09:00:03 +000071}
72
Jan Engelhardtd441ad62011-05-06 17:45:12 +020073static int parse_policy(const char *s)
Patrick McHardy524bb802005-11-19 09:00:03 +000074{
75 if (strcmp(s, "none") == 0)
Jan Engelhardtcd300542009-05-31 22:43:12 +020076 return XT_POLICY_MATCH_NONE;
Patrick McHardy524bb802005-11-19 09:00:03 +000077 if (strcmp(s, "ipsec") == 0)
78 return 0;
Jan Engelhardt1829ed42009-02-21 03:29:44 +010079 xtables_error(PARAMETER_PROBLEM, "policy match: invalid policy \"%s\"", s);
Patrick McHardy524bb802005-11-19 09:00:03 +000080}
81
Jan Engelhardtd441ad62011-05-06 17:45:12 +020082static int parse_mode(const char *s)
Patrick McHardy524bb802005-11-19 09:00:03 +000083{
84 if (strcmp(s, "transport") == 0)
Jan Engelhardtcd300542009-05-31 22:43:12 +020085 return XT_POLICY_MODE_TRANSPORT;
Patrick McHardy524bb802005-11-19 09:00:03 +000086 if (strcmp(s, "tunnel") == 0)
Jan Engelhardtcd300542009-05-31 22:43:12 +020087 return XT_POLICY_MODE_TUNNEL;
Jan Engelhardt1829ed42009-02-21 03:29:44 +010088 xtables_error(PARAMETER_PROBLEM, "policy match: invalid mode \"%s\"", s);
Patrick McHardy524bb802005-11-19 09:00:03 +000089}
90
Jan Engelhardtd441ad62011-05-06 17:45:12 +020091static void policy_parse(struct xt_option_call *cb)
Patrick McHardy524bb802005-11-19 09:00:03 +000092{
Jan Engelhardtd441ad62011-05-06 17:45:12 +020093 struct xt_policy_info *info = cb->data;
Jan Engelhardtcd300542009-05-31 22:43:12 +020094 struct xt_policy_elem *e = &info->pol[info->len];
Patrick McHardy524bb802005-11-19 09:00:03 +000095
Jan Engelhardtd441ad62011-05-06 17:45:12 +020096 xtables_option_parse(cb);
97 switch (cb->entry->id) {
98 case O_DIRECTION:
99 info->flags |= parse_direction(cb->arg);
Patrick McHardy524bb802005-11-19 09:00:03 +0000100 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200101 case O_POLICY:
102 info->flags |= parse_policy(cb->arg);
Patrick McHardy524bb802005-11-19 09:00:03 +0000103 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200104 case O_STRICT:
Jan Engelhardtcd300542009-05-31 22:43:12 +0200105 info->flags |= XT_POLICY_MATCH_STRICT;
Patrick McHardy524bb802005-11-19 09:00:03 +0000106 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200107 case O_REQID:
Patrick McHardy524bb802005-11-19 09:00:03 +0000108 if (e->match.reqid)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100109 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000110 "policy match: double --reqid option");
Patrick McHardy524bb802005-11-19 09:00:03 +0000111 e->match.reqid = 1;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200112 e->invert.reqid = cb->invert;
113 e->reqid = cb->val.u32;
Patrick McHardy524bb802005-11-19 09:00:03 +0000114 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200115 case O_SPI:
Patrick McHardy524bb802005-11-19 09:00:03 +0000116 if (e->match.spi)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100117 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000118 "policy match: double --spi option");
Patrick McHardy524bb802005-11-19 09:00:03 +0000119 e->match.spi = 1;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200120 e->invert.spi = cb->invert;
121 e->spi = cb->val.u32;
Patrick McHardy524bb802005-11-19 09:00:03 +0000122 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200123 case O_TUNNELSRC:
Patrick McHardy524bb802005-11-19 09:00:03 +0000124 if (e->match.saddr)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100125 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000126 "policy match: double --tunnel-src option");
127
Patrick McHardy524bb802005-11-19 09:00:03 +0000128 e->match.saddr = 1;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200129 e->invert.saddr = cb->invert;
130 memcpy(&e->saddr, &cb->val.haddr, sizeof(cb->val.haddr));
131 memcpy(&e->smask, &cb->val.hmask, sizeof(cb->val.hmask));
Patrick McHardy524bb802005-11-19 09:00:03 +0000132 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200133 case O_TUNNELDST:
Patrick McHardy524bb802005-11-19 09:00:03 +0000134 if (e->match.daddr)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100135 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000136 "policy match: double --tunnel-dst option");
Patrick McHardy524bb802005-11-19 09:00:03 +0000137 e->match.daddr = 1;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200138 e->invert.daddr = cb->invert;
139 memcpy(&e->daddr, &cb->val.haddr, sizeof(cb->val.haddr));
140 memcpy(&e->dmask, &cb->val.hmask, sizeof(cb->val.hmask));
Patrick McHardy524bb802005-11-19 09:00:03 +0000141 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200142 case O_PROTO:
Patrick McHardy524bb802005-11-19 09:00:03 +0000143 if (e->match.proto)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100144 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000145 "policy match: double --proto option");
Jan Engelhardtcdc8e0b2011-05-12 13:59:38 +0200146 e->proto = cb->val.protocol;
Patrick McHardy524bb802005-11-19 09:00:03 +0000147 if (e->proto != IPPROTO_AH && e->proto != IPPROTO_ESP &&
148 e->proto != IPPROTO_COMP)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100149 xtables_error(PARAMETER_PROBLEM,
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200150 "policy match: protocol must be ah/esp/ipcomp");
Patrick McHardy524bb802005-11-19 09:00:03 +0000151 e->match.proto = 1;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200152 e->invert.proto = cb->invert;
Patrick McHardy524bb802005-11-19 09:00:03 +0000153 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200154 case O_MODE:
Patrick McHardy524bb802005-11-19 09:00:03 +0000155 if (e->match.mode)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100156 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000157 "policy match: double --mode option");
Patrick McHardy524bb802005-11-19 09:00:03 +0000158 e->match.mode = 1;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200159 e->invert.mode = cb->invert;
160 e->mode = parse_mode(cb->arg);
Patrick McHardy524bb802005-11-19 09:00:03 +0000161 break;
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200162 case O_NEXT:
Jan Engelhardtcd300542009-05-31 22:43:12 +0200163 if (++info->len == XT_POLICY_MAX_ELEM)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100164 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000165 "policy match: maximum policy depth reached");
166 break;
Patrick McHardy524bb802005-11-19 09:00:03 +0000167 }
Patrick McHardy524bb802005-11-19 09:00:03 +0000168}
169
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200170static void policy_check(struct xt_fcheck_call *cb)
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200171{
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200172 struct xt_policy_info *info = cb->data;
173 const struct xt_policy_elem *e;
Patrick McHardy524bb802005-11-19 09:00:03 +0000174 int i;
175
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200176 /*
177 * The old "no parameters given" check is carried out
178 * by testing for --dir.
179 */
Jan Engelhardtcd300542009-05-31 22:43:12 +0200180 if (!(info->flags & (XT_POLICY_MATCH_IN | XT_POLICY_MATCH_OUT)))
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100181 xtables_error(PARAMETER_PROBLEM,
Jan Engelhardt028ad9e2010-01-31 16:49:50 +0100182 "policy match: neither --dir in nor --dir out specified");
Patrick McHardy524bb802005-11-19 09:00:03 +0000183
Jan Engelhardtcd300542009-05-31 22:43:12 +0200184 if (info->flags & XT_POLICY_MATCH_NONE) {
185 if (info->flags & XT_POLICY_MATCH_STRICT)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100186 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000187 "policy match: policy none but --strict given");
188
189 if (info->len != 0)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100190 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000191 "policy match: policy none but policy given");
192 } else
193 info->len++; /* increase len by 1, no --next after last element */
194
Jan Engelhardt15392932011-05-12 12:46:40 +0200195 /*
196 * This is already represented with O_NEXT requiring F_STRICT in the
197 * options table, but will keep this code as a comment for reference.
198 *
Jan Engelhardtcd300542009-05-31 22:43:12 +0200199 if (!(info->flags & XT_POLICY_MATCH_STRICT) && info->len > 1)
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100200 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000201 "policy match: multiple elements but no --strict");
Jan Engelhardt15392932011-05-12 12:46:40 +0200202 */
Patrick McHardy524bb802005-11-19 09:00:03 +0000203
204 for (i = 0; i < info->len; i++) {
205 e = &info->pol[i];
Patrick McHardya46d88d2006-01-12 09:43:18 +0000206
Jan Engelhardtcd300542009-05-31 22:43:12 +0200207 if (info->flags & XT_POLICY_MATCH_STRICT &&
Noticed by Tom Eastep3f347562006-01-22 13:47:07 +0000208 !(e->match.reqid || e->match.spi || e->match.saddr ||
Patrick McHardya46d88d2006-01-12 09:43:18 +0000209 e->match.daddr || e->match.proto || e->match.mode))
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100210 xtables_error(PARAMETER_PROBLEM,
Jan Engelhardt15392932011-05-12 12:46:40 +0200211 "policy match: empty policy element %u. "
212 "--strict is in effect, but at least one of "
213 "reqid, spi, tunnel-src, tunnel-dst, proto or "
214 "mode is required.", i);
Patrick McHardya46d88d2006-01-12 09:43:18 +0000215
Patrick McHardy524bb802005-11-19 09:00:03 +0000216 if ((e->match.saddr || e->match.daddr)
Jan Engelhardtcd300542009-05-31 22:43:12 +0200217 && ((e->mode == XT_POLICY_MODE_TUNNEL && e->invert.mode) ||
218 (e->mode == XT_POLICY_MODE_TRANSPORT && !e->invert.mode)))
Jan Engelhardt1829ed42009-02-21 03:29:44 +0100219 xtables_error(PARAMETER_PROBLEM,
Patrick McHardy524bb802005-11-19 09:00:03 +0000220 "policy match: --tunnel-src/--tunnel-dst "
221 "is only valid in tunnel mode");
222 }
223}
224
Jan Engelhardt7ac40522011-01-07 12:34:04 +0100225static void print_mode(const char *prefix, uint8_t mode, int numeric)
Patrick McHardy524bb802005-11-19 09:00:03 +0000226{
Jan Engelhardt73866352010-12-18 02:04:59 +0100227 printf(" %smode ", prefix);
Patrick McHardy524bb802005-11-19 09:00:03 +0000228
229 switch (mode) {
Jan Engelhardtcd300542009-05-31 22:43:12 +0200230 case XT_POLICY_MODE_TRANSPORT:
Jan Engelhardt73866352010-12-18 02:04:59 +0100231 printf("transport");
Patrick McHardy524bb802005-11-19 09:00:03 +0000232 break;
Jan Engelhardtcd300542009-05-31 22:43:12 +0200233 case XT_POLICY_MODE_TUNNEL:
Jan Engelhardt73866352010-12-18 02:04:59 +0100234 printf("tunnel");
Patrick McHardy524bb802005-11-19 09:00:03 +0000235 break;
236 default:
Jan Engelhardt73866352010-12-18 02:04:59 +0100237 printf("???");
Patrick McHardy524bb802005-11-19 09:00:03 +0000238 break;
239 }
240}
241
Jan Engelhardt7ac40522011-01-07 12:34:04 +0100242static void print_proto(const char *prefix, uint8_t proto, int numeric)
Patrick McHardy524bb802005-11-19 09:00:03 +0000243{
Jan Engelhardtdd6e4b92011-05-07 00:05:24 +0200244 const struct protoent *p = NULL;
Patrick McHardy524bb802005-11-19 09:00:03 +0000245
Jan Engelhardt73866352010-12-18 02:04:59 +0100246 printf(" %sproto ", prefix);
Patrick McHardy524bb802005-11-19 09:00:03 +0000247 if (!numeric)
248 p = getprotobynumber(proto);
249 if (p != NULL)
Jan Engelhardt73866352010-12-18 02:04:59 +0100250 printf("%s", p->p_name);
Patrick McHardy524bb802005-11-19 09:00:03 +0000251 else
Jan Engelhardt73866352010-12-18 02:04:59 +0100252 printf("%u", proto);
Patrick McHardy524bb802005-11-19 09:00:03 +0000253}
254
255#define PRINT_INVERT(x) \
256do { \
257 if (x) \
Jan Engelhardt73866352010-12-18 02:04:59 +0100258 printf(" !"); \
Patrick McHardy524bb802005-11-19 09:00:03 +0000259} while(0)
260
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200261static void print_entry(const char *prefix, const struct xt_policy_elem *e,
262 bool numeric, uint8_t family)
Patrick McHardy524bb802005-11-19 09:00:03 +0000263{
264 if (e->match.reqid) {
265 PRINT_INVERT(e->invert.reqid);
Jan Engelhardt73866352010-12-18 02:04:59 +0100266 printf(" %sreqid %u", prefix, e->reqid);
Patrick McHardy524bb802005-11-19 09:00:03 +0000267 }
268 if (e->match.spi) {
269 PRINT_INVERT(e->invert.spi);
Jan Engelhardt73866352010-12-18 02:04:59 +0100270 printf(" %sspi 0x%x", prefix, e->spi);
Patrick McHardy524bb802005-11-19 09:00:03 +0000271 }
272 if (e->match.proto) {
273 PRINT_INVERT(e->invert.proto);
274 print_proto(prefix, e->proto, numeric);
275 }
276 if (e->match.mode) {
277 PRINT_INVERT(e->invert.mode);
278 print_mode(prefix, e->mode, numeric);
279 }
280 if (e->match.daddr) {
281 PRINT_INVERT(e->invert.daddr);
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200282 if (family == NFPROTO_IPV6)
Jan Engelhardt73866352010-12-18 02:04:59 +0100283 printf(" %stunnel-dst %s%s", prefix,
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200284 xtables_ip6addr_to_numeric(&e->daddr.a6),
285 xtables_ip6mask_to_numeric(&e->dmask.a6));
286 else
Jan Engelhardt73866352010-12-18 02:04:59 +0100287 printf(" %stunnel-dst %s%s", prefix,
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200288 xtables_ipaddr_to_numeric(&e->daddr.a4),
289 xtables_ipmask_to_numeric(&e->dmask.a4));
Patrick McHardy524bb802005-11-19 09:00:03 +0000290 }
291 if (e->match.saddr) {
292 PRINT_INVERT(e->invert.saddr);
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200293 if (family == NFPROTO_IPV6)
Jan Engelhardt73866352010-12-18 02:04:59 +0100294 printf(" %stunnel-src %s%s", prefix,
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200295 xtables_ip6addr_to_numeric(&e->saddr.a6),
296 xtables_ip6mask_to_numeric(&e->smask.a6));
297 else
Jan Engelhardt73866352010-12-18 02:04:59 +0100298 printf(" %stunnel-src %s%s", prefix,
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200299 xtables_ipaddr_to_numeric(&e->saddr.a4),
300 xtables_ipmask_to_numeric(&e->smask.a4));
Patrick McHardy524bb802005-11-19 09:00:03 +0000301 }
302}
303
Jan Engelhardtdd6e4b92011-05-07 00:05:24 +0200304static void print_flags(const char *prefix, const struct xt_policy_info *info)
Patrick McHardy524bb802005-11-19 09:00:03 +0000305{
Jan Engelhardtcd300542009-05-31 22:43:12 +0200306 if (info->flags & XT_POLICY_MATCH_IN)
Jan Engelhardt73866352010-12-18 02:04:59 +0100307 printf(" %sdir in", prefix);
Patrick McHardy524bb802005-11-19 09:00:03 +0000308 else
Jan Engelhardt73866352010-12-18 02:04:59 +0100309 printf(" %sdir out", prefix);
Patrick McHardy524bb802005-11-19 09:00:03 +0000310
Jan Engelhardtcd300542009-05-31 22:43:12 +0200311 if (info->flags & XT_POLICY_MATCH_NONE)
Jan Engelhardt73866352010-12-18 02:04:59 +0100312 printf(" %spol none", prefix);
Patrick McHardy524bb802005-11-19 09:00:03 +0000313 else
Jan Engelhardt73866352010-12-18 02:04:59 +0100314 printf(" %spol ipsec", prefix);
Patrick McHardy524bb802005-11-19 09:00:03 +0000315
Jan Engelhardtcd300542009-05-31 22:43:12 +0200316 if (info->flags & XT_POLICY_MATCH_STRICT)
Jan Engelhardt73866352010-12-18 02:04:59 +0100317 printf(" %sstrict", prefix);
Patrick McHardy524bb802005-11-19 09:00:03 +0000318}
319
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200320static void policy4_print(const void *ip, const struct xt_entry_match *match,
321 int numeric)
Patrick McHardy524bb802005-11-19 09:00:03 +0000322{
Jan Engelhardtcd300542009-05-31 22:43:12 +0200323 const struct xt_policy_info *info = (void *)match->data;
Patrick McHardy524bb802005-11-19 09:00:03 +0000324 unsigned int i;
325
Jan Engelhardt73866352010-12-18 02:04:59 +0100326 printf(" policy match");
Patrick McHardy524bb802005-11-19 09:00:03 +0000327 print_flags("", info);
328 for (i = 0; i < info->len; i++) {
329 if (info->len > 1)
Jan Engelhardt73866352010-12-18 02:04:59 +0100330 printf(" [%u]", i);
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200331 print_entry("", &info->pol[i], numeric, NFPROTO_IPV4);
Patrick McHardy524bb802005-11-19 09:00:03 +0000332 }
333}
334
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200335static void policy6_print(const void *ip, const struct xt_entry_match *match,
336 int numeric)
337{
338 const struct xt_policy_info *info = (void *)match->data;
339 unsigned int i;
340
Jan Engelhardt73866352010-12-18 02:04:59 +0100341 printf(" policy match");
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200342 print_flags("", info);
343 for (i = 0; i < info->len; i++) {
344 if (info->len > 1)
Jan Engelhardt73866352010-12-18 02:04:59 +0100345 printf(" [%u]", i);
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200346 print_entry("", &info->pol[i], numeric, NFPROTO_IPV6);
347 }
348}
349
350static void policy4_save(const void *ip, const struct xt_entry_match *match)
Patrick McHardy524bb802005-11-19 09:00:03 +0000351{
Jan Engelhardtcd300542009-05-31 22:43:12 +0200352 const struct xt_policy_info *info = (void *)match->data;
Patrick McHardy524bb802005-11-19 09:00:03 +0000353 unsigned int i;
354
355 print_flags("--", info);
356 for (i = 0; i < info->len; i++) {
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200357 print_entry("--", &info->pol[i], false, NFPROTO_IPV4);
358 if (i + 1 < info->len)
Jan Engelhardt73866352010-12-18 02:04:59 +0100359 printf(" --next");
Jan Engelhardt67cf1a92009-06-01 11:46:12 +0200360 }
361}
362
363static void policy6_save(const void *ip, const struct xt_entry_match *match)
364{
365 const struct xt_policy_info *info = (void *)match->data;
366 unsigned int i;
367
368 print_flags("--", info);
369 for (i = 0; i < info->len; i++) {
370 print_entry("--", &info->pol[i], false, NFPROTO_IPV6);
Patrick McHardy524bb802005-11-19 09:00:03 +0000371 if (i + 1 < info->len)
Jan Engelhardt73866352010-12-18 02:04:59 +0100372 printf(" --next");
Patrick McHardy524bb802005-11-19 09:00:03 +0000373 }
374}
375
Jan Engelhardtf2a77522009-06-25 20:12:12 +0200376static struct xtables_match policy_mt_reg[] = {
377 {
378 .name = "policy",
379 .version = XTABLES_VERSION,
380 .family = NFPROTO_IPV4,
381 .size = XT_ALIGN(sizeof(struct xt_policy_info)),
382 .userspacesize = XT_ALIGN(sizeof(struct xt_policy_info)),
383 .help = policy_help,
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200384 .x6_parse = policy_parse,
385 .x6_fcheck = policy_check,
Jan Engelhardtf2a77522009-06-25 20:12:12 +0200386 .print = policy4_print,
387 .save = policy4_save,
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200388 .x6_options = policy_opts,
Jan Engelhardtf2a77522009-06-25 20:12:12 +0200389 },
390 {
391 .name = "policy",
392 .version = XTABLES_VERSION,
393 .family = NFPROTO_IPV6,
394 .size = XT_ALIGN(sizeof(struct xt_policy_info)),
395 .userspacesize = XT_ALIGN(sizeof(struct xt_policy_info)),
396 .help = policy_help,
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200397 .x6_parse = policy_parse,
398 .x6_fcheck = policy_check,
Jan Engelhardtf2a77522009-06-25 20:12:12 +0200399 .print = policy6_print,
400 .save = policy6_save,
Jan Engelhardtd441ad62011-05-06 17:45:12 +0200401 .x6_options = policy_opts,
Jan Engelhardtf2a77522009-06-25 20:12:12 +0200402 },
Patrick McHardy524bb802005-11-19 09:00:03 +0000403};
404
405void _init(void)
406{
Jan Engelhardtf2a77522009-06-25 20:12:12 +0200407 xtables_register_matches(policy_mt_reg, ARRAY_SIZE(policy_mt_reg));
Patrick McHardy524bb802005-11-19 09:00:03 +0000408}