]> git.proxmox.com Git - mirror_frr.git/blob - lib/distribute.c
Merge pull request #5793 from ton31337/fix/formatting_show_bgp_summary_failed
[mirror_frr.git] / lib / distribute.c
1 /* Distribute list functions
2 * Copyright (C) 1998, 1999 Kunihiro Ishiguro
3 *
4 * This file is part of GNU Zebra.
5 *
6 * GNU Zebra is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published
8 * by the Free Software Foundation; either version 2, or (at your
9 * option) any later version.
10 *
11 * GNU Zebra is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along
17 * with this program; see the file COPYING; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19 */
20
21 #include <zebra.h>
22
23 #include "hash.h"
24 #include "if.h"
25 #include "filter.h"
26 #include "command.h"
27 #include "distribute.h"
28 #include "memory.h"
29
30 DEFINE_MTYPE_STATIC(LIB, DISTRIBUTE_CTX, "Distribute ctx")
31 DEFINE_MTYPE_STATIC(LIB, DISTRIBUTE, "Distribute list")
32 DEFINE_MTYPE_STATIC(LIB, DISTRIBUTE_IFNAME, "Dist-list ifname")
33 DEFINE_MTYPE_STATIC(LIB, DISTRIBUTE_NAME, "Dist-list name")
34
35 static struct list *dist_ctx_list;
36
37 static struct distribute *distribute_new(void)
38 {
39 return XCALLOC(MTYPE_DISTRIBUTE, sizeof(struct distribute));
40 }
41
42 /* Free distribute object. */
43 static void distribute_free(struct distribute *dist)
44 {
45 int i = 0;
46
47 XFREE(MTYPE_DISTRIBUTE_IFNAME, dist->ifname);
48
49 for (i = 0; i < DISTRIBUTE_MAX; i++) {
50 XFREE(MTYPE_DISTRIBUTE_NAME, dist->list[i]);
51 }
52
53 for (i = 0; i < DISTRIBUTE_MAX; i++) {
54 XFREE(MTYPE_DISTRIBUTE_NAME, dist->prefix[i]);
55 }
56
57 XFREE(MTYPE_DISTRIBUTE, dist);
58 }
59
60 static void distribute_free_if_empty(struct distribute_ctx *ctx,
61 struct distribute *dist)
62 {
63 int i;
64
65 for (i = 0; i < DISTRIBUTE_MAX; i++)
66 if (dist->list[i] != NULL || dist->prefix[i] != NULL)
67 return;
68
69 hash_release(ctx->disthash, dist);
70 distribute_free(dist);
71 }
72
73 /* Lookup interface's distribute list. */
74 struct distribute *distribute_lookup(struct distribute_ctx *ctx,
75 const char *ifname)
76 {
77 struct distribute key;
78 struct distribute *dist;
79
80 /* temporary reference */
81 key.ifname = (ifname) ? XSTRDUP(MTYPE_DISTRIBUTE_IFNAME, ifname) : NULL;
82
83 dist = hash_lookup(ctx->disthash, &key);
84
85 XFREE(MTYPE_DISTRIBUTE_IFNAME, key.ifname);
86
87 return dist;
88 }
89
90 void distribute_list_add_hook(struct distribute_ctx *ctx,
91 void (*func)(struct distribute_ctx *ctx,
92 struct distribute *))
93 {
94 ctx->distribute_add_hook = func;
95 }
96
97 void distribute_list_delete_hook(struct distribute_ctx *ctx,
98 void (*func)(struct distribute_ctx *ctx,
99 struct distribute *))
100 {
101 ctx->distribute_delete_hook = func;
102 }
103
104 static void *distribute_hash_alloc(struct distribute *arg)
105 {
106 struct distribute *dist;
107
108 dist = distribute_new();
109 if (arg->ifname)
110 dist->ifname = XSTRDUP(MTYPE_DISTRIBUTE_IFNAME, arg->ifname);
111 else
112 dist->ifname = NULL;
113 return dist;
114 }
115
116 /* Make new distribute list and push into hash. */
117 static struct distribute *distribute_get(struct distribute_ctx *ctx,
118 const char *ifname)
119 {
120 struct distribute key;
121 struct distribute *ret;
122
123 /* temporary reference */
124 key.ifname = (ifname) ? XSTRDUP(MTYPE_DISTRIBUTE_IFNAME, ifname) : NULL;
125
126 ret = hash_get(ctx->disthash, &key,
127 (void *(*)(void *))distribute_hash_alloc);
128
129 XFREE(MTYPE_DISTRIBUTE_IFNAME, key.ifname);
130
131 return ret;
132 }
133
134 static unsigned int distribute_hash_make(const void *arg)
135 {
136 const struct distribute *dist = arg;
137
138 return dist->ifname ? string_hash_make(dist->ifname) : 0;
139 }
140
141 /* If two distribute-list have same value then return 1 else return
142 0. This function is used by hash package. */
143 static bool distribute_cmp(const struct distribute *dist1,
144 const struct distribute *dist2)
145 {
146 if (dist1->ifname && dist2->ifname)
147 if (strcmp(dist1->ifname, dist2->ifname) == 0)
148 return true;
149 if (!dist1->ifname && !dist2->ifname)
150 return true;
151 return false;
152 }
153
154 /* Set access-list name to the distribute list. */
155 static void distribute_list_set(struct distribute_ctx *ctx,
156 const char *ifname, enum distribute_type type,
157 const char *alist_name)
158 {
159 struct distribute *dist;
160
161 dist = distribute_get(ctx, ifname);
162
163 XFREE(MTYPE_DISTRIBUTE_NAME, dist->list[type]);
164 dist->list[type] = XSTRDUP(MTYPE_DISTRIBUTE_NAME, alist_name);
165
166 /* Apply this distribute-list to the interface. */
167 (ctx->distribute_add_hook)(ctx, dist);
168 }
169
170 /* Unset distribute-list. If matched distribute-list exist then
171 return 1. */
172 static int distribute_list_unset(struct distribute_ctx *ctx,
173 const char *ifname,
174 enum distribute_type type,
175 const char *alist_name)
176 {
177 struct distribute *dist;
178
179 dist = distribute_lookup(ctx, ifname);
180 if (!dist)
181 return 0;
182
183 if (!dist->list[type])
184 return 0;
185 if (strcmp(dist->list[type], alist_name) != 0)
186 return 0;
187
188 XFREE(MTYPE_DISTRIBUTE_NAME, dist->list[type]);
189
190 /* Apply this distribute-list to the interface. */
191 (ctx->distribute_delete_hook)(ctx, dist);
192
193 /* If all dist are NULL, then free distribute list. */
194 distribute_free_if_empty(ctx, dist);
195 return 1;
196 }
197
198 /* Set access-list name to the distribute list. */
199 static void distribute_list_prefix_set(struct distribute_ctx *ctx,
200 const char *ifname,
201 enum distribute_type type,
202 const char *plist_name)
203 {
204 struct distribute *dist;
205
206 dist = distribute_get(ctx, ifname);
207
208 XFREE(MTYPE_DISTRIBUTE_NAME, dist->prefix[type]);
209 dist->prefix[type] = XSTRDUP(MTYPE_DISTRIBUTE_NAME, plist_name);
210
211 /* Apply this distribute-list to the interface. */
212 (ctx->distribute_add_hook)(ctx, dist);
213 }
214
215 /* Unset distribute-list. If matched distribute-list exist then
216 return 1. */
217 static int distribute_list_prefix_unset(struct distribute_ctx *ctx,
218 const char *ifname,
219 enum distribute_type type,
220 const char *plist_name)
221 {
222 struct distribute *dist;
223
224 dist = distribute_lookup(ctx, ifname);
225 if (!dist)
226 return 0;
227
228 if (!dist->prefix[type])
229 return 0;
230 if (strcmp(dist->prefix[type], plist_name) != 0)
231 return 0;
232
233 XFREE(MTYPE_DISTRIBUTE_NAME, dist->prefix[type]);
234
235 /* Apply this distribute-list to the interface. */
236 (ctx->distribute_delete_hook)(ctx, dist);
237
238 /* If all dist are NULL, then free distribute list. */
239 distribute_free_if_empty(ctx, dist);
240 return 1;
241 }
242
243 DEFUN (distribute_list,
244 distribute_list_cmd,
245 "distribute-list [prefix] WORD <in|out> [WORD]",
246 "Filter networks in routing updates\n"
247 "Specify a prefix\n"
248 "Access-list name\n"
249 "Filter incoming routing updates\n"
250 "Filter outgoing routing updates\n"
251 "Interface name\n")
252 {
253 int prefix = (argv[1]->type == WORD_TKN) ? 1 : 0;
254 /* Check of distribute list type. */
255 enum distribute_type type = argv[2 + prefix]->arg[0] == 'i'
256 ? DISTRIBUTE_V4_IN
257 : DISTRIBUTE_V4_OUT;
258
259 /* Set appropriate function call */
260 void (*distfn)(struct distribute_ctx *, const char *,
261 enum distribute_type, const char *) =
262 prefix ? &distribute_list_prefix_set : &distribute_list_set;
263 struct distribute_ctx *ctx =
264 (struct distribute_ctx *)listnode_head(dist_ctx_list);
265
266 /* if interface is present, get name */
267 const char *ifname = NULL;
268 if (argv[argc - 1]->type == VARIABLE_TKN)
269 ifname = argv[argc - 1]->arg;
270
271 /* Get interface name corresponding distribute list. */
272 distfn(ctx, ifname, type, argv[1 + prefix]->arg);
273
274 return CMD_SUCCESS;
275 }
276
277 DEFUN (ipv6_distribute_list,
278 ipv6_distribute_list_cmd,
279 "ipv6 distribute-list [prefix] WORD <in|out> [WORD]",
280 "IPv6\n"
281 "Filter networks in routing updates\n"
282 "Specify a prefix\n"
283 "Access-list name\n"
284 "Filter incoming routing updates\n"
285 "Filter outgoing routing updates\n"
286 "Interface name\n")
287 {
288 int prefix = (argv[2]->type == WORD_TKN) ? 1 : 0;
289 /* Check of distribute list type. */
290 enum distribute_type type = argv[3 + prefix]->arg[0] == 'i'
291 ? DISTRIBUTE_V6_IN
292 : DISTRIBUTE_V6_OUT;
293
294 /* Set appropriate function call */
295 void (*distfn)(struct distribute_ctx *, const char *,
296 enum distribute_type, const char *) =
297 prefix ? &distribute_list_prefix_set : &distribute_list_set;
298 struct distribute_ctx *ctx = listnode_head(dist_ctx_list);
299
300 /* if interface is present, get name */
301 const char *ifname = NULL;
302 if (argv[argc - 1]->type == VARIABLE_TKN)
303 ifname = argv[argc - 1]->arg;
304
305 /* Get interface name corresponding distribute list. */
306 distfn(ctx, ifname, type, argv[2 + prefix]->arg);
307
308 return CMD_SUCCESS;
309 }
310
311 DEFUN (no_distribute_list,
312 no_distribute_list_cmd,
313 "no distribute-list [prefix] WORD <in|out> [WORD]",
314 NO_STR
315 "Filter networks in routing updates\n"
316 "Specify a prefix\n"
317 "Access-list name\n"
318 "Filter incoming routing updates\n"
319 "Filter outgoing routing updates\n"
320 "Interface name\n")
321 {
322 int prefix = (argv[2]->type == WORD_TKN) ? 1 : 0;
323 int idx_alname = 2 + prefix;
324 int idx_disttype = idx_alname + 1;
325 enum distribute_type type =
326 argv[idx_disttype]->arg[0] == 'i' ?
327 DISTRIBUTE_V4_IN : DISTRIBUTE_V4_OUT;
328
329 /* Set appropriate function call */
330 int (*distfn)(struct distribute_ctx *, const char *,
331 enum distribute_type, const char *) =
332 prefix ? &distribute_list_prefix_unset : &distribute_list_unset;
333 struct distribute_ctx *ctx = listnode_head(dist_ctx_list);
334
335 /* if interface is present, get name */
336 const char *ifname = NULL;
337 if (argv[argc - 1]->type == VARIABLE_TKN)
338 ifname = argv[argc - 1]->arg;
339 /* Get interface name corresponding distribute list. */
340 int ret = distfn(ctx, ifname, type, argv[2 + prefix]->arg);
341
342 if (!ret) {
343 vty_out(vty, "distribute list doesn't exist\n");
344 return CMD_WARNING_CONFIG_FAILED;
345 }
346 return CMD_SUCCESS;
347 }
348
349 DEFUN (no_ipv6_distribute_list,
350 no_ipv6_distribute_list_cmd,
351 "no ipv6 distribute-list [prefix] WORD <in|out> [WORD]",
352 NO_STR
353 "IPv6\n"
354 "Filter networks in routing updates\n"
355 "Specify a prefix\n"
356 "Access-list name\n"
357 "Filter incoming routing updates\n"
358 "Filter outgoing routing updates\n"
359 "Interface name\n")
360 {
361 int prefix = (argv[3]->type == WORD_TKN) ? 1 : 0;
362 int idx_alname = 3 + prefix;
363 int idx_disttype = idx_alname + 1;
364
365 enum distribute_type type =
366 argv[idx_disttype]->arg[0] == 'i' ?
367 DISTRIBUTE_V6_IN : DISTRIBUTE_V6_OUT;
368 struct distribute_ctx *ctx = listnode_head(dist_ctx_list);
369
370 /* Set appropriate function call */
371 int (*distfn)(struct distribute_ctx *, const char *,
372 enum distribute_type, const char *) =
373 prefix ? &distribute_list_prefix_unset : &distribute_list_unset;
374
375 /* if interface is present, get name */
376 const char *ifname = NULL;
377
378 if (argv[argc - 1]->type == VARIABLE_TKN)
379 ifname = argv[argc - 1]->arg;
380 /* Get interface name corresponding distribute list. */
381 int ret = distfn(ctx, ifname, type, argv[3 + prefix]->arg);
382
383 if (!ret) {
384 vty_out(vty, "distribute list doesn't exist\n");
385 return CMD_WARNING_CONFIG_FAILED;
386 }
387 return CMD_SUCCESS;
388 }
389
390 static int distribute_print(struct vty *vty, char *tab[], int is_prefix,
391 enum distribute_type type, int has_print)
392 {
393 if (tab[type]) {
394 vty_out(vty, "%s %s%s", has_print ? "," : "",
395 is_prefix ? "(prefix-list) " : "", tab[type]);
396 return 1;
397 }
398 return has_print;
399 }
400
401 int config_show_distribute(struct vty *vty, struct distribute_ctx *dist_ctxt)
402 {
403 unsigned int i;
404 int has_print = 0;
405 struct hash_bucket *mp;
406 struct distribute *dist;
407
408 /* Output filter configuration. */
409 dist = distribute_lookup(dist_ctxt, NULL);
410 vty_out(vty, " Outgoing update filter list for all interface is");
411 has_print = 0;
412 if (dist) {
413 has_print = distribute_print(vty, dist->list, 0,
414 DISTRIBUTE_V4_OUT, has_print);
415 has_print = distribute_print(vty, dist->prefix, 1,
416 DISTRIBUTE_V4_OUT, has_print);
417 has_print = distribute_print(vty, dist->list, 0,
418 DISTRIBUTE_V6_OUT, has_print);
419 has_print = distribute_print(vty, dist->prefix, 1,
420 DISTRIBUTE_V6_OUT, has_print);
421 }
422 if (has_print)
423 vty_out(vty, "\n");
424 else
425 vty_out(vty, " not set\n");
426
427 for (i = 0; i < dist_ctxt->disthash->size; i++)
428 for (mp = dist_ctxt->disthash->index[i]; mp; mp = mp->next) {
429 dist = mp->data;
430 if (dist->ifname) {
431 vty_out(vty, " %s filtered by",
432 dist->ifname);
433 has_print = 0;
434 has_print = distribute_print(vty, dist->list, 0,
435 DISTRIBUTE_V4_OUT,
436 has_print);
437 has_print = distribute_print(
438 vty, dist->prefix, 1, DISTRIBUTE_V4_OUT,
439 has_print);
440 has_print = distribute_print(vty, dist->list, 0,
441 DISTRIBUTE_V6_OUT,
442 has_print);
443 has_print = distribute_print(
444 vty, dist->prefix, 1, DISTRIBUTE_V6_OUT,
445 has_print);
446 if (has_print)
447 vty_out(vty, "\n");
448 else
449 vty_out(vty, " nothing\n");
450 }
451 }
452
453
454 /* Input filter configuration. */
455 dist = distribute_lookup(dist_ctxt, NULL);
456 vty_out(vty, " Incoming update filter list for all interface is");
457 has_print = 0;
458 if (dist) {
459 has_print = distribute_print(vty, dist->list, 0,
460 DISTRIBUTE_V4_IN, has_print);
461 has_print = distribute_print(vty, dist->prefix, 1,
462 DISTRIBUTE_V4_IN, has_print);
463 has_print = distribute_print(vty, dist->list, 0,
464 DISTRIBUTE_V6_IN, has_print);
465 has_print = distribute_print(vty, dist->prefix, 1,
466 DISTRIBUTE_V6_IN, has_print);
467 }
468 if (has_print)
469 vty_out(vty, "\n");
470 else
471 vty_out(vty, " not set\n");
472
473 for (i = 0; i < dist_ctxt->disthash->size; i++)
474 for (mp = dist_ctxt->disthash->index[i]; mp; mp = mp->next) {
475 dist = mp->data;
476 if (dist->ifname) {
477 vty_out(vty, " %s filtered by",
478 dist->ifname);
479 has_print = 0;
480 has_print = distribute_print(vty, dist->list, 0,
481 DISTRIBUTE_V4_IN,
482 has_print);
483 has_print = distribute_print(
484 vty, dist->prefix, 1, DISTRIBUTE_V4_IN,
485 has_print);
486 has_print = distribute_print(vty, dist->list, 0,
487 DISTRIBUTE_V6_IN,
488 has_print);
489 has_print = distribute_print(
490 vty, dist->prefix, 1, DISTRIBUTE_V6_IN,
491 has_print);
492 if (has_print)
493 vty_out(vty, "\n");
494 else
495 vty_out(vty, " nothing\n");
496 }
497 }
498 return 0;
499 }
500
501 /* Configuration write function. */
502 int config_write_distribute(struct vty *vty,
503 struct distribute_ctx *dist_ctxt)
504 {
505 unsigned int i;
506 int j;
507 int output, v6;
508 struct hash_bucket *mp;
509 int write = 0;
510
511 for (i = 0; i < dist_ctxt->disthash->size; i++)
512 for (mp = dist_ctxt->disthash->index[i]; mp; mp = mp->next) {
513 struct distribute *dist;
514
515 dist = mp->data;
516
517 for (j = 0; j < DISTRIBUTE_MAX; j++)
518 if (dist->list[j]) {
519 output = j == DISTRIBUTE_V4_OUT
520 || j == DISTRIBUTE_V6_OUT;
521 v6 = j == DISTRIBUTE_V6_IN
522 || j == DISTRIBUTE_V6_OUT;
523 vty_out(vty,
524 " %sdistribute-list %s %s %s\n",
525 v6 ? "ipv6 " : "",
526 dist->list[j],
527 output ? "out" : "in",
528 dist->ifname ? dist->ifname
529 : "");
530 write++;
531 }
532
533 for (j = 0; j < DISTRIBUTE_MAX; j++)
534 if (dist->prefix[j]) {
535 output = j == DISTRIBUTE_V4_OUT
536 || j == DISTRIBUTE_V6_OUT;
537 v6 = j == DISTRIBUTE_V6_IN
538 || j == DISTRIBUTE_V6_OUT;
539 vty_out(vty,
540 " %sdistribute-list prefix %s %s %s\n",
541 v6 ? "ipv6 " : "",
542 dist->prefix[j],
543 output ? "out" : "in",
544 dist->ifname ? dist->ifname
545 : "");
546 write++;
547 }
548 }
549 return write;
550 }
551
552 void distribute_list_delete(struct distribute_ctx **ctx)
553 {
554 if ((*ctx)->disthash) {
555 hash_clean((*ctx)->disthash, (void (*)(void *))distribute_free);
556 }
557 if (!dist_ctx_list)
558 dist_ctx_list = list_new();
559 listnode_delete(dist_ctx_list, *ctx);
560 if (list_isempty(dist_ctx_list))
561 list_delete(&dist_ctx_list);
562 XFREE(MTYPE_DISTRIBUTE_CTX, (*ctx));
563 }
564
565 /* Initialize distribute list container */
566 struct distribute_ctx *distribute_list_ctx_create(struct vrf *vrf)
567 {
568 struct distribute_ctx *ctx;
569
570 ctx = XCALLOC(MTYPE_DISTRIBUTE_CTX, sizeof(struct distribute_ctx));
571 ctx->vrf = vrf;
572 ctx->disthash = hash_create(
573 distribute_hash_make,
574 (bool (*)(const void *, const void *))distribute_cmp, NULL);
575 if (!dist_ctx_list)
576 dist_ctx_list = list_new();
577 listnode_add(dist_ctx_list, ctx);
578 return ctx;
579 }
580
581 /* Initialize distribute list vty commands */
582 void distribute_list_init(int node)
583 {
584 /* vtysh command-extraction doesn't grok install_element(node, ) */
585 if (node == RIP_NODE) {
586 install_element(RIP_NODE, &distribute_list_cmd);
587 install_element(RIP_NODE, &no_distribute_list_cmd);
588 } else if (node == RIPNG_NODE) {
589 install_element(RIPNG_NODE, &distribute_list_cmd);
590 install_element(RIPNG_NODE, &no_distribute_list_cmd);
591 /* install v6 */
592 install_element(RIPNG_NODE, &ipv6_distribute_list_cmd);
593 install_element(RIPNG_NODE, &no_ipv6_distribute_list_cmd);
594 }
595
596 /* TODO: install v4 syntax command for v6 only protocols. */
597 /* if (node == RIPNG_NODE) {
598 * install_element (node, &ipv6_as_v4_distribute_list_all_cmd);
599 * install_element (node, &no_ipv6_as_v4_distribute_list_all_cmd);
600 * install_element (node, &ipv6_as_v4_distribute_list_cmd);
601 * install_element (node, &no_ipv6_as_v4_distribute_list_cmd);
602 * install_element (node, &ipv6_as_v4_distribute_list_prefix_all_cmd);
603 * install_element (node,
604 &no_ipv6_as_v4_distribute_list_prefix_all_cmd);
605 * install_element (node, &ipv6_as_v4_distribute_list_prefix_cmd);
606 * install_element (node, &no_ipv6_as_v4_distribute_list_prefix_cmd);
607 }*/
608 }