1*a71a9546SAutomerger Merge Worker /*
2*a71a9546SAutomerger Merge Worker * Copyright (c) 2005-2013 Patrick McHardy <[email protected]>
3*a71a9546SAutomerger Merge Worker */
4*a71a9546SAutomerger Merge Worker
5*a71a9546SAutomerger Merge Worker #include <stdbool.h>
6*a71a9546SAutomerger Merge Worker #include <stdint.h>
7*a71a9546SAutomerger Merge Worker #include <stdio.h>
8*a71a9546SAutomerger Merge Worker #include <string.h>
9*a71a9546SAutomerger Merge Worker #include <netdb.h>
10*a71a9546SAutomerger Merge Worker #include <xtables.h>
11*a71a9546SAutomerger Merge Worker #include <linux/netfilter/xt_policy.h>
12*a71a9546SAutomerger Merge Worker
13*a71a9546SAutomerger Merge Worker enum {
14*a71a9546SAutomerger Merge Worker O_DIRECTION = 0,
15*a71a9546SAutomerger Merge Worker O_POLICY,
16*a71a9546SAutomerger Merge Worker O_STRICT,
17*a71a9546SAutomerger Merge Worker O_REQID,
18*a71a9546SAutomerger Merge Worker O_SPI,
19*a71a9546SAutomerger Merge Worker O_PROTO,
20*a71a9546SAutomerger Merge Worker O_MODE,
21*a71a9546SAutomerger Merge Worker O_TUNNELSRC,
22*a71a9546SAutomerger Merge Worker O_TUNNELDST,
23*a71a9546SAutomerger Merge Worker O_NEXT,
24*a71a9546SAutomerger Merge Worker F_STRICT = 1 << O_STRICT,
25*a71a9546SAutomerger Merge Worker };
26*a71a9546SAutomerger Merge Worker
policy_help(void)27*a71a9546SAutomerger Merge Worker static void policy_help(void)
28*a71a9546SAutomerger Merge Worker {
29*a71a9546SAutomerger Merge Worker printf(
30*a71a9546SAutomerger Merge Worker "policy match options:\n"
31*a71a9546SAutomerger Merge Worker " --dir in|out match policy applied during decapsulation/\n"
32*a71a9546SAutomerger Merge Worker " policy to be applied during encapsulation\n"
33*a71a9546SAutomerger Merge Worker " --pol none|ipsec match policy\n"
34*a71a9546SAutomerger Merge Worker " --strict match entire policy instead of single element\n"
35*a71a9546SAutomerger Merge Worker " at any position\n"
36*a71a9546SAutomerger Merge Worker "These options may be used repeatedly, to describe policy elements:\n"
37*a71a9546SAutomerger Merge Worker "[!] --reqid reqid match reqid\n"
38*a71a9546SAutomerger Merge Worker "[!] --spi spi match SPI\n"
39*a71a9546SAutomerger Merge Worker "[!] --proto proto match protocol (ah/esp/ipcomp)\n"
40*a71a9546SAutomerger Merge Worker "[!] --mode mode match mode (transport/tunnel)\n"
41*a71a9546SAutomerger Merge Worker "[!] --tunnel-src addr/mask match tunnel source\n"
42*a71a9546SAutomerger Merge Worker "[!] --tunnel-dst addr/mask match tunnel destination\n"
43*a71a9546SAutomerger Merge Worker " --next begin next element in policy\n");
44*a71a9546SAutomerger Merge Worker }
45*a71a9546SAutomerger Merge Worker
46*a71a9546SAutomerger Merge Worker static const struct xt_option_entry policy_opts[] = {
47*a71a9546SAutomerger Merge Worker {.name = "dir", .id = O_DIRECTION, .type = XTTYPE_STRING},
48*a71a9546SAutomerger Merge Worker {.name = "pol", .id = O_POLICY, .type = XTTYPE_STRING},
49*a71a9546SAutomerger Merge Worker {.name = "strict", .id = O_STRICT, .type = XTTYPE_NONE},
50*a71a9546SAutomerger Merge Worker {.name = "reqid", .id = O_REQID, .type = XTTYPE_UINT32,
51*a71a9546SAutomerger Merge Worker .flags = XTOPT_MULTI | XTOPT_INVERT},
52*a71a9546SAutomerger Merge Worker {.name = "spi", .id = O_SPI, .type = XTTYPE_UINT32,
53*a71a9546SAutomerger Merge Worker .flags = XTOPT_MULTI | XTOPT_INVERT},
54*a71a9546SAutomerger Merge Worker {.name = "tunnel-src", .id = O_TUNNELSRC, .type = XTTYPE_HOSTMASK,
55*a71a9546SAutomerger Merge Worker .flags = XTOPT_MULTI | XTOPT_INVERT},
56*a71a9546SAutomerger Merge Worker {.name = "tunnel-dst", .id = O_TUNNELDST, .type = XTTYPE_HOSTMASK,
57*a71a9546SAutomerger Merge Worker .flags = XTOPT_MULTI | XTOPT_INVERT},
58*a71a9546SAutomerger Merge Worker {.name = "proto", .id = O_PROTO, .type = XTTYPE_PROTOCOL,
59*a71a9546SAutomerger Merge Worker .flags = XTOPT_MULTI | XTOPT_INVERT},
60*a71a9546SAutomerger Merge Worker {.name = "mode", .id = O_MODE, .type = XTTYPE_STRING,
61*a71a9546SAutomerger Merge Worker .flags = XTOPT_MULTI | XTOPT_INVERT},
62*a71a9546SAutomerger Merge Worker {.name = "next", .id = O_NEXT, .type = XTTYPE_NONE,
63*a71a9546SAutomerger Merge Worker .flags = XTOPT_MULTI, .also = F_STRICT},
64*a71a9546SAutomerger Merge Worker XTOPT_TABLEEND,
65*a71a9546SAutomerger Merge Worker };
66*a71a9546SAutomerger Merge Worker
parse_direction(const char * s)67*a71a9546SAutomerger Merge Worker static int parse_direction(const char *s)
68*a71a9546SAutomerger Merge Worker {
69*a71a9546SAutomerger Merge Worker if (strcmp(s, "in") == 0)
70*a71a9546SAutomerger Merge Worker return XT_POLICY_MATCH_IN;
71*a71a9546SAutomerger Merge Worker if (strcmp(s, "out") == 0)
72*a71a9546SAutomerger Merge Worker return XT_POLICY_MATCH_OUT;
73*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM, "policy_match: invalid dir \"%s\"", s);
74*a71a9546SAutomerger Merge Worker }
75*a71a9546SAutomerger Merge Worker
parse_policy(const char * s)76*a71a9546SAutomerger Merge Worker static int parse_policy(const char *s)
77*a71a9546SAutomerger Merge Worker {
78*a71a9546SAutomerger Merge Worker if (strcmp(s, "none") == 0)
79*a71a9546SAutomerger Merge Worker return XT_POLICY_MATCH_NONE;
80*a71a9546SAutomerger Merge Worker if (strcmp(s, "ipsec") == 0)
81*a71a9546SAutomerger Merge Worker return 0;
82*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM, "policy match: invalid policy \"%s\"", s);
83*a71a9546SAutomerger Merge Worker }
84*a71a9546SAutomerger Merge Worker
parse_mode(const char * s)85*a71a9546SAutomerger Merge Worker static int parse_mode(const char *s)
86*a71a9546SAutomerger Merge Worker {
87*a71a9546SAutomerger Merge Worker if (strcmp(s, "transport") == 0)
88*a71a9546SAutomerger Merge Worker return XT_POLICY_MODE_TRANSPORT;
89*a71a9546SAutomerger Merge Worker if (strcmp(s, "tunnel") == 0)
90*a71a9546SAutomerger Merge Worker return XT_POLICY_MODE_TUNNEL;
91*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM, "policy match: invalid mode \"%s\"", s);
92*a71a9546SAutomerger Merge Worker }
93*a71a9546SAutomerger Merge Worker
policy_parse(struct xt_option_call * cb)94*a71a9546SAutomerger Merge Worker static void policy_parse(struct xt_option_call *cb)
95*a71a9546SAutomerger Merge Worker {
96*a71a9546SAutomerger Merge Worker struct xt_policy_info *info = cb->data;
97*a71a9546SAutomerger Merge Worker struct xt_policy_elem *e = &info->pol[info->len];
98*a71a9546SAutomerger Merge Worker
99*a71a9546SAutomerger Merge Worker xtables_option_parse(cb);
100*a71a9546SAutomerger Merge Worker switch (cb->entry->id) {
101*a71a9546SAutomerger Merge Worker case O_DIRECTION:
102*a71a9546SAutomerger Merge Worker info->flags |= parse_direction(cb->arg);
103*a71a9546SAutomerger Merge Worker break;
104*a71a9546SAutomerger Merge Worker case O_POLICY:
105*a71a9546SAutomerger Merge Worker info->flags |= parse_policy(cb->arg);
106*a71a9546SAutomerger Merge Worker break;
107*a71a9546SAutomerger Merge Worker case O_STRICT:
108*a71a9546SAutomerger Merge Worker info->flags |= XT_POLICY_MATCH_STRICT;
109*a71a9546SAutomerger Merge Worker break;
110*a71a9546SAutomerger Merge Worker case O_REQID:
111*a71a9546SAutomerger Merge Worker if (e->match.reqid)
112*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
113*a71a9546SAutomerger Merge Worker "policy match: double --reqid option");
114*a71a9546SAutomerger Merge Worker e->match.reqid = 1;
115*a71a9546SAutomerger Merge Worker e->invert.reqid = cb->invert;
116*a71a9546SAutomerger Merge Worker e->reqid = cb->val.u32;
117*a71a9546SAutomerger Merge Worker break;
118*a71a9546SAutomerger Merge Worker case O_SPI:
119*a71a9546SAutomerger Merge Worker if (e->match.spi)
120*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
121*a71a9546SAutomerger Merge Worker "policy match: double --spi option");
122*a71a9546SAutomerger Merge Worker e->match.spi = 1;
123*a71a9546SAutomerger Merge Worker e->invert.spi = cb->invert;
124*a71a9546SAutomerger Merge Worker e->spi = cb->val.u32;
125*a71a9546SAutomerger Merge Worker break;
126*a71a9546SAutomerger Merge Worker case O_TUNNELSRC:
127*a71a9546SAutomerger Merge Worker if (e->match.saddr)
128*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
129*a71a9546SAutomerger Merge Worker "policy match: double --tunnel-src option");
130*a71a9546SAutomerger Merge Worker
131*a71a9546SAutomerger Merge Worker e->match.saddr = 1;
132*a71a9546SAutomerger Merge Worker e->invert.saddr = cb->invert;
133*a71a9546SAutomerger Merge Worker memcpy(&e->saddr, &cb->val.haddr, sizeof(cb->val.haddr));
134*a71a9546SAutomerger Merge Worker memcpy(&e->smask, &cb->val.hmask, sizeof(cb->val.hmask));
135*a71a9546SAutomerger Merge Worker break;
136*a71a9546SAutomerger Merge Worker case O_TUNNELDST:
137*a71a9546SAutomerger Merge Worker if (e->match.daddr)
138*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
139*a71a9546SAutomerger Merge Worker "policy match: double --tunnel-dst option");
140*a71a9546SAutomerger Merge Worker e->match.daddr = 1;
141*a71a9546SAutomerger Merge Worker e->invert.daddr = cb->invert;
142*a71a9546SAutomerger Merge Worker memcpy(&e->daddr, &cb->val.haddr, sizeof(cb->val.haddr));
143*a71a9546SAutomerger Merge Worker memcpy(&e->dmask, &cb->val.hmask, sizeof(cb->val.hmask));
144*a71a9546SAutomerger Merge Worker break;
145*a71a9546SAutomerger Merge Worker case O_PROTO:
146*a71a9546SAutomerger Merge Worker if (e->match.proto)
147*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
148*a71a9546SAutomerger Merge Worker "policy match: double --proto option");
149*a71a9546SAutomerger Merge Worker e->proto = cb->val.protocol;
150*a71a9546SAutomerger Merge Worker if (e->proto != IPPROTO_AH && e->proto != IPPROTO_ESP &&
151*a71a9546SAutomerger Merge Worker e->proto != IPPROTO_COMP)
152*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
153*a71a9546SAutomerger Merge Worker "policy match: protocol must be ah/esp/ipcomp");
154*a71a9546SAutomerger Merge Worker e->match.proto = 1;
155*a71a9546SAutomerger Merge Worker e->invert.proto = cb->invert;
156*a71a9546SAutomerger Merge Worker break;
157*a71a9546SAutomerger Merge Worker case O_MODE:
158*a71a9546SAutomerger Merge Worker if (e->match.mode)
159*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
160*a71a9546SAutomerger Merge Worker "policy match: double --mode option");
161*a71a9546SAutomerger Merge Worker e->match.mode = 1;
162*a71a9546SAutomerger Merge Worker e->invert.mode = cb->invert;
163*a71a9546SAutomerger Merge Worker e->mode = parse_mode(cb->arg);
164*a71a9546SAutomerger Merge Worker break;
165*a71a9546SAutomerger Merge Worker case O_NEXT:
166*a71a9546SAutomerger Merge Worker if (++info->len == XT_POLICY_MAX_ELEM)
167*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
168*a71a9546SAutomerger Merge Worker "policy match: maximum policy depth reached");
169*a71a9546SAutomerger Merge Worker break;
170*a71a9546SAutomerger Merge Worker }
171*a71a9546SAutomerger Merge Worker }
172*a71a9546SAutomerger Merge Worker
policy_check(struct xt_fcheck_call * cb)173*a71a9546SAutomerger Merge Worker static void policy_check(struct xt_fcheck_call *cb)
174*a71a9546SAutomerger Merge Worker {
175*a71a9546SAutomerger Merge Worker struct xt_policy_info *info = cb->data;
176*a71a9546SAutomerger Merge Worker const struct xt_policy_elem *e;
177*a71a9546SAutomerger Merge Worker int i;
178*a71a9546SAutomerger Merge Worker
179*a71a9546SAutomerger Merge Worker /*
180*a71a9546SAutomerger Merge Worker * The old "no parameters given" check is carried out
181*a71a9546SAutomerger Merge Worker * by testing for --dir.
182*a71a9546SAutomerger Merge Worker */
183*a71a9546SAutomerger Merge Worker if (!(info->flags & (XT_POLICY_MATCH_IN | XT_POLICY_MATCH_OUT)))
184*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
185*a71a9546SAutomerger Merge Worker "policy match: neither --dir in nor --dir out specified");
186*a71a9546SAutomerger Merge Worker
187*a71a9546SAutomerger Merge Worker if (info->flags & XT_POLICY_MATCH_NONE) {
188*a71a9546SAutomerger Merge Worker if (info->flags & XT_POLICY_MATCH_STRICT)
189*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
190*a71a9546SAutomerger Merge Worker "policy match: policy none but --strict given");
191*a71a9546SAutomerger Merge Worker
192*a71a9546SAutomerger Merge Worker if (info->len != 0)
193*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
194*a71a9546SAutomerger Merge Worker "policy match: policy none but policy given");
195*a71a9546SAutomerger Merge Worker } else
196*a71a9546SAutomerger Merge Worker info->len++; /* increase len by 1, no --next after last element */
197*a71a9546SAutomerger Merge Worker
198*a71a9546SAutomerger Merge Worker /*
199*a71a9546SAutomerger Merge Worker * This is already represented with O_NEXT requiring F_STRICT in the
200*a71a9546SAutomerger Merge Worker * options table, but will keep this code as a comment for reference.
201*a71a9546SAutomerger Merge Worker *
202*a71a9546SAutomerger Merge Worker if (!(info->flags & XT_POLICY_MATCH_STRICT) && info->len > 1)
203*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
204*a71a9546SAutomerger Merge Worker "policy match: multiple elements but no --strict");
205*a71a9546SAutomerger Merge Worker */
206*a71a9546SAutomerger Merge Worker
207*a71a9546SAutomerger Merge Worker for (i = 0; i < info->len; i++) {
208*a71a9546SAutomerger Merge Worker e = &info->pol[i];
209*a71a9546SAutomerger Merge Worker
210*a71a9546SAutomerger Merge Worker if (info->flags & XT_POLICY_MATCH_STRICT &&
211*a71a9546SAutomerger Merge Worker !(e->match.reqid || e->match.spi || e->match.saddr ||
212*a71a9546SAutomerger Merge Worker e->match.daddr || e->match.proto || e->match.mode))
213*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
214*a71a9546SAutomerger Merge Worker "policy match: empty policy element %u. "
215*a71a9546SAutomerger Merge Worker "--strict is in effect, but at least one of "
216*a71a9546SAutomerger Merge Worker "reqid, spi, tunnel-src, tunnel-dst, proto or "
217*a71a9546SAutomerger Merge Worker "mode is required.", i);
218*a71a9546SAutomerger Merge Worker
219*a71a9546SAutomerger Merge Worker if ((e->match.saddr || e->match.daddr)
220*a71a9546SAutomerger Merge Worker && ((e->mode == XT_POLICY_MODE_TUNNEL && e->invert.mode) ||
221*a71a9546SAutomerger Merge Worker (e->mode == XT_POLICY_MODE_TRANSPORT && !e->invert.mode)))
222*a71a9546SAutomerger Merge Worker xtables_error(PARAMETER_PROBLEM,
223*a71a9546SAutomerger Merge Worker "policy match: --tunnel-src/--tunnel-dst "
224*a71a9546SAutomerger Merge Worker "is only valid in tunnel mode");
225*a71a9546SAutomerger Merge Worker }
226*a71a9546SAutomerger Merge Worker }
227*a71a9546SAutomerger Merge Worker
print_mode(const char * prefix,uint8_t mode,int numeric)228*a71a9546SAutomerger Merge Worker static void print_mode(const char *prefix, uint8_t mode, int numeric)
229*a71a9546SAutomerger Merge Worker {
230*a71a9546SAutomerger Merge Worker printf(" %smode ", prefix);
231*a71a9546SAutomerger Merge Worker
232*a71a9546SAutomerger Merge Worker switch (mode) {
233*a71a9546SAutomerger Merge Worker case XT_POLICY_MODE_TRANSPORT:
234*a71a9546SAutomerger Merge Worker printf("transport");
235*a71a9546SAutomerger Merge Worker break;
236*a71a9546SAutomerger Merge Worker case XT_POLICY_MODE_TUNNEL:
237*a71a9546SAutomerger Merge Worker printf("tunnel");
238*a71a9546SAutomerger Merge Worker break;
239*a71a9546SAutomerger Merge Worker default:
240*a71a9546SAutomerger Merge Worker printf("???");
241*a71a9546SAutomerger Merge Worker break;
242*a71a9546SAutomerger Merge Worker }
243*a71a9546SAutomerger Merge Worker }
244*a71a9546SAutomerger Merge Worker
print_proto(const char * prefix,uint8_t proto,int numeric)245*a71a9546SAutomerger Merge Worker static void print_proto(const char *prefix, uint8_t proto, int numeric)
246*a71a9546SAutomerger Merge Worker {
247*a71a9546SAutomerger Merge Worker const struct protoent *p = NULL;
248*a71a9546SAutomerger Merge Worker
249*a71a9546SAutomerger Merge Worker printf(" %sproto ", prefix);
250*a71a9546SAutomerger Merge Worker if (!numeric)
251*a71a9546SAutomerger Merge Worker p = getprotobynumber(proto);
252*a71a9546SAutomerger Merge Worker if (p != NULL)
253*a71a9546SAutomerger Merge Worker printf("%s", p->p_name);
254*a71a9546SAutomerger Merge Worker else
255*a71a9546SAutomerger Merge Worker printf("%u", proto);
256*a71a9546SAutomerger Merge Worker }
257*a71a9546SAutomerger Merge Worker
258*a71a9546SAutomerger Merge Worker #define PRINT_INVERT(x) \
259*a71a9546SAutomerger Merge Worker do { \
260*a71a9546SAutomerger Merge Worker if (x) \
261*a71a9546SAutomerger Merge Worker printf(" !"); \
262*a71a9546SAutomerger Merge Worker } while(0)
263*a71a9546SAutomerger Merge Worker
print_entry(const char * prefix,const struct xt_policy_elem * e,bool numeric,uint8_t family)264*a71a9546SAutomerger Merge Worker static void print_entry(const char *prefix, const struct xt_policy_elem *e,
265*a71a9546SAutomerger Merge Worker bool numeric, uint8_t family)
266*a71a9546SAutomerger Merge Worker {
267*a71a9546SAutomerger Merge Worker if (e->match.reqid) {
268*a71a9546SAutomerger Merge Worker PRINT_INVERT(e->invert.reqid);
269*a71a9546SAutomerger Merge Worker printf(" %sreqid %u", prefix, e->reqid);
270*a71a9546SAutomerger Merge Worker }
271*a71a9546SAutomerger Merge Worker if (e->match.spi) {
272*a71a9546SAutomerger Merge Worker PRINT_INVERT(e->invert.spi);
273*a71a9546SAutomerger Merge Worker printf(" %sspi 0x%x", prefix, e->spi);
274*a71a9546SAutomerger Merge Worker }
275*a71a9546SAutomerger Merge Worker if (e->match.proto) {
276*a71a9546SAutomerger Merge Worker PRINT_INVERT(e->invert.proto);
277*a71a9546SAutomerger Merge Worker print_proto(prefix, e->proto, numeric);
278*a71a9546SAutomerger Merge Worker }
279*a71a9546SAutomerger Merge Worker if (e->match.mode) {
280*a71a9546SAutomerger Merge Worker PRINT_INVERT(e->invert.mode);
281*a71a9546SAutomerger Merge Worker print_mode(prefix, e->mode, numeric);
282*a71a9546SAutomerger Merge Worker }
283*a71a9546SAutomerger Merge Worker if (e->match.daddr) {
284*a71a9546SAutomerger Merge Worker PRINT_INVERT(e->invert.daddr);
285*a71a9546SAutomerger Merge Worker if (family == NFPROTO_IPV6)
286*a71a9546SAutomerger Merge Worker printf(" %stunnel-dst %s%s", prefix,
287*a71a9546SAutomerger Merge Worker xtables_ip6addr_to_numeric(&e->daddr.a6),
288*a71a9546SAutomerger Merge Worker xtables_ip6mask_to_numeric(&e->dmask.a6));
289*a71a9546SAutomerger Merge Worker else
290*a71a9546SAutomerger Merge Worker printf(" %stunnel-dst %s%s", prefix,
291*a71a9546SAutomerger Merge Worker xtables_ipaddr_to_numeric(&e->daddr.a4),
292*a71a9546SAutomerger Merge Worker xtables_ipmask_to_numeric(&e->dmask.a4));
293*a71a9546SAutomerger Merge Worker }
294*a71a9546SAutomerger Merge Worker if (e->match.saddr) {
295*a71a9546SAutomerger Merge Worker PRINT_INVERT(e->invert.saddr);
296*a71a9546SAutomerger Merge Worker if (family == NFPROTO_IPV6)
297*a71a9546SAutomerger Merge Worker printf(" %stunnel-src %s%s", prefix,
298*a71a9546SAutomerger Merge Worker xtables_ip6addr_to_numeric(&e->saddr.a6),
299*a71a9546SAutomerger Merge Worker xtables_ip6mask_to_numeric(&e->smask.a6));
300*a71a9546SAutomerger Merge Worker else
301*a71a9546SAutomerger Merge Worker printf(" %stunnel-src %s%s", prefix,
302*a71a9546SAutomerger Merge Worker xtables_ipaddr_to_numeric(&e->saddr.a4),
303*a71a9546SAutomerger Merge Worker xtables_ipmask_to_numeric(&e->smask.a4));
304*a71a9546SAutomerger Merge Worker }
305*a71a9546SAutomerger Merge Worker }
306*a71a9546SAutomerger Merge Worker
print_flags(const char * prefix,const struct xt_policy_info * info)307*a71a9546SAutomerger Merge Worker static void print_flags(const char *prefix, const struct xt_policy_info *info)
308*a71a9546SAutomerger Merge Worker {
309*a71a9546SAutomerger Merge Worker if (info->flags & XT_POLICY_MATCH_IN)
310*a71a9546SAutomerger Merge Worker printf(" %sdir in", prefix);
311*a71a9546SAutomerger Merge Worker else
312*a71a9546SAutomerger Merge Worker printf(" %sdir out", prefix);
313*a71a9546SAutomerger Merge Worker
314*a71a9546SAutomerger Merge Worker if (info->flags & XT_POLICY_MATCH_NONE)
315*a71a9546SAutomerger Merge Worker printf(" %spol none", prefix);
316*a71a9546SAutomerger Merge Worker else
317*a71a9546SAutomerger Merge Worker printf(" %spol ipsec", prefix);
318*a71a9546SAutomerger Merge Worker
319*a71a9546SAutomerger Merge Worker if (info->flags & XT_POLICY_MATCH_STRICT)
320*a71a9546SAutomerger Merge Worker printf(" %sstrict", prefix);
321*a71a9546SAutomerger Merge Worker }
322*a71a9546SAutomerger Merge Worker
policy4_print(const void * ip,const struct xt_entry_match * match,int numeric)323*a71a9546SAutomerger Merge Worker static void policy4_print(const void *ip, const struct xt_entry_match *match,
324*a71a9546SAutomerger Merge Worker int numeric)
325*a71a9546SAutomerger Merge Worker {
326*a71a9546SAutomerger Merge Worker const struct xt_policy_info *info = (void *)match->data;
327*a71a9546SAutomerger Merge Worker unsigned int i;
328*a71a9546SAutomerger Merge Worker
329*a71a9546SAutomerger Merge Worker printf(" policy match");
330*a71a9546SAutomerger Merge Worker print_flags("", info);
331*a71a9546SAutomerger Merge Worker for (i = 0; i < info->len; i++) {
332*a71a9546SAutomerger Merge Worker if (info->len > 1)
333*a71a9546SAutomerger Merge Worker printf(" [%u]", i);
334*a71a9546SAutomerger Merge Worker print_entry("", &info->pol[i], numeric, NFPROTO_IPV4);
335*a71a9546SAutomerger Merge Worker }
336*a71a9546SAutomerger Merge Worker }
337*a71a9546SAutomerger Merge Worker
policy6_print(const void * ip,const struct xt_entry_match * match,int numeric)338*a71a9546SAutomerger Merge Worker static void policy6_print(const void *ip, const struct xt_entry_match *match,
339*a71a9546SAutomerger Merge Worker int numeric)
340*a71a9546SAutomerger Merge Worker {
341*a71a9546SAutomerger Merge Worker const struct xt_policy_info *info = (void *)match->data;
342*a71a9546SAutomerger Merge Worker unsigned int i;
343*a71a9546SAutomerger Merge Worker
344*a71a9546SAutomerger Merge Worker printf(" policy match");
345*a71a9546SAutomerger Merge Worker print_flags("", info);
346*a71a9546SAutomerger Merge Worker for (i = 0; i < info->len; i++) {
347*a71a9546SAutomerger Merge Worker if (info->len > 1)
348*a71a9546SAutomerger Merge Worker printf(" [%u]", i);
349*a71a9546SAutomerger Merge Worker print_entry("", &info->pol[i], numeric, NFPROTO_IPV6);
350*a71a9546SAutomerger Merge Worker }
351*a71a9546SAutomerger Merge Worker }
352*a71a9546SAutomerger Merge Worker
policy4_save(const void * ip,const struct xt_entry_match * match)353*a71a9546SAutomerger Merge Worker static void policy4_save(const void *ip, const struct xt_entry_match *match)
354*a71a9546SAutomerger Merge Worker {
355*a71a9546SAutomerger Merge Worker const struct xt_policy_info *info = (void *)match->data;
356*a71a9546SAutomerger Merge Worker unsigned int i;
357*a71a9546SAutomerger Merge Worker
358*a71a9546SAutomerger Merge Worker print_flags("--", info);
359*a71a9546SAutomerger Merge Worker for (i = 0; i < info->len; i++) {
360*a71a9546SAutomerger Merge Worker print_entry("--", &info->pol[i], false, NFPROTO_IPV4);
361*a71a9546SAutomerger Merge Worker if (i + 1 < info->len)
362*a71a9546SAutomerger Merge Worker printf(" --next");
363*a71a9546SAutomerger Merge Worker }
364*a71a9546SAutomerger Merge Worker }
365*a71a9546SAutomerger Merge Worker
policy6_save(const void * ip,const struct xt_entry_match * match)366*a71a9546SAutomerger Merge Worker static void policy6_save(const void *ip, const struct xt_entry_match *match)
367*a71a9546SAutomerger Merge Worker {
368*a71a9546SAutomerger Merge Worker const struct xt_policy_info *info = (void *)match->data;
369*a71a9546SAutomerger Merge Worker unsigned int i;
370*a71a9546SAutomerger Merge Worker
371*a71a9546SAutomerger Merge Worker print_flags("--", info);
372*a71a9546SAutomerger Merge Worker for (i = 0; i < info->len; i++) {
373*a71a9546SAutomerger Merge Worker print_entry("--", &info->pol[i], false, NFPROTO_IPV6);
374*a71a9546SAutomerger Merge Worker if (i + 1 < info->len)
375*a71a9546SAutomerger Merge Worker printf(" --next");
376*a71a9546SAutomerger Merge Worker }
377*a71a9546SAutomerger Merge Worker }
378*a71a9546SAutomerger Merge Worker
policy_xlate(struct xt_xlate * xl,const struct xt_xlate_mt_params * params)379*a71a9546SAutomerger Merge Worker static int policy_xlate(struct xt_xlate *xl,
380*a71a9546SAutomerger Merge Worker const struct xt_xlate_mt_params *params)
381*a71a9546SAutomerger Merge Worker {
382*a71a9546SAutomerger Merge Worker static const unsigned int allowed = XT_POLICY_MATCH_STRICT |
383*a71a9546SAutomerger Merge Worker XT_POLICY_MATCH_NONE |
384*a71a9546SAutomerger Merge Worker XT_POLICY_MATCH_IN;
385*a71a9546SAutomerger Merge Worker static const struct xt_policy_elem empty;
386*a71a9546SAutomerger Merge Worker const struct xt_policy_info *info = (const void *)params->match->data;
387*a71a9546SAutomerger Merge Worker
388*a71a9546SAutomerger Merge Worker if ((info->flags & ~allowed) || info->len > 1)
389*a71a9546SAutomerger Merge Worker return 0;
390*a71a9546SAutomerger Merge Worker
391*a71a9546SAutomerger Merge Worker if (memcmp(&info->pol[0], &empty, sizeof(empty)))
392*a71a9546SAutomerger Merge Worker return 0;
393*a71a9546SAutomerger Merge Worker
394*a71a9546SAutomerger Merge Worker xt_xlate_add(xl, "meta secpath ");
395*a71a9546SAutomerger Merge Worker
396*a71a9546SAutomerger Merge Worker if (info->flags & XT_POLICY_MATCH_NONE)
397*a71a9546SAutomerger Merge Worker xt_xlate_add(xl, "missing");
398*a71a9546SAutomerger Merge Worker else
399*a71a9546SAutomerger Merge Worker xt_xlate_add(xl, "exists");
400*a71a9546SAutomerger Merge Worker
401*a71a9546SAutomerger Merge Worker return 1;
402*a71a9546SAutomerger Merge Worker }
403*a71a9546SAutomerger Merge Worker
404*a71a9546SAutomerger Merge Worker static struct xtables_match policy_mt_reg[] = {
405*a71a9546SAutomerger Merge Worker {
406*a71a9546SAutomerger Merge Worker .name = "policy",
407*a71a9546SAutomerger Merge Worker .version = XTABLES_VERSION,
408*a71a9546SAutomerger Merge Worker .family = NFPROTO_IPV4,
409*a71a9546SAutomerger Merge Worker .size = XT_ALIGN(sizeof(struct xt_policy_info)),
410*a71a9546SAutomerger Merge Worker .userspacesize = XT_ALIGN(sizeof(struct xt_policy_info)),
411*a71a9546SAutomerger Merge Worker .help = policy_help,
412*a71a9546SAutomerger Merge Worker .x6_parse = policy_parse,
413*a71a9546SAutomerger Merge Worker .x6_fcheck = policy_check,
414*a71a9546SAutomerger Merge Worker .print = policy4_print,
415*a71a9546SAutomerger Merge Worker .save = policy4_save,
416*a71a9546SAutomerger Merge Worker .x6_options = policy_opts,
417*a71a9546SAutomerger Merge Worker .xlate = policy_xlate,
418*a71a9546SAutomerger Merge Worker },
419*a71a9546SAutomerger Merge Worker {
420*a71a9546SAutomerger Merge Worker .name = "policy",
421*a71a9546SAutomerger Merge Worker .version = XTABLES_VERSION,
422*a71a9546SAutomerger Merge Worker .family = NFPROTO_IPV6,
423*a71a9546SAutomerger Merge Worker .size = XT_ALIGN(sizeof(struct xt_policy_info)),
424*a71a9546SAutomerger Merge Worker .userspacesize = XT_ALIGN(sizeof(struct xt_policy_info)),
425*a71a9546SAutomerger Merge Worker .help = policy_help,
426*a71a9546SAutomerger Merge Worker .x6_parse = policy_parse,
427*a71a9546SAutomerger Merge Worker .x6_fcheck = policy_check,
428*a71a9546SAutomerger Merge Worker .print = policy6_print,
429*a71a9546SAutomerger Merge Worker .save = policy6_save,
430*a71a9546SAutomerger Merge Worker .x6_options = policy_opts,
431*a71a9546SAutomerger Merge Worker .xlate = policy_xlate,
432*a71a9546SAutomerger Merge Worker },
433*a71a9546SAutomerger Merge Worker };
434*a71a9546SAutomerger Merge Worker
_init(void)435*a71a9546SAutomerger Merge Worker void _init(void)
436*a71a9546SAutomerger Merge Worker {
437*a71a9546SAutomerger Merge Worker xtables_register_matches(policy_mt_reg, ARRAY_SIZE(policy_mt_reg));
438*a71a9546SAutomerger Merge Worker }
439