]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - sys/netpfil/pf/pf_ruleset.c
net: clean up empty lines in .c and .h files
[FreeBSD/FreeBSD.git] / sys / netpfil / pf / pf_ruleset.c
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2001 Daniel Hartmeier
5  * Copyright (c) 2002,2003 Henning Brauer
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  *
12  *    - Redistributions of source code must retain the above copyright
13  *      notice, this list of conditions and the following disclaimer.
14  *    - Redistributions in binary form must reproduce the above
15  *      copyright notice, this list of conditions and the following
16  *      disclaimer in the documentation and/or other materials provided
17  *      with the distribution.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
22  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
23  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
24  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
25  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
26  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
27  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
29  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30  * POSSIBILITY OF SUCH DAMAGE.
31  *
32  * Effort sponsored in part by the Defense Advanced Research Projects
33  * Agency (DARPA) and Air Force Research Laboratory, Air Force
34  * Materiel Command, USAF, under agreement number F30602-01-2-0537.
35  *
36  *      $OpenBSD: pf_ruleset.c,v 1.2 2008/12/18 15:31:37 dhill Exp $
37  */
38
39 #include <sys/cdefs.h>
40 __FBSDID("$FreeBSD$");
41
42 #include <sys/param.h>
43 #include <sys/socket.h>
44 #ifdef _KERNEL
45 # include <sys/systm.h>
46 # include <sys/refcount.h>
47 #endif /* _KERNEL */
48 #include <sys/mbuf.h>
49
50 #include <netinet/in.h>
51 #include <netinet/in_systm.h>
52 #include <netinet/ip.h>
53 #include <netinet/tcp.h>
54
55 #include <net/if.h>
56 #include <net/vnet.h>
57 #include <net/pfvar.h>
58
59 #ifdef INET6
60 #include <netinet/ip6.h>
61 #endif /* INET6 */
62
63 #ifdef _KERNEL
64 #define DPFPRINTF(format, x...)                         \
65         if (V_pf_status.debug >= PF_DEBUG_NOISY)        \
66                 printf(format , ##x)
67 #define rs_malloc(x)            malloc(x, M_TEMP, M_NOWAIT|M_ZERO)
68 #define rs_free(x)              free(x, M_TEMP)
69
70 #else
71 /* Userland equivalents so we can lend code to pfctl et al. */
72
73 #include <arpa/inet.h>
74 #include <errno.h>
75 #include <stdio.h>
76 #include <stdlib.h>
77 #include <string.h>
78 #define rs_malloc(x)             calloc(1, x)
79 #define rs_free(x)               free(x)
80
81 #ifdef PFDEBUG
82 #include <sys/stdarg.h>
83 #define DPFPRINTF(format, x...) fprintf(stderr, format , ##x)
84 #else
85 #define DPFPRINTF(format, x...) ((void)0)
86 #endif /* PFDEBUG */
87 #endif /* _KERNEL */
88
89 #ifdef _KERNEL
90 VNET_DEFINE(struct pf_anchor_global,    pf_anchors);
91 VNET_DEFINE(struct pf_anchor,           pf_main_anchor);
92 #else /* ! _KERNEL */
93 struct pf_anchor_global  pf_anchors;
94 struct pf_anchor         pf_main_anchor;
95 #undef V_pf_anchors
96 #define V_pf_anchors             pf_anchors
97 #undef pf_main_ruleset
98 #define pf_main_ruleset          pf_main_anchor.ruleset
99 #endif /* _KERNEL */
100
101 static __inline int pf_anchor_compare(struct pf_anchor *, struct pf_anchor *);
102
103 static struct pf_anchor         *pf_find_anchor(const char *);
104
105 RB_GENERATE(pf_anchor_global, pf_anchor, entry_global, pf_anchor_compare);
106 RB_GENERATE(pf_anchor_node, pf_anchor, entry_node, pf_anchor_compare);
107
108 static __inline int
109 pf_anchor_compare(struct pf_anchor *a, struct pf_anchor *b)
110 {
111         int c = strcmp(a->path, b->path);
112
113         return (c ? (c < 0 ? -1 : 1) : 0);
114 }
115
116 int
117 pf_get_ruleset_number(u_int8_t action)
118 {
119         switch (action) {
120         case PF_SCRUB:
121         case PF_NOSCRUB:
122                 return (PF_RULESET_SCRUB);
123                 break;
124         case PF_PASS:
125         case PF_DROP:
126                 return (PF_RULESET_FILTER);
127                 break;
128         case PF_NAT:
129         case PF_NONAT:
130                 return (PF_RULESET_NAT);
131                 break;
132         case PF_BINAT:
133         case PF_NOBINAT:
134                 return (PF_RULESET_BINAT);
135                 break;
136         case PF_RDR:
137         case PF_NORDR:
138                 return (PF_RULESET_RDR);
139                 break;
140         default:
141                 return (PF_RULESET_MAX);
142                 break;
143         }
144 }
145
146 void
147 pf_init_ruleset(struct pf_ruleset *ruleset)
148 {
149         int     i;
150
151         memset(ruleset, 0, sizeof(struct pf_ruleset));
152         for (i = 0; i < PF_RULESET_MAX; i++) {
153                 TAILQ_INIT(&ruleset->rules[i].queues[0]);
154                 TAILQ_INIT(&ruleset->rules[i].queues[1]);
155                 ruleset->rules[i].active.ptr = &ruleset->rules[i].queues[0];
156                 ruleset->rules[i].inactive.ptr = &ruleset->rules[i].queues[1];
157         }
158 }
159
160 static struct pf_anchor *
161 pf_find_anchor(const char *path)
162 {
163         struct pf_anchor        *key, *found;
164
165         key = (struct pf_anchor *)rs_malloc(sizeof(*key));
166         if (key == NULL)
167                 return (NULL);
168         strlcpy(key->path, path, sizeof(key->path));
169         found = RB_FIND(pf_anchor_global, &V_pf_anchors, key);
170         rs_free(key);
171         return (found);
172 }
173
174 struct pf_ruleset *
175 pf_find_ruleset(const char *path)
176 {
177         struct pf_anchor        *anchor;
178
179         while (*path == '/')
180                 path++;
181         if (!*path)
182                 return (&pf_main_ruleset);
183         anchor = pf_find_anchor(path);
184         if (anchor == NULL)
185                 return (NULL);
186         else
187                 return (&anchor->ruleset);
188 }
189
190 struct pf_ruleset *
191 pf_find_or_create_ruleset(const char *path)
192 {
193         char                    *p, *q, *r;
194         struct pf_ruleset       *ruleset;
195         struct pf_anchor        *anchor = NULL, *dup, *parent = NULL;
196
197         if (path[0] == 0)
198                 return (&pf_main_ruleset);
199         while (*path == '/')
200                 path++;
201         ruleset = pf_find_ruleset(path);
202         if (ruleset != NULL)
203                 return (ruleset);
204         p = (char *)rs_malloc(MAXPATHLEN);
205         if (p == NULL)
206                 return (NULL);
207         strlcpy(p, path, MAXPATHLEN);
208         while (parent == NULL && (q = strrchr(p, '/')) != NULL) {
209                 *q = 0;
210                 if ((ruleset = pf_find_ruleset(p)) != NULL) {
211                         parent = ruleset->anchor;
212                         break;
213                 }
214         }
215         if (q == NULL)
216                 q = p;
217         else
218                 q++;
219         strlcpy(p, path, MAXPATHLEN);
220         if (!*q) {
221                 rs_free(p);
222                 return (NULL);
223         }
224         while ((r = strchr(q, '/')) != NULL || *q) {
225                 if (r != NULL)
226                         *r = 0;
227                 if (!*q || strlen(q) >= PF_ANCHOR_NAME_SIZE ||
228                     (parent != NULL && strlen(parent->path) >=
229                     MAXPATHLEN - PF_ANCHOR_NAME_SIZE - 1)) {
230                         rs_free(p);
231                         return (NULL);
232                 }
233                 anchor = (struct pf_anchor *)rs_malloc(sizeof(*anchor));
234                 if (anchor == NULL) {
235                         rs_free(p);
236                         return (NULL);
237                 }
238                 RB_INIT(&anchor->children);
239                 strlcpy(anchor->name, q, sizeof(anchor->name));
240                 if (parent != NULL) {
241                         strlcpy(anchor->path, parent->path,
242                             sizeof(anchor->path));
243                         strlcat(anchor->path, "/", sizeof(anchor->path));
244                 }
245                 strlcat(anchor->path, anchor->name, sizeof(anchor->path));
246                 if ((dup = RB_INSERT(pf_anchor_global, &V_pf_anchors, anchor)) !=
247                     NULL) {
248                         printf("pf_find_or_create_ruleset: RB_INSERT1 "
249                             "'%s' '%s' collides with '%s' '%s'\n",
250                             anchor->path, anchor->name, dup->path, dup->name);
251                         rs_free(anchor);
252                         rs_free(p);
253                         return (NULL);
254                 }
255                 if (parent != NULL) {
256                         anchor->parent = parent;
257                         if ((dup = RB_INSERT(pf_anchor_node, &parent->children,
258                             anchor)) != NULL) {
259                                 printf("pf_find_or_create_ruleset: "
260                                     "RB_INSERT2 '%s' '%s' collides with "
261                                     "'%s' '%s'\n", anchor->path, anchor->name,
262                                     dup->path, dup->name);
263                                 RB_REMOVE(pf_anchor_global, &V_pf_anchors,
264                                     anchor);
265                                 rs_free(anchor);
266                                 rs_free(p);
267                                 return (NULL);
268                         }
269                 }
270                 pf_init_ruleset(&anchor->ruleset);
271                 anchor->ruleset.anchor = anchor;
272                 parent = anchor;
273                 if (r != NULL)
274                         q = r + 1;
275                 else
276                         *q = 0;
277         }
278         rs_free(p);
279         return (&anchor->ruleset);
280 }
281
282 void
283 pf_remove_if_empty_ruleset(struct pf_ruleset *ruleset)
284 {
285         struct pf_anchor        *parent;
286         int                      i;
287
288         while (ruleset != NULL) {
289                 if (ruleset == &pf_main_ruleset || ruleset->anchor == NULL ||
290                     !RB_EMPTY(&ruleset->anchor->children) ||
291                     ruleset->anchor->refcnt > 0 || ruleset->tables > 0 ||
292                     ruleset->topen)
293                         return;
294                 for (i = 0; i < PF_RULESET_MAX; ++i)
295                         if (!TAILQ_EMPTY(ruleset->rules[i].active.ptr) ||
296                             !TAILQ_EMPTY(ruleset->rules[i].inactive.ptr) ||
297                             ruleset->rules[i].inactive.open)
298                                 return;
299                 RB_REMOVE(pf_anchor_global, &V_pf_anchors, ruleset->anchor);
300                 if ((parent = ruleset->anchor->parent) != NULL)
301                         RB_REMOVE(pf_anchor_node, &parent->children,
302                             ruleset->anchor);
303                 rs_free(ruleset->anchor);
304                 if (parent == NULL)
305                         return;
306                 ruleset = &parent->ruleset;
307         }
308 }
309
310 int
311 pf_anchor_setup(struct pf_rule *r, const struct pf_ruleset *s,
312     const char *name)
313 {
314         char                    *p, *path;
315         struct pf_ruleset       *ruleset;
316
317         r->anchor = NULL;
318         r->anchor_relative = 0;
319         r->anchor_wildcard = 0;
320         if (!name[0])
321                 return (0);
322         path = (char *)rs_malloc(MAXPATHLEN);
323         if (path == NULL)
324                 return (1);
325         if (name[0] == '/')
326                 strlcpy(path, name + 1, MAXPATHLEN);
327         else {
328                 /* relative path */
329                 r->anchor_relative = 1;
330                 if (s->anchor == NULL || !s->anchor->path[0])
331                         path[0] = 0;
332                 else
333                         strlcpy(path, s->anchor->path, MAXPATHLEN);
334                 while (name[0] == '.' && name[1] == '.' && name[2] == '/') {
335                         if (!path[0]) {
336                                 printf("pf_anchor_setup: .. beyond root\n");
337                                 rs_free(path);
338                                 return (1);
339                         }
340                         if ((p = strrchr(path, '/')) != NULL)
341                                 *p = 0;
342                         else
343                                 path[0] = 0;
344                         r->anchor_relative++;
345                         name += 3;
346                 }
347                 if (path[0])
348                         strlcat(path, "/", MAXPATHLEN);
349                 strlcat(path, name, MAXPATHLEN);
350         }
351         if ((p = strrchr(path, '/')) != NULL && !strcmp(p, "/*")) {
352                 r->anchor_wildcard = 1;
353                 *p = 0;
354         }
355         ruleset = pf_find_or_create_ruleset(path);
356         rs_free(path);
357         if (ruleset == NULL || ruleset->anchor == NULL) {
358                 printf("pf_anchor_setup: ruleset\n");
359                 return (1);
360         }
361         r->anchor = ruleset->anchor;
362         r->anchor->refcnt++;
363         return (0);
364 }
365
366 int
367 pf_anchor_copyout(const struct pf_ruleset *rs, const struct pf_rule *r,
368     struct pfioc_rule *pr)
369 {
370         pr->anchor_call[0] = 0;
371         if (r->anchor == NULL)
372                 return (0);
373         if (!r->anchor_relative) {
374                 strlcpy(pr->anchor_call, "/", sizeof(pr->anchor_call));
375                 strlcat(pr->anchor_call, r->anchor->path,
376                     sizeof(pr->anchor_call));
377         } else {
378                 char    *a, *p;
379                 int      i;
380
381                 a = (char *)rs_malloc(MAXPATHLEN);
382                 if (a == NULL)
383                         return (1);
384                 if (rs->anchor == NULL)
385                         a[0] = 0;
386                 else
387                         strlcpy(a, rs->anchor->path, MAXPATHLEN);
388                 for (i = 1; i < r->anchor_relative; ++i) {
389                         if ((p = strrchr(a, '/')) == NULL)
390                                 p = a;
391                         *p = 0;
392                         strlcat(pr->anchor_call, "../",
393                             sizeof(pr->anchor_call));
394                 }
395                 if (strncmp(a, r->anchor->path, strlen(a))) {
396                         printf("pf_anchor_copyout: '%s' '%s'\n", a,
397                             r->anchor->path);
398                         rs_free(a);
399                         return (1);
400                 }
401                 if (strlen(r->anchor->path) > strlen(a))
402                         strlcat(pr->anchor_call, r->anchor->path + (a[0] ?
403                             strlen(a) + 1 : 0), sizeof(pr->anchor_call));
404                 rs_free(a);
405         }
406         if (r->anchor_wildcard)
407                 strlcat(pr->anchor_call, pr->anchor_call[0] ? "/*" : "*",
408                     sizeof(pr->anchor_call));
409         return (0);
410 }
411
412 void
413 pf_anchor_remove(struct pf_rule *r)
414 {
415         if (r->anchor == NULL)
416                 return;
417         if (r->anchor->refcnt <= 0) {
418                 printf("pf_anchor_remove: broken refcount\n");
419                 r->anchor = NULL;
420                 return;
421         }
422         if (!--r->anchor->refcnt)
423                 pf_remove_if_empty_ruleset(&r->anchor->ruleset);
424         r->anchor = NULL;
425 }