]> git.proxmox.com Git - mirror_frr.git/blob - pimd/pim_cmd.c
cb2ba87ec68fa55dcda2ef0dd778c399aa18ee6d
[mirror_frr.git] / pimd / pim_cmd.c
1 /*
2 * PIM for Quagga
3 * Copyright (C) 2008 Everton da Silva Marques
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License along
16 * with this program; see the file COPYING; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18 */
19
20 #include <zebra.h>
21
22 #include "lib/json.h"
23 #include "command.h"
24 #include "if.h"
25 #include "prefix.h"
26 #include "zclient.h"
27 #include "plist.h"
28 #include "hash.h"
29 #include "nexthop.h"
30 #include "vrf.h"
31 #include "ferr.h"
32
33 #include "pimd.h"
34 #include "pim_mroute.h"
35 #include "pim_cmd.h"
36 #include "pim_iface.h"
37 #include "pim_vty.h"
38 #include "pim_mroute.h"
39 #include "pim_str.h"
40 #include "pim_igmp.h"
41 #include "pim_igmpv3.h"
42 #include "pim_sock.h"
43 #include "pim_time.h"
44 #include "pim_util.h"
45 #include "pim_oil.h"
46 #include "pim_neighbor.h"
47 #include "pim_pim.h"
48 #include "pim_ifchannel.h"
49 #include "pim_hello.h"
50 #include "pim_msg.h"
51 #include "pim_upstream.h"
52 #include "pim_rpf.h"
53 #include "pim_macro.h"
54 #include "pim_ssmpingd.h"
55 #include "pim_zebra.h"
56 #include "pim_static.h"
57 #include "pim_rp.h"
58 #include "pim_zlookup.h"
59 #include "pim_msdp.h"
60 #include "pim_ssm.h"
61 #include "pim_nht.h"
62 #include "pim_bfd.h"
63 #include "pim_vxlan.h"
64 #include "bfd.h"
65
66 #ifndef VTYSH_EXTRACT_PL
67 #include "pimd/pim_cmd_clippy.c"
68 #endif
69
70 static struct cmd_node interface_node = {
71 INTERFACE_NODE, "%s(config-if)# ", 1 /* vtysh ? yes */
72 };
73
74 static struct cmd_node debug_node = {DEBUG_NODE, "", 1};
75
76 static struct vrf *pim_cmd_lookup_vrf(struct vty *vty, struct cmd_token *argv[],
77 const int argc, int *idx)
78 {
79 struct vrf *vrf;
80
81 if (argv_find(argv, argc, "NAME", idx))
82 vrf = vrf_lookup_by_name(argv[*idx]->arg);
83 else
84 vrf = vrf_lookup_by_id(VRF_DEFAULT);
85
86 if (!vrf)
87 vty_out(vty, "Specified VRF: %s does not exist\n",
88 argv[*idx]->arg);
89
90 return vrf;
91 }
92
93 static void pim_if_membership_clear(struct interface *ifp)
94 {
95 struct pim_interface *pim_ifp;
96
97 pim_ifp = ifp->info;
98 zassert(pim_ifp);
99
100 if (PIM_IF_TEST_PIM(pim_ifp->options)
101 && PIM_IF_TEST_IGMP(pim_ifp->options)) {
102 return;
103 }
104
105 pim_ifchannel_membership_clear(ifp);
106 }
107
108 /*
109 When PIM is disabled on interface, IGMPv3 local membership
110 information is not injected into PIM interface state.
111
112 The function pim_if_membership_refresh() fetches all IGMPv3 local
113 membership information into PIM. It is intented to be called
114 whenever PIM is enabled on the interface in order to collect missed
115 local membership information.
116 */
117 static void pim_if_membership_refresh(struct interface *ifp)
118 {
119 struct pim_interface *pim_ifp;
120 struct listnode *sock_node;
121 struct igmp_sock *igmp;
122
123 pim_ifp = ifp->info;
124 zassert(pim_ifp);
125
126 if (!PIM_IF_TEST_PIM(pim_ifp->options))
127 return;
128 if (!PIM_IF_TEST_IGMP(pim_ifp->options))
129 return;
130
131 /*
132 First clear off membership from all PIM (S,G) entries on the
133 interface
134 */
135
136 pim_ifchannel_membership_clear(ifp);
137
138 /*
139 Then restore PIM (S,G) membership from all IGMPv3 (S,G) entries on
140 the interface
141 */
142
143 /* scan igmp sockets */
144 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
145 struct listnode *grpnode;
146 struct igmp_group *grp;
147
148 /* scan igmp groups */
149 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list, grpnode,
150 grp)) {
151 struct listnode *srcnode;
152 struct igmp_source *src;
153
154 /* scan group sources */
155 for (ALL_LIST_ELEMENTS_RO(grp->group_source_list,
156 srcnode, src)) {
157
158 if (IGMP_SOURCE_TEST_FORWARDING(
159 src->source_flags)) {
160 struct prefix_sg sg;
161
162 memset(&sg, 0,
163 sizeof(struct prefix_sg));
164 sg.src = src->source_addr;
165 sg.grp = grp->group_addr;
166 pim_ifchannel_local_membership_add(ifp,
167 &sg);
168 }
169
170 } /* scan group sources */
171 } /* scan igmp groups */
172 } /* scan igmp sockets */
173
174 /*
175 Finally delete every PIM (S,G) entry lacking all state info
176 */
177
178 pim_ifchannel_delete_on_noinfo(ifp);
179 }
180
181 static void pim_show_assert_helper(struct vty *vty,
182 struct pim_interface *pim_ifp,
183 struct pim_ifchannel *ch, time_t now)
184 {
185 char ch_src_str[INET_ADDRSTRLEN];
186 char ch_grp_str[INET_ADDRSTRLEN];
187 char winner_str[INET_ADDRSTRLEN];
188 struct in_addr ifaddr;
189 char uptime[10];
190 char timer[10];
191
192 ifaddr = pim_ifp->primary_address;
193
194 pim_inet4_dump("<ch_src?>", ch->sg.src, ch_src_str, sizeof(ch_src_str));
195 pim_inet4_dump("<ch_grp?>", ch->sg.grp, ch_grp_str, sizeof(ch_grp_str));
196 pim_inet4_dump("<assrt_win?>", ch->ifassert_winner, winner_str,
197 sizeof(winner_str));
198
199 pim_time_uptime(uptime, sizeof(uptime), now - ch->ifassert_creation);
200 pim_time_timer_to_mmss(timer, sizeof(timer), ch->t_ifassert_timer);
201
202 vty_out(vty, "%-16s %-15s %-15s %-15s %-6s %-15s %-8s %-5s\n",
203 ch->interface->name, inet_ntoa(ifaddr), ch_src_str, ch_grp_str,
204 pim_ifchannel_ifassert_name(ch->ifassert_state), winner_str,
205 uptime, timer);
206 }
207
208 static void pim_show_assert(struct pim_instance *pim, struct vty *vty)
209 {
210 struct pim_interface *pim_ifp;
211 struct pim_ifchannel *ch;
212 struct interface *ifp;
213 time_t now;
214
215 now = pim_time_monotonic_sec();
216
217 vty_out(vty,
218 "Interface Address Source Group State Winner Uptime Timer\n");
219
220 FOR_ALL_INTERFACES (pim->vrf, ifp) {
221 pim_ifp = ifp->info;
222 if (!pim_ifp)
223 continue;
224
225 RB_FOREACH (ch, pim_ifchannel_rb, &pim_ifp->ifchannel_rb) {
226 pim_show_assert_helper(vty, pim_ifp, ch, now);
227 } /* scan interface channels */
228 }
229 }
230
231 static void pim_show_assert_internal_helper(struct vty *vty,
232 struct pim_interface *pim_ifp,
233 struct pim_ifchannel *ch)
234 {
235 char ch_src_str[INET_ADDRSTRLEN];
236 char ch_grp_str[INET_ADDRSTRLEN];
237 struct in_addr ifaddr;
238
239 ifaddr = pim_ifp->primary_address;
240
241 pim_inet4_dump("<ch_src?>", ch->sg.src, ch_src_str, sizeof(ch_src_str));
242 pim_inet4_dump("<ch_grp?>", ch->sg.grp, ch_grp_str, sizeof(ch_grp_str));
243 vty_out(vty, "%-16s %-15s %-15s %-15s %-3s %-3s %-3s %-4s\n",
244 ch->interface->name, inet_ntoa(ifaddr), ch_src_str, ch_grp_str,
245 PIM_IF_FLAG_TEST_COULD_ASSERT(ch->flags) ? "yes" : "no",
246 pim_macro_ch_could_assert_eval(ch) ? "yes" : "no",
247 PIM_IF_FLAG_TEST_ASSERT_TRACKING_DESIRED(ch->flags) ? "yes"
248 : "no",
249 pim_macro_assert_tracking_desired_eval(ch) ? "yes" : "no");
250 }
251
252 static void pim_show_assert_internal(struct pim_instance *pim, struct vty *vty)
253 {
254 struct pim_interface *pim_ifp;
255 struct pim_ifchannel *ch;
256 struct interface *ifp;
257
258 vty_out(vty,
259 "CA: CouldAssert\n"
260 "ECA: Evaluate CouldAssert\n"
261 "ATD: AssertTrackingDesired\n"
262 "eATD: Evaluate AssertTrackingDesired\n\n");
263
264 vty_out(vty,
265 "Interface Address Source Group CA eCA ATD eATD\n");
266 FOR_ALL_INTERFACES (pim->vrf, ifp) {
267 pim_ifp = ifp->info;
268 if (!pim_ifp)
269 continue;
270
271 RB_FOREACH (ch, pim_ifchannel_rb, &pim_ifp->ifchannel_rb) {
272 pim_show_assert_internal_helper(vty, pim_ifp, ch);
273 } /* scan interface channels */
274 }
275 }
276
277 static void pim_show_assert_metric_helper(struct vty *vty,
278 struct pim_interface *pim_ifp,
279 struct pim_ifchannel *ch)
280 {
281 char ch_src_str[INET_ADDRSTRLEN];
282 char ch_grp_str[INET_ADDRSTRLEN];
283 char addr_str[INET_ADDRSTRLEN];
284 struct pim_assert_metric am;
285 struct in_addr ifaddr;
286
287 ifaddr = pim_ifp->primary_address;
288
289 am = pim_macro_spt_assert_metric(&ch->upstream->rpf,
290 pim_ifp->primary_address);
291
292 pim_inet4_dump("<ch_src?>", ch->sg.src, ch_src_str, sizeof(ch_src_str));
293 pim_inet4_dump("<ch_grp?>", ch->sg.grp, ch_grp_str, sizeof(ch_grp_str));
294 pim_inet4_dump("<addr?>", am.ip_address, addr_str, sizeof(addr_str));
295
296 vty_out(vty, "%-16s %-15s %-15s %-15s %-3s %4u %6u %-15s\n",
297 ch->interface->name, inet_ntoa(ifaddr), ch_src_str, ch_grp_str,
298 am.rpt_bit_flag ? "yes" : "no", am.metric_preference,
299 am.route_metric, addr_str);
300 }
301
302 static void pim_show_assert_metric(struct pim_instance *pim, struct vty *vty)
303 {
304 struct pim_interface *pim_ifp;
305 struct pim_ifchannel *ch;
306 struct interface *ifp;
307
308 vty_out(vty,
309 "Interface Address Source Group RPT Pref Metric Address \n");
310
311 FOR_ALL_INTERFACES (pim->vrf, ifp) {
312 pim_ifp = ifp->info;
313 if (!pim_ifp)
314 continue;
315
316 RB_FOREACH (ch, pim_ifchannel_rb, &pim_ifp->ifchannel_rb) {
317 pim_show_assert_metric_helper(vty, pim_ifp, ch);
318 } /* scan interface channels */
319 }
320 }
321
322 static void pim_show_assert_winner_metric_helper(struct vty *vty,
323 struct pim_interface *pim_ifp,
324 struct pim_ifchannel *ch)
325 {
326 char ch_src_str[INET_ADDRSTRLEN];
327 char ch_grp_str[INET_ADDRSTRLEN];
328 char addr_str[INET_ADDRSTRLEN];
329 struct pim_assert_metric *am;
330 struct in_addr ifaddr;
331 char pref_str[16];
332 char metr_str[16];
333
334 ifaddr = pim_ifp->primary_address;
335
336 am = &ch->ifassert_winner_metric;
337
338 pim_inet4_dump("<ch_src?>", ch->sg.src, ch_src_str, sizeof(ch_src_str));
339 pim_inet4_dump("<ch_grp?>", ch->sg.grp, ch_grp_str, sizeof(ch_grp_str));
340 pim_inet4_dump("<addr?>", am->ip_address, addr_str, sizeof(addr_str));
341
342 if (am->metric_preference == PIM_ASSERT_METRIC_PREFERENCE_MAX)
343 snprintf(pref_str, sizeof(pref_str), "INFI");
344 else
345 snprintf(pref_str, sizeof(pref_str), "%4u",
346 am->metric_preference);
347
348 if (am->route_metric == PIM_ASSERT_ROUTE_METRIC_MAX)
349 snprintf(metr_str, sizeof(metr_str), "INFI");
350 else
351 snprintf(metr_str, sizeof(metr_str), "%6u", am->route_metric);
352
353 vty_out(vty, "%-16s %-15s %-15s %-15s %-3s %-4s %-6s %-15s\n",
354 ch->interface->name, inet_ntoa(ifaddr), ch_src_str, ch_grp_str,
355 am->rpt_bit_flag ? "yes" : "no", pref_str, metr_str, addr_str);
356 }
357
358 static void pim_show_assert_winner_metric(struct pim_instance *pim,
359 struct vty *vty)
360 {
361 struct pim_interface *pim_ifp;
362 struct pim_ifchannel *ch;
363 struct interface *ifp;
364
365 vty_out(vty,
366 "Interface Address Source Group RPT Pref Metric Address \n");
367
368 FOR_ALL_INTERFACES (pim->vrf, ifp) {
369 pim_ifp = ifp->info;
370 if (!pim_ifp)
371 continue;
372
373 RB_FOREACH (ch, pim_ifchannel_rb, &pim_ifp->ifchannel_rb) {
374 pim_show_assert_winner_metric_helper(vty, pim_ifp, ch);
375 } /* scan interface channels */
376 }
377 }
378
379 static void json_object_pim_ifp_add(struct json_object *json,
380 struct interface *ifp)
381 {
382 struct pim_interface *pim_ifp;
383
384 pim_ifp = ifp->info;
385 json_object_string_add(json, "name", ifp->name);
386 json_object_string_add(json, "state", if_is_up(ifp) ? "up" : "down");
387 json_object_string_add(json, "address",
388 inet_ntoa(pim_ifp->primary_address));
389 json_object_int_add(json, "index", ifp->ifindex);
390
391 if (if_is_multicast(ifp))
392 json_object_boolean_true_add(json, "flagMulticast");
393
394 if (if_is_broadcast(ifp))
395 json_object_boolean_true_add(json, "flagBroadcast");
396
397 if (ifp->flags & IFF_ALLMULTI)
398 json_object_boolean_true_add(json, "flagAllMulticast");
399
400 if (ifp->flags & IFF_PROMISC)
401 json_object_boolean_true_add(json, "flagPromiscuous");
402
403 if (PIM_IF_IS_DELETED(ifp))
404 json_object_boolean_true_add(json, "flagDeleted");
405
406 if (pim_if_lan_delay_enabled(ifp))
407 json_object_boolean_true_add(json, "lanDelayEnabled");
408 }
409
410 static void pim_show_membership_helper(struct vty *vty,
411 struct pim_interface *pim_ifp,
412 struct pim_ifchannel *ch,
413 struct json_object *json)
414 {
415 char ch_src_str[INET_ADDRSTRLEN];
416 char ch_grp_str[INET_ADDRSTRLEN];
417 json_object *json_iface = NULL;
418 json_object *json_row = NULL;
419
420 pim_inet4_dump("<ch_src?>", ch->sg.src, ch_src_str, sizeof(ch_src_str));
421 pim_inet4_dump("<ch_grp?>", ch->sg.grp, ch_grp_str, sizeof(ch_grp_str));
422
423 json_object_object_get_ex(json, ch->interface->name, &json_iface);
424 if (!json_iface) {
425 json_iface = json_object_new_object();
426 json_object_pim_ifp_add(json_iface, ch->interface);
427 json_object_object_add(json, ch->interface->name, json_iface);
428 }
429
430 json_row = json_object_new_object();
431 json_object_string_add(json_row, "source", ch_src_str);
432 json_object_string_add(json_row, "group", ch_grp_str);
433 json_object_string_add(json_row, "localMembership",
434 ch->local_ifmembership == PIM_IFMEMBERSHIP_NOINFO
435 ? "NOINFO"
436 : "INCLUDE");
437 json_object_object_add(json_iface, ch_grp_str, json_row);
438 }
439 static void pim_show_membership(struct pim_instance *pim, struct vty *vty,
440 bool uj)
441 {
442 struct pim_interface *pim_ifp;
443 struct pim_ifchannel *ch;
444 struct interface *ifp;
445 enum json_type type;
446 json_object *json = NULL;
447 json_object *json_tmp = NULL;
448
449 json = json_object_new_object();
450
451 FOR_ALL_INTERFACES (pim->vrf, ifp) {
452 pim_ifp = ifp->info;
453 if (!pim_ifp)
454 continue;
455
456 RB_FOREACH (ch, pim_ifchannel_rb, &pim_ifp->ifchannel_rb) {
457 pim_show_membership_helper(vty, pim_ifp, ch, json);
458 } /* scan interface channels */
459 }
460
461 if (uj) {
462 vty_out(vty, "%s\n", json_object_to_json_string_ext(
463 json, JSON_C_TO_STRING_PRETTY));
464 } else {
465 vty_out(vty,
466 "Interface Address Source Group Membership\n");
467
468 /*
469 * Example of the json data we are traversing
470 *
471 * {
472 * "swp3":{
473 * "name":"swp3",
474 * "state":"up",
475 * "address":"10.1.20.1",
476 * "index":5,
477 * "flagMulticast":true,
478 * "flagBroadcast":true,
479 * "lanDelayEnabled":true,
480 * "226.10.10.10":{
481 * "source":"*",
482 * "group":"226.10.10.10",
483 * "localMembership":"INCLUDE"
484 * }
485 * }
486 * }
487 */
488
489 /* foreach interface */
490 json_object_object_foreach(json, key, val)
491 {
492
493 /* Find all of the keys where the val is an object. In
494 * the example
495 * above the only one is 226.10.10.10
496 */
497 json_object_object_foreach(val, if_field_key,
498 if_field_val)
499 {
500 type = json_object_get_type(if_field_val);
501
502 if (type == json_type_object) {
503 vty_out(vty, "%-16s ", key);
504
505 json_object_object_get_ex(
506 val, "address", &json_tmp);
507 vty_out(vty, "%-15s ",
508 json_object_get_string(
509 json_tmp));
510
511 json_object_object_get_ex(if_field_val,
512 "source",
513 &json_tmp);
514 vty_out(vty, "%-15s ",
515 json_object_get_string(
516 json_tmp));
517
518 /* Group */
519 vty_out(vty, "%-15s ", if_field_key);
520
521 json_object_object_get_ex(
522 if_field_val, "localMembership",
523 &json_tmp);
524 vty_out(vty, "%-10s\n",
525 json_object_get_string(
526 json_tmp));
527 }
528 }
529 }
530 }
531
532 json_object_free(json);
533 }
534
535 static void pim_print_ifp_flags(struct vty *vty, struct interface *ifp,
536 int mloop)
537 {
538 vty_out(vty, "Flags\n");
539 vty_out(vty, "-----\n");
540 vty_out(vty, "All Multicast : %s\n",
541 (ifp->flags & IFF_ALLMULTI) ? "yes" : "no");
542 vty_out(vty, "Broadcast : %s\n",
543 if_is_broadcast(ifp) ? "yes" : "no");
544 vty_out(vty, "Deleted : %s\n",
545 PIM_IF_IS_DELETED(ifp) ? "yes" : "no");
546 vty_out(vty, "Interface Index : %d\n", ifp->ifindex);
547 vty_out(vty, "Multicast : %s\n",
548 if_is_multicast(ifp) ? "yes" : "no");
549 vty_out(vty, "Multicast Loop : %d\n", mloop);
550 vty_out(vty, "Promiscuous : %s\n",
551 (ifp->flags & IFF_PROMISC) ? "yes" : "no");
552 vty_out(vty, "\n");
553 vty_out(vty, "\n");
554 }
555
556 static void igmp_show_interfaces(struct pim_instance *pim, struct vty *vty,
557 bool uj)
558 {
559 struct interface *ifp;
560 time_t now;
561 json_object *json = NULL;
562 json_object *json_row = NULL;
563
564 now = pim_time_monotonic_sec();
565
566 if (uj)
567 json = json_object_new_object();
568 else
569 vty_out(vty,
570 "Interface State Address V Querier Query Timer Uptime\n");
571
572 FOR_ALL_INTERFACES (pim->vrf, ifp) {
573 struct pim_interface *pim_ifp;
574 struct listnode *sock_node;
575 struct igmp_sock *igmp;
576
577 pim_ifp = ifp->info;
578
579 if (!pim_ifp)
580 continue;
581
582 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node,
583 igmp)) {
584 char uptime[10];
585 char query_hhmmss[10];
586
587 pim_time_uptime(uptime, sizeof(uptime),
588 now - igmp->sock_creation);
589 pim_time_timer_to_hhmmss(query_hhmmss,
590 sizeof(query_hhmmss),
591 igmp->t_igmp_query_timer);
592
593 if (uj) {
594 json_row = json_object_new_object();
595 json_object_pim_ifp_add(json_row, ifp);
596 json_object_string_add(json_row, "upTime",
597 uptime);
598 json_object_int_add(json_row, "version",
599 pim_ifp->igmp_version);
600
601 if (igmp->t_igmp_query_timer) {
602 json_object_boolean_true_add(json_row,
603 "querier");
604 json_object_string_add(json_row,
605 "queryTimer",
606 query_hhmmss);
607 }
608
609 json_object_object_add(json, ifp->name,
610 json_row);
611
612 if (igmp->mtrace_only) {
613 json_object_boolean_true_add(
614 json_row, "mtraceOnly");
615 }
616 } else {
617 vty_out(vty,
618 "%-16s %5s %15s %d %7s %11s %8s\n",
619 ifp->name,
620 if_is_up(ifp)
621 ? (igmp->mtrace_only ? "mtrc"
622 : "up")
623 : "down",
624 inet_ntoa(igmp->ifaddr),
625 pim_ifp->igmp_version,
626 igmp->t_igmp_query_timer ? "local"
627 : "other",
628 query_hhmmss, uptime);
629 }
630 }
631 }
632
633 if (uj) {
634 vty_out(vty, "%s\n", json_object_to_json_string_ext(
635 json, JSON_C_TO_STRING_PRETTY));
636 json_object_free(json);
637 }
638 }
639
640 static void igmp_show_interfaces_single(struct pim_instance *pim,
641 struct vty *vty, const char *ifname,
642 bool uj)
643 {
644 struct igmp_sock *igmp;
645 struct interface *ifp;
646 struct listnode *sock_node;
647 struct pim_interface *pim_ifp;
648 char uptime[10];
649 char query_hhmmss[10];
650 char other_hhmmss[10];
651 int found_ifname = 0;
652 int sqi;
653 int mloop = 0;
654 long gmi_msec; /* Group Membership Interval */
655 long lmqt_msec;
656 long ohpi_msec;
657 long oqpi_msec; /* Other Querier Present Interval */
658 long qri_msec;
659 time_t now;
660
661 json_object *json = NULL;
662 json_object *json_row = NULL;
663
664 if (uj)
665 json = json_object_new_object();
666
667 now = pim_time_monotonic_sec();
668
669 FOR_ALL_INTERFACES (pim->vrf, ifp) {
670 pim_ifp = ifp->info;
671
672 if (!pim_ifp)
673 continue;
674
675 if (strcmp(ifname, "detail") && strcmp(ifname, ifp->name))
676 continue;
677
678 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node,
679 igmp)) {
680 found_ifname = 1;
681 pim_time_uptime(uptime, sizeof(uptime),
682 now - igmp->sock_creation);
683 pim_time_timer_to_hhmmss(query_hhmmss,
684 sizeof(query_hhmmss),
685 igmp->t_igmp_query_timer);
686 pim_time_timer_to_hhmmss(other_hhmmss,
687 sizeof(other_hhmmss),
688 igmp->t_other_querier_timer);
689
690 gmi_msec = PIM_IGMP_GMI_MSEC(
691 igmp->querier_robustness_variable,
692 igmp->querier_query_interval,
693 pim_ifp->igmp_query_max_response_time_dsec);
694
695 sqi = PIM_IGMP_SQI(
696 pim_ifp->igmp_default_query_interval);
697
698 oqpi_msec = PIM_IGMP_OQPI_MSEC(
699 igmp->querier_robustness_variable,
700 igmp->querier_query_interval,
701 pim_ifp->igmp_query_max_response_time_dsec);
702
703 lmqt_msec = PIM_IGMP_LMQT_MSEC(
704 pim_ifp->igmp_query_max_response_time_dsec,
705 igmp->querier_robustness_variable);
706
707 ohpi_msec =
708 PIM_IGMP_OHPI_DSEC(
709 igmp->querier_robustness_variable,
710 igmp->querier_query_interval,
711 pim_ifp->igmp_query_max_response_time_dsec)
712 * 100;
713
714 qri_msec = pim_ifp->igmp_query_max_response_time_dsec
715 * 100;
716 if (pim_ifp->pim_sock_fd >= 0)
717 mloop = pim_socket_mcastloop_get(
718 pim_ifp->pim_sock_fd);
719 else
720 mloop = 0;
721
722 if (uj) {
723 json_row = json_object_new_object();
724 json_object_pim_ifp_add(json_row, ifp);
725 json_object_string_add(json_row, "upTime",
726 uptime);
727 json_object_string_add(json_row, "querier",
728 igmp->t_igmp_query_timer
729 ? "local"
730 : "other");
731 json_object_int_add(json_row, "queryStartCount",
732 igmp->startup_query_count);
733 json_object_string_add(json_row,
734 "queryQueryTimer",
735 query_hhmmss);
736 json_object_string_add(json_row,
737 "queryOtherTimer",
738 other_hhmmss);
739 json_object_int_add(json_row, "version",
740 pim_ifp->igmp_version);
741 json_object_int_add(
742 json_row,
743 "timerGroupMembershipIntervalMsec",
744 gmi_msec);
745 json_object_int_add(json_row,
746 "timerLastMemberQueryMsec",
747 lmqt_msec);
748 json_object_int_add(
749 json_row,
750 "timerOlderHostPresentIntervalMsec",
751 ohpi_msec);
752 json_object_int_add(
753 json_row,
754 "timerOtherQuerierPresentIntervalMsec",
755 oqpi_msec);
756 json_object_int_add(
757 json_row, "timerQueryInterval",
758 igmp->querier_query_interval);
759 json_object_int_add(
760 json_row,
761 "timerQueryResponseIntervalMsec",
762 qri_msec);
763 json_object_int_add(
764 json_row, "timerRobustnessVariable",
765 igmp->querier_robustness_variable);
766 json_object_int_add(json_row,
767 "timerStartupQueryInterval",
768 sqi);
769
770 json_object_object_add(json, ifp->name,
771 json_row);
772
773 if (igmp->mtrace_only) {
774 json_object_boolean_true_add(
775 json_row, "mtraceOnly");
776 }
777 } else {
778 vty_out(vty, "Interface : %s\n", ifp->name);
779 vty_out(vty, "State : %s\n",
780 if_is_up(ifp)
781 ? (igmp->mtrace_only ? "mtrace"
782 : "up")
783 : "down");
784 vty_out(vty, "Address : %s\n",
785 inet_ntoa(pim_ifp->primary_address));
786 vty_out(vty, "Uptime : %s\n", uptime);
787 vty_out(vty, "Version : %d\n",
788 pim_ifp->igmp_version);
789 vty_out(vty, "\n");
790 vty_out(vty, "\n");
791
792 vty_out(vty, "Querier\n");
793 vty_out(vty, "-------\n");
794 vty_out(vty, "Querier : %s\n",
795 igmp->t_igmp_query_timer ? "local"
796 : "other");
797 vty_out(vty, "Start Count : %d\n",
798 igmp->startup_query_count);
799 vty_out(vty, "Query Timer : %s\n",
800 query_hhmmss);
801 vty_out(vty, "Other Timer : %s\n",
802 other_hhmmss);
803 vty_out(vty, "\n");
804 vty_out(vty, "\n");
805
806 vty_out(vty, "Timers\n");
807 vty_out(vty, "------\n");
808 vty_out(vty,
809 "Group Membership Interval : %lis\n",
810 gmi_msec / 1000);
811 vty_out(vty,
812 "Last Member Query Time : %lis\n",
813 lmqt_msec / 1000);
814 vty_out(vty,
815 "Older Host Present Interval : %lis\n",
816 ohpi_msec / 1000);
817 vty_out(vty,
818 "Other Querier Present Interval : %lis\n",
819 oqpi_msec / 1000);
820 vty_out(vty,
821 "Query Interval : %ds\n",
822 igmp->querier_query_interval);
823 vty_out(vty,
824 "Query Response Interval : %lis\n",
825 qri_msec / 1000);
826 vty_out(vty,
827 "Robustness Variable : %d\n",
828 igmp->querier_robustness_variable);
829 vty_out(vty,
830 "Startup Query Interval : %ds\n",
831 sqi);
832 vty_out(vty, "\n");
833 vty_out(vty, "\n");
834
835 pim_print_ifp_flags(vty, ifp, mloop);
836 }
837 }
838 }
839
840 if (uj) {
841 vty_out(vty, "%s\n", json_object_to_json_string_ext(
842 json, JSON_C_TO_STRING_PRETTY));
843 json_object_free(json);
844 } else {
845 if (!found_ifname)
846 vty_out(vty, "%% No such interface\n");
847 }
848 }
849
850 static void igmp_show_interface_join(struct pim_instance *pim, struct vty *vty)
851 {
852 struct interface *ifp;
853 time_t now;
854
855 now = pim_time_monotonic_sec();
856
857 vty_out(vty,
858 "Interface Address Source Group Socket Uptime \n");
859
860 FOR_ALL_INTERFACES (pim->vrf, ifp) {
861 struct pim_interface *pim_ifp;
862 struct listnode *join_node;
863 struct igmp_join *ij;
864 struct in_addr pri_addr;
865 char pri_addr_str[INET_ADDRSTRLEN];
866
867 pim_ifp = ifp->info;
868
869 if (!pim_ifp)
870 continue;
871
872 if (!pim_ifp->igmp_join_list)
873 continue;
874
875 pri_addr = pim_find_primary_addr(ifp);
876 pim_inet4_dump("<pri?>", pri_addr, pri_addr_str,
877 sizeof(pri_addr_str));
878
879 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_join_list, join_node,
880 ij)) {
881 char group_str[INET_ADDRSTRLEN];
882 char source_str[INET_ADDRSTRLEN];
883 char uptime[10];
884
885 pim_time_uptime(uptime, sizeof(uptime),
886 now - ij->sock_creation);
887 pim_inet4_dump("<grp?>", ij->group_addr, group_str,
888 sizeof(group_str));
889 pim_inet4_dump("<src?>", ij->source_addr, source_str,
890 sizeof(source_str));
891
892 vty_out(vty, "%-16s %-15s %-15s %-15s %6d %8s\n",
893 ifp->name, pri_addr_str, source_str, group_str,
894 ij->sock_fd, uptime);
895 } /* for (pim_ifp->igmp_join_list) */
896
897 } /* for (iflist) */
898 }
899
900 static void pim_show_interfaces_single(struct pim_instance *pim,
901 struct vty *vty, const char *ifname,
902 bool uj)
903 {
904 struct in_addr ifaddr;
905 struct interface *ifp;
906 struct listnode *neighnode;
907 struct listnode *upnode;
908 struct pim_interface *pim_ifp;
909 struct pim_neighbor *neigh;
910 struct pim_upstream *up;
911 time_t now;
912 char dr_str[INET_ADDRSTRLEN];
913 char dr_uptime[10];
914 char expire[10];
915 char grp_str[INET_ADDRSTRLEN];
916 char hello_period[10];
917 char hello_timer[10];
918 char neigh_src_str[INET_ADDRSTRLEN];
919 char src_str[INET_ADDRSTRLEN];
920 char stat_uptime[10];
921 char uptime[10];
922 int mloop = 0;
923 int found_ifname = 0;
924 int print_header;
925 json_object *json = NULL;
926 json_object *json_row = NULL;
927 json_object *json_pim_neighbor = NULL;
928 json_object *json_pim_neighbors = NULL;
929 json_object *json_group = NULL;
930 json_object *json_group_source = NULL;
931 json_object *json_fhr_sources = NULL;
932 struct pim_secondary_addr *sec_addr;
933 struct listnode *sec_node;
934
935 now = pim_time_monotonic_sec();
936
937 if (uj)
938 json = json_object_new_object();
939
940 FOR_ALL_INTERFACES (pim->vrf, ifp) {
941 pim_ifp = ifp->info;
942
943 if (!pim_ifp)
944 continue;
945
946 if (strcmp(ifname, "detail") && strcmp(ifname, ifp->name))
947 continue;
948
949 found_ifname = 1;
950 ifaddr = pim_ifp->primary_address;
951 pim_inet4_dump("<dr?>", pim_ifp->pim_dr_addr, dr_str,
952 sizeof(dr_str));
953 pim_time_uptime_begin(dr_uptime, sizeof(dr_uptime), now,
954 pim_ifp->pim_dr_election_last);
955 pim_time_timer_to_hhmmss(hello_timer, sizeof(hello_timer),
956 pim_ifp->t_pim_hello_timer);
957 pim_time_mmss(hello_period, sizeof(hello_period),
958 pim_ifp->pim_hello_period);
959 pim_time_uptime(stat_uptime, sizeof(stat_uptime),
960 now - pim_ifp->pim_ifstat_start);
961 if (pim_ifp->pim_sock_fd >= 0)
962 mloop = pim_socket_mcastloop_get(pim_ifp->pim_sock_fd);
963 else
964 mloop = 0;
965
966 if (uj) {
967 char pbuf[PREFIX2STR_BUFFER];
968 json_row = json_object_new_object();
969 json_object_pim_ifp_add(json_row, ifp);
970
971 if (pim_ifp->update_source.s_addr != INADDR_ANY) {
972 json_object_string_add(
973 json_row, "useSource",
974 inet_ntoa(pim_ifp->update_source));
975 }
976 if (pim_ifp->sec_addr_list) {
977 json_object *sec_list = NULL;
978
979 sec_list = json_object_new_array();
980 for (ALL_LIST_ELEMENTS_RO(
981 pim_ifp->sec_addr_list, sec_node,
982 sec_addr)) {
983 json_object_array_add(
984 sec_list,
985 json_object_new_string(
986 prefix2str(
987 &sec_addr->addr,
988 pbuf,
989 sizeof(pbuf))));
990 }
991 json_object_object_add(json_row,
992 "secondaryAddressList",
993 sec_list);
994 }
995
996 // PIM neighbors
997 if (pim_ifp->pim_neighbor_list->count) {
998 json_pim_neighbors = json_object_new_object();
999
1000 for (ALL_LIST_ELEMENTS_RO(
1001 pim_ifp->pim_neighbor_list,
1002 neighnode, neigh)) {
1003 json_pim_neighbor =
1004 json_object_new_object();
1005 pim_inet4_dump("<src?>",
1006 neigh->source_addr,
1007 neigh_src_str,
1008 sizeof(neigh_src_str));
1009 pim_time_uptime(uptime, sizeof(uptime),
1010 now - neigh->creation);
1011 pim_time_timer_to_hhmmss(
1012 expire, sizeof(expire),
1013 neigh->t_expire_timer);
1014
1015 json_object_string_add(
1016 json_pim_neighbor, "address",
1017 neigh_src_str);
1018 json_object_string_add(
1019 json_pim_neighbor, "upTime",
1020 uptime);
1021 json_object_string_add(
1022 json_pim_neighbor, "holdtime",
1023 expire);
1024
1025 json_object_object_add(
1026 json_pim_neighbors,
1027 neigh_src_str,
1028 json_pim_neighbor);
1029 }
1030
1031 json_object_object_add(json_row, "neighbors",
1032 json_pim_neighbors);
1033 }
1034
1035 json_object_string_add(json_row, "drAddress", dr_str);
1036 json_object_int_add(json_row, "drPriority",
1037 pim_ifp->pim_dr_priority);
1038 json_object_string_add(json_row, "drUptime", dr_uptime);
1039 json_object_int_add(json_row, "drElections",
1040 pim_ifp->pim_dr_election_count);
1041 json_object_int_add(json_row, "drChanges",
1042 pim_ifp->pim_dr_election_changes);
1043
1044 // FHR
1045 for (ALL_LIST_ELEMENTS_RO(pim->upstream_list, upnode,
1046 up)) {
1047 if (ifp != up->rpf.source_nexthop.interface)
1048 continue;
1049
1050 if (!(up->flags & PIM_UPSTREAM_FLAG_MASK_FHR))
1051 continue;
1052
1053 if (!json_fhr_sources)
1054 json_fhr_sources =
1055 json_object_new_object();
1056
1057 pim_inet4_dump("<src?>", up->sg.src, src_str,
1058 sizeof(src_str));
1059 pim_inet4_dump("<grp?>", up->sg.grp, grp_str,
1060 sizeof(grp_str));
1061 pim_time_uptime(uptime, sizeof(uptime),
1062 now - up->state_transition);
1063
1064 /*
1065 * Does this group live in json_fhr_sources?
1066 * If not create it.
1067 */
1068 json_object_object_get_ex(json_fhr_sources,
1069 grp_str, &json_group);
1070
1071 if (!json_group) {
1072 json_group = json_object_new_object();
1073 json_object_object_add(json_fhr_sources,
1074 grp_str,
1075 json_group);
1076 }
1077
1078 json_group_source = json_object_new_object();
1079 json_object_string_add(json_group_source,
1080 "source", src_str);
1081 json_object_string_add(json_group_source,
1082 "group", grp_str);
1083 json_object_string_add(json_group_source,
1084 "upTime", uptime);
1085 json_object_object_add(json_group, src_str,
1086 json_group_source);
1087 }
1088
1089 if (json_fhr_sources) {
1090 json_object_object_add(json_row,
1091 "firstHopRouter",
1092 json_fhr_sources);
1093 }
1094
1095 json_object_int_add(json_row, "helloPeriod",
1096 pim_ifp->pim_hello_period);
1097 json_object_string_add(json_row, "helloTimer",
1098 hello_timer);
1099 json_object_string_add(json_row, "helloStatStart",
1100 stat_uptime);
1101 json_object_int_add(json_row, "helloReceived",
1102 pim_ifp->pim_ifstat_hello_recv);
1103 json_object_int_add(json_row, "helloReceivedFailed",
1104 pim_ifp->pim_ifstat_hello_recvfail);
1105 json_object_int_add(json_row, "helloSend",
1106 pim_ifp->pim_ifstat_hello_sent);
1107 json_object_int_add(json_row, "hellosendFailed",
1108 pim_ifp->pim_ifstat_hello_sendfail);
1109 json_object_int_add(json_row, "helloGenerationId",
1110 pim_ifp->pim_generation_id);
1111 json_object_int_add(json_row, "flagMulticastLoop",
1112 mloop);
1113
1114 json_object_int_add(
1115 json_row, "effectivePropagationDelay",
1116 pim_if_effective_propagation_delay_msec(ifp));
1117 json_object_int_add(
1118 json_row, "effectiveOverrideInterval",
1119 pim_if_effective_override_interval_msec(ifp));
1120 json_object_int_add(
1121 json_row, "joinPruneOverrideInterval",
1122 pim_if_jp_override_interval_msec(ifp));
1123
1124 json_object_int_add(
1125 json_row, "propagationDelay",
1126 pim_ifp->pim_propagation_delay_msec);
1127 json_object_int_add(
1128 json_row, "propagationDelayHighest",
1129 pim_ifp->pim_neighbors_highest_propagation_delay_msec);
1130 json_object_int_add(
1131 json_row, "overrideInterval",
1132 pim_ifp->pim_override_interval_msec);
1133 json_object_int_add(
1134 json_row, "overrideIntervalHighest",
1135 pim_ifp->pim_neighbors_highest_override_interval_msec);
1136 json_object_object_add(json, ifp->name, json_row);
1137
1138 } else {
1139 vty_out(vty, "Interface : %s\n", ifp->name);
1140 vty_out(vty, "State : %s\n",
1141 if_is_up(ifp) ? "up" : "down");
1142 if (pim_ifp->update_source.s_addr != INADDR_ANY) {
1143 vty_out(vty, "Use Source : %s\n",
1144 inet_ntoa(pim_ifp->update_source));
1145 }
1146 if (pim_ifp->sec_addr_list) {
1147 char pbuf[PREFIX2STR_BUFFER];
1148 vty_out(vty, "Address : %s (primary)\n",
1149 inet_ntoa(ifaddr));
1150 for (ALL_LIST_ELEMENTS_RO(
1151 pim_ifp->sec_addr_list, sec_node,
1152 sec_addr)) {
1153 vty_out(vty, " %s\n",
1154 prefix2str(&sec_addr->addr,
1155 pbuf, sizeof(pbuf)));
1156 }
1157 } else {
1158 vty_out(vty, "Address : %s\n",
1159 inet_ntoa(ifaddr));
1160 }
1161 vty_out(vty, "\n");
1162
1163 // PIM neighbors
1164 print_header = 1;
1165
1166 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list,
1167 neighnode, neigh)) {
1168
1169 if (print_header) {
1170 vty_out(vty, "PIM Neighbors\n");
1171 vty_out(vty, "-------------\n");
1172 print_header = 0;
1173 }
1174
1175 pim_inet4_dump("<src?>", neigh->source_addr,
1176 neigh_src_str,
1177 sizeof(neigh_src_str));
1178 pim_time_uptime(uptime, sizeof(uptime),
1179 now - neigh->creation);
1180 pim_time_timer_to_hhmmss(expire, sizeof(expire),
1181 neigh->t_expire_timer);
1182 vty_out(vty,
1183 "%-15s : up for %s, holdtime expires in %s\n",
1184 neigh_src_str, uptime, expire);
1185 }
1186
1187 if (!print_header) {
1188 vty_out(vty, "\n");
1189 vty_out(vty, "\n");
1190 }
1191
1192 vty_out(vty, "Designated Router\n");
1193 vty_out(vty, "-----------------\n");
1194 vty_out(vty, "Address : %s\n", dr_str);
1195 vty_out(vty, "Priority : %u(%d)\n",
1196 pim_ifp->pim_dr_priority,
1197 pim_ifp->pim_dr_num_nondrpri_neighbors);
1198 vty_out(vty, "Uptime : %s\n", dr_uptime);
1199 vty_out(vty, "Elections : %d\n",
1200 pim_ifp->pim_dr_election_count);
1201 vty_out(vty, "Changes : %d\n",
1202 pim_ifp->pim_dr_election_changes);
1203 vty_out(vty, "\n");
1204 vty_out(vty, "\n");
1205
1206 // FHR
1207 print_header = 1;
1208 for (ALL_LIST_ELEMENTS_RO(pim->upstream_list, upnode,
1209 up)) {
1210 if (!up->rpf.source_nexthop.interface)
1211 continue;
1212
1213 if (strcmp(ifp->name,
1214 up->rpf.source_nexthop
1215 .interface->name)
1216 != 0)
1217 continue;
1218
1219 if (!(up->flags & PIM_UPSTREAM_FLAG_MASK_FHR))
1220 continue;
1221
1222 if (print_header) {
1223 vty_out(vty,
1224 "FHR - First Hop Router\n");
1225 vty_out(vty,
1226 "----------------------\n");
1227 print_header = 0;
1228 }
1229
1230 pim_inet4_dump("<src?>", up->sg.src, src_str,
1231 sizeof(src_str));
1232 pim_inet4_dump("<grp?>", up->sg.grp, grp_str,
1233 sizeof(grp_str));
1234 pim_time_uptime(uptime, sizeof(uptime),
1235 now - up->state_transition);
1236 vty_out(vty,
1237 "%s : %s is a source, uptime is %s\n",
1238 grp_str, src_str, uptime);
1239 }
1240
1241 if (!print_header) {
1242 vty_out(vty, "\n");
1243 vty_out(vty, "\n");
1244 }
1245
1246 vty_out(vty, "Hellos\n");
1247 vty_out(vty, "------\n");
1248 vty_out(vty, "Period : %d\n",
1249 pim_ifp->pim_hello_period);
1250 vty_out(vty, "Timer : %s\n", hello_timer);
1251 vty_out(vty, "StatStart : %s\n", stat_uptime);
1252 vty_out(vty, "Receive : %d\n",
1253 pim_ifp->pim_ifstat_hello_recv);
1254 vty_out(vty, "Receive Failed : %d\n",
1255 pim_ifp->pim_ifstat_hello_recvfail);
1256 vty_out(vty, "Send : %d\n",
1257 pim_ifp->pim_ifstat_hello_sent);
1258 vty_out(vty, "Send Failed : %d\n",
1259 pim_ifp->pim_ifstat_hello_sendfail);
1260 vty_out(vty, "Generation ID : %08x\n",
1261 pim_ifp->pim_generation_id);
1262 vty_out(vty, "\n");
1263 vty_out(vty, "\n");
1264
1265 pim_print_ifp_flags(vty, ifp, mloop);
1266
1267 vty_out(vty, "Join Prune Interval\n");
1268 vty_out(vty, "-------------------\n");
1269 vty_out(vty, "LAN Delay : %s\n",
1270 pim_if_lan_delay_enabled(ifp) ? "yes" : "no");
1271 vty_out(vty, "Effective Propagation Delay : %d msec\n",
1272 pim_if_effective_propagation_delay_msec(ifp));
1273 vty_out(vty, "Effective Override Interval : %d msec\n",
1274 pim_if_effective_override_interval_msec(ifp));
1275 vty_out(vty, "Join Prune Override Interval : %d msec\n",
1276 pim_if_jp_override_interval_msec(ifp));
1277 vty_out(vty, "\n");
1278 vty_out(vty, "\n");
1279
1280 vty_out(vty, "LAN Prune Delay\n");
1281 vty_out(vty, "---------------\n");
1282 vty_out(vty, "Propagation Delay : %d msec\n",
1283 pim_ifp->pim_propagation_delay_msec);
1284 vty_out(vty, "Propagation Delay (Highest) : %d msec\n",
1285 pim_ifp->pim_neighbors_highest_propagation_delay_msec);
1286 vty_out(vty, "Override Interval : %d msec\n",
1287 pim_ifp->pim_override_interval_msec);
1288 vty_out(vty, "Override Interval (Highest) : %d msec\n",
1289 pim_ifp->pim_neighbors_highest_override_interval_msec);
1290 vty_out(vty, "\n");
1291 vty_out(vty, "\n");
1292 }
1293 }
1294
1295 if (uj) {
1296 vty_out(vty, "%s\n", json_object_to_json_string_ext(
1297 json, JSON_C_TO_STRING_PRETTY));
1298 json_object_free(json);
1299 } else {
1300 if (!found_ifname)
1301 vty_out(vty, "%% No such interface\n");
1302 }
1303 }
1304
1305 static void igmp_show_statistics(struct pim_instance *pim, struct vty *vty,
1306 const char *ifname, bool uj)
1307 {
1308 struct interface *ifp;
1309 struct igmp_stats rx_stats;
1310
1311 igmp_stats_init(&rx_stats);
1312
1313 FOR_ALL_INTERFACES (pim->vrf, ifp) {
1314 struct pim_interface *pim_ifp;
1315 struct listnode *sock_node;
1316 struct igmp_sock *igmp;
1317
1318 pim_ifp = ifp->info;
1319
1320 if (!pim_ifp)
1321 continue;
1322
1323 if (ifname && strcmp(ifname, ifp->name))
1324 continue;
1325
1326 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node,
1327 igmp)) {
1328 igmp_stats_add(&rx_stats, &igmp->rx_stats);
1329 }
1330 }
1331 if (uj) {
1332 json_object *json = NULL;
1333 json_object *json_row = NULL;
1334
1335 json = json_object_new_object();
1336 json_row = json_object_new_object();
1337
1338 json_object_string_add(json_row, "name", ifname ? ifname :
1339 "global");
1340 json_object_int_add(json_row, "queryV1", rx_stats.query_v1);
1341 json_object_int_add(json_row, "queryV2", rx_stats.query_v2);
1342 json_object_int_add(json_row, "queryV3", rx_stats.query_v3);
1343 json_object_int_add(json_row, "leaveV3", rx_stats.leave_v2);
1344 json_object_int_add(json_row, "reportV1", rx_stats.report_v1);
1345 json_object_int_add(json_row, "reportV2", rx_stats.report_v2);
1346 json_object_int_add(json_row, "reportV3", rx_stats.report_v3);
1347 json_object_int_add(json_row, "mtraceResponse",
1348 rx_stats.mtrace_rsp);
1349 json_object_int_add(json_row, "mtraceRequest",
1350 rx_stats.mtrace_req);
1351 json_object_int_add(json_row, "unsupported",
1352 rx_stats.unsupported);
1353 json_object_object_add(json, ifname ? ifname : "global",
1354 json_row);
1355 vty_out(vty, "%s\n", json_object_to_json_string_ext(
1356 json, JSON_C_TO_STRING_PRETTY));
1357 json_object_free(json);
1358 } else {
1359 vty_out(vty, "IGMP RX statistics\n");
1360 vty_out(vty, "Interface : %s\n",
1361 ifname ? ifname : "global");
1362 vty_out(vty, "V1 query : %u\n", rx_stats.query_v1);
1363 vty_out(vty, "V2 query : %u\n", rx_stats.query_v2);
1364 vty_out(vty, "V3 query : %u\n", rx_stats.query_v3);
1365 vty_out(vty, "V2 leave : %u\n", rx_stats.leave_v2);
1366 vty_out(vty, "V1 report : %u\n", rx_stats.report_v1);
1367 vty_out(vty, "V2 report : %u\n", rx_stats.report_v2);
1368 vty_out(vty, "V3 report : %u\n", rx_stats.report_v3);
1369 vty_out(vty, "mtrace response : %u\n", rx_stats.mtrace_rsp);
1370 vty_out(vty, "mtrace request : %u\n", rx_stats.mtrace_req);
1371 vty_out(vty, "unsupported : %u\n", rx_stats.unsupported);
1372 }
1373 }
1374
1375 static void pim_show_interfaces(struct pim_instance *pim, struct vty *vty,
1376 bool uj)
1377 {
1378 struct interface *ifp;
1379 struct listnode *upnode;
1380 struct pim_interface *pim_ifp;
1381 struct pim_upstream *up;
1382 int fhr = 0;
1383 int pim_nbrs = 0;
1384 int pim_ifchannels = 0;
1385 json_object *json = NULL;
1386 json_object *json_row = NULL;
1387 json_object *json_tmp;
1388
1389 json = json_object_new_object();
1390
1391 FOR_ALL_INTERFACES (pim->vrf, ifp) {
1392 pim_ifp = ifp->info;
1393
1394 if (!pim_ifp)
1395 continue;
1396
1397 pim_nbrs = pim_ifp->pim_neighbor_list->count;
1398 pim_ifchannels = pim_if_ifchannel_count(pim_ifp);
1399 fhr = 0;
1400
1401 for (ALL_LIST_ELEMENTS_RO(pim->upstream_list, upnode, up))
1402 if (ifp == up->rpf.source_nexthop.interface)
1403 if (up->flags & PIM_UPSTREAM_FLAG_MASK_FHR)
1404 fhr++;
1405
1406 json_row = json_object_new_object();
1407 json_object_pim_ifp_add(json_row, ifp);
1408 json_object_int_add(json_row, "pimNeighbors", pim_nbrs);
1409 json_object_int_add(json_row, "pimIfChannels", pim_ifchannels);
1410 json_object_int_add(json_row, "firstHopRouterCount", fhr);
1411 json_object_string_add(json_row, "pimDesignatedRouter",
1412 inet_ntoa(pim_ifp->pim_dr_addr));
1413
1414 if (pim_ifp->pim_dr_addr.s_addr
1415 == pim_ifp->primary_address.s_addr)
1416 json_object_boolean_true_add(
1417 json_row, "pimDesignatedRouterLocal");
1418
1419 json_object_object_add(json, ifp->name, json_row);
1420 }
1421
1422 if (uj) {
1423 vty_out(vty, "%s\n", json_object_to_json_string_ext(
1424 json, JSON_C_TO_STRING_PRETTY));
1425 } else {
1426 vty_out(vty,
1427 "Interface State Address PIM Nbrs PIM DR FHR IfChannels\n");
1428
1429 json_object_object_foreach(json, key, val)
1430 {
1431 vty_out(vty, "%-16s ", key);
1432
1433 json_object_object_get_ex(val, "state", &json_tmp);
1434 vty_out(vty, "%5s ", json_object_get_string(json_tmp));
1435
1436 json_object_object_get_ex(val, "address", &json_tmp);
1437 vty_out(vty, "%15s ",
1438 json_object_get_string(json_tmp));
1439
1440 json_object_object_get_ex(val, "pimNeighbors",
1441 &json_tmp);
1442 vty_out(vty, "%8d ", json_object_get_int(json_tmp));
1443
1444 if (json_object_object_get_ex(
1445 val, "pimDesignatedRouterLocal",
1446 &json_tmp)) {
1447 vty_out(vty, "%15s ", "local");
1448 } else {
1449 json_object_object_get_ex(
1450 val, "pimDesignatedRouter", &json_tmp);
1451 vty_out(vty, "%15s ",
1452 json_object_get_string(json_tmp));
1453 }
1454
1455 json_object_object_get_ex(val, "firstHopRouter",
1456 &json_tmp);
1457 vty_out(vty, "%3d ", json_object_get_int(json_tmp));
1458
1459 json_object_object_get_ex(val, "pimIfChannels",
1460 &json_tmp);
1461 vty_out(vty, "%9d\n", json_object_get_int(json_tmp));
1462 }
1463 }
1464
1465 json_object_free(json);
1466 }
1467
1468 static void pim_show_interface_traffic(struct pim_instance *pim,
1469 struct vty *vty, bool uj)
1470 {
1471 struct interface *ifp = NULL;
1472 struct pim_interface *pim_ifp = NULL;
1473 json_object *json = NULL;
1474 json_object *json_row = NULL;
1475
1476 if (uj)
1477 json = json_object_new_object();
1478 else {
1479 vty_out(vty, "\n");
1480 vty_out(vty, "%-16s%-17s%-17s%-17s%-17s%-17s%-17s\n",
1481 "Interface", " HELLO", " JOIN",
1482 " PRUNE", " REGISTER", "REGISTER-STOP",
1483 " ASSERT");
1484 vty_out(vty, "%-16s%-17s%-17s%-17s%-17s%-17s%-17s\n", "",
1485 " Rx/Tx", " Rx/Tx", " Rx/Tx",
1486 " Rx/Tx", " Rx/Tx", " Rx/Tx");
1487 vty_out(vty,
1488 "---------------------------------------------------------------------------------------------------------------\n");
1489 }
1490
1491 FOR_ALL_INTERFACES (pim->vrf, ifp) {
1492 pim_ifp = ifp->info;
1493
1494 if (!pim_ifp)
1495 continue;
1496
1497 if (pim_ifp->pim_sock_fd < 0)
1498 continue;
1499 if (uj) {
1500 json_row = json_object_new_object();
1501 json_object_pim_ifp_add(json_row, ifp);
1502 json_object_int_add(json_row, "helloRx",
1503 pim_ifp->pim_ifstat_hello_recv);
1504 json_object_int_add(json_row, "helloTx",
1505 pim_ifp->pim_ifstat_hello_sent);
1506 json_object_int_add(json_row, "joinRx",
1507 pim_ifp->pim_ifstat_join_recv);
1508 json_object_int_add(json_row, "joinTx",
1509 pim_ifp->pim_ifstat_join_send);
1510 json_object_int_add(json_row, "registerRx",
1511 pim_ifp->pim_ifstat_reg_recv);
1512 json_object_int_add(json_row, "registerTx",
1513 pim_ifp->pim_ifstat_reg_recv);
1514 json_object_int_add(json_row, "registerStopRx",
1515 pim_ifp->pim_ifstat_reg_stop_recv);
1516 json_object_int_add(json_row, "registerStopTx",
1517 pim_ifp->pim_ifstat_reg_stop_send);
1518 json_object_int_add(json_row, "assertRx",
1519 pim_ifp->pim_ifstat_assert_recv);
1520 json_object_int_add(json_row, "assertTx",
1521 pim_ifp->pim_ifstat_assert_send);
1522
1523 json_object_object_add(json, ifp->name, json_row);
1524 } else {
1525 vty_out(vty,
1526 "%-16s %8u/%-8u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u \n",
1527 ifp->name, pim_ifp->pim_ifstat_hello_recv,
1528 pim_ifp->pim_ifstat_hello_sent,
1529 pim_ifp->pim_ifstat_join_recv,
1530 pim_ifp->pim_ifstat_join_send,
1531 pim_ifp->pim_ifstat_prune_recv,
1532 pim_ifp->pim_ifstat_prune_send,
1533 pim_ifp->pim_ifstat_reg_recv,
1534 pim_ifp->pim_ifstat_reg_send,
1535 pim_ifp->pim_ifstat_reg_stop_recv,
1536 pim_ifp->pim_ifstat_reg_stop_send,
1537 pim_ifp->pim_ifstat_assert_recv,
1538 pim_ifp->pim_ifstat_assert_send);
1539 }
1540 }
1541 if (uj) {
1542 vty_out(vty, "%s\n", json_object_to_json_string_ext(
1543 json, JSON_C_TO_STRING_PRETTY));
1544 json_object_free(json);
1545 }
1546 }
1547
1548 static void pim_show_interface_traffic_single(struct pim_instance *pim,
1549 struct vty *vty,
1550 const char *ifname, bool uj)
1551 {
1552 struct interface *ifp = NULL;
1553 struct pim_interface *pim_ifp = NULL;
1554 json_object *json = NULL;
1555 json_object *json_row = NULL;
1556 uint8_t found_ifname = 0;
1557
1558 if (uj)
1559 json = json_object_new_object();
1560 else {
1561 vty_out(vty, "\n");
1562 vty_out(vty, "%-16s%-17s%-17s%-17s%-17s%-17s%-17s\n",
1563 "Interface", " HELLO", " JOIN", " PRUNE",
1564 " REGISTER", " REGISTER-STOP", " ASSERT");
1565 vty_out(vty, "%-14s%-18s%-17s%-17s%-17s%-17s%-17s\n", "",
1566 " Rx/Tx", " Rx/Tx", " Rx/Tx", " Rx/Tx",
1567 " Rx/Tx", " Rx/Tx");
1568 vty_out(vty,
1569 "---------------------------------------------------------------------------------------------------------------------\n");
1570 }
1571
1572 FOR_ALL_INTERFACES (pim->vrf, ifp) {
1573 if (strcmp(ifname, ifp->name))
1574 continue;
1575
1576 pim_ifp = ifp->info;
1577
1578 if (!pim_ifp)
1579 continue;
1580
1581 if (pim_ifp->pim_sock_fd < 0)
1582 continue;
1583
1584 found_ifname = 1;
1585 if (uj) {
1586 json_row = json_object_new_object();
1587 json_object_pim_ifp_add(json_row, ifp);
1588 json_object_int_add(json_row, "helloRx",
1589 pim_ifp->pim_ifstat_hello_recv);
1590 json_object_int_add(json_row, "helloTx",
1591 pim_ifp->pim_ifstat_hello_sent);
1592 json_object_int_add(json_row, "joinRx",
1593 pim_ifp->pim_ifstat_join_recv);
1594 json_object_int_add(json_row, "joinTx",
1595 pim_ifp->pim_ifstat_join_send);
1596 json_object_int_add(json_row, "registerRx",
1597 pim_ifp->pim_ifstat_reg_recv);
1598 json_object_int_add(json_row, "registerTx",
1599 pim_ifp->pim_ifstat_reg_recv);
1600 json_object_int_add(json_row, "registerStopRx",
1601 pim_ifp->pim_ifstat_reg_stop_recv);
1602 json_object_int_add(json_row, "registerStopTx",
1603 pim_ifp->pim_ifstat_reg_stop_send);
1604 json_object_int_add(json_row, "assertRx",
1605 pim_ifp->pim_ifstat_assert_recv);
1606 json_object_int_add(json_row, "assertTx",
1607 pim_ifp->pim_ifstat_assert_send);
1608
1609 json_object_object_add(json, ifp->name, json_row);
1610 } else {
1611 vty_out(vty,
1612 "%-16s %8u/%-8u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u \n",
1613 ifp->name, pim_ifp->pim_ifstat_hello_recv,
1614 pim_ifp->pim_ifstat_hello_sent,
1615 pim_ifp->pim_ifstat_join_recv,
1616 pim_ifp->pim_ifstat_join_send,
1617 pim_ifp->pim_ifstat_prune_recv,
1618 pim_ifp->pim_ifstat_prune_send,
1619 pim_ifp->pim_ifstat_reg_recv,
1620 pim_ifp->pim_ifstat_reg_send,
1621 pim_ifp->pim_ifstat_reg_stop_recv,
1622 pim_ifp->pim_ifstat_reg_stop_send,
1623 pim_ifp->pim_ifstat_assert_recv,
1624 pim_ifp->pim_ifstat_assert_send);
1625 }
1626 }
1627 if (uj) {
1628 vty_out(vty, "%s\n", json_object_to_json_string_ext(
1629 json, JSON_C_TO_STRING_PRETTY));
1630 json_object_free(json);
1631 } else {
1632 if (!found_ifname)
1633 vty_out(vty, "%% No such interface\n");
1634 }
1635 }
1636
1637 static void pim_show_join_helper(struct vty *vty, struct pim_interface *pim_ifp,
1638 struct pim_ifchannel *ch, json_object *json,
1639 time_t now, bool uj)
1640 {
1641 char ch_src_str[INET_ADDRSTRLEN];
1642 char ch_grp_str[INET_ADDRSTRLEN];
1643 json_object *json_iface = NULL;
1644 json_object *json_row = NULL;
1645 json_object *json_grp = NULL;
1646 struct in_addr ifaddr;
1647 char uptime[10];
1648 char expire[10];
1649 char prune[10];
1650
1651 ifaddr = pim_ifp->primary_address;
1652
1653 pim_inet4_dump("<ch_src?>", ch->sg.src, ch_src_str, sizeof(ch_src_str));
1654 pim_inet4_dump("<ch_grp?>", ch->sg.grp, ch_grp_str, sizeof(ch_grp_str));
1655
1656 pim_time_uptime_begin(uptime, sizeof(uptime), now, ch->ifjoin_creation);
1657 pim_time_timer_to_mmss(expire, sizeof(expire),
1658 ch->t_ifjoin_expiry_timer);
1659 pim_time_timer_to_mmss(prune, sizeof(prune),
1660 ch->t_ifjoin_prune_pending_timer);
1661
1662 if (uj) {
1663 json_object_object_get_ex(json, ch->interface->name,
1664 &json_iface);
1665
1666 if (!json_iface) {
1667 json_iface = json_object_new_object();
1668 json_object_pim_ifp_add(json_iface, ch->interface);
1669 json_object_object_add(json, ch->interface->name,
1670 json_iface);
1671 }
1672
1673 json_row = json_object_new_object();
1674 json_object_string_add(json_row, "source", ch_src_str);
1675 json_object_string_add(json_row, "group", ch_grp_str);
1676 json_object_string_add(json_row, "upTime", uptime);
1677 json_object_string_add(json_row, "expire", expire);
1678 json_object_string_add(json_row, "prune", prune);
1679 json_object_string_add(
1680 json_row, "channelJoinName",
1681 pim_ifchannel_ifjoin_name(ch->ifjoin_state, ch->flags));
1682 if (PIM_IF_FLAG_TEST_S_G_RPT(ch->flags))
1683 json_object_int_add(json_row, "SGRpt", 1);
1684
1685 json_object_object_get_ex(json_iface, ch_grp_str, &json_grp);
1686 if (!json_grp) {
1687 json_grp = json_object_new_object();
1688 json_object_object_add(json_grp, ch_src_str, json_row);
1689 json_object_object_add(json_iface, ch_grp_str,
1690 json_grp);
1691 } else
1692 json_object_object_add(json_grp, ch_src_str, json_row);
1693 } else {
1694 vty_out(vty, "%-16s %-15s %-15s %-15s %-10s %8s %-6s %5s\n",
1695 ch->interface->name, inet_ntoa(ifaddr), ch_src_str,
1696 ch_grp_str,
1697 pim_ifchannel_ifjoin_name(ch->ifjoin_state, ch->flags),
1698 uptime, expire, prune);
1699 }
1700 }
1701
1702 static void pim_show_join(struct pim_instance *pim, struct vty *vty,
1703 struct prefix_sg *sg, bool uj)
1704 {
1705 struct pim_interface *pim_ifp;
1706 struct pim_ifchannel *ch;
1707 struct interface *ifp;
1708 time_t now;
1709 json_object *json = NULL;
1710
1711 now = pim_time_monotonic_sec();
1712
1713 if (uj)
1714 json = json_object_new_object();
1715 else
1716 vty_out(vty,
1717 "Interface Address Source Group State Uptime Expire Prune\n");
1718
1719 FOR_ALL_INTERFACES (pim->vrf, ifp) {
1720 pim_ifp = ifp->info;
1721 if (!pim_ifp)
1722 continue;
1723
1724 RB_FOREACH (ch, pim_ifchannel_rb, &pim_ifp->ifchannel_rb) {
1725 if (sg->grp.s_addr != 0
1726 && sg->grp.s_addr != ch->sg.grp.s_addr)
1727 continue;
1728 if (sg->src.s_addr != 0
1729 && sg->src.s_addr != ch->sg.src.s_addr)
1730 continue;
1731 pim_show_join_helper(vty, pim_ifp, ch, json, now, uj);
1732 } /* scan interface channels */
1733 }
1734
1735 if (uj) {
1736 vty_out(vty, "%s\n", json_object_to_json_string_ext(
1737 json, JSON_C_TO_STRING_PRETTY));
1738 json_object_free(json);
1739 }
1740 }
1741
1742 static void pim_show_neighbors_single(struct pim_instance *pim, struct vty *vty,
1743 const char *neighbor, bool uj)
1744 {
1745 struct listnode *neighnode;
1746 struct interface *ifp;
1747 struct pim_interface *pim_ifp;
1748 struct pim_neighbor *neigh;
1749 time_t now;
1750 int found_neighbor = 0;
1751 int option_address_list;
1752 int option_dr_priority;
1753 int option_generation_id;
1754 int option_holdtime;
1755 int option_lan_prune_delay;
1756 int option_t_bit;
1757 char uptime[10];
1758 char expire[10];
1759 char neigh_src_str[INET_ADDRSTRLEN];
1760
1761 json_object *json = NULL;
1762 json_object *json_ifp = NULL;
1763 json_object *json_row = NULL;
1764
1765 now = pim_time_monotonic_sec();
1766
1767 if (uj)
1768 json = json_object_new_object();
1769
1770 FOR_ALL_INTERFACES (pim->vrf, ifp) {
1771 pim_ifp = ifp->info;
1772
1773 if (!pim_ifp)
1774 continue;
1775
1776 if (pim_ifp->pim_sock_fd < 0)
1777 continue;
1778
1779 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list, neighnode,
1780 neigh)) {
1781 pim_inet4_dump("<src?>", neigh->source_addr,
1782 neigh_src_str, sizeof(neigh_src_str));
1783
1784 /*
1785 * The user can specify either the interface name or the
1786 * PIM neighbor IP.
1787 * If this pim_ifp matches neither then skip.
1788 */
1789 if (strcmp(neighbor, "detail")
1790 && strcmp(neighbor, ifp->name)
1791 && strcmp(neighbor, neigh_src_str))
1792 continue;
1793
1794 found_neighbor = 1;
1795 pim_time_uptime(uptime, sizeof(uptime),
1796 now - neigh->creation);
1797 pim_time_timer_to_hhmmss(expire, sizeof(expire),
1798 neigh->t_expire_timer);
1799
1800 option_address_list = 0;
1801 option_dr_priority = 0;
1802 option_generation_id = 0;
1803 option_holdtime = 0;
1804 option_lan_prune_delay = 0;
1805 option_t_bit = 0;
1806
1807 if (PIM_OPTION_IS_SET(neigh->hello_options,
1808 PIM_OPTION_MASK_ADDRESS_LIST))
1809 option_address_list = 1;
1810
1811 if (PIM_OPTION_IS_SET(neigh->hello_options,
1812 PIM_OPTION_MASK_DR_PRIORITY))
1813 option_dr_priority = 1;
1814
1815 if (PIM_OPTION_IS_SET(neigh->hello_options,
1816 PIM_OPTION_MASK_GENERATION_ID))
1817 option_generation_id = 1;
1818
1819 if (PIM_OPTION_IS_SET(neigh->hello_options,
1820 PIM_OPTION_MASK_HOLDTIME))
1821 option_holdtime = 1;
1822
1823 if (PIM_OPTION_IS_SET(neigh->hello_options,
1824 PIM_OPTION_MASK_LAN_PRUNE_DELAY))
1825 option_lan_prune_delay = 1;
1826
1827 if (PIM_OPTION_IS_SET(
1828 neigh->hello_options,
1829 PIM_OPTION_MASK_CAN_DISABLE_JOIN_SUPPRESSION))
1830 option_t_bit = 1;
1831
1832 if (uj) {
1833
1834 /* Does this ifp live in json? If not create
1835 * it. */
1836 json_object_object_get_ex(json, ifp->name,
1837 &json_ifp);
1838
1839 if (!json_ifp) {
1840 json_ifp = json_object_new_object();
1841 json_object_pim_ifp_add(json_ifp, ifp);
1842 json_object_object_add(json, ifp->name,
1843 json_ifp);
1844 }
1845
1846 json_row = json_object_new_object();
1847 json_object_string_add(json_row, "interface",
1848 ifp->name);
1849 json_object_string_add(json_row, "address",
1850 neigh_src_str);
1851 json_object_string_add(json_row, "upTime",
1852 uptime);
1853 json_object_string_add(json_row, "holdtime",
1854 expire);
1855 json_object_int_add(json_row, "drPriority",
1856 neigh->dr_priority);
1857 json_object_int_add(json_row, "generationId",
1858 neigh->generation_id);
1859
1860 if (option_address_list)
1861 json_object_boolean_true_add(
1862 json_row,
1863 "helloOptionAddressList");
1864
1865 if (option_dr_priority)
1866 json_object_boolean_true_add(
1867 json_row,
1868 "helloOptionDrPriority");
1869
1870 if (option_generation_id)
1871 json_object_boolean_true_add(
1872 json_row,
1873 "helloOptionGenerationId");
1874
1875 if (option_holdtime)
1876 json_object_boolean_true_add(
1877 json_row,
1878 "helloOptionHoldtime");
1879
1880 if (option_lan_prune_delay)
1881 json_object_boolean_true_add(
1882 json_row,
1883 "helloOptionLanPruneDelay");
1884
1885 if (option_t_bit)
1886 json_object_boolean_true_add(
1887 json_row, "helloOptionTBit");
1888
1889 json_object_object_add(json_ifp, neigh_src_str,
1890 json_row);
1891
1892 } else {
1893 vty_out(vty, "Interface : %s\n", ifp->name);
1894 vty_out(vty, "Neighbor : %s\n", neigh_src_str);
1895 vty_out(vty,
1896 " Uptime : %s\n",
1897 uptime);
1898 vty_out(vty,
1899 " Holdtime : %s\n",
1900 expire);
1901 vty_out(vty,
1902 " DR Priority : %d\n",
1903 neigh->dr_priority);
1904 vty_out(vty,
1905 " Generation ID : %08x\n",
1906 neigh->generation_id);
1907 vty_out(vty,
1908 " Override Interval (msec) : %d\n",
1909 neigh->override_interval_msec);
1910 vty_out(vty,
1911 " Propagation Delay (msec) : %d\n",
1912 neigh->propagation_delay_msec);
1913 vty_out(vty,
1914 " Hello Option - Address List : %s\n",
1915 option_address_list ? "yes" : "no");
1916 vty_out(vty,
1917 " Hello Option - DR Priority : %s\n",
1918 option_dr_priority ? "yes" : "no");
1919 vty_out(vty,
1920 " Hello Option - Generation ID : %s\n",
1921 option_generation_id ? "yes" : "no");
1922 vty_out(vty,
1923 " Hello Option - Holdtime : %s\n",
1924 option_holdtime ? "yes" : "no");
1925 vty_out(vty,
1926 " Hello Option - LAN Prune Delay : %s\n",
1927 option_lan_prune_delay ? "yes" : "no");
1928 vty_out(vty,
1929 " Hello Option - T-bit : %s\n",
1930 option_t_bit ? "yes" : "no");
1931 pim_bfd_show_info(vty, neigh->bfd_info,
1932 json_ifp, uj, 0);
1933 vty_out(vty, "\n");
1934 }
1935 }
1936 }
1937
1938 if (uj) {
1939 vty_out(vty, "%s\n", json_object_to_json_string_ext(
1940 json, JSON_C_TO_STRING_PRETTY));
1941 json_object_free(json);
1942 } else {
1943 {
1944 if (!found_neighbor)
1945 vty_out(vty,
1946 "%% No such interface or neighbor\n");
1947 }
1948 }
1949 }
1950
1951 static void pim_show_state(struct pim_instance *pim, struct vty *vty,
1952 const char *src_or_group, const char *group, bool uj)
1953 {
1954 struct channel_oil *c_oil;
1955 struct listnode *node;
1956 json_object *json = NULL;
1957 json_object *json_group = NULL;
1958 json_object *json_ifp_in = NULL;
1959 json_object *json_ifp_out = NULL;
1960 json_object *json_source = NULL;
1961 time_t now;
1962 int first_oif;
1963 now = pim_time_monotonic_sec();
1964
1965 if (uj) {
1966 json = json_object_new_object();
1967 } else {
1968 vty_out(vty,
1969 "Codes: J -> Pim Join, I -> IGMP Report, S -> Source, * -> Inherited from (*,G), V -> VxLAN");
1970 vty_out(vty,
1971 "\nInstalled Source Group IIF OIL\n");
1972 }
1973
1974 for (ALL_LIST_ELEMENTS_RO(pim->channel_oil_list, node, c_oil)) {
1975 char grp_str[INET_ADDRSTRLEN];
1976 char src_str[INET_ADDRSTRLEN];
1977 char in_ifname[INTERFACE_NAMSIZ + 1];
1978 char out_ifname[INTERFACE_NAMSIZ + 1];
1979 int oif_vif_index;
1980 struct interface *ifp_in;
1981 first_oif = 1;
1982
1983 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp, grp_str,
1984 sizeof(grp_str));
1985 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin, src_str,
1986 sizeof(src_str));
1987 ifp_in = pim_if_find_by_vif_index(pim, c_oil->oil.mfcc_parent);
1988
1989 if (ifp_in)
1990 strcpy(in_ifname, ifp_in->name);
1991 else
1992 strcpy(in_ifname, "<iif?>");
1993
1994 if (src_or_group) {
1995 if (strcmp(src_or_group, src_str)
1996 && strcmp(src_or_group, grp_str))
1997 continue;
1998
1999 if (group && strcmp(group, grp_str))
2000 continue;
2001 }
2002
2003 if (uj) {
2004
2005 /* Find the group, create it if it doesn't exist */
2006 json_object_object_get_ex(json, grp_str, &json_group);
2007
2008 if (!json_group) {
2009 json_group = json_object_new_object();
2010 json_object_object_add(json, grp_str,
2011 json_group);
2012 }
2013
2014 /* Find the source nested under the group, create it if
2015 * it doesn't exist */
2016 json_object_object_get_ex(json_group, src_str,
2017 &json_source);
2018
2019 if (!json_source) {
2020 json_source = json_object_new_object();
2021 json_object_object_add(json_group, src_str,
2022 json_source);
2023 }
2024
2025 /* Find the inbound interface nested under the source,
2026 * create it if it doesn't exist */
2027 json_object_object_get_ex(json_source, in_ifname,
2028 &json_ifp_in);
2029
2030 if (!json_ifp_in) {
2031 json_ifp_in = json_object_new_object();
2032 json_object_object_add(json_source, in_ifname,
2033 json_ifp_in);
2034 json_object_int_add(json_source, "Installed",
2035 c_oil->installed);
2036 json_object_int_add(json_source, "RefCount",
2037 c_oil->oil_ref_count);
2038 json_object_int_add(json_source, "OilListSize",
2039 c_oil->oil_size);
2040 json_object_int_add(
2041 json_source, "OilRescan",
2042 c_oil->oil_inherited_rescan);
2043 json_object_int_add(json_source, "LastUsed",
2044 c_oil->cc.lastused);
2045 json_object_int_add(json_source, "PacketCount",
2046 c_oil->cc.pktcnt);
2047 json_object_int_add(json_source, "ByteCount",
2048 c_oil->cc.bytecnt);
2049 json_object_int_add(json_source,
2050 "WrongInterface",
2051 c_oil->cc.wrong_if);
2052 }
2053 } else {
2054 vty_out(vty, "%-9d %-15s %-15s %-16s ",
2055 c_oil->installed, src_str, grp_str, in_ifname);
2056 }
2057
2058 for (oif_vif_index = 0; oif_vif_index < MAXVIFS;
2059 ++oif_vif_index) {
2060 struct interface *ifp_out;
2061 char oif_uptime[10];
2062 int ttl;
2063
2064 ttl = c_oil->oil.mfcc_ttls[oif_vif_index];
2065 if (ttl < 1)
2066 continue;
2067
2068 ifp_out = pim_if_find_by_vif_index(pim, oif_vif_index);
2069 pim_time_uptime(
2070 oif_uptime, sizeof(oif_uptime),
2071 now - c_oil->oif_creation[oif_vif_index]);
2072
2073 if (ifp_out)
2074 strcpy(out_ifname, ifp_out->name);
2075 else
2076 strcpy(out_ifname, "<oif?>");
2077
2078 if (uj) {
2079 json_ifp_out = json_object_new_object();
2080 json_object_string_add(json_ifp_out, "source",
2081 src_str);
2082 json_object_string_add(json_ifp_out, "group",
2083 grp_str);
2084 json_object_string_add(json_ifp_out,
2085 "inboundInterface",
2086 in_ifname);
2087 json_object_string_add(json_ifp_out,
2088 "outboundInterface",
2089 out_ifname);
2090 json_object_int_add(json_ifp_out, "installed",
2091 c_oil->installed);
2092
2093 json_object_object_add(json_ifp_in, out_ifname,
2094 json_ifp_out);
2095 } else {
2096 if (first_oif) {
2097 first_oif = 0;
2098 vty_out(vty, "%s(%c%c%c%c%c)", out_ifname,
2099 (c_oil->oif_flags[oif_vif_index]
2100 & PIM_OIF_FLAG_PROTO_IGMP)
2101 ? 'I'
2102 : ' ',
2103 (c_oil->oif_flags[oif_vif_index]
2104 & PIM_OIF_FLAG_PROTO_PIM)
2105 ? 'J'
2106 : ' ',
2107 (c_oil->oif_flags[oif_vif_index]
2108 & PIM_OIF_FLAG_PROTO_VXLAN)
2109 ? 'V'
2110 : ' ',
2111 (c_oil->oif_flags[oif_vif_index]
2112 & PIM_OIF_FLAG_PROTO_SOURCE)
2113 ? 'S'
2114 : ' ',
2115 (c_oil->oif_flags[oif_vif_index]
2116 & PIM_OIF_FLAG_PROTO_STAR)
2117 ? '*'
2118 : ' ');
2119 } else
2120 vty_out(vty, ", %s(%c%c%c%c%c)",
2121 out_ifname,
2122 (c_oil->oif_flags[oif_vif_index]
2123 & PIM_OIF_FLAG_PROTO_IGMP)
2124 ? 'I'
2125 : ' ',
2126 (c_oil->oif_flags[oif_vif_index]
2127 & PIM_OIF_FLAG_PROTO_PIM)
2128 ? 'J'
2129 : ' ',
2130 (c_oil->oif_flags[oif_vif_index]
2131 & PIM_OIF_FLAG_PROTO_VXLAN)
2132 ? 'V'
2133 : ' ',
2134 (c_oil->oif_flags[oif_vif_index]
2135 & PIM_OIF_FLAG_PROTO_SOURCE)
2136 ? 'S'
2137 : ' ',
2138 (c_oil->oif_flags[oif_vif_index]
2139 & PIM_OIF_FLAG_PROTO_STAR)
2140 ? '*'
2141 : ' ');
2142 }
2143 }
2144
2145 if (!uj)
2146 vty_out(vty, "\n");
2147 }
2148
2149
2150 if (uj) {
2151 vty_out(vty, "%s\n", json_object_to_json_string_ext(
2152 json, JSON_C_TO_STRING_PRETTY));
2153 json_object_free(json);
2154 } else {
2155 vty_out(vty, "\n");
2156 }
2157 }
2158
2159 static void pim_show_neighbors(struct pim_instance *pim, struct vty *vty,
2160 bool uj)
2161 {
2162 struct listnode *neighnode;
2163 struct interface *ifp;
2164 struct pim_interface *pim_ifp;
2165 struct pim_neighbor *neigh;
2166 time_t now;
2167 char uptime[10];
2168 char expire[10];
2169 char neigh_src_str[INET_ADDRSTRLEN];
2170 json_object *json = NULL;
2171 json_object *json_ifp_rows = NULL;
2172 json_object *json_row = NULL;
2173
2174 now = pim_time_monotonic_sec();
2175
2176 if (uj) {
2177 json = json_object_new_object();
2178 } else {
2179 vty_out(vty,
2180 "Interface Neighbor Uptime Holdtime DR Pri\n");
2181 }
2182
2183 FOR_ALL_INTERFACES (pim->vrf, ifp) {
2184 pim_ifp = ifp->info;
2185
2186 if (!pim_ifp)
2187 continue;
2188
2189 if (pim_ifp->pim_sock_fd < 0)
2190 continue;
2191
2192 if (uj)
2193 json_ifp_rows = json_object_new_object();
2194
2195 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list, neighnode,
2196 neigh)) {
2197 pim_inet4_dump("<src?>", neigh->source_addr,
2198 neigh_src_str, sizeof(neigh_src_str));
2199 pim_time_uptime(uptime, sizeof(uptime),
2200 now - neigh->creation);
2201 pim_time_timer_to_hhmmss(expire, sizeof(expire),
2202 neigh->t_expire_timer);
2203
2204 if (uj) {
2205 json_row = json_object_new_object();
2206 json_object_string_add(json_row, "interface",
2207 ifp->name);
2208 json_object_string_add(json_row, "neighbor",
2209 neigh_src_str);
2210 json_object_string_add(json_row, "upTime",
2211 uptime);
2212 json_object_string_add(json_row, "holdTime",
2213 expire);
2214 json_object_int_add(json_row, "holdTimeMax",
2215 neigh->holdtime);
2216 json_object_int_add(json_row, "drPriority",
2217 neigh->dr_priority);
2218 json_object_object_add(json_ifp_rows,
2219 neigh_src_str, json_row);
2220
2221 } else {
2222 vty_out(vty, "%-16s %15s %8s %8s %6d\n",
2223 ifp->name, neigh_src_str, uptime,
2224 expire, neigh->dr_priority);
2225 }
2226 }
2227
2228 if (uj) {
2229 json_object_object_add(json, ifp->name, json_ifp_rows);
2230 json_ifp_rows = NULL;
2231 }
2232 }
2233
2234 if (uj) {
2235 vty_out(vty, "%s\n", json_object_to_json_string_ext(
2236 json, JSON_C_TO_STRING_PRETTY));
2237 json_object_free(json);
2238 }
2239 }
2240
2241 static void pim_show_neighbors_secondary(struct pim_instance *pim,
2242 struct vty *vty)
2243 {
2244 struct interface *ifp;
2245
2246 vty_out(vty,
2247 "Interface Address Neighbor Secondary \n");
2248
2249 FOR_ALL_INTERFACES (pim->vrf, ifp) {
2250 struct pim_interface *pim_ifp;
2251 struct in_addr ifaddr;
2252 struct listnode *neighnode;
2253 struct pim_neighbor *neigh;
2254
2255 pim_ifp = ifp->info;
2256
2257 if (!pim_ifp)
2258 continue;
2259
2260 if (pim_ifp->pim_sock_fd < 0)
2261 continue;
2262
2263 ifaddr = pim_ifp->primary_address;
2264
2265 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list, neighnode,
2266 neigh)) {
2267 char neigh_src_str[INET_ADDRSTRLEN];
2268 struct listnode *prefix_node;
2269 struct prefix *p;
2270
2271 if (!neigh->prefix_list)
2272 continue;
2273
2274 pim_inet4_dump("<src?>", neigh->source_addr,
2275 neigh_src_str, sizeof(neigh_src_str));
2276
2277 for (ALL_LIST_ELEMENTS_RO(neigh->prefix_list,
2278 prefix_node, p)) {
2279 char neigh_sec_str[PREFIX2STR_BUFFER];
2280
2281 prefix2str(p, neigh_sec_str,
2282 sizeof(neigh_sec_str));
2283
2284 vty_out(vty, "%-16s %-15s %-15s %-15s\n",
2285 ifp->name, inet_ntoa(ifaddr),
2286 neigh_src_str, neigh_sec_str);
2287 }
2288 }
2289 }
2290 }
2291
2292 static void json_object_pim_upstream_add(json_object *json,
2293 struct pim_upstream *up)
2294 {
2295 if (up->flags & PIM_UPSTREAM_FLAG_MASK_DR_JOIN_DESIRED)
2296 json_object_boolean_true_add(json, "drJoinDesired");
2297
2298 if (up->flags & PIM_UPSTREAM_FLAG_MASK_DR_JOIN_DESIRED_UPDATED)
2299 json_object_boolean_true_add(json, "drJoinDesiredUpdated");
2300
2301 if (up->flags & PIM_UPSTREAM_FLAG_MASK_FHR)
2302 json_object_boolean_true_add(json, "firstHopRouter");
2303
2304 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_IGMP)
2305 json_object_boolean_true_add(json, "sourceIgmp");
2306
2307 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_PIM)
2308 json_object_boolean_true_add(json, "sourcePim");
2309
2310 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_STREAM)
2311 json_object_boolean_true_add(json, "sourceStream");
2312
2313 /* XXX: need to print ths flag in the plain text display as well */
2314 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_MSDP)
2315 json_object_boolean_true_add(json, "sourceMsdp");
2316
2317 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SEND_SG_RPT_PRUNE)
2318 json_object_boolean_true_add(json, "sendSGRptPrune");
2319
2320 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_LHR)
2321 json_object_boolean_true_add(json, "lastHopRouter");
2322
2323 if (up->flags & PIM_UPSTREAM_FLAG_MASK_DISABLE_KAT_EXPIRY)
2324 json_object_boolean_true_add(json, "disableKATExpiry");
2325
2326 if (up->flags & PIM_UPSTREAM_FLAG_MASK_STATIC_IIF)
2327 json_object_boolean_true_add(json, "staticIncomingInterface");
2328
2329 if (up->flags & PIM_UPSTREAM_FLAG_MASK_ALLOW_IIF_IN_OIL)
2330 json_object_boolean_true_add(json,
2331 "allowIncomingInterfaceinOil");
2332
2333 if (up->flags & PIM_UPSTREAM_FLAG_MASK_NO_PIMREG_DATA)
2334 json_object_boolean_true_add(json, "noPimRegistrationData");
2335
2336 if (up->flags & PIM_UPSTREAM_FLAG_MASK_FORCE_PIMREG)
2337 json_object_boolean_true_add(json, "forcePimRegistration");
2338
2339 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_VXLAN_ORIG)
2340 json_object_boolean_true_add(json, "sourceVxlanOrigination");
2341
2342 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_VXLAN_TERM)
2343 json_object_boolean_true_add(json, "sourceVxlanTermination");
2344
2345 if (up->flags & PIM_UPSTREAM_FLAG_MASK_MLAG_VXLAN)
2346 json_object_boolean_true_add(json, "mlagVxlan");
2347
2348 if (up->flags & PIM_UPSTREAM_FLAG_MASK_MLAG_NON_DF)
2349 json_object_boolean_true_add(json,
2350 "mlagNonDesignatedForwarder");
2351 }
2352
2353 static const char *
2354 pim_upstream_state2brief_str(enum pim_upstream_state join_state,
2355 char *state_str)
2356 {
2357 switch (join_state) {
2358 case PIM_UPSTREAM_NOTJOINED:
2359 strcpy(state_str, "NotJ");
2360 break;
2361 case PIM_UPSTREAM_JOINED:
2362 strcpy(state_str, "J");
2363 break;
2364 default:
2365 strcpy(state_str, "Unk");
2366 }
2367 return state_str;
2368 }
2369
2370 static const char *pim_reg_state2brief_str(enum pim_reg_state reg_state,
2371 char *state_str)
2372 {
2373 switch (reg_state) {
2374 case PIM_REG_NOINFO:
2375 strcpy(state_str, "RegNI");
2376 break;
2377 case PIM_REG_JOIN:
2378 strcpy(state_str, "RegJ");
2379 break;
2380 case PIM_REG_JOIN_PENDING:
2381 case PIM_REG_PRUNE:
2382 strcpy(state_str, "RegP");
2383 break;
2384 default:
2385 strcpy(state_str, "Unk");
2386 }
2387 return state_str;
2388 }
2389
2390 static void pim_show_upstream(struct pim_instance *pim, struct vty *vty,
2391 struct prefix_sg *sg, bool uj)
2392 {
2393 struct listnode *upnode;
2394 struct pim_upstream *up;
2395 time_t now;
2396 json_object *json = NULL;
2397 json_object *json_group = NULL;
2398 json_object *json_row = NULL;
2399
2400 now = pim_time_monotonic_sec();
2401
2402 if (uj)
2403 json = json_object_new_object();
2404 else
2405 vty_out(vty,
2406 "Iif Source Group State Uptime JoinTimer RSTimer KATimer RefCnt\n");
2407
2408 for (ALL_LIST_ELEMENTS_RO(pim->upstream_list, upnode, up)) {
2409 char src_str[INET_ADDRSTRLEN];
2410 char grp_str[INET_ADDRSTRLEN];
2411 char uptime[10];
2412 char join_timer[10];
2413 char rs_timer[10];
2414 char ka_timer[10];
2415 char msdp_reg_timer[10];
2416 char state_str[PIM_REG_STATE_STR_LEN];
2417
2418 if (sg->grp.s_addr != 0 && sg->grp.s_addr != up->sg.grp.s_addr)
2419 continue;
2420 if (sg->src.s_addr != 0 && sg->src.s_addr != up->sg.src.s_addr)
2421 continue;
2422
2423 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
2424 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
2425 pim_time_uptime(uptime, sizeof(uptime),
2426 now - up->state_transition);
2427 pim_time_timer_to_hhmmss(join_timer, sizeof(join_timer),
2428 up->t_join_timer);
2429
2430 /*
2431 * If the upstream is not dummy and it has a J/P timer for the
2432 * neighbor display that
2433 */
2434 if (!up->t_join_timer && up->rpf.source_nexthop.interface) {
2435 struct pim_neighbor *nbr;
2436
2437 nbr = pim_neighbor_find(
2438 up->rpf.source_nexthop.interface,
2439 up->rpf.rpf_addr.u.prefix4);
2440 if (nbr)
2441 pim_time_timer_to_hhmmss(join_timer,
2442 sizeof(join_timer),
2443 nbr->jp_timer);
2444 }
2445
2446 pim_time_timer_to_hhmmss(rs_timer, sizeof(rs_timer),
2447 up->t_rs_timer);
2448 pim_time_timer_to_hhmmss(ka_timer, sizeof(ka_timer),
2449 up->t_ka_timer);
2450 pim_time_timer_to_hhmmss(msdp_reg_timer, sizeof(msdp_reg_timer),
2451 up->t_msdp_reg_timer);
2452
2453 pim_upstream_state2brief_str(up->join_state, state_str);
2454 if (up->reg_state != PIM_REG_NOINFO) {
2455 char tmp_str[PIM_REG_STATE_STR_LEN];
2456
2457 sprintf(state_str + strlen(state_str), ",%s",
2458 pim_reg_state2brief_str(up->reg_state,
2459 tmp_str));
2460 }
2461
2462 if (uj) {
2463 json_object_object_get_ex(json, grp_str, &json_group);
2464
2465 if (!json_group) {
2466 json_group = json_object_new_object();
2467 json_object_object_add(json, grp_str,
2468 json_group);
2469 }
2470
2471 json_row = json_object_new_object();
2472 json_object_pim_upstream_add(json_row, up);
2473 json_object_string_add(
2474 json_row, "inboundInterface",
2475 up->rpf.source_nexthop.interface
2476 ? up->rpf.source_nexthop.interface->name
2477 : "Unknown");
2478
2479 /*
2480 * The RPF address we use is slightly different
2481 * based upon what we are looking up.
2482 * If we have a S, list that unless
2483 * we are the FHR, else we just put
2484 * the RP as the rpfAddress
2485 */
2486 if (up->flags & PIM_UPSTREAM_FLAG_MASK_FHR
2487 || up->sg.src.s_addr == INADDR_ANY) {
2488 char rpf[PREFIX_STRLEN];
2489 struct pim_rpf *rpg;
2490
2491 rpg = RP(pim, up->sg.grp);
2492 pim_inet4_dump("<rpf?>",
2493 rpg->rpf_addr.u.prefix4, rpf,
2494 sizeof(rpf));
2495 json_object_string_add(json_row, "rpfAddress",
2496 rpf);
2497 } else {
2498 json_object_string_add(json_row, "rpfAddress",
2499 src_str);
2500 }
2501
2502 json_object_string_add(json_row, "source", src_str);
2503 json_object_string_add(json_row, "group", grp_str);
2504 json_object_string_add(json_row, "state", state_str);
2505 json_object_string_add(
2506 json_row, "joinState",
2507 pim_upstream_state2str(up->join_state));
2508 json_object_string_add(
2509 json_row, "regState",
2510 pim_reg_state2str(up->reg_state, state_str));
2511 json_object_string_add(json_row, "upTime", uptime);
2512 json_object_string_add(json_row, "joinTimer",
2513 join_timer);
2514 json_object_string_add(json_row, "resetTimer",
2515 rs_timer);
2516 json_object_string_add(json_row, "keepaliveTimer",
2517 ka_timer);
2518 json_object_string_add(json_row, "msdpRegTimer",
2519 msdp_reg_timer);
2520 json_object_int_add(json_row, "refCount",
2521 up->ref_count);
2522 json_object_int_add(json_row, "sptBit", up->sptbit);
2523 json_object_object_add(json_group, src_str, json_row);
2524 } else {
2525 vty_out(vty,
2526 "%-16s%-15s %-15s %-11s %-8s %-9s %-9s %-9s %6d\n",
2527 up->rpf.source_nexthop.interface
2528 ? up->rpf.source_nexthop.interface->name
2529 : "Unknown",
2530 src_str, grp_str, state_str, uptime, join_timer,
2531 rs_timer, ka_timer, up->ref_count);
2532 }
2533 }
2534
2535 if (uj) {
2536 vty_out(vty, "%s\n", json_object_to_json_string_ext(
2537 json, JSON_C_TO_STRING_PRETTY));
2538 json_object_free(json);
2539 }
2540 }
2541
2542 static void pim_show_join_desired_helper(struct pim_instance *pim,
2543 struct vty *vty,
2544 struct pim_interface *pim_ifp,
2545 struct pim_ifchannel *ch,
2546 json_object *json, bool uj)
2547 {
2548 struct pim_upstream *up = ch->upstream;
2549 json_object *json_group = NULL;
2550 char src_str[INET_ADDRSTRLEN];
2551 char grp_str[INET_ADDRSTRLEN];
2552 json_object *json_row = NULL;
2553
2554 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
2555 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
2556
2557 if (uj) {
2558 json_object_object_get_ex(json, grp_str, &json_group);
2559
2560 if (!json_group) {
2561 json_group = json_object_new_object();
2562 json_object_object_add(json, grp_str, json_group);
2563 }
2564
2565 json_row = json_object_new_object();
2566 json_object_pim_upstream_add(json_row, up);
2567 json_object_string_add(json_row, "interface",
2568 ch->interface->name);
2569 json_object_string_add(json_row, "source", src_str);
2570 json_object_string_add(json_row, "group", grp_str);
2571
2572 if (pim_macro_ch_lost_assert(ch))
2573 json_object_boolean_true_add(json_row, "lostAssert");
2574
2575 if (pim_macro_chisin_joins(ch))
2576 json_object_boolean_true_add(json_row, "joins");
2577
2578 if (pim_macro_chisin_pim_include(ch))
2579 json_object_boolean_true_add(json_row, "pimInclude");
2580
2581 if (pim_upstream_evaluate_join_desired(pim, up))
2582 json_object_boolean_true_add(json_row,
2583 "evaluateJoinDesired");
2584
2585 json_object_object_add(json_group, src_str, json_row);
2586
2587 } else {
2588 vty_out(vty, "%-16s %-15s %-15s %-10s %-5s %-10s %-11s %-6s\n",
2589 ch->interface->name, src_str, grp_str,
2590 pim_macro_ch_lost_assert(ch) ? "yes" : "no",
2591 pim_macro_chisin_joins(ch) ? "yes" : "no",
2592 pim_macro_chisin_pim_include(ch) ? "yes" : "no",
2593 PIM_UPSTREAM_FLAG_TEST_DR_JOIN_DESIRED(up->flags)
2594 ? "yes"
2595 : "no",
2596 pim_upstream_evaluate_join_desired(pim, up) ? "yes"
2597 : "no");
2598 }
2599 }
2600
2601 static void pim_show_join_desired(struct pim_instance *pim, struct vty *vty,
2602 bool uj)
2603 {
2604 struct pim_interface *pim_ifp;
2605 struct pim_ifchannel *ch;
2606 struct interface *ifp;
2607
2608 json_object *json = NULL;
2609
2610 if (uj)
2611 json = json_object_new_object();
2612 else
2613 vty_out(vty,
2614 "Interface Source Group LostAssert Joins PimInclude JoinDesired EvalJD\n");
2615
2616 /* scan per-interface (S,G) state */
2617 FOR_ALL_INTERFACES (pim->vrf, ifp) {
2618 pim_ifp = ifp->info;
2619 if (!pim_ifp)
2620 continue;
2621
2622
2623 RB_FOREACH (ch, pim_ifchannel_rb, &pim_ifp->ifchannel_rb) {
2624 /* scan all interfaces */
2625 pim_show_join_desired_helper(pim, vty, pim_ifp, ch,
2626 json, uj);
2627 }
2628 }
2629
2630 if (uj) {
2631 vty_out(vty, "%s\n", json_object_to_json_string_ext(
2632 json, JSON_C_TO_STRING_PRETTY));
2633 json_object_free(json);
2634 }
2635 }
2636
2637 static void pim_show_upstream_rpf(struct pim_instance *pim, struct vty *vty,
2638 bool uj)
2639 {
2640 struct listnode *upnode;
2641 struct pim_upstream *up;
2642 json_object *json = NULL;
2643 json_object *json_group = NULL;
2644 json_object *json_row = NULL;
2645
2646 if (uj)
2647 json = json_object_new_object();
2648 else
2649 vty_out(vty,
2650 "Source Group RpfIface RibNextHop RpfAddress \n");
2651
2652 for (ALL_LIST_ELEMENTS_RO(pim->upstream_list, upnode, up)) {
2653 char src_str[INET_ADDRSTRLEN];
2654 char grp_str[INET_ADDRSTRLEN];
2655 char rpf_nexthop_str[PREFIX_STRLEN];
2656 char rpf_addr_str[PREFIX_STRLEN];
2657 struct pim_rpf *rpf;
2658 const char *rpf_ifname;
2659
2660 rpf = &up->rpf;
2661
2662 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
2663 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
2664 pim_addr_dump("<nexthop?>",
2665 &rpf->source_nexthop.mrib_nexthop_addr,
2666 rpf_nexthop_str, sizeof(rpf_nexthop_str));
2667 pim_addr_dump("<rpf?>", &rpf->rpf_addr, rpf_addr_str,
2668 sizeof(rpf_addr_str));
2669
2670 rpf_ifname = rpf->source_nexthop.interface ? rpf->source_nexthop.interface->name : "<ifname?>";
2671
2672 if (uj) {
2673 json_object_object_get_ex(json, grp_str, &json_group);
2674
2675 if (!json_group) {
2676 json_group = json_object_new_object();
2677 json_object_object_add(json, grp_str,
2678 json_group);
2679 }
2680
2681 json_row = json_object_new_object();
2682 json_object_pim_upstream_add(json_row, up);
2683 json_object_string_add(json_row, "source", src_str);
2684 json_object_string_add(json_row, "group", grp_str);
2685 json_object_string_add(json_row, "rpfInterface",
2686 rpf_ifname);
2687 json_object_string_add(json_row, "ribNexthop",
2688 rpf_nexthop_str);
2689 json_object_string_add(json_row, "rpfAddress",
2690 rpf_addr_str);
2691 json_object_object_add(json_group, src_str, json_row);
2692 } else {
2693 vty_out(vty, "%-15s %-15s %-16s %-15s %-15s\n", src_str,
2694 grp_str, rpf_ifname, rpf_nexthop_str,
2695 rpf_addr_str);
2696 }
2697 }
2698
2699 if (uj) {
2700 vty_out(vty, "%s\n", json_object_to_json_string_ext(
2701 json, JSON_C_TO_STRING_PRETTY));
2702 json_object_free(json);
2703 }
2704 }
2705
2706 static void show_rpf_refresh_stats(struct vty *vty, struct pim_instance *pim,
2707 time_t now, json_object *json)
2708 {
2709 char refresh_uptime[10];
2710
2711 pim_time_uptime_begin(refresh_uptime, sizeof(refresh_uptime), now,
2712 pim->rpf_cache_refresh_last);
2713
2714 if (json) {
2715 json_object_int_add(json, "rpfCacheRefreshDelayMsecs",
2716 router->rpf_cache_refresh_delay_msec);
2717 json_object_int_add(
2718 json, "rpfCacheRefreshTimer",
2719 pim_time_timer_remain_msec(pim->rpf_cache_refresher));
2720 json_object_int_add(json, "rpfCacheRefreshRequests",
2721 pim->rpf_cache_refresh_requests);
2722 json_object_int_add(json, "rpfCacheRefreshEvents",
2723 pim->rpf_cache_refresh_events);
2724 json_object_string_add(json, "rpfCacheRefreshLast",
2725 refresh_uptime);
2726 json_object_int_add(json, "nexthopLookups",
2727 pim->nexthop_lookups);
2728 json_object_int_add(json, "nexthopLookupsAvoided",
2729 pim->nexthop_lookups_avoided);
2730 } else {
2731 vty_out(vty,
2732 "RPF Cache Refresh Delay: %ld msecs\n"
2733 "RPF Cache Refresh Timer: %ld msecs\n"
2734 "RPF Cache Refresh Requests: %lld\n"
2735 "RPF Cache Refresh Events: %lld\n"
2736 "RPF Cache Refresh Last: %s\n"
2737 "Nexthop Lookups: %lld\n"
2738 "Nexthop Lookups Avoided: %lld\n",
2739 router->rpf_cache_refresh_delay_msec,
2740 pim_time_timer_remain_msec(pim->rpf_cache_refresher),
2741 (long long)pim->rpf_cache_refresh_requests,
2742 (long long)pim->rpf_cache_refresh_events,
2743 refresh_uptime, (long long)pim->nexthop_lookups,
2744 (long long)pim->nexthop_lookups_avoided);
2745 }
2746 }
2747
2748 static void show_scan_oil_stats(struct pim_instance *pim, struct vty *vty,
2749 time_t now)
2750 {
2751 char uptime_scan_oil[10];
2752 char uptime_mroute_add[10];
2753 char uptime_mroute_del[10];
2754
2755 pim_time_uptime_begin(uptime_scan_oil, sizeof(uptime_scan_oil), now,
2756 pim->scan_oil_last);
2757 pim_time_uptime_begin(uptime_mroute_add, sizeof(uptime_mroute_add), now,
2758 pim->mroute_add_last);
2759 pim_time_uptime_begin(uptime_mroute_del, sizeof(uptime_mroute_del), now,
2760 pim->mroute_del_last);
2761
2762 vty_out(vty,
2763 "Scan OIL - Last: %s Events: %lld\n"
2764 "MFC Add - Last: %s Events: %lld\n"
2765 "MFC Del - Last: %s Events: %lld\n",
2766 uptime_scan_oil, (long long)pim->scan_oil_events,
2767 uptime_mroute_add, (long long)pim->mroute_add_events,
2768 uptime_mroute_del, (long long)pim->mroute_del_events);
2769 }
2770
2771 static void pim_show_rpf(struct pim_instance *pim, struct vty *vty, bool uj)
2772 {
2773 struct listnode *up_node;
2774 struct pim_upstream *up;
2775 time_t now = pim_time_monotonic_sec();
2776 json_object *json = NULL;
2777 json_object *json_group = NULL;
2778 json_object *json_row = NULL;
2779
2780 if (uj) {
2781 json = json_object_new_object();
2782 show_rpf_refresh_stats(vty, pim, now, json);
2783 } else {
2784 show_rpf_refresh_stats(vty, pim, now, json);
2785 vty_out(vty, "\n");
2786 vty_out(vty,
2787 "Source Group RpfIface RpfAddress RibNextHop Metric Pref\n");
2788 }
2789
2790 for (ALL_LIST_ELEMENTS_RO(pim->upstream_list, up_node, up)) {
2791 char src_str[INET_ADDRSTRLEN];
2792 char grp_str[INET_ADDRSTRLEN];
2793 char rpf_addr_str[PREFIX_STRLEN];
2794 char rib_nexthop_str[PREFIX_STRLEN];
2795 const char *rpf_ifname;
2796 struct pim_rpf *rpf = &up->rpf;
2797
2798 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
2799 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
2800 pim_addr_dump("<rpf?>", &rpf->rpf_addr, rpf_addr_str,
2801 sizeof(rpf_addr_str));
2802 pim_addr_dump("<nexthop?>",
2803 &rpf->source_nexthop.mrib_nexthop_addr,
2804 rib_nexthop_str, sizeof(rib_nexthop_str));
2805
2806 rpf_ifname = rpf->source_nexthop.interface ? rpf->source_nexthop.interface->name : "<ifname?>";
2807
2808 if (uj) {
2809 json_object_object_get_ex(json, grp_str, &json_group);
2810
2811 if (!json_group) {
2812 json_group = json_object_new_object();
2813 json_object_object_add(json, grp_str,
2814 json_group);
2815 }
2816
2817 json_row = json_object_new_object();
2818 json_object_string_add(json_row, "source", src_str);
2819 json_object_string_add(json_row, "group", grp_str);
2820 json_object_string_add(json_row, "rpfInterface",
2821 rpf_ifname);
2822 json_object_string_add(json_row, "rpfAddress",
2823 rpf_addr_str);
2824 json_object_string_add(json_row, "ribNexthop",
2825 rib_nexthop_str);
2826 json_object_int_add(
2827 json_row, "routeMetric",
2828 rpf->source_nexthop.mrib_route_metric);
2829 json_object_int_add(
2830 json_row, "routePreference",
2831 rpf->source_nexthop.mrib_metric_preference);
2832 json_object_object_add(json_group, src_str, json_row);
2833
2834 } else {
2835 vty_out(vty, "%-15s %-15s %-16s %-15s %-15s %6d %4d\n",
2836 src_str, grp_str, rpf_ifname, rpf_addr_str,
2837 rib_nexthop_str,
2838 rpf->source_nexthop.mrib_route_metric,
2839 rpf->source_nexthop.mrib_metric_preference);
2840 }
2841 }
2842
2843 if (uj) {
2844 vty_out(vty, "%s\n", json_object_to_json_string_ext(
2845 json, JSON_C_TO_STRING_PRETTY));
2846 json_object_free(json);
2847 }
2848 }
2849
2850 struct pnc_cache_walk_data {
2851 struct vty *vty;
2852 struct pim_instance *pim;
2853 };
2854
2855 static int pim_print_pnc_cache_walkcb(struct hash_bucket *bucket, void *arg)
2856 {
2857 struct pim_nexthop_cache *pnc = bucket->data;
2858 struct pnc_cache_walk_data *cwd = arg;
2859 struct vty *vty = cwd->vty;
2860 struct pim_instance *pim = cwd->pim;
2861 struct nexthop *nh_node = NULL;
2862 ifindex_t first_ifindex;
2863 struct interface *ifp = NULL;
2864
2865 for (nh_node = pnc->nexthop; nh_node; nh_node = nh_node->next) {
2866 first_ifindex = nh_node->ifindex;
2867 ifp = if_lookup_by_index(first_ifindex, pim->vrf_id);
2868
2869 vty_out(vty, "%-15s ", inet_ntoa(pnc->rpf.rpf_addr.u.prefix4));
2870 vty_out(vty, "%-16s ", ifp ? ifp->name : "NULL");
2871 vty_out(vty, "%s ", inet_ntoa(nh_node->gate.ipv4));
2872 vty_out(vty, "\n");
2873 }
2874 return CMD_SUCCESS;
2875 }
2876
2877 static void pim_show_nexthop(struct pim_instance *pim, struct vty *vty)
2878 {
2879 struct pnc_cache_walk_data cwd;
2880
2881 cwd.vty = vty;
2882 cwd.pim = pim;
2883 vty_out(vty, "Number of registered addresses: %lu\n",
2884 pim->rpf_hash->count);
2885 vty_out(vty, "Address Interface Nexthop\n");
2886 vty_out(vty, "---------------------------------------------\n");
2887
2888 hash_walk(pim->rpf_hash, pim_print_pnc_cache_walkcb, &cwd);
2889 }
2890
2891 static void igmp_show_groups(struct pim_instance *pim, struct vty *vty, bool uj)
2892 {
2893 struct interface *ifp;
2894 time_t now;
2895 json_object *json = NULL;
2896 json_object *json_iface = NULL;
2897 json_object *json_row = NULL;
2898
2899 now = pim_time_monotonic_sec();
2900
2901 if (uj)
2902 json = json_object_new_object();
2903 else
2904 vty_out(vty,
2905 "Interface Address Group Mode Timer Srcs V Uptime \n");
2906
2907 /* scan interfaces */
2908 FOR_ALL_INTERFACES (pim->vrf, ifp) {
2909 struct pim_interface *pim_ifp = ifp->info;
2910 struct listnode *sock_node;
2911 struct igmp_sock *igmp;
2912
2913 if (!pim_ifp)
2914 continue;
2915
2916 /* scan igmp sockets */
2917 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node,
2918 igmp)) {
2919 char ifaddr_str[INET_ADDRSTRLEN];
2920 struct listnode *grpnode;
2921 struct igmp_group *grp;
2922
2923 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str,
2924 sizeof(ifaddr_str));
2925
2926 /* scan igmp groups */
2927 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list,
2928 grpnode, grp)) {
2929 char group_str[INET_ADDRSTRLEN];
2930 char hhmmss[10];
2931 char uptime[10];
2932
2933 pim_inet4_dump("<group?>", grp->group_addr,
2934 group_str, sizeof(group_str));
2935 pim_time_timer_to_hhmmss(hhmmss, sizeof(hhmmss),
2936 grp->t_group_timer);
2937 pim_time_uptime(uptime, sizeof(uptime),
2938 now - grp->group_creation);
2939
2940 if (uj) {
2941 json_object_object_get_ex(
2942 json, ifp->name, &json_iface);
2943
2944 if (!json_iface) {
2945 json_iface =
2946 json_object_new_object();
2947 json_object_pim_ifp_add(
2948 json_iface, ifp);
2949 json_object_object_add(
2950 json, ifp->name,
2951 json_iface);
2952 }
2953
2954 json_row = json_object_new_object();
2955 json_object_string_add(
2956 json_row, "source", ifaddr_str);
2957 json_object_string_add(
2958 json_row, "group", group_str);
2959
2960 if (grp->igmp_version == 3)
2961 json_object_string_add(
2962 json_row, "mode",
2963 grp->group_filtermode_isexcl
2964 ? "EXCLUDE"
2965 : "INCLUDE");
2966
2967 json_object_string_add(json_row,
2968 "timer", hhmmss);
2969 json_object_int_add(
2970 json_row, "sourcesCount",
2971 grp->group_source_list
2972 ? listcount(
2973 grp->group_source_list)
2974 : 0);
2975 json_object_int_add(json_row, "version",
2976 grp->igmp_version);
2977 json_object_string_add(
2978 json_row, "uptime", uptime);
2979 json_object_object_add(json_iface,
2980 group_str,
2981 json_row);
2982
2983 } else {
2984 vty_out(vty,
2985 "%-16s %-15s %-15s %4s %8s %4d %d %8s\n",
2986 ifp->name, ifaddr_str,
2987 group_str,
2988 grp->igmp_version == 3
2989 ? (grp->group_filtermode_isexcl
2990 ? "EXCL"
2991 : "INCL")
2992 : "----",
2993 hhmmss,
2994 grp->group_source_list
2995 ? listcount(
2996 grp->group_source_list)
2997 : 0,
2998 grp->igmp_version, uptime);
2999 }
3000 } /* scan igmp groups */
3001 } /* scan igmp sockets */
3002 } /* scan interfaces */
3003
3004 if (uj) {
3005 vty_out(vty, "%s\n", json_object_to_json_string_ext(
3006 json, JSON_C_TO_STRING_PRETTY));
3007 json_object_free(json);
3008 }
3009 }
3010
3011 static void igmp_show_group_retransmission(struct pim_instance *pim,
3012 struct vty *vty)
3013 {
3014 struct interface *ifp;
3015
3016 vty_out(vty,
3017 "Interface Address Group RetTimer Counter RetSrcs\n");
3018
3019 /* scan interfaces */
3020 FOR_ALL_INTERFACES (pim->vrf, ifp) {
3021 struct pim_interface *pim_ifp = ifp->info;
3022 struct listnode *sock_node;
3023 struct igmp_sock *igmp;
3024
3025 if (!pim_ifp)
3026 continue;
3027
3028 /* scan igmp sockets */
3029 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node,
3030 igmp)) {
3031 char ifaddr_str[INET_ADDRSTRLEN];
3032 struct listnode *grpnode;
3033 struct igmp_group *grp;
3034
3035 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str,
3036 sizeof(ifaddr_str));
3037
3038 /* scan igmp groups */
3039 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list,
3040 grpnode, grp)) {
3041 char group_str[INET_ADDRSTRLEN];
3042 char grp_retr_mmss[10];
3043 struct listnode *src_node;
3044 struct igmp_source *src;
3045 int grp_retr_sources = 0;
3046
3047 pim_inet4_dump("<group?>", grp->group_addr,
3048 group_str, sizeof(group_str));
3049 pim_time_timer_to_mmss(
3050 grp_retr_mmss, sizeof(grp_retr_mmss),
3051 grp->t_group_query_retransmit_timer);
3052
3053
3054 /* count group sources with retransmission state
3055 */
3056 for (ALL_LIST_ELEMENTS_RO(
3057 grp->group_source_list, src_node,
3058 src)) {
3059 if (src->source_query_retransmit_count
3060 > 0) {
3061 ++grp_retr_sources;
3062 }
3063 }
3064
3065 vty_out(vty, "%-16s %-15s %-15s %-8s %7d %7d\n",
3066 ifp->name, ifaddr_str, group_str,
3067 grp_retr_mmss,
3068 grp->group_specific_query_retransmit_count,
3069 grp_retr_sources);
3070
3071 } /* scan igmp groups */
3072 } /* scan igmp sockets */
3073 } /* scan interfaces */
3074 }
3075
3076 static void igmp_show_sources(struct pim_instance *pim, struct vty *vty)
3077 {
3078 struct interface *ifp;
3079 time_t now;
3080
3081 now = pim_time_monotonic_sec();
3082
3083 vty_out(vty,
3084 "Interface Address Group Source Timer Fwd Uptime \n");
3085
3086 /* scan interfaces */
3087 FOR_ALL_INTERFACES (pim->vrf, ifp) {
3088 struct pim_interface *pim_ifp = ifp->info;
3089 struct listnode *sock_node;
3090 struct igmp_sock *igmp;
3091
3092 if (!pim_ifp)
3093 continue;
3094
3095 /* scan igmp sockets */
3096 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node,
3097 igmp)) {
3098 char ifaddr_str[INET_ADDRSTRLEN];
3099 struct listnode *grpnode;
3100 struct igmp_group *grp;
3101
3102 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str,
3103 sizeof(ifaddr_str));
3104
3105 /* scan igmp groups */
3106 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list,
3107 grpnode, grp)) {
3108 char group_str[INET_ADDRSTRLEN];
3109 struct listnode *srcnode;
3110 struct igmp_source *src;
3111
3112 pim_inet4_dump("<group?>", grp->group_addr,
3113 group_str, sizeof(group_str));
3114
3115 /* scan group sources */
3116 for (ALL_LIST_ELEMENTS_RO(
3117 grp->group_source_list, srcnode,
3118 src)) {
3119 char source_str[INET_ADDRSTRLEN];
3120 char mmss[10];
3121 char uptime[10];
3122
3123 pim_inet4_dump(
3124 "<source?>", src->source_addr,
3125 source_str, sizeof(source_str));
3126
3127 pim_time_timer_to_mmss(
3128 mmss, sizeof(mmss),
3129 src->t_source_timer);
3130
3131 pim_time_uptime(
3132 uptime, sizeof(uptime),
3133 now - src->source_creation);
3134
3135 vty_out(vty,
3136 "%-16s %-15s %-15s %-15s %5s %3s %8s\n",
3137 ifp->name, ifaddr_str,
3138 group_str, source_str, mmss,
3139 IGMP_SOURCE_TEST_FORWARDING(
3140 src->source_flags)
3141 ? "Y"
3142 : "N",
3143 uptime);
3144
3145 } /* scan group sources */
3146 } /* scan igmp groups */
3147 } /* scan igmp sockets */
3148 } /* scan interfaces */
3149 }
3150
3151 static void igmp_show_source_retransmission(struct pim_instance *pim,
3152 struct vty *vty)
3153 {
3154 struct interface *ifp;
3155
3156 vty_out(vty,
3157 "Interface Address Group Source Counter\n");
3158
3159 /* scan interfaces */
3160 FOR_ALL_INTERFACES (pim->vrf, ifp) {
3161 struct pim_interface *pim_ifp = ifp->info;
3162 struct listnode *sock_node;
3163 struct igmp_sock *igmp;
3164
3165 if (!pim_ifp)
3166 continue;
3167
3168 /* scan igmp sockets */
3169 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node,
3170 igmp)) {
3171 char ifaddr_str[INET_ADDRSTRLEN];
3172 struct listnode *grpnode;
3173 struct igmp_group *grp;
3174
3175 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str,
3176 sizeof(ifaddr_str));
3177
3178 /* scan igmp groups */
3179 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list,
3180 grpnode, grp)) {
3181 char group_str[INET_ADDRSTRLEN];
3182 struct listnode *srcnode;
3183 struct igmp_source *src;
3184
3185 pim_inet4_dump("<group?>", grp->group_addr,
3186 group_str, sizeof(group_str));
3187
3188 /* scan group sources */
3189 for (ALL_LIST_ELEMENTS_RO(
3190 grp->group_source_list, srcnode,
3191 src)) {
3192 char source_str[INET_ADDRSTRLEN];
3193
3194 pim_inet4_dump(
3195 "<source?>", src->source_addr,
3196 source_str, sizeof(source_str));
3197
3198 vty_out(vty,
3199 "%-16s %-15s %-15s %-15s %7d\n",
3200 ifp->name, ifaddr_str,
3201 group_str, source_str,
3202 src->source_query_retransmit_count);
3203
3204 } /* scan group sources */
3205 } /* scan igmp groups */
3206 } /* scan igmp sockets */
3207 } /* scan interfaces */
3208 }
3209
3210 static void clear_igmp_interfaces(struct pim_instance *pim)
3211 {
3212 struct interface *ifp;
3213
3214 FOR_ALL_INTERFACES (pim->vrf, ifp)
3215 pim_if_addr_del_all_igmp(ifp);
3216
3217 FOR_ALL_INTERFACES (pim->vrf, ifp)
3218 pim_if_addr_add_all(ifp);
3219 }
3220
3221 static void clear_pim_interfaces(struct pim_instance *pim)
3222 {
3223 struct interface *ifp;
3224
3225 FOR_ALL_INTERFACES (pim->vrf, ifp) {
3226 if (ifp->info) {
3227 pim_neighbor_delete_all(ifp, "interface cleared");
3228 }
3229 }
3230 }
3231
3232 static void clear_interfaces(struct pim_instance *pim)
3233 {
3234 clear_igmp_interfaces(pim);
3235 clear_pim_interfaces(pim);
3236 }
3237
3238 #define PIM_GET_PIM_INTERFACE(pim_ifp, ifp) \
3239 pim_ifp = ifp->info; \
3240 if (!pim_ifp) { \
3241 vty_out(vty, \
3242 "%% Enable PIM and/or IGMP on this interface first\n"); \
3243 return CMD_WARNING_CONFIG_FAILED; \
3244 }
3245
3246 DEFUN (clear_ip_interfaces,
3247 clear_ip_interfaces_cmd,
3248 "clear ip interfaces [vrf NAME]",
3249 CLEAR_STR
3250 IP_STR
3251 "Reset interfaces\n"
3252 VRF_CMD_HELP_STR)
3253 {
3254 int idx = 2;
3255 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3256
3257 if (!vrf)
3258 return CMD_WARNING;
3259
3260 clear_interfaces(vrf->info);
3261
3262 return CMD_SUCCESS;
3263 }
3264
3265 DEFUN (clear_ip_igmp_interfaces,
3266 clear_ip_igmp_interfaces_cmd,
3267 "clear ip igmp [vrf NAME] interfaces",
3268 CLEAR_STR
3269 IP_STR
3270 CLEAR_IP_IGMP_STR
3271 VRF_CMD_HELP_STR
3272 "Reset IGMP interfaces\n")
3273 {
3274 int idx = 2;
3275 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3276
3277 if (!vrf)
3278 return CMD_WARNING;
3279
3280 clear_igmp_interfaces(vrf->info);
3281
3282 return CMD_SUCCESS;
3283 }
3284
3285 static void mroute_add_all(struct pim_instance *pim)
3286 {
3287 struct listnode *node;
3288 struct channel_oil *c_oil;
3289
3290 for (ALL_LIST_ELEMENTS_RO(pim->channel_oil_list, node, c_oil)) {
3291 if (pim_mroute_add(c_oil, __PRETTY_FUNCTION__)) {
3292 /* just log warning */
3293 char source_str[INET_ADDRSTRLEN];
3294 char group_str[INET_ADDRSTRLEN];
3295 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin,
3296 source_str, sizeof(source_str));
3297 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp,
3298 group_str, sizeof(group_str));
3299 zlog_warn("%s %s: (S,G)=(%s,%s) failure writing MFC",
3300 __FILE__, __PRETTY_FUNCTION__, source_str,
3301 group_str);
3302 }
3303 }
3304 }
3305
3306 static void mroute_del_all(struct pim_instance *pim)
3307 {
3308 struct listnode *node;
3309 struct channel_oil *c_oil;
3310
3311 for (ALL_LIST_ELEMENTS_RO(pim->channel_oil_list, node, c_oil)) {
3312 if (pim_mroute_del(c_oil, __PRETTY_FUNCTION__)) {
3313 /* just log warning */
3314 char source_str[INET_ADDRSTRLEN];
3315 char group_str[INET_ADDRSTRLEN];
3316 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin,
3317 source_str, sizeof(source_str));
3318 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp,
3319 group_str, sizeof(group_str));
3320 zlog_warn("%s %s: (S,G)=(%s,%s) failure clearing MFC",
3321 __FILE__, __PRETTY_FUNCTION__, source_str,
3322 group_str);
3323 }
3324 }
3325 }
3326
3327 DEFUN (clear_ip_mroute,
3328 clear_ip_mroute_cmd,
3329 "clear ip mroute [vrf NAME]",
3330 CLEAR_STR
3331 IP_STR
3332 "Reset multicast routes\n"
3333 VRF_CMD_HELP_STR)
3334 {
3335 int idx = 2;
3336 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3337
3338 if (!vrf)
3339 return CMD_WARNING;
3340
3341 mroute_del_all(vrf->info);
3342 mroute_add_all(vrf->info);
3343
3344 return CMD_SUCCESS;
3345 }
3346
3347 DEFUN (clear_ip_pim_interfaces,
3348 clear_ip_pim_interfaces_cmd,
3349 "clear ip pim [vrf NAME] interfaces",
3350 CLEAR_STR
3351 IP_STR
3352 CLEAR_IP_PIM_STR
3353 VRF_CMD_HELP_STR
3354 "Reset PIM interfaces\n")
3355 {
3356 int idx = 2;
3357 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3358
3359 if (!vrf)
3360 return CMD_WARNING;
3361
3362 clear_pim_interfaces(vrf->info);
3363
3364 return CMD_SUCCESS;
3365 }
3366
3367 DEFUN (clear_ip_pim_interface_traffic,
3368 clear_ip_pim_interface_traffic_cmd,
3369 "clear ip pim [vrf NAME] interface traffic",
3370 "Reset functions\n"
3371 "IP information\n"
3372 "PIM clear commands\n"
3373 VRF_CMD_HELP_STR
3374 "Reset PIM interfaces\n"
3375 "Reset Protocol Packet counters\n")
3376 {
3377 int idx = 2;
3378 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3379 struct interface *ifp = NULL;
3380 struct pim_interface *pim_ifp = NULL;
3381
3382 if (!vrf)
3383 return CMD_WARNING;
3384
3385 FOR_ALL_INTERFACES (vrf, ifp) {
3386 pim_ifp = ifp->info;
3387
3388 if (!pim_ifp)
3389 continue;
3390
3391 pim_ifp->pim_ifstat_hello_recv = 0;
3392 pim_ifp->pim_ifstat_hello_sent = 0;
3393 pim_ifp->pim_ifstat_join_recv = 0;
3394 pim_ifp->pim_ifstat_join_send = 0;
3395 pim_ifp->pim_ifstat_prune_recv = 0;
3396 pim_ifp->pim_ifstat_prune_send = 0;
3397 pim_ifp->pim_ifstat_reg_recv = 0;
3398 pim_ifp->pim_ifstat_reg_send = 0;
3399 pim_ifp->pim_ifstat_reg_stop_recv = 0;
3400 pim_ifp->pim_ifstat_reg_stop_send = 0;
3401 pim_ifp->pim_ifstat_assert_recv = 0;
3402 pim_ifp->pim_ifstat_assert_send = 0;
3403 }
3404
3405 return CMD_SUCCESS;
3406 }
3407
3408 DEFUN (clear_ip_pim_oil,
3409 clear_ip_pim_oil_cmd,
3410 "clear ip pim [vrf NAME] oil",
3411 CLEAR_STR
3412 IP_STR
3413 CLEAR_IP_PIM_STR
3414 VRF_CMD_HELP_STR
3415 "Rescan PIM OIL (output interface list)\n")
3416 {
3417 int idx = 2;
3418 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3419
3420 if (!vrf)
3421 return CMD_WARNING;
3422
3423 pim_scan_oil(vrf->info);
3424
3425 return CMD_SUCCESS;
3426 }
3427
3428 DEFUN (show_ip_igmp_interface,
3429 show_ip_igmp_interface_cmd,
3430 "show ip igmp [vrf NAME] interface [detail|WORD] [json]",
3431 SHOW_STR
3432 IP_STR
3433 IGMP_STR
3434 VRF_CMD_HELP_STR
3435 "IGMP interface information\n"
3436 "Detailed output\n"
3437 "interface name\n"
3438 JSON_STR)
3439 {
3440 int idx = 2;
3441 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3442 bool uj = use_json(argc, argv);
3443
3444 if (!vrf)
3445 return CMD_WARNING;
3446
3447 if (argv_find(argv, argc, "detail", &idx)
3448 || argv_find(argv, argc, "WORD", &idx))
3449 igmp_show_interfaces_single(vrf->info, vty, argv[idx]->arg, uj);
3450 else
3451 igmp_show_interfaces(vrf->info, vty, uj);
3452
3453 return CMD_SUCCESS;
3454 }
3455
3456 DEFUN (show_ip_igmp_interface_vrf_all,
3457 show_ip_igmp_interface_vrf_all_cmd,
3458 "show ip igmp vrf all interface [detail|WORD] [json]",
3459 SHOW_STR
3460 IP_STR
3461 IGMP_STR
3462 VRF_CMD_HELP_STR
3463 "IGMP interface information\n"
3464 "Detailed output\n"
3465 "interface name\n"
3466 JSON_STR)
3467 {
3468 int idx = 2;
3469 bool uj = use_json(argc, argv);
3470 struct vrf *vrf;
3471 bool first = true;
3472
3473 if (uj)
3474 vty_out(vty, "{ ");
3475 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
3476 if (uj) {
3477 if (!first)
3478 vty_out(vty, ", ");
3479 vty_out(vty, " \"%s\": ", vrf->name);
3480 first = false;
3481 } else
3482 vty_out(vty, "VRF: %s\n", vrf->name);
3483 if (argv_find(argv, argc, "detail", &idx)
3484 || argv_find(argv, argc, "WORD", &idx))
3485 igmp_show_interfaces_single(vrf->info, vty,
3486 argv[idx]->arg, uj);
3487 else
3488 igmp_show_interfaces(vrf->info, vty, uj);
3489 }
3490 if (uj)
3491 vty_out(vty, "}\n");
3492
3493 return CMD_SUCCESS;
3494 }
3495
3496 DEFUN (show_ip_igmp_join,
3497 show_ip_igmp_join_cmd,
3498 "show ip igmp [vrf NAME] join",
3499 SHOW_STR
3500 IP_STR
3501 IGMP_STR
3502 VRF_CMD_HELP_STR
3503 "IGMP static join information\n")
3504 {
3505 int idx = 2;
3506 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3507
3508 if (!vrf)
3509 return CMD_WARNING;
3510
3511 igmp_show_interface_join(vrf->info, vty);
3512
3513 return CMD_SUCCESS;
3514 }
3515
3516 DEFUN (show_ip_igmp_join_vrf_all,
3517 show_ip_igmp_join_vrf_all_cmd,
3518 "show ip igmp vrf all join",
3519 SHOW_STR
3520 IP_STR
3521 IGMP_STR
3522 VRF_CMD_HELP_STR
3523 "IGMP static join information\n")
3524 {
3525 bool uj = use_json(argc, argv);
3526 struct vrf *vrf;
3527 bool first = true;
3528
3529 if (uj)
3530 vty_out(vty, "{ ");
3531 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
3532 if (uj) {
3533 if (!first)
3534 vty_out(vty, ", ");
3535 vty_out(vty, " \"%s\": ", vrf->name);
3536 first = false;
3537 } else
3538 vty_out(vty, "VRF: %s\n", vrf->name);
3539 igmp_show_interface_join(vrf->info, vty);
3540 }
3541 if (uj)
3542 vty_out(vty, "}\n");
3543
3544 return CMD_SUCCESS;
3545 }
3546
3547 DEFUN (show_ip_igmp_groups,
3548 show_ip_igmp_groups_cmd,
3549 "show ip igmp [vrf NAME] groups [json]",
3550 SHOW_STR
3551 IP_STR
3552 IGMP_STR
3553 VRF_CMD_HELP_STR
3554 IGMP_GROUP_STR
3555 JSON_STR)
3556 {
3557 int idx = 2;
3558 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3559 bool uj = use_json(argc, argv);
3560
3561 if (!vrf)
3562 return CMD_WARNING;
3563
3564 igmp_show_groups(vrf->info, vty, uj);
3565
3566 return CMD_SUCCESS;
3567 }
3568
3569 DEFUN (show_ip_igmp_groups_vrf_all,
3570 show_ip_igmp_groups_vrf_all_cmd,
3571 "show ip igmp vrf all groups [json]",
3572 SHOW_STR
3573 IP_STR
3574 IGMP_STR
3575 VRF_CMD_HELP_STR
3576 IGMP_GROUP_STR
3577 JSON_STR)
3578 {
3579 bool uj = use_json(argc, argv);
3580 struct vrf *vrf;
3581 bool first = true;
3582
3583 if (uj)
3584 vty_out(vty, "{ ");
3585 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
3586 if (uj) {
3587 if (!first)
3588 vty_out(vty, ", ");
3589 vty_out(vty, " \"%s\": ", vrf->name);
3590 first = false;
3591 } else
3592 vty_out(vty, "VRF: %s\n", vrf->name);
3593 igmp_show_groups(vrf->info, vty, uj);
3594 }
3595 if (uj)
3596 vty_out(vty, "}\n");
3597
3598 return CMD_SUCCESS;
3599 }
3600
3601 DEFUN (show_ip_igmp_groups_retransmissions,
3602 show_ip_igmp_groups_retransmissions_cmd,
3603 "show ip igmp [vrf NAME] groups retransmissions",
3604 SHOW_STR
3605 IP_STR
3606 IGMP_STR
3607 VRF_CMD_HELP_STR
3608 IGMP_GROUP_STR
3609 "IGMP group retransmissions\n")
3610 {
3611 int idx = 2;
3612 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3613
3614 if (!vrf)
3615 return CMD_WARNING;
3616
3617 igmp_show_group_retransmission(vrf->info, vty);
3618
3619 return CMD_SUCCESS;
3620 }
3621
3622 DEFUN (show_ip_igmp_sources,
3623 show_ip_igmp_sources_cmd,
3624 "show ip igmp [vrf NAME] sources",
3625 SHOW_STR
3626 IP_STR
3627 IGMP_STR
3628 VRF_CMD_HELP_STR
3629 IGMP_SOURCE_STR)
3630 {
3631 int idx = 2;
3632 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3633
3634 if (!vrf)
3635 return CMD_WARNING;
3636
3637 igmp_show_sources(vrf->info, vty);
3638
3639 return CMD_SUCCESS;
3640 }
3641
3642 DEFUN (show_ip_igmp_sources_retransmissions,
3643 show_ip_igmp_sources_retransmissions_cmd,
3644 "show ip igmp [vrf NAME] sources retransmissions",
3645 SHOW_STR
3646 IP_STR
3647 IGMP_STR
3648 VRF_CMD_HELP_STR
3649 IGMP_SOURCE_STR
3650 "IGMP source retransmissions\n")
3651 {
3652 int idx = 2;
3653 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3654
3655 if (!vrf)
3656 return CMD_WARNING;
3657
3658 igmp_show_source_retransmission(vrf->info, vty);
3659
3660 return CMD_SUCCESS;
3661 }
3662
3663 DEFUN (show_ip_igmp_statistics,
3664 show_ip_igmp_statistics_cmd,
3665 "show ip igmp [vrf NAME] statistics [interface WORD] [json]",
3666 SHOW_STR
3667 IP_STR
3668 IGMP_STR
3669 VRF_CMD_HELP_STR
3670 "IGMP statistics\n"
3671 "interface\n"
3672 "IGMP interface\n"
3673 JSON_STR)
3674 {
3675 int idx = 2;
3676 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3677 bool uj = use_json(argc, argv);
3678
3679 if (!vrf)
3680 return CMD_WARNING;
3681
3682 if (argv_find(argv, argc, "WORD", &idx))
3683 igmp_show_statistics(vrf->info, vty, argv[idx]->arg, uj);
3684 else
3685 igmp_show_statistics(vrf->info, vty, NULL, uj);
3686
3687 return CMD_SUCCESS;
3688 }
3689
3690 DEFUN (show_ip_pim_assert,
3691 show_ip_pim_assert_cmd,
3692 "show ip pim [vrf NAME] assert",
3693 SHOW_STR
3694 IP_STR
3695 PIM_STR
3696 VRF_CMD_HELP_STR
3697 "PIM interface assert\n")
3698 {
3699 int idx = 2;
3700 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3701
3702 if (!vrf)
3703 return CMD_WARNING;
3704
3705 pim_show_assert(vrf->info, vty);
3706
3707 return CMD_SUCCESS;
3708 }
3709
3710 DEFUN (show_ip_pim_assert_internal,
3711 show_ip_pim_assert_internal_cmd,
3712 "show ip pim [vrf NAME] assert-internal",
3713 SHOW_STR
3714 IP_STR
3715 PIM_STR
3716 VRF_CMD_HELP_STR
3717 "PIM interface internal assert state\n")
3718 {
3719 int idx = 2;
3720 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3721
3722 if (!vrf)
3723 return CMD_WARNING;
3724
3725 pim_show_assert_internal(vrf->info, vty);
3726
3727 return CMD_SUCCESS;
3728 }
3729
3730 DEFUN (show_ip_pim_assert_metric,
3731 show_ip_pim_assert_metric_cmd,
3732 "show ip pim [vrf NAME] assert-metric",
3733 SHOW_STR
3734 IP_STR
3735 PIM_STR
3736 VRF_CMD_HELP_STR
3737 "PIM interface assert metric\n")
3738 {
3739 int idx = 2;
3740 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3741
3742 if (!vrf)
3743 return CMD_WARNING;
3744
3745 pim_show_assert_metric(vrf->info, vty);
3746
3747 return CMD_SUCCESS;
3748 }
3749
3750 DEFUN (show_ip_pim_assert_winner_metric,
3751 show_ip_pim_assert_winner_metric_cmd,
3752 "show ip pim [vrf NAME] assert-winner-metric",
3753 SHOW_STR
3754 IP_STR
3755 PIM_STR
3756 VRF_CMD_HELP_STR
3757 "PIM interface assert winner metric\n")
3758 {
3759 int idx = 2;
3760 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3761
3762 if (!vrf)
3763 return CMD_WARNING;
3764
3765 pim_show_assert_winner_metric(vrf->info, vty);
3766
3767 return CMD_SUCCESS;
3768 }
3769
3770 DEFUN (show_ip_pim_interface,
3771 show_ip_pim_interface_cmd,
3772 "show ip pim [vrf NAME] interface [detail|WORD] [json]",
3773 SHOW_STR
3774 IP_STR
3775 PIM_STR
3776 VRF_CMD_HELP_STR
3777 "PIM interface information\n"
3778 "Detailed output\n"
3779 "interface name\n"
3780 JSON_STR)
3781 {
3782 int idx = 2;
3783 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3784 bool uj = use_json(argc, argv);
3785
3786 if (!vrf)
3787 return CMD_WARNING;
3788
3789 if (argv_find(argv, argc, "WORD", &idx)
3790 || argv_find(argv, argc, "detail", &idx))
3791 pim_show_interfaces_single(vrf->info, vty, argv[idx]->arg, uj);
3792 else
3793 pim_show_interfaces(vrf->info, vty, uj);
3794
3795 return CMD_SUCCESS;
3796 }
3797
3798 DEFUN (show_ip_pim_interface_vrf_all,
3799 show_ip_pim_interface_vrf_all_cmd,
3800 "show ip pim vrf all interface [detail|WORD] [json]",
3801 SHOW_STR
3802 IP_STR
3803 PIM_STR
3804 VRF_CMD_HELP_STR
3805 "PIM interface information\n"
3806 "Detailed output\n"
3807 "interface name\n"
3808 JSON_STR)
3809 {
3810 int idx = 6;
3811 bool uj = use_json(argc, argv);
3812 struct vrf *vrf;
3813 bool first = true;
3814
3815 if (uj)
3816 vty_out(vty, "{ ");
3817 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
3818 if (uj) {
3819 if (!first)
3820 vty_out(vty, ", ");
3821 vty_out(vty, " \"%s\": ", vrf->name);
3822 first = false;
3823 } else
3824 vty_out(vty, "VRF: %s\n", vrf->name);
3825 if (argv_find(argv, argc, "WORD", &idx)
3826 || argv_find(argv, argc, "detail", &idx))
3827 pim_show_interfaces_single(vrf->info, vty,
3828 argv[idx]->arg, uj);
3829 else
3830 pim_show_interfaces(vrf->info, vty, uj);
3831 }
3832 if (uj)
3833 vty_out(vty, "}\n");
3834
3835 return CMD_SUCCESS;
3836 }
3837
3838 DEFPY (show_ip_pim_join,
3839 show_ip_pim_join_cmd,
3840 "show ip pim [vrf NAME] join [A.B.C.D$s_or_g [A.B.C.D$g]] [json$json]",
3841 SHOW_STR
3842 IP_STR
3843 PIM_STR
3844 VRF_CMD_HELP_STR
3845 "PIM interface join information\n"
3846 "The Source or Group\n"
3847 "The Group\n"
3848 JSON_STR)
3849 {
3850 struct prefix_sg sg = {0};
3851 struct vrf *v;
3852 bool uj = !!json;
3853 struct pim_instance *pim;
3854
3855 v = vrf_lookup_by_name(vrf ? vrf : VRF_DEFAULT_NAME);
3856
3857 if (!v) {
3858 vty_out(vty, "%% Vrf specified: %s does not exist\n", vrf);
3859 return CMD_WARNING;
3860 }
3861 pim = pim_get_pim_instance(v->vrf_id);
3862
3863 if (!pim) {
3864 vty_out(vty, "%% Unable to find pim instance\n");
3865 return CMD_WARNING;
3866 }
3867
3868 if (s_or_g.s_addr != 0) {
3869 if (g.s_addr != 0) {
3870 sg.src = s_or_g;
3871 sg.grp = g;
3872 } else
3873 sg.grp = s_or_g;
3874 }
3875
3876 pim_show_join(pim, vty, &sg, uj);
3877
3878 return CMD_SUCCESS;
3879 }
3880
3881 DEFUN (show_ip_pim_join_vrf_all,
3882 show_ip_pim_join_vrf_all_cmd,
3883 "show ip pim vrf all join [json]",
3884 SHOW_STR
3885 IP_STR
3886 PIM_STR
3887 VRF_CMD_HELP_STR
3888 "PIM interface join information\n"
3889 JSON_STR)
3890 {
3891 struct prefix_sg sg = {0};
3892 bool uj = use_json(argc, argv);
3893 struct vrf *vrf;
3894 bool first = true;
3895
3896 if (uj)
3897 vty_out(vty, "{ ");
3898 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
3899 if (uj) {
3900 if (!first)
3901 vty_out(vty, ", ");
3902 vty_out(vty, " \"%s\": ", vrf->name);
3903 first = false;
3904 } else
3905 vty_out(vty, "VRF: %s\n", vrf->name);
3906 pim_show_join(vrf->info, vty, &sg, uj);
3907 }
3908 if (uj)
3909 vty_out(vty, "}\n");
3910
3911 return CMD_WARNING;
3912 }
3913
3914 DEFUN (show_ip_pim_local_membership,
3915 show_ip_pim_local_membership_cmd,
3916 "show ip pim [vrf NAME] local-membership [json]",
3917 SHOW_STR
3918 IP_STR
3919 PIM_STR
3920 VRF_CMD_HELP_STR
3921 "PIM interface local-membership\n"
3922 JSON_STR)
3923 {
3924 int idx = 2;
3925 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3926 bool uj = use_json(argc, argv);
3927
3928 if (!vrf)
3929 return CMD_WARNING;
3930
3931 pim_show_membership(vrf->info, vty, uj);
3932
3933 return CMD_SUCCESS;
3934 }
3935
3936 DEFUN (show_ip_pim_neighbor,
3937 show_ip_pim_neighbor_cmd,
3938 "show ip pim [vrf NAME] neighbor [detail|WORD] [json]",
3939 SHOW_STR
3940 IP_STR
3941 PIM_STR
3942 VRF_CMD_HELP_STR
3943 "PIM neighbor information\n"
3944 "Detailed output\n"
3945 "Name of interface or neighbor\n"
3946 JSON_STR)
3947 {
3948 int idx = 2;
3949 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
3950 bool uj = use_json(argc, argv);
3951
3952 if (!vrf)
3953 return CMD_WARNING;
3954
3955 if (argv_find(argv, argc, "detail", &idx)
3956 || argv_find(argv, argc, "WORD", &idx))
3957 pim_show_neighbors_single(vrf->info, vty, argv[idx]->arg, uj);
3958 else
3959 pim_show_neighbors(vrf->info, vty, uj);
3960
3961 return CMD_SUCCESS;
3962 }
3963
3964 DEFUN (show_ip_pim_neighbor_vrf_all,
3965 show_ip_pim_neighbor_vrf_all_cmd,
3966 "show ip pim vrf all neighbor [detail|WORD] [json]",
3967 SHOW_STR
3968 IP_STR
3969 PIM_STR
3970 VRF_CMD_HELP_STR
3971 "PIM neighbor information\n"
3972 "Detailed output\n"
3973 "Name of interface or neighbor\n"
3974 JSON_STR)
3975 {
3976 int idx = 2;
3977 bool uj = use_json(argc, argv);
3978 struct vrf *vrf;
3979 bool first = true;
3980
3981 if (uj)
3982 vty_out(vty, "{ ");
3983 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
3984 if (uj) {
3985 if (!first)
3986 vty_out(vty, ", ");
3987 vty_out(vty, " \"%s\": ", vrf->name);
3988 first = false;
3989 } else
3990 vty_out(vty, "VRF: %s\n", vrf->name);
3991 if (argv_find(argv, argc, "detail", &idx)
3992 || argv_find(argv, argc, "WORD", &idx))
3993 pim_show_neighbors_single(vrf->info, vty,
3994 argv[idx]->arg, uj);
3995 else
3996 pim_show_neighbors(vrf->info, vty, uj);
3997 }
3998 if (uj)
3999 vty_out(vty, "}\n");
4000
4001 return CMD_SUCCESS;
4002 }
4003
4004 DEFUN (show_ip_pim_secondary,
4005 show_ip_pim_secondary_cmd,
4006 "show ip pim [vrf NAME] secondary",
4007 SHOW_STR
4008 IP_STR
4009 PIM_STR
4010 VRF_CMD_HELP_STR
4011 "PIM neighbor addresses\n")
4012 {
4013 int idx = 2;
4014 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4015
4016 if (!vrf)
4017 return CMD_WARNING;
4018
4019 pim_show_neighbors_secondary(vrf->info, vty);
4020
4021 return CMD_SUCCESS;
4022 }
4023
4024 DEFUN (show_ip_pim_state,
4025 show_ip_pim_state_cmd,
4026 "show ip pim [vrf NAME] state [A.B.C.D [A.B.C.D]] [json]",
4027 SHOW_STR
4028 IP_STR
4029 PIM_STR
4030 VRF_CMD_HELP_STR
4031 "PIM state information\n"
4032 "Unicast or Multicast address\n"
4033 "Multicast address\n"
4034 JSON_STR)
4035 {
4036 const char *src_or_group = NULL;
4037 const char *group = NULL;
4038 int idx = 2;
4039 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4040 bool uj = use_json(argc, argv);
4041
4042 if (!vrf)
4043 return CMD_WARNING;
4044
4045 if (uj)
4046 argc--;
4047
4048 if (argv_find(argv, argc, "A.B.C.D", &idx)) {
4049 src_or_group = argv[idx]->arg;
4050 if (idx + 1 < argc)
4051 group = argv[idx + 1]->arg;
4052 }
4053
4054 pim_show_state(vrf->info, vty, src_or_group, group, uj);
4055
4056 return CMD_SUCCESS;
4057 }
4058
4059 DEFUN (show_ip_pim_state_vrf_all,
4060 show_ip_pim_state_vrf_all_cmd,
4061 "show ip pim vrf all state [A.B.C.D [A.B.C.D]] [json]",
4062 SHOW_STR
4063 IP_STR
4064 PIM_STR
4065 VRF_CMD_HELP_STR
4066 "PIM state information\n"
4067 "Unicast or Multicast address\n"
4068 "Multicast address\n"
4069 JSON_STR)
4070 {
4071 const char *src_or_group = NULL;
4072 const char *group = NULL;
4073 int idx = 2;
4074 bool uj = use_json(argc, argv);
4075 struct vrf *vrf;
4076 bool first = true;
4077
4078 if (uj) {
4079 vty_out(vty, "{ ");
4080 argc--;
4081 }
4082
4083 if (argv_find(argv, argc, "A.B.C.D", &idx)) {
4084 src_or_group = argv[idx]->arg;
4085 if (idx + 1 < argc)
4086 group = argv[idx + 1]->arg;
4087 }
4088
4089 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
4090 if (uj) {
4091 if (!first)
4092 vty_out(vty, ", ");
4093 vty_out(vty, " \"%s\": ", vrf->name);
4094 first = false;
4095 } else
4096 vty_out(vty, "VRF: %s\n", vrf->name);
4097 pim_show_state(vrf->info, vty, src_or_group, group, uj);
4098 }
4099 if (uj)
4100 vty_out(vty, "}\n");
4101
4102 return CMD_SUCCESS;
4103 }
4104
4105 DEFPY (show_ip_pim_upstream,
4106 show_ip_pim_upstream_cmd,
4107 "show ip pim [vrf NAME] upstream [A.B.C.D$s_or_g [A.B.C.D$g]] [json$json]",
4108 SHOW_STR
4109 IP_STR
4110 PIM_STR
4111 VRF_CMD_HELP_STR
4112 "PIM upstream information\n"
4113 "The Source or Group\n"
4114 "The Group\n"
4115 JSON_STR)
4116 {
4117 struct prefix_sg sg = {0};
4118 struct vrf *v;
4119 bool uj = !!json;
4120 struct pim_instance *pim;
4121
4122 v = vrf_lookup_by_name(vrf ? vrf : VRF_DEFAULT_NAME);
4123
4124 if (!v) {
4125 vty_out(vty, "%% Vrf specified: %s does not exist\n", vrf);
4126 return CMD_WARNING;
4127 }
4128 pim = pim_get_pim_instance(v->vrf_id);
4129
4130 if (!pim) {
4131 vty_out(vty, "%% Unable to find pim instance\n");
4132 return CMD_WARNING;
4133 }
4134
4135 if (s_or_g.s_addr != 0) {
4136 if (g.s_addr != 0) {
4137 sg.src = s_or_g;
4138 sg.grp = g;
4139 } else
4140 sg.grp = s_or_g;
4141 }
4142 pim_show_upstream(pim, vty, &sg, uj);
4143
4144 return CMD_SUCCESS;
4145 }
4146
4147 DEFUN (show_ip_pim_upstream_vrf_all,
4148 show_ip_pim_upstream_vrf_all_cmd,
4149 "show ip pim vrf all upstream [json]",
4150 SHOW_STR
4151 IP_STR
4152 PIM_STR
4153 VRF_CMD_HELP_STR
4154 "PIM upstream information\n"
4155 JSON_STR)
4156 {
4157 struct prefix_sg sg = {0};
4158 bool uj = use_json(argc, argv);
4159 struct vrf *vrf;
4160 bool first = true;
4161
4162 if (uj)
4163 vty_out(vty, "{ ");
4164 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
4165 if (uj) {
4166 if (!first)
4167 vty_out(vty, ", ");
4168 vty_out(vty, " \"%s\": ", vrf->name);
4169 first = false;
4170 } else
4171 vty_out(vty, "VRF: %s\n", vrf->name);
4172 pim_show_upstream(vrf->info, vty, &sg, uj);
4173 }
4174
4175 return CMD_SUCCESS;
4176 }
4177
4178 DEFUN (show_ip_pim_upstream_join_desired,
4179 show_ip_pim_upstream_join_desired_cmd,
4180 "show ip pim [vrf NAME] upstream-join-desired [json]",
4181 SHOW_STR
4182 IP_STR
4183 PIM_STR
4184 VRF_CMD_HELP_STR
4185 "PIM upstream join-desired\n"
4186 JSON_STR)
4187 {
4188 int idx = 2;
4189 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4190 bool uj = use_json(argc, argv);
4191
4192 if (!vrf)
4193 return CMD_WARNING;
4194
4195 pim_show_join_desired(vrf->info, vty, uj);
4196
4197 return CMD_SUCCESS;
4198 }
4199
4200 DEFUN (show_ip_pim_upstream_rpf,
4201 show_ip_pim_upstream_rpf_cmd,
4202 "show ip pim [vrf NAME] upstream-rpf [json]",
4203 SHOW_STR
4204 IP_STR
4205 PIM_STR
4206 VRF_CMD_HELP_STR
4207 "PIM upstream source rpf\n"
4208 JSON_STR)
4209 {
4210 int idx = 2;
4211 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4212 bool uj = use_json(argc, argv);
4213
4214 if (!vrf)
4215 return CMD_WARNING;
4216
4217 pim_show_upstream_rpf(vrf->info, vty, uj);
4218
4219 return CMD_SUCCESS;
4220 }
4221
4222 DEFUN (show_ip_pim_rp,
4223 show_ip_pim_rp_cmd,
4224 "show ip pim [vrf NAME] rp-info [json]",
4225 SHOW_STR
4226 IP_STR
4227 PIM_STR
4228 VRF_CMD_HELP_STR
4229 "PIM RP information\n"
4230 JSON_STR)
4231 {
4232 int idx = 2;
4233 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4234 bool uj = use_json(argc, argv);
4235
4236 if (!vrf)
4237 return CMD_WARNING;
4238
4239 pim_rp_show_information(vrf->info, vty, uj);
4240
4241 return CMD_SUCCESS;
4242 }
4243
4244 DEFUN (show_ip_pim_rp_vrf_all,
4245 show_ip_pim_rp_vrf_all_cmd,
4246 "show ip pim vrf all rp-info [json]",
4247 SHOW_STR
4248 IP_STR
4249 PIM_STR
4250 VRF_CMD_HELP_STR
4251 "PIM RP information\n"
4252 JSON_STR)
4253 {
4254 bool uj = use_json(argc, argv);
4255 struct vrf *vrf;
4256 bool first = true;
4257
4258 if (uj)
4259 vty_out(vty, "{ ");
4260 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
4261 if (uj) {
4262 if (!first)
4263 vty_out(vty, ", ");
4264 vty_out(vty, " \"%s\": ", vrf->name);
4265 first = false;
4266 } else
4267 vty_out(vty, "VRF: %s\n", vrf->name);
4268 pim_rp_show_information(vrf->info, vty, uj);
4269 }
4270 if (uj)
4271 vty_out(vty, "}\n");
4272
4273 return CMD_SUCCESS;
4274 }
4275
4276 DEFUN (show_ip_pim_rpf,
4277 show_ip_pim_rpf_cmd,
4278 "show ip pim [vrf NAME] rpf [json]",
4279 SHOW_STR
4280 IP_STR
4281 PIM_STR
4282 VRF_CMD_HELP_STR
4283 "PIM cached source rpf information\n"
4284 JSON_STR)
4285 {
4286 int idx = 2;
4287 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4288 bool uj = use_json(argc, argv);
4289
4290 if (!vrf)
4291 return CMD_WARNING;
4292
4293 pim_show_rpf(vrf->info, vty, uj);
4294
4295 return CMD_SUCCESS;
4296 }
4297
4298 DEFUN (show_ip_pim_rpf_vrf_all,
4299 show_ip_pim_rpf_vrf_all_cmd,
4300 "show ip pim vrf all rpf [json]",
4301 SHOW_STR
4302 IP_STR
4303 PIM_STR
4304 VRF_CMD_HELP_STR
4305 "PIM cached source rpf information\n"
4306 JSON_STR)
4307 {
4308 bool uj = use_json(argc, argv);
4309 struct vrf *vrf;
4310 bool first = true;
4311
4312 if (uj)
4313 vty_out(vty, "{ ");
4314 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
4315 if (uj) {
4316 if (!first)
4317 vty_out(vty, ", ");
4318 vty_out(vty, " \"%s\": ", vrf->name);
4319 first = false;
4320 } else
4321 vty_out(vty, "VRF: %s\n", vrf->name);
4322 pim_show_rpf(vrf->info, vty, uj);
4323 }
4324 if (uj)
4325 vty_out(vty, "}\n");
4326
4327 return CMD_SUCCESS;
4328 }
4329
4330 DEFUN (show_ip_pim_nexthop,
4331 show_ip_pim_nexthop_cmd,
4332 "show ip pim [vrf NAME] nexthop",
4333 SHOW_STR
4334 IP_STR
4335 PIM_STR
4336 VRF_CMD_HELP_STR
4337 "PIM cached nexthop rpf information\n")
4338 {
4339 int idx = 2;
4340 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4341
4342 if (!vrf)
4343 return CMD_WARNING;
4344
4345 pim_show_nexthop(vrf->info, vty);
4346
4347 return CMD_SUCCESS;
4348 }
4349
4350 DEFUN (show_ip_pim_nexthop_lookup,
4351 show_ip_pim_nexthop_lookup_cmd,
4352 "show ip pim [vrf NAME] nexthop-lookup A.B.C.D A.B.C.D",
4353 SHOW_STR
4354 IP_STR
4355 PIM_STR
4356 VRF_CMD_HELP_STR
4357 "PIM cached nexthop rpf lookup\n"
4358 "Source/RP address\n"
4359 "Multicast Group address\n")
4360 {
4361 struct prefix nht_p;
4362 int result = 0;
4363 struct in_addr src_addr, grp_addr;
4364 struct in_addr vif_source;
4365 const char *addr_str, *addr_str1;
4366 struct prefix grp;
4367 struct pim_nexthop nexthop;
4368 char nexthop_addr_str[PREFIX_STRLEN];
4369 char grp_str[PREFIX_STRLEN];
4370 int idx = 2;
4371 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4372
4373 if (!vrf)
4374 return CMD_WARNING;
4375
4376 argv_find(argv, argc, "A.B.C.D", &idx);
4377 addr_str = argv[idx]->arg;
4378 result = inet_pton(AF_INET, addr_str, &src_addr);
4379 if (result <= 0) {
4380 vty_out(vty, "Bad unicast address %s: errno=%d: %s\n", addr_str,
4381 errno, safe_strerror(errno));
4382 return CMD_WARNING;
4383 }
4384
4385 if (pim_is_group_224_4(src_addr)) {
4386 vty_out(vty,
4387 "Invalid argument. Expected Valid Source Address.\n");
4388 return CMD_WARNING;
4389 }
4390
4391 addr_str1 = argv[idx + 1]->arg;
4392 result = inet_pton(AF_INET, addr_str1, &grp_addr);
4393 if (result <= 0) {
4394 vty_out(vty, "Bad unicast address %s: errno=%d: %s\n", addr_str,
4395 errno, safe_strerror(errno));
4396 return CMD_WARNING;
4397 }
4398
4399 if (!pim_is_group_224_4(grp_addr)) {
4400 vty_out(vty,
4401 "Invalid argument. Expected Valid Multicast Group Address.\n");
4402 return CMD_WARNING;
4403 }
4404
4405 if (!pim_rp_set_upstream_addr(vrf->info, &vif_source, src_addr,
4406 grp_addr))
4407 return CMD_SUCCESS;
4408
4409 nht_p.family = AF_INET;
4410 nht_p.prefixlen = IPV4_MAX_BITLEN;
4411 nht_p.u.prefix4 = vif_source;
4412 grp.family = AF_INET;
4413 grp.prefixlen = IPV4_MAX_BITLEN;
4414 grp.u.prefix4 = grp_addr;
4415 memset(&nexthop, 0, sizeof(nexthop));
4416
4417 result = pim_ecmp_nexthop_lookup(vrf->info, &nexthop, &nht_p, &grp, 0);
4418
4419 if (!result) {
4420 vty_out(vty,
4421 "Nexthop Lookup failed, no usable routes returned.\n");
4422 return CMD_SUCCESS;
4423 }
4424
4425 pim_addr_dump("<grp?>", &grp, grp_str, sizeof(grp_str));
4426 pim_addr_dump("<nexthop?>", &nexthop.mrib_nexthop_addr,
4427 nexthop_addr_str, sizeof(nexthop_addr_str));
4428 vty_out(vty, "Group %s --- Nexthop %s Interface %s \n", grp_str,
4429 nexthop_addr_str, nexthop.interface->name);
4430
4431 return CMD_SUCCESS;
4432 }
4433
4434 DEFUN (show_ip_pim_interface_traffic,
4435 show_ip_pim_interface_traffic_cmd,
4436 "show ip pim [vrf NAME] interface traffic [WORD] [json]",
4437 SHOW_STR
4438 IP_STR
4439 PIM_STR
4440 VRF_CMD_HELP_STR
4441 "PIM interface information\n"
4442 "Protocol Packet counters\n"
4443 "Interface name\n"
4444 JSON_STR)
4445 {
4446 int idx = 2;
4447 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4448 bool uj = use_json(argc, argv);
4449
4450 if (!vrf)
4451 return CMD_WARNING;
4452
4453 if (argv_find(argv, argc, "WORD", &idx))
4454 pim_show_interface_traffic_single(vrf->info, vty,
4455 argv[idx]->arg, uj);
4456 else
4457 pim_show_interface_traffic(vrf->info, vty, uj);
4458
4459 return CMD_SUCCESS;
4460 }
4461
4462 static void show_multicast_interfaces(struct pim_instance *pim, struct vty *vty)
4463 {
4464 struct interface *ifp;
4465
4466 vty_out(vty, "\n");
4467
4468 vty_out(vty,
4469 "Interface Address ifi Vif PktsIn PktsOut BytesIn BytesOut\n");
4470
4471 FOR_ALL_INTERFACES (pim->vrf, ifp) {
4472 struct pim_interface *pim_ifp;
4473 struct in_addr ifaddr;
4474 struct sioc_vif_req vreq;
4475
4476 pim_ifp = ifp->info;
4477
4478 if (!pim_ifp)
4479 continue;
4480
4481 memset(&vreq, 0, sizeof(vreq));
4482 vreq.vifi = pim_ifp->mroute_vif_index;
4483
4484 if (ioctl(pim->mroute_socket, SIOCGETVIFCNT, &vreq)) {
4485 zlog_warn(
4486 "ioctl(SIOCGETVIFCNT=%lu) failure for interface %s vif_index=%d: errno=%d: %s",
4487 (unsigned long)SIOCGETVIFCNT, ifp->name,
4488 pim_ifp->mroute_vif_index, errno,
4489 safe_strerror(errno));
4490 }
4491
4492 ifaddr = pim_ifp->primary_address;
4493
4494 vty_out(vty, "%-16s %-15s %3d %3d %7lu %7lu %10lu %10lu\n",
4495 ifp->name, inet_ntoa(ifaddr), ifp->ifindex,
4496 pim_ifp->mroute_vif_index, (unsigned long)vreq.icount,
4497 (unsigned long)vreq.ocount, (unsigned long)vreq.ibytes,
4498 (unsigned long)vreq.obytes);
4499 }
4500 }
4501
4502 static void pim_cmd_show_ip_multicast_helper(struct pim_instance *pim,
4503 struct vty *vty)
4504 {
4505 struct vrf *vrf = pim->vrf;
4506 time_t now = pim_time_monotonic_sec();
4507 char uptime[10];
4508 char mlag_role[80];
4509
4510 pim = vrf->info;
4511
4512 vty_out(vty, "Router MLAG Role: %s\n",
4513 mlag_role2str(router->role, mlag_role, sizeof(mlag_role)));
4514 vty_out(vty, "Mroute socket descriptor:");
4515
4516 vty_out(vty, " %d(%s)\n", pim->mroute_socket, vrf->name);
4517
4518 pim_time_uptime(uptime, sizeof(uptime),
4519 now - pim->mroute_socket_creation);
4520 vty_out(vty, "Mroute socket uptime: %s\n", uptime);
4521
4522 vty_out(vty, "\n");
4523
4524 pim_zebra_zclient_update(vty);
4525 pim_zlookup_show_ip_multicast(vty);
4526
4527 vty_out(vty, "\n");
4528 vty_out(vty, "Maximum highest VifIndex: %d\n", PIM_MAX_USABLE_VIFS);
4529
4530 vty_out(vty, "\n");
4531 vty_out(vty, "Upstream Join Timer: %d secs\n", router->t_periodic);
4532 vty_out(vty, "Join/Prune Holdtime: %d secs\n", PIM_JP_HOLDTIME);
4533 vty_out(vty, "PIM ECMP: %s\n", pim->ecmp_enable ? "Enable" : "Disable");
4534 vty_out(vty, "PIM ECMP Rebalance: %s\n",
4535 pim->ecmp_rebalance_enable ? "Enable" : "Disable");
4536
4537 vty_out(vty, "\n");
4538
4539 show_rpf_refresh_stats(vty, pim, now, NULL);
4540
4541 vty_out(vty, "\n");
4542
4543 show_scan_oil_stats(pim, vty, now);
4544
4545 show_multicast_interfaces(pim, vty);
4546 }
4547
4548 DEFUN (show_ip_multicast,
4549 show_ip_multicast_cmd,
4550 "show ip multicast [vrf NAME]",
4551 SHOW_STR
4552 IP_STR
4553 VRF_CMD_HELP_STR
4554 "Multicast global information\n")
4555 {
4556 int idx = 2;
4557 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
4558
4559 if (!vrf)
4560 return CMD_WARNING;
4561
4562 pim_cmd_show_ip_multicast_helper(vrf->info, vty);
4563
4564 return CMD_SUCCESS;
4565 }
4566
4567 DEFUN (show_ip_multicast_vrf_all,
4568 show_ip_multicast_vrf_all_cmd,
4569 "show ip multicast vrf all",
4570 SHOW_STR
4571 IP_STR
4572 VRF_CMD_HELP_STR
4573 "Multicast global information\n")
4574 {
4575 bool uj = use_json(argc, argv);
4576 struct vrf *vrf;
4577 bool first = true;
4578
4579 if (uj)
4580 vty_out(vty, "{ ");
4581 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
4582 if (uj) {
4583 if (!first)
4584 vty_out(vty, ", ");
4585 vty_out(vty, " \"%s\": ", vrf->name);
4586 first = false;
4587 } else
4588 vty_out(vty, "VRF: %s\n", vrf->name);
4589 pim_cmd_show_ip_multicast_helper(vrf->info, vty);
4590 }
4591 if (uj)
4592 vty_out(vty, "}\n");
4593
4594 return CMD_SUCCESS;
4595 }
4596
4597 static void show_mroute(struct pim_instance *pim, struct vty *vty,
4598 struct prefix_sg *sg, bool fill, bool uj)
4599 {
4600 struct listnode *node;
4601 struct channel_oil *c_oil;
4602 struct static_route *s_route;
4603 time_t now;
4604 json_object *json = NULL;
4605 json_object *json_group = NULL;
4606 json_object *json_source = NULL;
4607 json_object *json_oil = NULL;
4608 json_object *json_ifp_out = NULL;
4609 int found_oif;
4610 int first;
4611 char grp_str[INET_ADDRSTRLEN];
4612 char src_str[INET_ADDRSTRLEN];
4613 char in_ifname[INTERFACE_NAMSIZ + 1];
4614 char out_ifname[INTERFACE_NAMSIZ + 1];
4615 int oif_vif_index;
4616 struct interface *ifp_in;
4617 char proto[100];
4618
4619 if (uj) {
4620 json = json_object_new_object();
4621 } else {
4622 vty_out(vty,
4623 "Source Group Proto Input Output TTL Uptime\n");
4624 }
4625
4626 now = pim_time_monotonic_sec();
4627
4628 /* print list of PIM and IGMP routes */
4629 for (ALL_LIST_ELEMENTS_RO(pim->channel_oil_list, node, c_oil)) {
4630 found_oif = 0;
4631 first = 1;
4632 if (!c_oil->installed && !uj)
4633 continue;
4634
4635 if (sg->grp.s_addr != 0 &&
4636 sg->grp.s_addr != c_oil->oil.mfcc_mcastgrp.s_addr)
4637 continue;
4638 if (sg->src.s_addr != 0 &&
4639 sg->src.s_addr != c_oil->oil.mfcc_origin.s_addr)
4640 continue;
4641
4642 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp, grp_str,
4643 sizeof(grp_str));
4644 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin, src_str,
4645 sizeof(src_str));
4646 ifp_in = pim_if_find_by_vif_index(pim, c_oil->oil.mfcc_parent);
4647
4648 if (ifp_in)
4649 strcpy(in_ifname, ifp_in->name);
4650 else
4651 strcpy(in_ifname, "<iif?>");
4652
4653 if (uj) {
4654
4655 /* Find the group, create it if it doesn't exist */
4656 json_object_object_get_ex(json, grp_str, &json_group);
4657
4658 if (!json_group) {
4659 json_group = json_object_new_object();
4660 json_object_object_add(json, grp_str,
4661 json_group);
4662 }
4663
4664 /* Find the source nested under the group, create it if
4665 * it doesn't exist */
4666 json_object_object_get_ex(json_group, src_str,
4667 &json_source);
4668
4669 if (!json_source) {
4670 json_source = json_object_new_object();
4671 json_object_object_add(json_group, src_str,
4672 json_source);
4673 }
4674
4675 /* Find the inbound interface nested under the source,
4676 * create it if it doesn't exist */
4677 json_object_int_add(json_source, "installed",
4678 c_oil->installed);
4679 json_object_int_add(json_source, "refCount",
4680 c_oil->oil_ref_count);
4681 json_object_int_add(json_source, "oilSize",
4682 c_oil->oil_size);
4683 json_object_int_add(json_source, "OilInheritedRescan",
4684 c_oil->oil_inherited_rescan);
4685 json_object_string_add(json_source, "iif", in_ifname);
4686 json_oil = NULL;
4687 }
4688
4689 for (oif_vif_index = 0; oif_vif_index < MAXVIFS;
4690 ++oif_vif_index) {
4691 struct interface *ifp_out;
4692 char mroute_uptime[10];
4693 int ttl;
4694
4695 ttl = c_oil->oil.mfcc_ttls[oif_vif_index];
4696 if (ttl < 1)
4697 continue;
4698
4699 ifp_out = pim_if_find_by_vif_index(pim, oif_vif_index);
4700 pim_time_uptime(
4701 mroute_uptime, sizeof(mroute_uptime),
4702 now - c_oil->mroute_creation);
4703 found_oif = 1;
4704
4705 if (ifp_out)
4706 strcpy(out_ifname, ifp_out->name);
4707 else
4708 strcpy(out_ifname, "<oif?>");
4709
4710 if (uj) {
4711 json_ifp_out = json_object_new_object();
4712 json_object_string_add(json_ifp_out, "source",
4713 src_str);
4714 json_object_string_add(json_ifp_out, "group",
4715 grp_str);
4716
4717 if (c_oil->oif_flags[oif_vif_index]
4718 & PIM_OIF_FLAG_PROTO_PIM)
4719 json_object_boolean_true_add(
4720 json_ifp_out, "protocolPim");
4721
4722 if (c_oil->oif_flags[oif_vif_index]
4723 & PIM_OIF_FLAG_PROTO_IGMP)
4724 json_object_boolean_true_add(
4725 json_ifp_out, "protocolIgmp");
4726
4727 if (c_oil->oif_flags[oif_vif_index]
4728 & PIM_OIF_FLAG_PROTO_VXLAN)
4729 json_object_boolean_true_add(
4730 json_ifp_out, "protocolVxlan");
4731
4732 if (c_oil->oif_flags[oif_vif_index]
4733 & PIM_OIF_FLAG_PROTO_SOURCE)
4734 json_object_boolean_true_add(
4735 json_ifp_out, "protocolSource");
4736
4737 if (c_oil->oif_flags[oif_vif_index]
4738 & PIM_OIF_FLAG_PROTO_STAR)
4739 json_object_boolean_true_add(
4740 json_ifp_out,
4741 "protocolInherited");
4742
4743 json_object_string_add(json_ifp_out,
4744 "inboundInterface",
4745 in_ifname);
4746 json_object_int_add(json_ifp_out, "iVifI",
4747 c_oil->oil.mfcc_parent);
4748 json_object_string_add(json_ifp_out,
4749 "outboundInterface",
4750 out_ifname);
4751 json_object_int_add(json_ifp_out, "oVifI",
4752 oif_vif_index);
4753 json_object_int_add(json_ifp_out, "ttl", ttl);
4754 json_object_string_add(json_ifp_out, "upTime",
4755 mroute_uptime);
4756 if (!json_oil) {
4757 json_oil = json_object_new_object();
4758 json_object_object_add(json_source,
4759 "oil", json_oil);
4760 }
4761 json_object_object_add(json_oil, out_ifname,
4762 json_ifp_out);
4763 } else {
4764 if (c_oil->oif_flags[oif_vif_index]
4765 & PIM_OIF_FLAG_PROTO_PIM) {
4766 strcpy(proto, "PIM");
4767 }
4768
4769 if (c_oil->oif_flags[oif_vif_index]
4770 & PIM_OIF_FLAG_PROTO_IGMP) {
4771 strcpy(proto, "IGMP");
4772 }
4773
4774 if (c_oil->oif_flags[oif_vif_index]
4775 & PIM_OIF_FLAG_PROTO_VXLAN) {
4776 strcpy(proto, "VxLAN");
4777 }
4778
4779 if (c_oil->oif_flags[oif_vif_index]
4780 & PIM_OIF_FLAG_PROTO_SOURCE) {
4781 strcpy(proto, "SRC");
4782 }
4783
4784 if (c_oil->oif_flags[oif_vif_index]
4785 & PIM_OIF_FLAG_PROTO_STAR) {
4786 strcpy(proto, "STAR");
4787 }
4788
4789 vty_out(vty,
4790 "%-15s %-15s %-6s %-16s %-16s %-3d %8s\n",
4791 src_str, grp_str, proto, in_ifname,
4792 out_ifname, ttl, mroute_uptime);
4793
4794 if (first) {
4795 src_str[0] = '\0';
4796 grp_str[0] = '\0';
4797 in_ifname[0] = '\0';
4798 first = 0;
4799 }
4800 }
4801 }
4802
4803 if (!uj && !found_oif) {
4804 vty_out(vty, "%-15s %-15s %-6s %-16s %-16s %-3d %8s\n",
4805 src_str, grp_str, "none", in_ifname, "none", 0,
4806 "--:--:--");
4807 }
4808 }
4809
4810 /* Print list of static routes */
4811 for (ALL_LIST_ELEMENTS_RO(pim->static_routes, node, s_route)) {
4812 first = 1;
4813
4814 if (!s_route->c_oil.installed)
4815 continue;
4816
4817 pim_inet4_dump("<group?>", s_route->group, grp_str,
4818 sizeof(grp_str));
4819 pim_inet4_dump("<source?>", s_route->source, src_str,
4820 sizeof(src_str));
4821 ifp_in = pim_if_find_by_vif_index(pim, s_route->iif);
4822 found_oif = 0;
4823
4824 if (ifp_in)
4825 strcpy(in_ifname, ifp_in->name);
4826 else
4827 strcpy(in_ifname, "<iif?>");
4828
4829 if (uj) {
4830
4831 /* Find the group, create it if it doesn't exist */
4832 json_object_object_get_ex(json, grp_str, &json_group);
4833
4834 if (!json_group) {
4835 json_group = json_object_new_object();
4836 json_object_object_add(json, grp_str,
4837 json_group);
4838 }
4839
4840 /* Find the source nested under the group, create it if
4841 * it doesn't exist */
4842 json_object_object_get_ex(json_group, src_str,
4843 &json_source);
4844
4845 if (!json_source) {
4846 json_source = json_object_new_object();
4847 json_object_object_add(json_group, src_str,
4848 json_source);
4849 }
4850
4851 json_object_string_add(json_source, "iif", in_ifname);
4852 json_oil = NULL;
4853 } else {
4854 strcpy(proto, "STATIC");
4855 }
4856
4857 for (oif_vif_index = 0; oif_vif_index < MAXVIFS;
4858 ++oif_vif_index) {
4859 struct interface *ifp_out;
4860 char oif_uptime[10];
4861 int ttl;
4862
4863 ttl = s_route->oif_ttls[oif_vif_index];
4864 if (ttl < 1)
4865 continue;
4866
4867 ifp_out = pim_if_find_by_vif_index(pim, oif_vif_index);
4868 pim_time_uptime(
4869 oif_uptime, sizeof(oif_uptime),
4870 now
4871 - s_route->c_oil
4872 .oif_creation[oif_vif_index]);
4873 found_oif = 1;
4874
4875 if (ifp_out)
4876 strcpy(out_ifname, ifp_out->name);
4877 else
4878 strcpy(out_ifname, "<oif?>");
4879
4880 if (uj) {
4881 json_ifp_out = json_object_new_object();
4882 json_object_string_add(json_ifp_out, "source",
4883 src_str);
4884 json_object_string_add(json_ifp_out, "group",
4885 grp_str);
4886 json_object_boolean_true_add(json_ifp_out,
4887 "protocolStatic");
4888 json_object_string_add(json_ifp_out,
4889 "inboundInterface",
4890 in_ifname);
4891 json_object_int_add(
4892 json_ifp_out, "iVifI",
4893 s_route->c_oil.oil.mfcc_parent);
4894 json_object_string_add(json_ifp_out,
4895 "outboundInterface",
4896 out_ifname);
4897 json_object_int_add(json_ifp_out, "oVifI",
4898 oif_vif_index);
4899 json_object_int_add(json_ifp_out, "ttl", ttl);
4900 json_object_string_add(json_ifp_out, "upTime",
4901 oif_uptime);
4902 if (!json_oil) {
4903 json_oil = json_object_new_object();
4904 json_object_object_add(json_source,
4905 "oil", json_oil);
4906 }
4907 json_object_object_add(json_oil, out_ifname,
4908 json_ifp_out);
4909 } else {
4910 vty_out(vty,
4911 "%-15s %-15s %-6s %-16s %-16s %-3d %8s %s\n",
4912 src_str, grp_str, proto, in_ifname,
4913 out_ifname, ttl, oif_uptime,
4914 pim->vrf->name);
4915 if (first && !fill) {
4916 src_str[0] = '\0';
4917 grp_str[0] = '\0';
4918 in_ifname[0] = '\0';
4919 first = 0;
4920 }
4921 }
4922 }
4923
4924 if (!uj && !found_oif) {
4925 vty_out(vty,
4926 "%-15s %-15s %-6s %-16s %-16s %-3d %8s %s\n",
4927 src_str, grp_str, proto, in_ifname, "none", 0,
4928 "--:--:--", pim->vrf->name);
4929 }
4930 }
4931
4932 if (uj) {
4933 vty_out(vty, "%s\n", json_object_to_json_string_ext(
4934 json, JSON_C_TO_STRING_PRETTY));
4935 json_object_free(json);
4936 }
4937 }
4938
4939 DEFPY (show_ip_mroute,
4940 show_ip_mroute_cmd,
4941 "show ip mroute [vrf NAME] [A.B.C.D$s_or_g [A.B.C.D$g]] [fill$fill] [json$json]",
4942 SHOW_STR
4943 IP_STR
4944 MROUTE_STR
4945 VRF_CMD_HELP_STR
4946 "The Source or Group\n"
4947 "The Group\n"
4948 "Fill in Assumed data\n"
4949 JSON_STR)
4950 {
4951 struct prefix_sg sg = {0};
4952 struct pim_instance *pim;
4953 struct vrf *v;
4954
4955 v = vrf_lookup_by_name(vrf ? vrf : VRF_DEFAULT_NAME);
4956
4957 if (!v) {
4958 vty_out(vty, "%% Vrf specified: %s does not exist\n", vrf);
4959 return CMD_WARNING;
4960 }
4961 pim = pim_get_pim_instance(v->vrf_id);
4962
4963 if (!pim) {
4964 vty_out(vty, "%% Unable to find pim instance\n");
4965 return CMD_WARNING;
4966 }
4967
4968 if (s_or_g.s_addr != 0) {
4969 if (g.s_addr != 0) {
4970 sg.src = s_or_g;
4971 sg.grp = g;
4972 } else
4973 sg.grp = s_or_g;
4974 }
4975 show_mroute(pim, vty, &sg, !!fill, !!json);
4976 return CMD_SUCCESS;
4977 }
4978
4979 DEFUN (show_ip_mroute_vrf_all,
4980 show_ip_mroute_vrf_all_cmd,
4981 "show ip mroute vrf all [fill] [json]",
4982 SHOW_STR
4983 IP_STR
4984 MROUTE_STR
4985 VRF_CMD_HELP_STR
4986 "Fill in Assumed data\n"
4987 JSON_STR)
4988 {
4989 struct prefix_sg sg = {0};
4990 bool uj = use_json(argc, argv);
4991 int idx = 4;
4992 struct vrf *vrf;
4993 bool first = true;
4994 bool fill = false;
4995
4996 if (argv_find(argv, argc, "fill", &idx))
4997 fill = true;
4998
4999 if (uj)
5000 vty_out(vty, "{ ");
5001 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
5002 if (uj) {
5003 if (!first)
5004 vty_out(vty, ", ");
5005 vty_out(vty, " \"%s\": ", vrf->name);
5006 first = false;
5007 } else
5008 vty_out(vty, "VRF: %s\n", vrf->name);
5009 show_mroute(vrf->info, vty, &sg, fill, uj);
5010 }
5011 if (uj)
5012 vty_out(vty, "}\n");
5013
5014 return CMD_SUCCESS;
5015 }
5016
5017 static void show_mroute_count(struct pim_instance *pim, struct vty *vty)
5018 {
5019 struct listnode *node;
5020 struct channel_oil *c_oil;
5021 struct static_route *s_route;
5022
5023 vty_out(vty, "\n");
5024
5025 vty_out(vty,
5026 "Source Group LastUsed Packets Bytes WrongIf \n");
5027
5028 /* Print PIM and IGMP route counts */
5029 for (ALL_LIST_ELEMENTS_RO(pim->channel_oil_list, node, c_oil)) {
5030 char group_str[INET_ADDRSTRLEN];
5031 char source_str[INET_ADDRSTRLEN];
5032
5033 if (!c_oil->installed)
5034 continue;
5035
5036 pim_mroute_update_counters(c_oil);
5037
5038 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp, group_str,
5039 sizeof(group_str));
5040 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin, source_str,
5041 sizeof(source_str));
5042
5043 vty_out(vty, "%-15s %-15s %-8llu %-7ld %-10ld %-7ld\n",
5044 source_str, group_str, c_oil->cc.lastused / 100,
5045 c_oil->cc.pktcnt, c_oil->cc.bytecnt,
5046 c_oil->cc.wrong_if);
5047 }
5048
5049 for (ALL_LIST_ELEMENTS_RO(pim->static_routes, node, s_route)) {
5050 char group_str[INET_ADDRSTRLEN];
5051 char source_str[INET_ADDRSTRLEN];
5052
5053 if (!s_route->c_oil.installed)
5054 continue;
5055
5056 pim_mroute_update_counters(&s_route->c_oil);
5057
5058 pim_inet4_dump("<group?>", s_route->c_oil.oil.mfcc_mcastgrp,
5059 group_str, sizeof(group_str));
5060 pim_inet4_dump("<source?>", s_route->c_oil.oil.mfcc_origin,
5061 source_str, sizeof(source_str));
5062
5063 vty_out(vty, "%-15s %-15s %-8llu %-7ld %-10ld %-7ld\n",
5064 source_str, group_str, s_route->c_oil.cc.lastused,
5065 s_route->c_oil.cc.pktcnt, s_route->c_oil.cc.bytecnt,
5066 s_route->c_oil.cc.wrong_if);
5067 }
5068 }
5069
5070 DEFUN (show_ip_mroute_count,
5071 show_ip_mroute_count_cmd,
5072 "show ip mroute [vrf NAME] count",
5073 SHOW_STR
5074 IP_STR
5075 MROUTE_STR
5076 VRF_CMD_HELP_STR
5077 "Route and packet count data\n")
5078 {
5079 int idx = 2;
5080 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
5081
5082 if (!vrf)
5083 return CMD_WARNING;
5084
5085 show_mroute_count(vrf->info, vty);
5086 return CMD_SUCCESS;
5087 }
5088
5089 DEFUN (show_ip_mroute_count_vrf_all,
5090 show_ip_mroute_count_vrf_all_cmd,
5091 "show ip mroute vrf all count",
5092 SHOW_STR
5093 IP_STR
5094 MROUTE_STR
5095 VRF_CMD_HELP_STR
5096 "Route and packet count data\n")
5097 {
5098 bool uj = use_json(argc, argv);
5099 struct vrf *vrf;
5100 bool first = true;
5101
5102 if (uj)
5103 vty_out(vty, "{ ");
5104 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
5105 if (uj) {
5106 if (!first)
5107 vty_out(vty, ", ");
5108 vty_out(vty, " \"%s\": ", vrf->name);
5109 first = false;
5110 } else
5111 vty_out(vty, "VRF: %s\n", vrf->name);
5112 show_mroute_count(vrf->info, vty);
5113 }
5114 if (uj)
5115 vty_out(vty, "}\n");
5116
5117 return CMD_SUCCESS;
5118 }
5119
5120 DEFUN (show_ip_rib,
5121 show_ip_rib_cmd,
5122 "show ip rib [vrf NAME] A.B.C.D",
5123 SHOW_STR
5124 IP_STR
5125 RIB_STR
5126 VRF_CMD_HELP_STR
5127 "Unicast address\n")
5128 {
5129 int idx = 2;
5130 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
5131 struct in_addr addr;
5132 const char *addr_str;
5133 struct pim_nexthop nexthop;
5134 char nexthop_addr_str[PREFIX_STRLEN];
5135 int result;
5136
5137 if (!vrf)
5138 return CMD_WARNING;
5139
5140 memset(&nexthop, 0, sizeof(nexthop));
5141 argv_find(argv, argc, "A.B.C.D", &idx);
5142 addr_str = argv[idx]->arg;
5143 result = inet_pton(AF_INET, addr_str, &addr);
5144 if (result <= 0) {
5145 vty_out(vty, "Bad unicast address %s: errno=%d: %s\n", addr_str,
5146 errno, safe_strerror(errno));
5147 return CMD_WARNING;
5148 }
5149
5150 if (!pim_nexthop_lookup(vrf->info, &nexthop, addr, 0)) {
5151 vty_out(vty,
5152 "Failure querying RIB nexthop for unicast address %s\n",
5153 addr_str);
5154 return CMD_WARNING;
5155 }
5156
5157 vty_out(vty,
5158 "Address NextHop Interface Metric Preference\n");
5159
5160 pim_addr_dump("<nexthop?>", &nexthop.mrib_nexthop_addr,
5161 nexthop_addr_str, sizeof(nexthop_addr_str));
5162
5163 vty_out(vty, "%-15s %-15s %-9s %6d %10d\n", addr_str, nexthop_addr_str,
5164 nexthop.interface ? nexthop.interface->name : "<ifname?>",
5165 nexthop.mrib_route_metric, nexthop.mrib_metric_preference);
5166
5167 return CMD_SUCCESS;
5168 }
5169
5170 static void show_ssmpingd(struct pim_instance *pim, struct vty *vty)
5171 {
5172 struct listnode *node;
5173 struct ssmpingd_sock *ss;
5174 time_t now;
5175
5176 vty_out(vty,
5177 "Source Socket Address Port Uptime Requests\n");
5178
5179 if (!pim->ssmpingd_list)
5180 return;
5181
5182 now = pim_time_monotonic_sec();
5183
5184 for (ALL_LIST_ELEMENTS_RO(pim->ssmpingd_list, node, ss)) {
5185 char source_str[INET_ADDRSTRLEN];
5186 char ss_uptime[10];
5187 struct sockaddr_in bind_addr;
5188 socklen_t len = sizeof(bind_addr);
5189 char bind_addr_str[INET_ADDRSTRLEN];
5190
5191 pim_inet4_dump("<src?>", ss->source_addr, source_str,
5192 sizeof(source_str));
5193
5194 if (pim_socket_getsockname(
5195 ss->sock_fd, (struct sockaddr *)&bind_addr, &len)) {
5196 vty_out(vty,
5197 "%% Failure reading socket name for ssmpingd source %s on fd=%d\n",
5198 source_str, ss->sock_fd);
5199 }
5200
5201 pim_inet4_dump("<addr?>", bind_addr.sin_addr, bind_addr_str,
5202 sizeof(bind_addr_str));
5203 pim_time_uptime(ss_uptime, sizeof(ss_uptime),
5204 now - ss->creation);
5205
5206 vty_out(vty, "%-15s %6d %-15s %5d %8s %8lld\n", source_str,
5207 ss->sock_fd, bind_addr_str, ntohs(bind_addr.sin_port),
5208 ss_uptime, (long long)ss->requests);
5209 }
5210 }
5211
5212 DEFUN (show_ip_ssmpingd,
5213 show_ip_ssmpingd_cmd,
5214 "show ip ssmpingd [vrf NAME]",
5215 SHOW_STR
5216 IP_STR
5217 SHOW_SSMPINGD_STR
5218 VRF_CMD_HELP_STR)
5219 {
5220 int idx = 2;
5221 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
5222
5223 if (!vrf)
5224 return CMD_WARNING;
5225
5226 show_ssmpingd(vrf->info, vty);
5227 return CMD_SUCCESS;
5228 }
5229
5230 static int pim_rp_cmd_worker(struct pim_instance *pim, struct vty *vty,
5231 const char *rp, const char *group,
5232 const char *plist)
5233 {
5234 int result;
5235
5236 result = pim_rp_new(pim, rp, group, plist);
5237
5238 if (result == PIM_GROUP_BAD_ADDRESS) {
5239 vty_out(vty, "%% Bad group address specified: %s\n", group);
5240 return CMD_WARNING_CONFIG_FAILED;
5241 }
5242
5243 if (result == PIM_RP_BAD_ADDRESS) {
5244 vty_out(vty, "%% Bad RP address specified: %s\n", rp);
5245 return CMD_WARNING_CONFIG_FAILED;
5246 }
5247
5248 if (result == PIM_RP_NO_PATH) {
5249 vty_out(vty, "%% No Path to RP address specified: %s\n", rp);
5250 return CMD_WARNING;
5251 }
5252
5253 if (result == PIM_GROUP_OVERLAP) {
5254 vty_out(vty,
5255 "%% Group range specified cannot exact match another\n");
5256 return CMD_WARNING_CONFIG_FAILED;
5257 }
5258
5259 if (result == PIM_GROUP_PFXLIST_OVERLAP) {
5260 vty_out(vty,
5261 "%% This group is already covered by a RP prefix-list\n");
5262 return CMD_WARNING_CONFIG_FAILED;
5263 }
5264
5265 if (result == PIM_RP_PFXLIST_IN_USE) {
5266 vty_out(vty,
5267 "%% The same prefix-list cannot be applied to multiple RPs\n");
5268 return CMD_WARNING_CONFIG_FAILED;
5269 }
5270
5271 if (result == PIM_GROUP_BAD_ADDR_MASK_COMBO) {
5272 vty_out(vty, "%% Inconsistent address and mask: %s\n",
5273 group);
5274 return CMD_WARNING_CONFIG_FAILED;
5275 }
5276
5277 return CMD_SUCCESS;
5278 }
5279
5280 static int pim_cmd_spt_switchover(struct pim_instance *pim,
5281 enum pim_spt_switchover spt,
5282 const char *plist)
5283 {
5284 pim->spt.switchover = spt;
5285
5286 switch (pim->spt.switchover) {
5287 case PIM_SPT_IMMEDIATE:
5288 XFREE(MTYPE_PIM_SPT_PLIST_NAME, pim->spt.plist);
5289
5290 pim_upstream_add_lhr_star_pimreg(pim);
5291 break;
5292 case PIM_SPT_INFINITY:
5293 pim_upstream_remove_lhr_star_pimreg(pim, plist);
5294
5295 XFREE(MTYPE_PIM_SPT_PLIST_NAME, pim->spt.plist);
5296
5297 if (plist)
5298 pim->spt.plist =
5299 XSTRDUP(MTYPE_PIM_SPT_PLIST_NAME, plist);
5300 break;
5301 }
5302
5303 return CMD_SUCCESS;
5304 }
5305
5306 DEFUN (ip_pim_spt_switchover_infinity,
5307 ip_pim_spt_switchover_infinity_cmd,
5308 "ip pim spt-switchover infinity-and-beyond",
5309 IP_STR
5310 PIM_STR
5311 "SPT-Switchover\n"
5312 "Never switch to SPT Tree\n")
5313 {
5314 PIM_DECLVAR_CONTEXT(vrf, pim);
5315 return pim_cmd_spt_switchover(pim, PIM_SPT_INFINITY, NULL);
5316 }
5317
5318 DEFUN (ip_pim_spt_switchover_infinity_plist,
5319 ip_pim_spt_switchover_infinity_plist_cmd,
5320 "ip pim spt-switchover infinity-and-beyond prefix-list WORD",
5321 IP_STR
5322 PIM_STR
5323 "SPT-Switchover\n"
5324 "Never switch to SPT Tree\n"
5325 "Prefix-List to control which groups to switch\n"
5326 "Prefix-List name\n")
5327 {
5328 PIM_DECLVAR_CONTEXT(vrf, pim);
5329 return pim_cmd_spt_switchover(pim, PIM_SPT_INFINITY, argv[5]->arg);
5330 }
5331
5332 DEFUN (no_ip_pim_spt_switchover_infinity,
5333 no_ip_pim_spt_switchover_infinity_cmd,
5334 "no ip pim spt-switchover infinity-and-beyond",
5335 NO_STR
5336 IP_STR
5337 PIM_STR
5338 "SPT_Switchover\n"
5339 "Never switch to SPT Tree\n")
5340 {
5341 PIM_DECLVAR_CONTEXT(vrf, pim);
5342 return pim_cmd_spt_switchover(pim, PIM_SPT_IMMEDIATE, NULL);
5343 }
5344
5345 DEFUN (no_ip_pim_spt_switchover_infinity_plist,
5346 no_ip_pim_spt_switchover_infinity_plist_cmd,
5347 "no ip pim spt-switchover infinity-and-beyond prefix-list WORD",
5348 NO_STR
5349 IP_STR
5350 PIM_STR
5351 "SPT_Switchover\n"
5352 "Never switch to SPT Tree\n"
5353 "Prefix-List to control which groups to switch\n"
5354 "Prefix-List name\n")
5355 {
5356 PIM_DECLVAR_CONTEXT(vrf, pim);
5357 return pim_cmd_spt_switchover(pim, PIM_SPT_IMMEDIATE, NULL);
5358 }
5359
5360 DEFUN (ip_pim_joinprune_time,
5361 ip_pim_joinprune_time_cmd,
5362 "ip pim join-prune-interval (60-600)",
5363 IP_STR
5364 "pim multicast routing\n"
5365 "Join Prune Send Interval\n"
5366 "Seconds\n")
5367 {
5368 PIM_DECLVAR_CONTEXT(vrf, pim);
5369 router->t_periodic = atoi(argv[3]->arg);
5370 return CMD_SUCCESS;
5371 }
5372
5373 DEFUN (no_ip_pim_joinprune_time,
5374 no_ip_pim_joinprune_time_cmd,
5375 "no ip pim join-prune-interval (60-600)",
5376 NO_STR
5377 IP_STR
5378 "pim multicast routing\n"
5379 "Join Prune Send Interval\n"
5380 "Seconds\n")
5381 {
5382 PIM_DECLVAR_CONTEXT(vrf, pim);
5383 router->t_periodic = PIM_DEFAULT_T_PERIODIC;
5384 return CMD_SUCCESS;
5385 }
5386
5387 DEFUN (ip_pim_register_suppress,
5388 ip_pim_register_suppress_cmd,
5389 "ip pim register-suppress-time (5-60000)",
5390 IP_STR
5391 "pim multicast routing\n"
5392 "Register Suppress Timer\n"
5393 "Seconds\n")
5394 {
5395 PIM_DECLVAR_CONTEXT(vrf, pim);
5396 router->register_suppress_time = atoi(argv[3]->arg);
5397 return CMD_SUCCESS;
5398 }
5399
5400 DEFUN (no_ip_pim_register_suppress,
5401 no_ip_pim_register_suppress_cmd,
5402 "no ip pim register-suppress-time (5-60000)",
5403 NO_STR
5404 IP_STR
5405 "pim multicast routing\n"
5406 "Register Suppress Timer\n"
5407 "Seconds\n")
5408 {
5409 PIM_DECLVAR_CONTEXT(vrf, pim);
5410 router->register_suppress_time = PIM_REGISTER_SUPPRESSION_TIME_DEFAULT;
5411 return CMD_SUCCESS;
5412 }
5413
5414 DEFUN (ip_pim_rp_keep_alive,
5415 ip_pim_rp_keep_alive_cmd,
5416 "ip pim rp keep-alive-timer (31-60000)",
5417 IP_STR
5418 "pim multicast routing\n"
5419 "Rendevous Point\n"
5420 "Keep alive Timer\n"
5421 "Seconds\n")
5422 {
5423 PIM_DECLVAR_CONTEXT(vrf, pim);
5424 pim->rp_keep_alive_time = atoi(argv[4]->arg);
5425 return CMD_SUCCESS;
5426 }
5427
5428 DEFUN (no_ip_pim_rp_keep_alive,
5429 no_ip_pim_rp_keep_alive_cmd,
5430 "no ip pim rp keep-alive-timer (31-60000)",
5431 NO_STR
5432 IP_STR
5433 "pim multicast routing\n"
5434 "Rendevous Point\n"
5435 "Keep alive Timer\n"
5436 "Seconds\n")
5437 {
5438 PIM_DECLVAR_CONTEXT(vrf, pim);
5439 pim->rp_keep_alive_time = PIM_KEEPALIVE_PERIOD;
5440 return CMD_SUCCESS;
5441 }
5442
5443 DEFUN (ip_pim_keep_alive,
5444 ip_pim_keep_alive_cmd,
5445 "ip pim keep-alive-timer (31-60000)",
5446 IP_STR
5447 "pim multicast routing\n"
5448 "Keep alive Timer\n"
5449 "Seconds\n")
5450 {
5451 PIM_DECLVAR_CONTEXT(vrf, pim);
5452 pim->keep_alive_time = atoi(argv[3]->arg);
5453 return CMD_SUCCESS;
5454 }
5455
5456 DEFUN (no_ip_pim_keep_alive,
5457 no_ip_pim_keep_alive_cmd,
5458 "no ip pim keep-alive-timer (31-60000)",
5459 NO_STR
5460 IP_STR
5461 "pim multicast routing\n"
5462 "Keep alive Timer\n"
5463 "Seconds\n")
5464 {
5465 PIM_DECLVAR_CONTEXT(vrf, pim);
5466 pim->keep_alive_time = PIM_KEEPALIVE_PERIOD;
5467 return CMD_SUCCESS;
5468 }
5469
5470 DEFUN (ip_pim_packets,
5471 ip_pim_packets_cmd,
5472 "ip pim packets (1-100)",
5473 IP_STR
5474 "pim multicast routing\n"
5475 "packets to process at one time per fd\n"
5476 "Number of packets\n")
5477 {
5478 PIM_DECLVAR_CONTEXT(vrf, pim);
5479 router->packet_process = atoi(argv[3]->arg);
5480 return CMD_SUCCESS;
5481 }
5482
5483 DEFUN (no_ip_pim_packets,
5484 no_ip_pim_packets_cmd,
5485 "no ip pim packets (1-100)",
5486 NO_STR
5487 IP_STR
5488 "pim multicast routing\n"
5489 "packets to process at one time per fd\n"
5490 "Number of packets\n")
5491 {
5492 PIM_DECLVAR_CONTEXT(vrf, pim);
5493 router->packet_process = PIM_DEFAULT_PACKET_PROCESS;
5494 return CMD_SUCCESS;
5495 }
5496
5497 DEFUN (ip_pim_v6_secondary,
5498 ip_pim_v6_secondary_cmd,
5499 "ip pim send-v6-secondary",
5500 IP_STR
5501 "pim multicast routing\n"
5502 "Send v6 secondary addresses\n")
5503 {
5504 PIM_DECLVAR_CONTEXT(vrf, pim);
5505 pim->send_v6_secondary = 1;
5506
5507 return CMD_SUCCESS;
5508 }
5509
5510 DEFUN (no_ip_pim_v6_secondary,
5511 no_ip_pim_v6_secondary_cmd,
5512 "no ip pim send-v6-secondary",
5513 NO_STR
5514 IP_STR
5515 "pim multicast routing\n"
5516 "Send v6 secondary addresses\n")
5517 {
5518 PIM_DECLVAR_CONTEXT(vrf, pim);
5519 pim->send_v6_secondary = 0;
5520
5521 return CMD_SUCCESS;
5522 }
5523
5524 DEFUN (ip_pim_rp,
5525 ip_pim_rp_cmd,
5526 "ip pim rp A.B.C.D [A.B.C.D/M]",
5527 IP_STR
5528 "pim multicast routing\n"
5529 "Rendevous Point\n"
5530 "ip address of RP\n"
5531 "Group Address range to cover\n")
5532 {
5533 PIM_DECLVAR_CONTEXT(vrf, pim);
5534 int idx_ipv4 = 3;
5535
5536 if (argc == (idx_ipv4 + 1))
5537 return pim_rp_cmd_worker(pim, vty, argv[idx_ipv4]->arg, NULL,
5538 NULL);
5539 else
5540 return pim_rp_cmd_worker(pim, vty, argv[idx_ipv4]->arg,
5541 argv[idx_ipv4 + 1]->arg, NULL);
5542 }
5543
5544 DEFUN (ip_pim_rp_prefix_list,
5545 ip_pim_rp_prefix_list_cmd,
5546 "ip pim rp A.B.C.D prefix-list WORD",
5547 IP_STR
5548 "pim multicast routing\n"
5549 "Rendevous Point\n"
5550 "ip address of RP\n"
5551 "group prefix-list filter\n"
5552 "Name of a prefix-list\n")
5553 {
5554 PIM_DECLVAR_CONTEXT(vrf, pim);
5555 return pim_rp_cmd_worker(pim, vty, argv[3]->arg, NULL, argv[5]->arg);
5556 }
5557
5558 static int pim_no_rp_cmd_worker(struct pim_instance *pim, struct vty *vty,
5559 const char *rp, const char *group,
5560 const char *plist)
5561 {
5562 int result = pim_rp_del(pim, rp, group, plist);
5563
5564 if (result == PIM_GROUP_BAD_ADDRESS) {
5565 vty_out(vty, "%% Bad group address specified: %s\n", group);
5566 return CMD_WARNING_CONFIG_FAILED;
5567 }
5568
5569 if (result == PIM_RP_BAD_ADDRESS) {
5570 vty_out(vty, "%% Bad RP address specified: %s\n", rp);
5571 return CMD_WARNING_CONFIG_FAILED;
5572 }
5573
5574 if (result == PIM_RP_NOT_FOUND) {
5575 vty_out(vty, "%% Unable to find specified RP\n");
5576 return CMD_WARNING_CONFIG_FAILED;
5577 }
5578
5579 return CMD_SUCCESS;
5580 }
5581
5582 DEFUN (no_ip_pim_rp,
5583 no_ip_pim_rp_cmd,
5584 "no ip pim rp A.B.C.D [A.B.C.D/M]",
5585 NO_STR
5586 IP_STR
5587 "pim multicast routing\n"
5588 "Rendevous Point\n"
5589 "ip address of RP\n"
5590 "Group Address range to cover\n")
5591 {
5592 PIM_DECLVAR_CONTEXT(vrf, pim);
5593 int idx_ipv4 = 4, idx_group = 0;
5594
5595 if (argv_find(argv, argc, "A.B.C.D/M", &idx_group))
5596 return pim_no_rp_cmd_worker(pim, vty, argv[idx_ipv4]->arg,
5597 argv[idx_group]->arg, NULL);
5598 else
5599 return pim_no_rp_cmd_worker(pim, vty, argv[idx_ipv4]->arg, NULL,
5600 NULL);
5601 }
5602
5603 DEFUN (no_ip_pim_rp_prefix_list,
5604 no_ip_pim_rp_prefix_list_cmd,
5605 "no ip pim rp A.B.C.D prefix-list WORD",
5606 NO_STR
5607 IP_STR
5608 "pim multicast routing\n"
5609 "Rendevous Point\n"
5610 "ip address of RP\n"
5611 "group prefix-list filter\n"
5612 "Name of a prefix-list\n")
5613 {
5614 PIM_DECLVAR_CONTEXT(vrf, pim);
5615 return pim_no_rp_cmd_worker(pim, vty, argv[4]->arg, NULL, argv[6]->arg);
5616 }
5617
5618 static int pim_ssm_cmd_worker(struct pim_instance *pim, struct vty *vty,
5619 const char *plist)
5620 {
5621 int result = pim_ssm_range_set(pim, pim->vrf_id, plist);
5622
5623 if (result == PIM_SSM_ERR_NONE)
5624 return CMD_SUCCESS;
5625
5626 switch (result) {
5627 case PIM_SSM_ERR_NO_VRF:
5628 vty_out(vty, "%% VRF doesn't exist\n");
5629 break;
5630 case PIM_SSM_ERR_DUP:
5631 vty_out(vty, "%% duplicate config\n");
5632 break;
5633 default:
5634 vty_out(vty, "%% ssm range config failed\n");
5635 }
5636
5637 return CMD_WARNING_CONFIG_FAILED;
5638 }
5639
5640 DEFUN (ip_pim_ssm_prefix_list,
5641 ip_pim_ssm_prefix_list_cmd,
5642 "ip pim ssm prefix-list WORD",
5643 IP_STR
5644 "pim multicast routing\n"
5645 "Source Specific Multicast\n"
5646 "group range prefix-list filter\n"
5647 "Name of a prefix-list\n")
5648 {
5649 PIM_DECLVAR_CONTEXT(vrf, pim);
5650 return pim_ssm_cmd_worker(pim, vty, argv[4]->arg);
5651 }
5652
5653 DEFUN (no_ip_pim_ssm_prefix_list,
5654 no_ip_pim_ssm_prefix_list_cmd,
5655 "no ip pim ssm prefix-list",
5656 NO_STR
5657 IP_STR
5658 "pim multicast routing\n"
5659 "Source Specific Multicast\n"
5660 "group range prefix-list filter\n")
5661 {
5662 PIM_DECLVAR_CONTEXT(vrf, pim);
5663 return pim_ssm_cmd_worker(pim, vty, NULL);
5664 }
5665
5666 DEFUN (no_ip_pim_ssm_prefix_list_name,
5667 no_ip_pim_ssm_prefix_list_name_cmd,
5668 "no ip pim ssm prefix-list WORD",
5669 NO_STR
5670 IP_STR
5671 "pim multicast routing\n"
5672 "Source Specific Multicast\n"
5673 "group range prefix-list filter\n"
5674 "Name of a prefix-list\n")
5675 {
5676 PIM_DECLVAR_CONTEXT(vrf, pim);
5677 struct pim_ssm *ssm = pim->ssm_info;
5678
5679 if (ssm->plist_name && !strcmp(ssm->plist_name, argv[5]->arg))
5680 return pim_ssm_cmd_worker(pim, vty, NULL);
5681
5682 vty_out(vty, "%% pim ssm prefix-list %s doesn't exist\n", argv[5]->arg);
5683
5684 return CMD_WARNING_CONFIG_FAILED;
5685 }
5686
5687 static void ip_pim_ssm_show_group_range(struct pim_instance *pim,
5688 struct vty *vty, bool uj)
5689 {
5690 struct pim_ssm *ssm = pim->ssm_info;
5691 const char *range_str =
5692 ssm->plist_name ? ssm->plist_name : PIM_SSM_STANDARD_RANGE;
5693
5694 if (uj) {
5695 json_object *json;
5696 json = json_object_new_object();
5697 json_object_string_add(json, "ssmGroups", range_str);
5698 vty_out(vty, "%s\n", json_object_to_json_string_ext(
5699 json, JSON_C_TO_STRING_PRETTY));
5700 json_object_free(json);
5701 } else
5702 vty_out(vty, "SSM group range : %s\n", range_str);
5703 }
5704
5705 DEFUN (show_ip_pim_ssm_range,
5706 show_ip_pim_ssm_range_cmd,
5707 "show ip pim [vrf NAME] group-type [json]",
5708 SHOW_STR
5709 IP_STR
5710 PIM_STR
5711 VRF_CMD_HELP_STR
5712 "PIM group type\n"
5713 JSON_STR)
5714 {
5715 int idx = 2;
5716 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
5717 bool uj = use_json(argc, argv);
5718
5719 if (!vrf)
5720 return CMD_WARNING;
5721
5722 ip_pim_ssm_show_group_range(vrf->info, vty, uj);
5723
5724 return CMD_SUCCESS;
5725 }
5726
5727 static void ip_pim_ssm_show_group_type(struct pim_instance *pim,
5728 struct vty *vty, bool uj,
5729 const char *group)
5730 {
5731 struct in_addr group_addr;
5732 const char *type_str;
5733 int result;
5734
5735 result = inet_pton(AF_INET, group, &group_addr);
5736 if (result <= 0)
5737 type_str = "invalid";
5738 else {
5739 if (pim_is_group_224_4(group_addr))
5740 type_str =
5741 pim_is_grp_ssm(pim, group_addr) ? "SSM" : "ASM";
5742 else
5743 type_str = "not-multicast";
5744 }
5745
5746 if (uj) {
5747 json_object *json;
5748 json = json_object_new_object();
5749 json_object_string_add(json, "groupType", type_str);
5750 vty_out(vty, "%s\n", json_object_to_json_string_ext(
5751 json, JSON_C_TO_STRING_PRETTY));
5752 json_object_free(json);
5753 } else
5754 vty_out(vty, "Group type : %s\n", type_str);
5755 }
5756
5757 DEFUN (show_ip_pim_group_type,
5758 show_ip_pim_group_type_cmd,
5759 "show ip pim [vrf NAME] group-type A.B.C.D [json]",
5760 SHOW_STR
5761 IP_STR
5762 PIM_STR
5763 VRF_CMD_HELP_STR
5764 "multicast group type\n"
5765 "group address\n"
5766 JSON_STR)
5767 {
5768 int idx = 2;
5769 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
5770 bool uj = use_json(argc, argv);
5771
5772 if (!vrf)
5773 return CMD_WARNING;
5774
5775 argv_find(argv, argc, "A.B.C.D", &idx);
5776 ip_pim_ssm_show_group_type(vrf->info, vty, uj, argv[idx]->arg);
5777
5778 return CMD_SUCCESS;
5779 }
5780
5781 DEFUN (ip_ssmpingd,
5782 ip_ssmpingd_cmd,
5783 "ip ssmpingd [A.B.C.D]",
5784 IP_STR
5785 CONF_SSMPINGD_STR
5786 "Source address\n")
5787 {
5788 PIM_DECLVAR_CONTEXT(vrf, pim);
5789 int idx_ipv4 = 2;
5790 int result;
5791 struct in_addr source_addr;
5792 const char *source_str = (argc == 3) ? argv[idx_ipv4]->arg : "0.0.0.0";
5793
5794 result = inet_pton(AF_INET, source_str, &source_addr);
5795 if (result <= 0) {
5796 vty_out(vty, "%% Bad source address %s: errno=%d: %s\n",
5797 source_str, errno, safe_strerror(errno));
5798 return CMD_WARNING_CONFIG_FAILED;
5799 }
5800
5801 result = pim_ssmpingd_start(pim, source_addr);
5802 if (result) {
5803 vty_out(vty, "%% Failure starting ssmpingd for source %s: %d\n",
5804 source_str, result);
5805 return CMD_WARNING_CONFIG_FAILED;
5806 }
5807
5808 return CMD_SUCCESS;
5809 }
5810
5811 DEFUN (no_ip_ssmpingd,
5812 no_ip_ssmpingd_cmd,
5813 "no ip ssmpingd [A.B.C.D]",
5814 NO_STR
5815 IP_STR
5816 CONF_SSMPINGD_STR
5817 "Source address\n")
5818 {
5819 PIM_DECLVAR_CONTEXT(vrf, pim);
5820 int idx_ipv4 = 3;
5821 int result;
5822 struct in_addr source_addr;
5823 const char *source_str = (argc == 4) ? argv[idx_ipv4]->arg : "0.0.0.0";
5824
5825 result = inet_pton(AF_INET, source_str, &source_addr);
5826 if (result <= 0) {
5827 vty_out(vty, "%% Bad source address %s: errno=%d: %s\n",
5828 source_str, errno, safe_strerror(errno));
5829 return CMD_WARNING_CONFIG_FAILED;
5830 }
5831
5832 result = pim_ssmpingd_stop(pim, source_addr);
5833 if (result) {
5834 vty_out(vty, "%% Failure stopping ssmpingd for source %s: %d\n",
5835 source_str, result);
5836 return CMD_WARNING_CONFIG_FAILED;
5837 }
5838
5839 return CMD_SUCCESS;
5840 }
5841
5842 DEFUN (ip_pim_ecmp,
5843 ip_pim_ecmp_cmd,
5844 "ip pim ecmp",
5845 IP_STR
5846 "pim multicast routing\n"
5847 "Enable PIM ECMP \n")
5848 {
5849 PIM_DECLVAR_CONTEXT(vrf, pim);
5850 pim->ecmp_enable = true;
5851
5852 return CMD_SUCCESS;
5853 }
5854
5855 DEFUN (no_ip_pim_ecmp,
5856 no_ip_pim_ecmp_cmd,
5857 "no ip pim ecmp",
5858 NO_STR
5859 IP_STR
5860 "pim multicast routing\n"
5861 "Disable PIM ECMP \n")
5862 {
5863 PIM_DECLVAR_CONTEXT(vrf, pim);
5864 pim->ecmp_enable = false;
5865
5866 return CMD_SUCCESS;
5867 }
5868
5869 DEFUN (ip_pim_ecmp_rebalance,
5870 ip_pim_ecmp_rebalance_cmd,
5871 "ip pim ecmp rebalance",
5872 IP_STR
5873 "pim multicast routing\n"
5874 "Enable PIM ECMP \n"
5875 "Enable PIM ECMP Rebalance\n")
5876 {
5877 PIM_DECLVAR_CONTEXT(vrf, pim);
5878 pim->ecmp_enable = true;
5879 pim->ecmp_rebalance_enable = true;
5880
5881 return CMD_SUCCESS;
5882 }
5883
5884 DEFUN (no_ip_pim_ecmp_rebalance,
5885 no_ip_pim_ecmp_rebalance_cmd,
5886 "no ip pim ecmp rebalance",
5887 NO_STR
5888 IP_STR
5889 "pim multicast routing\n"
5890 "Disable PIM ECMP \n"
5891 "Disable PIM ECMP Rebalance\n")
5892 {
5893 PIM_DECLVAR_CONTEXT(vrf, pim);
5894 pim->ecmp_rebalance_enable = false;
5895
5896 return CMD_SUCCESS;
5897 }
5898
5899 static int pim_cmd_igmp_start(struct vty *vty, struct interface *ifp)
5900 {
5901 struct pim_interface *pim_ifp;
5902 uint8_t need_startup = 0;
5903
5904 pim_ifp = ifp->info;
5905
5906 if (!pim_ifp) {
5907 pim_ifp = pim_if_new(ifp, true, false, false,
5908 false /*vxlan_term*/);
5909 if (!pim_ifp) {
5910 vty_out(vty, "Could not enable IGMP on interface %s\n",
5911 ifp->name);
5912 return CMD_WARNING_CONFIG_FAILED;
5913 }
5914 need_startup = 1;
5915 } else {
5916 if (!PIM_IF_TEST_IGMP(pim_ifp->options)) {
5917 PIM_IF_DO_IGMP(pim_ifp->options);
5918 need_startup = 1;
5919 }
5920 }
5921
5922 /* 'ip igmp' executed multiple times, with need_startup
5923 avoid multiple if add all and membership refresh */
5924 if (need_startup) {
5925 pim_if_addr_add_all(ifp);
5926 pim_if_membership_refresh(ifp);
5927 }
5928
5929 return CMD_SUCCESS;
5930 }
5931
5932 DEFUN (interface_ip_igmp,
5933 interface_ip_igmp_cmd,
5934 "ip igmp",
5935 IP_STR
5936 IFACE_IGMP_STR)
5937 {
5938 VTY_DECLVAR_CONTEXT(interface, ifp);
5939
5940 return pim_cmd_igmp_start(vty, ifp);
5941 }
5942
5943 DEFUN (interface_no_ip_igmp,
5944 interface_no_ip_igmp_cmd,
5945 "no ip igmp",
5946 NO_STR
5947 IP_STR
5948 IFACE_IGMP_STR)
5949 {
5950 VTY_DECLVAR_CONTEXT(interface, ifp);
5951 struct pim_interface *pim_ifp = ifp->info;
5952
5953 if (!pim_ifp)
5954 return CMD_SUCCESS;
5955
5956 PIM_IF_DONT_IGMP(pim_ifp->options);
5957
5958 pim_if_membership_clear(ifp);
5959
5960 pim_if_addr_del_all_igmp(ifp);
5961
5962 if (!PIM_IF_TEST_PIM(pim_ifp->options)) {
5963 pim_if_delete(ifp);
5964 }
5965
5966 return CMD_SUCCESS;
5967 }
5968
5969 DEFUN (interface_ip_igmp_join,
5970 interface_ip_igmp_join_cmd,
5971 "ip igmp join A.B.C.D A.B.C.D",
5972 IP_STR
5973 IFACE_IGMP_STR
5974 "IGMP join multicast group\n"
5975 "Multicast group address\n"
5976 "Source address\n")
5977 {
5978 VTY_DECLVAR_CONTEXT(interface, ifp);
5979 int idx_ipv4 = 3;
5980 int idx_ipv4_2 = 4;
5981 const char *group_str;
5982 const char *source_str;
5983 struct in_addr group_addr;
5984 struct in_addr source_addr;
5985 int result;
5986
5987 /* Group address */
5988 group_str = argv[idx_ipv4]->arg;
5989 result = inet_pton(AF_INET, group_str, &group_addr);
5990 if (result <= 0) {
5991 vty_out(vty, "Bad group address %s: errno=%d: %s\n", group_str,
5992 errno, safe_strerror(errno));
5993 return CMD_WARNING_CONFIG_FAILED;
5994 }
5995
5996 /* Source address */
5997 source_str = argv[idx_ipv4_2]->arg;
5998 result = inet_pton(AF_INET, source_str, &source_addr);
5999 if (result <= 0) {
6000 vty_out(vty, "Bad source address %s: errno=%d: %s\n",
6001 source_str, errno, safe_strerror(errno));
6002 return CMD_WARNING_CONFIG_FAILED;
6003 }
6004
6005 CMD_FERR_RETURN(pim_if_igmp_join_add(ifp, group_addr, source_addr),
6006 "Failure joining IGMP group: $ERR");
6007
6008 return CMD_SUCCESS;
6009 }
6010
6011 DEFUN (interface_no_ip_igmp_join,
6012 interface_no_ip_igmp_join_cmd,
6013 "no ip igmp join A.B.C.D A.B.C.D",
6014 NO_STR
6015 IP_STR
6016 IFACE_IGMP_STR
6017 "IGMP join multicast group\n"
6018 "Multicast group address\n"
6019 "Source address\n")
6020 {
6021 VTY_DECLVAR_CONTEXT(interface, ifp);
6022 int idx_ipv4 = 4;
6023 int idx_ipv4_2 = 5;
6024 const char *group_str;
6025 const char *source_str;
6026 struct in_addr group_addr;
6027 struct in_addr source_addr;
6028 int result;
6029
6030 /* Group address */
6031 group_str = argv[idx_ipv4]->arg;
6032 result = inet_pton(AF_INET, group_str, &group_addr);
6033 if (result <= 0) {
6034 vty_out(vty, "Bad group address %s: errno=%d: %s\n", group_str,
6035 errno, safe_strerror(errno));
6036 return CMD_WARNING_CONFIG_FAILED;
6037 }
6038
6039 /* Source address */
6040 source_str = argv[idx_ipv4_2]->arg;
6041 result = inet_pton(AF_INET, source_str, &source_addr);
6042 if (result <= 0) {
6043 vty_out(vty, "Bad source address %s: errno=%d: %s\n",
6044 source_str, errno, safe_strerror(errno));
6045 return CMD_WARNING_CONFIG_FAILED;
6046 }
6047
6048 result = pim_if_igmp_join_del(ifp, group_addr, source_addr);
6049 if (result) {
6050 vty_out(vty,
6051 "%% Failure leaving IGMP group %s source %s on interface %s: %d\n",
6052 group_str, source_str, ifp->name, result);
6053 return CMD_WARNING_CONFIG_FAILED;
6054 }
6055
6056 return CMD_SUCCESS;
6057 }
6058
6059 /*
6060 CLI reconfiguration affects the interface level (struct pim_interface).
6061 This function propagates the reconfiguration to every active socket
6062 for that interface.
6063 */
6064 static void igmp_sock_query_interval_reconfig(struct igmp_sock *igmp)
6065 {
6066 struct interface *ifp;
6067 struct pim_interface *pim_ifp;
6068
6069 zassert(igmp);
6070
6071 /* other querier present? */
6072
6073 if (igmp->t_other_querier_timer)
6074 return;
6075
6076 /* this is the querier */
6077
6078 zassert(igmp->interface);
6079 zassert(igmp->interface->info);
6080
6081 ifp = igmp->interface;
6082 pim_ifp = ifp->info;
6083
6084 if (PIM_DEBUG_IGMP_TRACE) {
6085 char ifaddr_str[INET_ADDRSTRLEN];
6086 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str,
6087 sizeof(ifaddr_str));
6088 zlog_debug("%s: Querier %s on %s reconfig query_interval=%d",
6089 __PRETTY_FUNCTION__, ifaddr_str, ifp->name,
6090 pim_ifp->igmp_default_query_interval);
6091 }
6092
6093 /*
6094 igmp_startup_mode_on() will reset QQI:
6095
6096 igmp->querier_query_interval = pim_ifp->igmp_default_query_interval;
6097 */
6098 igmp_startup_mode_on(igmp);
6099 }
6100
6101 static void igmp_sock_query_reschedule(struct igmp_sock *igmp)
6102 {
6103 if (igmp->t_igmp_query_timer) {
6104 /* other querier present */
6105 zassert(igmp->t_igmp_query_timer);
6106 zassert(!igmp->t_other_querier_timer);
6107
6108 pim_igmp_general_query_off(igmp);
6109 pim_igmp_general_query_on(igmp);
6110
6111 zassert(igmp->t_igmp_query_timer);
6112 zassert(!igmp->t_other_querier_timer);
6113 } else {
6114 /* this is the querier */
6115
6116 zassert(!igmp->t_igmp_query_timer);
6117 zassert(igmp->t_other_querier_timer);
6118
6119 pim_igmp_other_querier_timer_off(igmp);
6120 pim_igmp_other_querier_timer_on(igmp);
6121
6122 zassert(!igmp->t_igmp_query_timer);
6123 zassert(igmp->t_other_querier_timer);
6124 }
6125 }
6126
6127 static void change_query_interval(struct pim_interface *pim_ifp,
6128 int query_interval)
6129 {
6130 struct listnode *sock_node;
6131 struct igmp_sock *igmp;
6132
6133 pim_ifp->igmp_default_query_interval = query_interval;
6134
6135 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
6136 igmp_sock_query_interval_reconfig(igmp);
6137 igmp_sock_query_reschedule(igmp);
6138 }
6139 }
6140
6141 static void change_query_max_response_time(struct pim_interface *pim_ifp,
6142 int query_max_response_time_dsec)
6143 {
6144 struct listnode *sock_node;
6145 struct igmp_sock *igmp;
6146
6147 pim_ifp->igmp_query_max_response_time_dsec =
6148 query_max_response_time_dsec;
6149
6150 /*
6151 Below we modify socket/group/source timers in order to quickly
6152 reflect the change. Otherwise, those timers would eventually catch
6153 up.
6154 */
6155
6156 /* scan all sockets */
6157 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
6158 struct listnode *grp_node;
6159 struct igmp_group *grp;
6160
6161 /* reschedule socket general query */
6162 igmp_sock_query_reschedule(igmp);
6163
6164 /* scan socket groups */
6165 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list, grp_node,
6166 grp)) {
6167 struct listnode *src_node;
6168 struct igmp_source *src;
6169
6170 /* reset group timers for groups in EXCLUDE mode */
6171 if (grp->group_filtermode_isexcl) {
6172 igmp_group_reset_gmi(grp);
6173 }
6174
6175 /* scan group sources */
6176 for (ALL_LIST_ELEMENTS_RO(grp->group_source_list,
6177 src_node, src)) {
6178
6179 /* reset source timers for sources with running
6180 * timers */
6181 if (src->t_source_timer) {
6182 igmp_source_reset_gmi(igmp, grp, src);
6183 }
6184 }
6185 }
6186 }
6187 }
6188
6189 #define IGMP_QUERY_INTERVAL_MIN (1)
6190 #define IGMP_QUERY_INTERVAL_MAX (1800)
6191
6192 DEFUN (interface_ip_igmp_query_interval,
6193 interface_ip_igmp_query_interval_cmd,
6194 "ip igmp query-interval (1-1800)",
6195 IP_STR
6196 IFACE_IGMP_STR
6197 IFACE_IGMP_QUERY_INTERVAL_STR
6198 "Query interval in seconds\n")
6199 {
6200 VTY_DECLVAR_CONTEXT(interface, ifp);
6201 struct pim_interface *pim_ifp = ifp->info;
6202 int query_interval;
6203 int query_interval_dsec;
6204 int ret;
6205
6206 if (!pim_ifp) {
6207 ret = pim_cmd_igmp_start(vty, ifp);
6208 if (ret != CMD_SUCCESS)
6209 return ret;
6210 pim_ifp = ifp->info;
6211 }
6212
6213 query_interval = atoi(argv[3]->arg);
6214 query_interval_dsec = 10 * query_interval;
6215
6216 /*
6217 It seems we don't need to check bounds since command.c does it
6218 already, but we verify them anyway for extra safety.
6219 */
6220 if (query_interval < IGMP_QUERY_INTERVAL_MIN) {
6221 vty_out(vty,
6222 "General query interval %d lower than minimum %d\n",
6223 query_interval, IGMP_QUERY_INTERVAL_MIN);
6224 return CMD_WARNING_CONFIG_FAILED;
6225 }
6226 if (query_interval > IGMP_QUERY_INTERVAL_MAX) {
6227 vty_out(vty,
6228 "General query interval %d higher than maximum %d\n",
6229 query_interval, IGMP_QUERY_INTERVAL_MAX);
6230 return CMD_WARNING_CONFIG_FAILED;
6231 }
6232
6233 if (query_interval_dsec <= pim_ifp->igmp_query_max_response_time_dsec) {
6234 vty_out(vty,
6235 "Can't set general query interval %d dsec <= query max response time %d dsec.\n",
6236 query_interval_dsec,
6237 pim_ifp->igmp_query_max_response_time_dsec);
6238 return CMD_WARNING_CONFIG_FAILED;
6239 }
6240
6241 change_query_interval(pim_ifp, query_interval);
6242
6243 return CMD_SUCCESS;
6244 }
6245
6246 DEFUN (interface_no_ip_igmp_query_interval,
6247 interface_no_ip_igmp_query_interval_cmd,
6248 "no ip igmp query-interval",
6249 NO_STR
6250 IP_STR
6251 IFACE_IGMP_STR
6252 IFACE_IGMP_QUERY_INTERVAL_STR)
6253 {
6254 VTY_DECLVAR_CONTEXT(interface, ifp);
6255 struct pim_interface *pim_ifp = ifp->info;
6256 int default_query_interval_dsec;
6257
6258 if (!pim_ifp)
6259 return CMD_SUCCESS;
6260
6261 default_query_interval_dsec = IGMP_GENERAL_QUERY_INTERVAL * 10;
6262
6263 if (default_query_interval_dsec
6264 <= pim_ifp->igmp_query_max_response_time_dsec) {
6265 vty_out(vty,
6266 "Can't set default general query interval %d dsec <= query max response time %d dsec.\n",
6267 default_query_interval_dsec,
6268 pim_ifp->igmp_query_max_response_time_dsec);
6269 return CMD_WARNING_CONFIG_FAILED;
6270 }
6271
6272 change_query_interval(pim_ifp, IGMP_GENERAL_QUERY_INTERVAL);
6273
6274 return CMD_SUCCESS;
6275 }
6276
6277 DEFUN (interface_ip_igmp_version,
6278 interface_ip_igmp_version_cmd,
6279 "ip igmp version (2-3)",
6280 IP_STR
6281 IFACE_IGMP_STR
6282 "IGMP version\n"
6283 "IGMP version number\n")
6284 {
6285 VTY_DECLVAR_CONTEXT(interface, ifp);
6286 struct pim_interface *pim_ifp = ifp->info;
6287 int igmp_version, old_version = 0;
6288 int ret;
6289
6290 if (!pim_ifp) {
6291 ret = pim_cmd_igmp_start(vty, ifp);
6292 if (ret != CMD_SUCCESS)
6293 return ret;
6294 pim_ifp = ifp->info;
6295 }
6296
6297 igmp_version = atoi(argv[3]->arg);
6298 old_version = pim_ifp->igmp_version;
6299 pim_ifp->igmp_version = igmp_version;
6300
6301 // Check if IGMP is Enabled otherwise, enable on interface
6302 if (!PIM_IF_TEST_IGMP(pim_ifp->options)) {
6303 PIM_IF_DO_IGMP(pim_ifp->options);
6304 pim_if_addr_add_all(ifp);
6305 pim_if_membership_refresh(ifp);
6306 old_version = igmp_version;
6307 // avoid refreshing membership again.
6308 }
6309 /* Current and new version is different refresh existing
6310 membership. Going from 3 -> 2 or 2 -> 3. */
6311 if (old_version != igmp_version)
6312 pim_if_membership_refresh(ifp);
6313
6314 return CMD_SUCCESS;
6315 }
6316
6317 DEFUN (interface_no_ip_igmp_version,
6318 interface_no_ip_igmp_version_cmd,
6319 "no ip igmp version (2-3)",
6320 NO_STR
6321 IP_STR
6322 IFACE_IGMP_STR
6323 "IGMP version\n"
6324 "IGMP version number\n")
6325 {
6326 VTY_DECLVAR_CONTEXT(interface, ifp);
6327 struct pim_interface *pim_ifp = ifp->info;
6328
6329 if (!pim_ifp)
6330 return CMD_SUCCESS;
6331
6332 pim_ifp->igmp_version = IGMP_DEFAULT_VERSION;
6333
6334 return CMD_SUCCESS;
6335 }
6336
6337 #define IGMP_QUERY_MAX_RESPONSE_TIME_MIN_DSEC (10)
6338 #define IGMP_QUERY_MAX_RESPONSE_TIME_MAX_DSEC (250)
6339
6340 DEFUN (interface_ip_igmp_query_max_response_time,
6341 interface_ip_igmp_query_max_response_time_cmd,
6342 "ip igmp query-max-response-time (10-250)",
6343 IP_STR
6344 IFACE_IGMP_STR
6345 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_STR
6346 "Query response value in deci-seconds\n")
6347 {
6348 VTY_DECLVAR_CONTEXT(interface, ifp);
6349 struct pim_interface *pim_ifp = ifp->info;
6350 int query_max_response_time;
6351 int ret;
6352
6353 if (!pim_ifp) {
6354 ret = pim_cmd_igmp_start(vty, ifp);
6355 if (ret != CMD_SUCCESS)
6356 return ret;
6357 pim_ifp = ifp->info;
6358 }
6359
6360 query_max_response_time = atoi(argv[3]->arg);
6361
6362 if (query_max_response_time
6363 >= pim_ifp->igmp_default_query_interval * 10) {
6364 vty_out(vty,
6365 "Can't set query max response time %d sec >= general query interval %d sec\n",
6366 query_max_response_time,
6367 pim_ifp->igmp_default_query_interval);
6368 return CMD_WARNING_CONFIG_FAILED;
6369 }
6370
6371 change_query_max_response_time(pim_ifp, query_max_response_time);
6372
6373 return CMD_SUCCESS;
6374 }
6375
6376 DEFUN (interface_no_ip_igmp_query_max_response_time,
6377 interface_no_ip_igmp_query_max_response_time_cmd,
6378 "no ip igmp query-max-response-time (10-250)",
6379 NO_STR
6380 IP_STR
6381 IFACE_IGMP_STR
6382 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_STR
6383 "Time for response in deci-seconds\n")
6384 {
6385 VTY_DECLVAR_CONTEXT(interface, ifp);
6386 struct pim_interface *pim_ifp = ifp->info;
6387
6388 if (!pim_ifp)
6389 return CMD_SUCCESS;
6390
6391 change_query_max_response_time(pim_ifp,
6392 IGMP_QUERY_MAX_RESPONSE_TIME_DSEC);
6393
6394 return CMD_SUCCESS;
6395 }
6396
6397 #define IGMP_QUERY_MAX_RESPONSE_TIME_MIN_DSEC (10)
6398 #define IGMP_QUERY_MAX_RESPONSE_TIME_MAX_DSEC (250)
6399
6400 DEFUN_HIDDEN (interface_ip_igmp_query_max_response_time_dsec,
6401 interface_ip_igmp_query_max_response_time_dsec_cmd,
6402 "ip igmp query-max-response-time-dsec (10-250)",
6403 IP_STR
6404 IFACE_IGMP_STR
6405 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_DSEC_STR
6406 "Query response value in deciseconds\n")
6407 {
6408 VTY_DECLVAR_CONTEXT(interface, ifp);
6409 struct pim_interface *pim_ifp = ifp->info;
6410 int query_max_response_time_dsec;
6411 int default_query_interval_dsec;
6412 int ret;
6413
6414 if (!pim_ifp) {
6415 ret = pim_cmd_igmp_start(vty, ifp);
6416 if (ret != CMD_SUCCESS)
6417 return ret;
6418 pim_ifp = ifp->info;
6419 }
6420
6421 query_max_response_time_dsec = atoi(argv[4]->arg);
6422
6423 default_query_interval_dsec = 10 * pim_ifp->igmp_default_query_interval;
6424
6425 if (query_max_response_time_dsec >= default_query_interval_dsec) {
6426 vty_out(vty,
6427 "Can't set query max response time %d dsec >= general query interval %d dsec\n",
6428 query_max_response_time_dsec,
6429 default_query_interval_dsec);
6430 return CMD_WARNING_CONFIG_FAILED;
6431 }
6432
6433 change_query_max_response_time(pim_ifp, query_max_response_time_dsec);
6434
6435 return CMD_SUCCESS;
6436 }
6437
6438 DEFUN_HIDDEN (interface_no_ip_igmp_query_max_response_time_dsec,
6439 interface_no_ip_igmp_query_max_response_time_dsec_cmd,
6440 "no ip igmp query-max-response-time-dsec",
6441 NO_STR
6442 IP_STR
6443 IFACE_IGMP_STR
6444 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_DSEC_STR)
6445 {
6446 VTY_DECLVAR_CONTEXT(interface, ifp);
6447 struct pim_interface *pim_ifp = ifp->info;
6448
6449 if (!pim_ifp)
6450 return CMD_SUCCESS;
6451
6452 change_query_max_response_time(pim_ifp,
6453 IGMP_QUERY_MAX_RESPONSE_TIME_DSEC);
6454
6455 return CMD_SUCCESS;
6456 }
6457
6458 DEFUN (interface_ip_pim_drprio,
6459 interface_ip_pim_drprio_cmd,
6460 "ip pim drpriority (1-4294967295)",
6461 IP_STR
6462 PIM_STR
6463 "Set the Designated Router Election Priority\n"
6464 "Value of the new DR Priority\n")
6465 {
6466 VTY_DECLVAR_CONTEXT(interface, ifp);
6467 int idx_number = 3;
6468 struct pim_interface *pim_ifp = ifp->info;
6469 uint32_t old_dr_prio;
6470
6471 if (!pim_ifp) {
6472 vty_out(vty, "Please enable PIM on interface, first\n");
6473 return CMD_WARNING_CONFIG_FAILED;
6474 }
6475
6476 old_dr_prio = pim_ifp->pim_dr_priority;
6477
6478 pim_ifp->pim_dr_priority = strtol(argv[idx_number]->arg, NULL, 10);
6479
6480 if (old_dr_prio != pim_ifp->pim_dr_priority) {
6481 pim_if_dr_election(ifp);
6482 pim_hello_restart_now(ifp);
6483 }
6484
6485 return CMD_SUCCESS;
6486 }
6487
6488 DEFUN (interface_no_ip_pim_drprio,
6489 interface_no_ip_pim_drprio_cmd,
6490 "no ip pim drpriority [(1-4294967295)]",
6491 NO_STR
6492 IP_STR
6493 PIM_STR
6494 "Revert the Designated Router Priority to default\n"
6495 "Old Value of the Priority\n")
6496 {
6497 VTY_DECLVAR_CONTEXT(interface, ifp);
6498 struct pim_interface *pim_ifp = ifp->info;
6499
6500 if (!pim_ifp) {
6501 vty_out(vty, "Pim not enabled on this interface\n");
6502 return CMD_WARNING_CONFIG_FAILED;
6503 }
6504
6505 if (pim_ifp->pim_dr_priority != PIM_DEFAULT_DR_PRIORITY) {
6506 pim_ifp->pim_dr_priority = PIM_DEFAULT_DR_PRIORITY;
6507 pim_if_dr_election(ifp);
6508 pim_hello_restart_now(ifp);
6509 }
6510
6511 return CMD_SUCCESS;
6512 }
6513
6514 static int pim_cmd_interface_add(struct interface *ifp)
6515 {
6516 struct pim_interface *pim_ifp = ifp->info;
6517
6518 if (!pim_ifp) {
6519 pim_ifp = pim_if_new(ifp, false, true, false,
6520 false /*vxlan_term*/);
6521 if (!pim_ifp) {
6522 return 0;
6523 }
6524 } else {
6525 PIM_IF_DO_PIM(pim_ifp->options);
6526 }
6527
6528 pim_if_addr_add_all(ifp);
6529 pim_if_membership_refresh(ifp);
6530 return 1;
6531 }
6532
6533 DEFPY_HIDDEN (pim_test_sg_keepalive,
6534 pim_test_sg_keepalive_cmd,
6535 "test pim [vrf NAME$name] keepalive-reset A.B.C.D$source A.B.C.D$group",
6536 "Test code\n"
6537 PIM_STR
6538 VRF_CMD_HELP_STR
6539 "Reset the Keepalive Timer\n"
6540 "The Source we are resetting\n"
6541 "The Group we are resetting\n")
6542 {
6543 struct pim_upstream *up;
6544 struct pim_instance *pim;
6545 struct prefix_sg sg;
6546
6547 sg.src = source;
6548 sg.grp = group;
6549
6550 if (!name)
6551 pim = pim_get_pim_instance(VRF_DEFAULT);
6552 else {
6553 struct vrf *vrf = vrf_lookup_by_name(name);
6554
6555 if (!vrf) {
6556 vty_out(vty, "%% Vrf specified: %s does not exist\n",
6557 name);
6558 return CMD_WARNING;
6559 }
6560
6561 pim = pim_get_pim_instance(vrf->vrf_id);
6562 }
6563
6564 if (!pim) {
6565 vty_out(vty, "%% Unable to find pim instance\n");
6566 return CMD_WARNING;
6567 }
6568
6569 up = pim_upstream_find(pim, &sg);
6570 if (!up) {
6571 vty_out(vty, "%% Unable to find %s specified\n",
6572 pim_str_sg_dump(&sg));
6573 return CMD_WARNING;
6574 }
6575
6576 vty_out(vty, "Setting %s to current keep alive time: %d\n",
6577 pim_str_sg_dump(&sg), pim->keep_alive_time);
6578 pim_upstream_keep_alive_timer_start(up, pim->keep_alive_time);
6579
6580 return CMD_SUCCESS;
6581 }
6582
6583 DEFPY_HIDDEN (interface_ip_pim_activeactive,
6584 interface_ip_pim_activeactive_cmd,
6585 "[no$no] ip pim active-active",
6586 NO_STR
6587 IP_STR
6588 PIM_STR
6589 "Mark interface as Active-Active for MLAG operations, Hidden because not finished yet\n")
6590 {
6591 VTY_DECLVAR_CONTEXT(interface, ifp);
6592 struct pim_interface *pim_ifp;
6593
6594 if (!no && !pim_cmd_interface_add(ifp)) {
6595 vty_out(vty, "Could not enable PIM SM active-active on interface\n");
6596 return CMD_WARNING_CONFIG_FAILED;
6597 }
6598
6599 pim_ifp = ifp->info;
6600 if (no)
6601 pim_ifp->activeactive = false;
6602 else
6603 pim_ifp->activeactive = true;
6604
6605 return CMD_SUCCESS;
6606 }
6607
6608 DEFUN_HIDDEN (interface_ip_pim_ssm,
6609 interface_ip_pim_ssm_cmd,
6610 "ip pim ssm",
6611 IP_STR
6612 PIM_STR
6613 IFACE_PIM_STR)
6614 {
6615 VTY_DECLVAR_CONTEXT(interface, ifp);
6616
6617 if (!pim_cmd_interface_add(ifp)) {
6618 vty_out(vty, "Could not enable PIM SM on interface\n");
6619 return CMD_WARNING_CONFIG_FAILED;
6620 }
6621
6622 vty_out(vty,
6623 "WARN: Enabled PIM SM on interface; configure PIM SSM "
6624 "range if needed\n");
6625 return CMD_SUCCESS;
6626 }
6627
6628 static int interface_ip_pim_helper(struct vty *vty)
6629 {
6630 struct pim_interface *pim_ifp;
6631
6632 VTY_DECLVAR_CONTEXT(interface, ifp);
6633
6634 if (!pim_cmd_interface_add(ifp)) {
6635 vty_out(vty, "Could not enable PIM SM on interface\n");
6636 return CMD_WARNING_CONFIG_FAILED;
6637 }
6638
6639 pim_ifp = ifp->info;
6640
6641 pim_if_create_pimreg(pim_ifp->pim);
6642
6643 return CMD_SUCCESS;
6644 }
6645
6646 DEFUN_HIDDEN (interface_ip_pim_sm,
6647 interface_ip_pim_sm_cmd,
6648 "ip pim sm",
6649 IP_STR
6650 PIM_STR
6651 IFACE_PIM_SM_STR)
6652 {
6653 return interface_ip_pim_helper(vty);
6654 }
6655
6656 DEFUN (interface_ip_pim,
6657 interface_ip_pim_cmd,
6658 "ip pim",
6659 IP_STR
6660 PIM_STR)
6661 {
6662 return interface_ip_pim_helper(vty);
6663 }
6664
6665 static int pim_cmd_interface_delete(struct interface *ifp)
6666 {
6667 struct pim_interface *pim_ifp = ifp->info;
6668
6669 if (!pim_ifp)
6670 return 1;
6671
6672 PIM_IF_DONT_PIM(pim_ifp->options);
6673
6674 pim_if_membership_clear(ifp);
6675
6676 /*
6677 pim_sock_delete() removes all neighbors from
6678 pim_ifp->pim_neighbor_list.
6679 */
6680 pim_sock_delete(ifp, "pim unconfigured on interface");
6681
6682 if (!PIM_IF_TEST_IGMP(pim_ifp->options)) {
6683 pim_if_addr_del_all(ifp);
6684 pim_if_delete(ifp);
6685 }
6686
6687 return 1;
6688 }
6689
6690 static int interface_no_ip_pim_helper(struct vty *vty)
6691 {
6692 VTY_DECLVAR_CONTEXT(interface, ifp);
6693 if (!pim_cmd_interface_delete(ifp)) {
6694 vty_out(vty, "Unable to delete interface information\n");
6695 return CMD_WARNING_CONFIG_FAILED;
6696 }
6697
6698 return CMD_SUCCESS;
6699 }
6700
6701 DEFUN_HIDDEN (interface_no_ip_pim_ssm,
6702 interface_no_ip_pim_ssm_cmd,
6703 "no ip pim ssm",
6704 NO_STR
6705 IP_STR
6706 PIM_STR
6707 IFACE_PIM_STR)
6708 {
6709 return interface_no_ip_pim_helper(vty);
6710 }
6711
6712 DEFUN_HIDDEN (interface_no_ip_pim_sm,
6713 interface_no_ip_pim_sm_cmd,
6714 "no ip pim sm",
6715 NO_STR
6716 IP_STR
6717 PIM_STR
6718 IFACE_PIM_SM_STR)
6719 {
6720 return interface_no_ip_pim_helper(vty);
6721 }
6722
6723 DEFUN (interface_no_ip_pim,
6724 interface_no_ip_pim_cmd,
6725 "no ip pim",
6726 NO_STR
6727 IP_STR
6728 PIM_STR)
6729 {
6730 return interface_no_ip_pim_helper(vty);
6731 }
6732
6733 /* boundaries */
6734 DEFUN(interface_ip_pim_boundary_oil,
6735 interface_ip_pim_boundary_oil_cmd,
6736 "ip multicast boundary oil WORD",
6737 IP_STR
6738 "Generic multicast configuration options\n"
6739 "Define multicast boundary\n"
6740 "Filter OIL by group using prefix list\n"
6741 "Prefix list to filter OIL with\n")
6742 {
6743 VTY_DECLVAR_CONTEXT(interface, iif);
6744 struct pim_interface *pim_ifp;
6745 int idx = 0;
6746
6747 argv_find(argv, argc, "WORD", &idx);
6748
6749 PIM_GET_PIM_INTERFACE(pim_ifp, iif);
6750
6751 if (pim_ifp->boundary_oil_plist)
6752 XFREE(MTYPE_PIM_INTERFACE, pim_ifp->boundary_oil_plist);
6753
6754 pim_ifp->boundary_oil_plist =
6755 XSTRDUP(MTYPE_PIM_INTERFACE, argv[idx]->arg);
6756
6757 /* Interface will be pruned from OIL on next Join */
6758 return CMD_SUCCESS;
6759 }
6760
6761 DEFUN(interface_no_ip_pim_boundary_oil,
6762 interface_no_ip_pim_boundary_oil_cmd,
6763 "no ip multicast boundary oil [WORD]",
6764 NO_STR
6765 IP_STR
6766 "Generic multicast configuration options\n"
6767 "Define multicast boundary\n"
6768 "Filter OIL by group using prefix list\n"
6769 "Prefix list to filter OIL with\n")
6770 {
6771 VTY_DECLVAR_CONTEXT(interface, iif);
6772 struct pim_interface *pim_ifp;
6773 int idx = 0;
6774
6775 argv_find(argv, argc, "WORD", &idx);
6776
6777 PIM_GET_PIM_INTERFACE(pim_ifp, iif);
6778
6779 if (pim_ifp->boundary_oil_plist)
6780 XFREE(MTYPE_PIM_INTERFACE, pim_ifp->boundary_oil_plist);
6781
6782 return CMD_SUCCESS;
6783 }
6784
6785 DEFUN (interface_ip_mroute,
6786 interface_ip_mroute_cmd,
6787 "ip mroute INTERFACE A.B.C.D",
6788 IP_STR
6789 "Add multicast route\n"
6790 "Outgoing interface name\n"
6791 "Group address\n")
6792 {
6793 VTY_DECLVAR_CONTEXT(interface, iif);
6794 struct pim_interface *pim_ifp;
6795 struct pim_instance *pim;
6796 int idx_interface = 2;
6797 int idx_ipv4 = 3;
6798 struct interface *oif;
6799 const char *oifname;
6800 const char *grp_str;
6801 struct in_addr grp_addr;
6802 struct in_addr src_addr;
6803 int result;
6804
6805 PIM_GET_PIM_INTERFACE(pim_ifp, iif);
6806 pim = pim_ifp->pim;
6807
6808 oifname = argv[idx_interface]->arg;
6809 oif = if_lookup_by_name(oifname, pim->vrf_id);
6810 if (!oif) {
6811 vty_out(vty, "No such interface name %s\n", oifname);
6812 return CMD_WARNING;
6813 }
6814
6815 grp_str = argv[idx_ipv4]->arg;
6816 result = inet_pton(AF_INET, grp_str, &grp_addr);
6817 if (result <= 0) {
6818 vty_out(vty, "Bad group address %s: errno=%d: %s\n", grp_str,
6819 errno, safe_strerror(errno));
6820 return CMD_WARNING;
6821 }
6822
6823 src_addr.s_addr = INADDR_ANY;
6824
6825 if (pim_static_add(pim, iif, oif, grp_addr, src_addr)) {
6826 vty_out(vty, "Failed to add route\n");
6827 return CMD_WARNING;
6828 }
6829
6830 return CMD_SUCCESS;
6831 }
6832
6833 DEFUN (interface_ip_mroute_source,
6834 interface_ip_mroute_source_cmd,
6835 "ip mroute INTERFACE A.B.C.D A.B.C.D",
6836 IP_STR
6837 "Add multicast route\n"
6838 "Outgoing interface name\n"
6839 "Group address\n"
6840 "Source address\n")
6841 {
6842 VTY_DECLVAR_CONTEXT(interface, iif);
6843 struct pim_interface *pim_ifp;
6844 struct pim_instance *pim;
6845 int idx_interface = 2;
6846 int idx_ipv4 = 3;
6847 int idx_ipv4_2 = 4;
6848 struct interface *oif;
6849 const char *oifname;
6850 const char *grp_str;
6851 struct in_addr grp_addr;
6852 const char *src_str;
6853 struct in_addr src_addr;
6854 int result;
6855
6856 PIM_GET_PIM_INTERFACE(pim_ifp, iif);
6857 pim = pim_ifp->pim;
6858
6859 oifname = argv[idx_interface]->arg;
6860 oif = if_lookup_by_name(oifname, pim->vrf_id);
6861 if (!oif) {
6862 vty_out(vty, "No such interface name %s\n", oifname);
6863 return CMD_WARNING;
6864 }
6865
6866 grp_str = argv[idx_ipv4]->arg;
6867 result = inet_pton(AF_INET, grp_str, &grp_addr);
6868 if (result <= 0) {
6869 vty_out(vty, "Bad group address %s: errno=%d: %s\n", grp_str,
6870 errno, safe_strerror(errno));
6871 return CMD_WARNING;
6872 }
6873
6874 src_str = argv[idx_ipv4_2]->arg;
6875 result = inet_pton(AF_INET, src_str, &src_addr);
6876 if (result <= 0) {
6877 vty_out(vty, "Bad source address %s: errno=%d: %s\n", src_str,
6878 errno, safe_strerror(errno));
6879 return CMD_WARNING;
6880 }
6881
6882 if (pim_static_add(pim, iif, oif, grp_addr, src_addr)) {
6883 vty_out(vty, "Failed to add route\n");
6884 return CMD_WARNING;
6885 }
6886
6887 return CMD_SUCCESS;
6888 }
6889
6890 DEFUN (interface_no_ip_mroute,
6891 interface_no_ip_mroute_cmd,
6892 "no ip mroute INTERFACE A.B.C.D",
6893 NO_STR
6894 IP_STR
6895 "Add multicast route\n"
6896 "Outgoing interface name\n"
6897 "Group Address\n")
6898 {
6899 VTY_DECLVAR_CONTEXT(interface, iif);
6900 struct pim_interface *pim_ifp;
6901 struct pim_instance *pim;
6902 int idx_interface = 3;
6903 int idx_ipv4 = 4;
6904 struct interface *oif;
6905 const char *oifname;
6906 const char *grp_str;
6907 struct in_addr grp_addr;
6908 struct in_addr src_addr;
6909 int result;
6910
6911 PIM_GET_PIM_INTERFACE(pim_ifp, iif);
6912 pim = pim_ifp->pim;
6913
6914 oifname = argv[idx_interface]->arg;
6915 oif = if_lookup_by_name(oifname, pim->vrf_id);
6916 if (!oif) {
6917 vty_out(vty, "No such interface name %s\n", oifname);
6918 return CMD_WARNING;
6919 }
6920
6921 grp_str = argv[idx_ipv4]->arg;
6922 result = inet_pton(AF_INET, grp_str, &grp_addr);
6923 if (result <= 0) {
6924 vty_out(vty, "Bad group address %s: errno=%d: %s\n", grp_str,
6925 errno, safe_strerror(errno));
6926 return CMD_WARNING;
6927 }
6928
6929 src_addr.s_addr = INADDR_ANY;
6930
6931 if (pim_static_del(pim, iif, oif, grp_addr, src_addr)) {
6932 vty_out(vty, "Failed to remove route\n");
6933 return CMD_WARNING;
6934 }
6935
6936 return CMD_SUCCESS;
6937 }
6938
6939 DEFUN (interface_no_ip_mroute_source,
6940 interface_no_ip_mroute_source_cmd,
6941 "no ip mroute INTERFACE A.B.C.D A.B.C.D",
6942 NO_STR
6943 IP_STR
6944 "Add multicast route\n"
6945 "Outgoing interface name\n"
6946 "Group Address\n"
6947 "Source Address\n")
6948 {
6949 VTY_DECLVAR_CONTEXT(interface, iif);
6950 struct pim_interface *pim_ifp;
6951 struct pim_instance *pim;
6952 int idx_interface = 3;
6953 int idx_ipv4 = 4;
6954 int idx_ipv4_2 = 5;
6955 struct interface *oif;
6956 const char *oifname;
6957 const char *grp_str;
6958 struct in_addr grp_addr;
6959 const char *src_str;
6960 struct in_addr src_addr;
6961 int result;
6962
6963 PIM_GET_PIM_INTERFACE(pim_ifp, iif);
6964 pim = pim_ifp->pim;
6965
6966 oifname = argv[idx_interface]->arg;
6967 oif = if_lookup_by_name(oifname, pim->vrf_id);
6968 if (!oif) {
6969 vty_out(vty, "No such interface name %s\n", oifname);
6970 return CMD_WARNING;
6971 }
6972
6973 grp_str = argv[idx_ipv4]->arg;
6974 result = inet_pton(AF_INET, grp_str, &grp_addr);
6975 if (result <= 0) {
6976 vty_out(vty, "Bad group address %s: errno=%d: %s\n", grp_str,
6977 errno, safe_strerror(errno));
6978 return CMD_WARNING;
6979 }
6980
6981 src_str = argv[idx_ipv4_2]->arg;
6982 result = inet_pton(AF_INET, src_str, &src_addr);
6983 if (result <= 0) {
6984 vty_out(vty, "Bad source address %s: errno=%d: %s\n", src_str,
6985 errno, safe_strerror(errno));
6986 return CMD_WARNING;
6987 }
6988
6989 if (pim_static_del(pim, iif, oif, grp_addr, src_addr)) {
6990 vty_out(vty, "Failed to remove route\n");
6991 return CMD_WARNING;
6992 }
6993
6994 return CMD_SUCCESS;
6995 }
6996
6997 DEFUN (interface_ip_pim_hello,
6998 interface_ip_pim_hello_cmd,
6999 "ip pim hello (1-180) [(1-180)]",
7000 IP_STR
7001 PIM_STR
7002 IFACE_PIM_HELLO_STR
7003 IFACE_PIM_HELLO_TIME_STR
7004 IFACE_PIM_HELLO_HOLD_STR)
7005 {
7006 VTY_DECLVAR_CONTEXT(interface, ifp);
7007 int idx_time = 3;
7008 int idx_hold = 4;
7009 struct pim_interface *pim_ifp = ifp->info;
7010
7011 if (!pim_ifp) {
7012 if (!pim_cmd_interface_add(ifp)) {
7013 vty_out(vty, "Could not enable PIM SM on interface\n");
7014 return CMD_WARNING_CONFIG_FAILED;
7015 }
7016 }
7017
7018 pim_ifp = ifp->info;
7019 pim_ifp->pim_hello_period = strtol(argv[idx_time]->arg, NULL, 10);
7020
7021 if (argc == idx_hold + 1)
7022 pim_ifp->pim_default_holdtime =
7023 strtol(argv[idx_hold]->arg, NULL, 10);
7024
7025 return CMD_SUCCESS;
7026 }
7027
7028 DEFUN (interface_no_ip_pim_hello,
7029 interface_no_ip_pim_hello_cmd,
7030 "no ip pim hello [(1-180) (1-180)]",
7031 NO_STR
7032 IP_STR
7033 PIM_STR
7034 IFACE_PIM_HELLO_STR
7035 IFACE_PIM_HELLO_TIME_STR
7036 IFACE_PIM_HELLO_HOLD_STR)
7037 {
7038 VTY_DECLVAR_CONTEXT(interface, ifp);
7039 struct pim_interface *pim_ifp = ifp->info;
7040
7041 if (!pim_ifp) {
7042 vty_out(vty, "Pim not enabled on this interface\n");
7043 return CMD_WARNING_CONFIG_FAILED;
7044 }
7045
7046 pim_ifp->pim_hello_period = PIM_DEFAULT_HELLO_PERIOD;
7047 pim_ifp->pim_default_holdtime = -1;
7048
7049 return CMD_SUCCESS;
7050 }
7051
7052 DEFUN (debug_igmp,
7053 debug_igmp_cmd,
7054 "debug igmp",
7055 DEBUG_STR
7056 DEBUG_IGMP_STR)
7057 {
7058 PIM_DO_DEBUG_IGMP_EVENTS;
7059 PIM_DO_DEBUG_IGMP_PACKETS;
7060 PIM_DO_DEBUG_IGMP_TRACE;
7061 return CMD_SUCCESS;
7062 }
7063
7064 DEFUN (no_debug_igmp,
7065 no_debug_igmp_cmd,
7066 "no debug igmp",
7067 NO_STR
7068 DEBUG_STR
7069 DEBUG_IGMP_STR)
7070 {
7071 PIM_DONT_DEBUG_IGMP_EVENTS;
7072 PIM_DONT_DEBUG_IGMP_PACKETS;
7073 PIM_DONT_DEBUG_IGMP_TRACE;
7074 return CMD_SUCCESS;
7075 }
7076
7077
7078 DEFUN (debug_igmp_events,
7079 debug_igmp_events_cmd,
7080 "debug igmp events",
7081 DEBUG_STR
7082 DEBUG_IGMP_STR
7083 DEBUG_IGMP_EVENTS_STR)
7084 {
7085 PIM_DO_DEBUG_IGMP_EVENTS;
7086 return CMD_SUCCESS;
7087 }
7088
7089 DEFUN (no_debug_igmp_events,
7090 no_debug_igmp_events_cmd,
7091 "no debug igmp events",
7092 NO_STR
7093 DEBUG_STR
7094 DEBUG_IGMP_STR
7095 DEBUG_IGMP_EVENTS_STR)
7096 {
7097 PIM_DONT_DEBUG_IGMP_EVENTS;
7098 return CMD_SUCCESS;
7099 }
7100
7101
7102 DEFUN (debug_igmp_packets,
7103 debug_igmp_packets_cmd,
7104 "debug igmp packets",
7105 DEBUG_STR
7106 DEBUG_IGMP_STR
7107 DEBUG_IGMP_PACKETS_STR)
7108 {
7109 PIM_DO_DEBUG_IGMP_PACKETS;
7110 return CMD_SUCCESS;
7111 }
7112
7113 DEFUN (no_debug_igmp_packets,
7114 no_debug_igmp_packets_cmd,
7115 "no debug igmp packets",
7116 NO_STR
7117 DEBUG_STR
7118 DEBUG_IGMP_STR
7119 DEBUG_IGMP_PACKETS_STR)
7120 {
7121 PIM_DONT_DEBUG_IGMP_PACKETS;
7122 return CMD_SUCCESS;
7123 }
7124
7125
7126 DEFUN (debug_igmp_trace,
7127 debug_igmp_trace_cmd,
7128 "debug igmp trace",
7129 DEBUG_STR
7130 DEBUG_IGMP_STR
7131 DEBUG_IGMP_TRACE_STR)
7132 {
7133 PIM_DO_DEBUG_IGMP_TRACE;
7134 return CMD_SUCCESS;
7135 }
7136
7137 DEFUN (no_debug_igmp_trace,
7138 no_debug_igmp_trace_cmd,
7139 "no debug igmp trace",
7140 NO_STR
7141 DEBUG_STR
7142 DEBUG_IGMP_STR
7143 DEBUG_IGMP_TRACE_STR)
7144 {
7145 PIM_DONT_DEBUG_IGMP_TRACE;
7146 return CMD_SUCCESS;
7147 }
7148
7149
7150 DEFUN (debug_mroute,
7151 debug_mroute_cmd,
7152 "debug mroute",
7153 DEBUG_STR
7154 DEBUG_MROUTE_STR)
7155 {
7156 PIM_DO_DEBUG_MROUTE;
7157 return CMD_SUCCESS;
7158 }
7159
7160 DEFUN (debug_mroute_detail,
7161 debug_mroute_detail_cmd,
7162 "debug mroute detail",
7163 DEBUG_STR
7164 DEBUG_MROUTE_STR
7165 "detailed\n")
7166 {
7167 PIM_DO_DEBUG_MROUTE_DETAIL;
7168 return CMD_SUCCESS;
7169 }
7170
7171 DEFUN (no_debug_mroute,
7172 no_debug_mroute_cmd,
7173 "no debug mroute",
7174 NO_STR
7175 DEBUG_STR
7176 DEBUG_MROUTE_STR)
7177 {
7178 PIM_DONT_DEBUG_MROUTE;
7179 return CMD_SUCCESS;
7180 }
7181
7182 DEFUN (no_debug_mroute_detail,
7183 no_debug_mroute_detail_cmd,
7184 "no debug mroute detail",
7185 NO_STR
7186 DEBUG_STR
7187 DEBUG_MROUTE_STR
7188 "detailed\n")
7189 {
7190 PIM_DONT_DEBUG_MROUTE_DETAIL;
7191 return CMD_SUCCESS;
7192 }
7193
7194 DEFUN (debug_pim_static,
7195 debug_pim_static_cmd,
7196 "debug pim static",
7197 DEBUG_STR
7198 DEBUG_PIM_STR
7199 DEBUG_STATIC_STR)
7200 {
7201 PIM_DO_DEBUG_STATIC;
7202 return CMD_SUCCESS;
7203 }
7204
7205 DEFUN (no_debug_pim_static,
7206 no_debug_pim_static_cmd,
7207 "no debug pim static",
7208 NO_STR
7209 DEBUG_STR
7210 DEBUG_PIM_STR
7211 DEBUG_STATIC_STR)
7212 {
7213 PIM_DONT_DEBUG_STATIC;
7214 return CMD_SUCCESS;
7215 }
7216
7217
7218 DEFUN (debug_pim,
7219 debug_pim_cmd,
7220 "debug pim",
7221 DEBUG_STR
7222 DEBUG_PIM_STR)
7223 {
7224 PIM_DO_DEBUG_PIM_EVENTS;
7225 PIM_DO_DEBUG_PIM_PACKETS;
7226 PIM_DO_DEBUG_PIM_TRACE;
7227 PIM_DO_DEBUG_MSDP_EVENTS;
7228 PIM_DO_DEBUG_MSDP_PACKETS;
7229 return CMD_SUCCESS;
7230 }
7231
7232 DEFUN (no_debug_pim,
7233 no_debug_pim_cmd,
7234 "no debug pim",
7235 NO_STR
7236 DEBUG_STR
7237 DEBUG_PIM_STR)
7238 {
7239 PIM_DONT_DEBUG_PIM_EVENTS;
7240 PIM_DONT_DEBUG_PIM_PACKETS;
7241 PIM_DONT_DEBUG_PIM_TRACE;
7242 PIM_DONT_DEBUG_MSDP_EVENTS;
7243 PIM_DONT_DEBUG_MSDP_PACKETS;
7244
7245 PIM_DONT_DEBUG_PIM_PACKETDUMP_SEND;
7246 PIM_DONT_DEBUG_PIM_PACKETDUMP_RECV;
7247
7248 return CMD_SUCCESS;
7249 }
7250
7251 DEFUN (debug_pim_nht,
7252 debug_pim_nht_cmd,
7253 "debug pim nht",
7254 DEBUG_STR
7255 DEBUG_PIM_STR
7256 "Nexthop Tracking\n")
7257 {
7258 PIM_DO_DEBUG_PIM_NHT;
7259 return CMD_SUCCESS;
7260 }
7261
7262 DEFUN (no_debug_pim_nht,
7263 no_debug_pim_nht_cmd,
7264 "no debug pim nht",
7265 NO_STR
7266 DEBUG_STR
7267 DEBUG_PIM_STR
7268 "Nexthop Tracking\n")
7269 {
7270 PIM_DONT_DEBUG_PIM_NHT;
7271 return CMD_SUCCESS;
7272 }
7273
7274 DEFUN (debug_pim_nht_rp,
7275 debug_pim_nht_rp_cmd,
7276 "debug pim nht rp",
7277 DEBUG_STR
7278 DEBUG_PIM_STR
7279 "Nexthop Tracking\n"
7280 "RP Nexthop Tracking\n")
7281 {
7282 PIM_DO_DEBUG_PIM_NHT_RP;
7283 return CMD_SUCCESS;
7284 }
7285
7286 DEFUN (no_debug_pim_nht_rp,
7287 no_debug_pim_nht_rp_cmd,
7288 "no debug pim nht rp",
7289 NO_STR
7290 DEBUG_STR
7291 DEBUG_PIM_STR
7292 "Nexthop Tracking\n"
7293 "RP Nexthop Tracking\n")
7294 {
7295 PIM_DONT_DEBUG_PIM_NHT_RP;
7296 return CMD_SUCCESS;
7297 }
7298
7299 DEFUN (debug_pim_events,
7300 debug_pim_events_cmd,
7301 "debug pim events",
7302 DEBUG_STR
7303 DEBUG_PIM_STR
7304 DEBUG_PIM_EVENTS_STR)
7305 {
7306 PIM_DO_DEBUG_PIM_EVENTS;
7307 return CMD_SUCCESS;
7308 }
7309
7310 DEFUN (no_debug_pim_events,
7311 no_debug_pim_events_cmd,
7312 "no debug pim events",
7313 NO_STR
7314 DEBUG_STR
7315 DEBUG_PIM_STR
7316 DEBUG_PIM_EVENTS_STR)
7317 {
7318 PIM_DONT_DEBUG_PIM_EVENTS;
7319 return CMD_SUCCESS;
7320 }
7321
7322 DEFUN (debug_pim_packets,
7323 debug_pim_packets_cmd,
7324 "debug pim packets [<hello|joins|register>]",
7325 DEBUG_STR
7326 DEBUG_PIM_STR
7327 DEBUG_PIM_PACKETS_STR
7328 DEBUG_PIM_HELLO_PACKETS_STR
7329 DEBUG_PIM_J_P_PACKETS_STR
7330 DEBUG_PIM_PIM_REG_PACKETS_STR)
7331 {
7332 int idx = 0;
7333 if (argv_find(argv, argc, "hello", &idx)) {
7334 PIM_DO_DEBUG_PIM_HELLO;
7335 vty_out(vty, "PIM Hello debugging is on\n");
7336 } else if (argv_find(argv, argc, "joins", &idx)) {
7337 PIM_DO_DEBUG_PIM_J_P;
7338 vty_out(vty, "PIM Join/Prune debugging is on\n");
7339 } else if (argv_find(argv, argc, "register", &idx)) {
7340 PIM_DO_DEBUG_PIM_REG;
7341 vty_out(vty, "PIM Register debugging is on\n");
7342 } else {
7343 PIM_DO_DEBUG_PIM_PACKETS;
7344 vty_out(vty, "PIM Packet debugging is on \n");
7345 }
7346 return CMD_SUCCESS;
7347 }
7348
7349 DEFUN (no_debug_pim_packets,
7350 no_debug_pim_packets_cmd,
7351 "no debug pim packets [<hello|joins|register>]",
7352 NO_STR
7353 DEBUG_STR
7354 DEBUG_PIM_STR
7355 DEBUG_PIM_PACKETS_STR
7356 DEBUG_PIM_HELLO_PACKETS_STR
7357 DEBUG_PIM_J_P_PACKETS_STR
7358 DEBUG_PIM_PIM_REG_PACKETS_STR)
7359 {
7360 int idx = 0;
7361 if (argv_find(argv, argc, "hello", &idx)) {
7362 PIM_DONT_DEBUG_PIM_HELLO;
7363 vty_out(vty, "PIM Hello debugging is off \n");
7364 } else if (argv_find(argv, argc, "joins", &idx)) {
7365 PIM_DONT_DEBUG_PIM_J_P;
7366 vty_out(vty, "PIM Join/Prune debugging is off \n");
7367 } else if (argv_find(argv, argc, "register", &idx)) {
7368 PIM_DONT_DEBUG_PIM_REG;
7369 vty_out(vty, "PIM Register debugging is off\n");
7370 } else
7371 PIM_DONT_DEBUG_PIM_PACKETS;
7372
7373 return CMD_SUCCESS;
7374 }
7375
7376
7377 DEFUN (debug_pim_packetdump_send,
7378 debug_pim_packetdump_send_cmd,
7379 "debug pim packet-dump send",
7380 DEBUG_STR
7381 DEBUG_PIM_STR
7382 DEBUG_PIM_PACKETDUMP_STR
7383 DEBUG_PIM_PACKETDUMP_SEND_STR)
7384 {
7385 PIM_DO_DEBUG_PIM_PACKETDUMP_SEND;
7386 return CMD_SUCCESS;
7387 }
7388
7389 DEFUN (no_debug_pim_packetdump_send,
7390 no_debug_pim_packetdump_send_cmd,
7391 "no debug pim packet-dump send",
7392 NO_STR
7393 DEBUG_STR
7394 DEBUG_PIM_STR
7395 DEBUG_PIM_PACKETDUMP_STR
7396 DEBUG_PIM_PACKETDUMP_SEND_STR)
7397 {
7398 PIM_DONT_DEBUG_PIM_PACKETDUMP_SEND;
7399 return CMD_SUCCESS;
7400 }
7401
7402 DEFUN (debug_pim_packetdump_recv,
7403 debug_pim_packetdump_recv_cmd,
7404 "debug pim packet-dump receive",
7405 DEBUG_STR
7406 DEBUG_PIM_STR
7407 DEBUG_PIM_PACKETDUMP_STR
7408 DEBUG_PIM_PACKETDUMP_RECV_STR)
7409 {
7410 PIM_DO_DEBUG_PIM_PACKETDUMP_RECV;
7411 return CMD_SUCCESS;
7412 }
7413
7414 DEFUN (no_debug_pim_packetdump_recv,
7415 no_debug_pim_packetdump_recv_cmd,
7416 "no debug pim packet-dump receive",
7417 NO_STR
7418 DEBUG_STR
7419 DEBUG_PIM_STR
7420 DEBUG_PIM_PACKETDUMP_STR
7421 DEBUG_PIM_PACKETDUMP_RECV_STR)
7422 {
7423 PIM_DONT_DEBUG_PIM_PACKETDUMP_RECV;
7424 return CMD_SUCCESS;
7425 }
7426
7427 DEFUN (debug_pim_trace,
7428 debug_pim_trace_cmd,
7429 "debug pim trace",
7430 DEBUG_STR
7431 DEBUG_PIM_STR
7432 DEBUG_PIM_TRACE_STR)
7433 {
7434 PIM_DO_DEBUG_PIM_TRACE;
7435 return CMD_SUCCESS;
7436 }
7437
7438 DEFUN (debug_pim_trace_detail,
7439 debug_pim_trace_detail_cmd,
7440 "debug pim trace detail",
7441 DEBUG_STR
7442 DEBUG_PIM_STR
7443 DEBUG_PIM_TRACE_STR
7444 "Detailed Information\n")
7445 {
7446 PIM_DO_DEBUG_PIM_TRACE_DETAIL;
7447 return CMD_SUCCESS;
7448 }
7449
7450 DEFUN (no_debug_pim_trace,
7451 no_debug_pim_trace_cmd,
7452 "no debug pim trace",
7453 NO_STR
7454 DEBUG_STR
7455 DEBUG_PIM_STR
7456 DEBUG_PIM_TRACE_STR)
7457 {
7458 PIM_DONT_DEBUG_PIM_TRACE;
7459 return CMD_SUCCESS;
7460 }
7461
7462 DEFUN (no_debug_pim_trace_detail,
7463 no_debug_pim_trace_detail_cmd,
7464 "no debug pim trace detail",
7465 NO_STR
7466 DEBUG_STR
7467 DEBUG_PIM_STR
7468 DEBUG_PIM_TRACE_STR
7469 "Detailed Information\n")
7470 {
7471 PIM_DONT_DEBUG_PIM_TRACE_DETAIL;
7472 return CMD_SUCCESS;
7473 }
7474
7475 DEFUN (debug_ssmpingd,
7476 debug_ssmpingd_cmd,
7477 "debug ssmpingd",
7478 DEBUG_STR
7479 DEBUG_SSMPINGD_STR)
7480 {
7481 PIM_DO_DEBUG_SSMPINGD;
7482 return CMD_SUCCESS;
7483 }
7484
7485 DEFUN (no_debug_ssmpingd,
7486 no_debug_ssmpingd_cmd,
7487 "no debug ssmpingd",
7488 NO_STR
7489 DEBUG_STR
7490 DEBUG_SSMPINGD_STR)
7491 {
7492 PIM_DONT_DEBUG_SSMPINGD;
7493 return CMD_SUCCESS;
7494 }
7495
7496 DEFUN (debug_pim_zebra,
7497 debug_pim_zebra_cmd,
7498 "debug pim zebra",
7499 DEBUG_STR
7500 DEBUG_PIM_STR
7501 DEBUG_PIM_ZEBRA_STR)
7502 {
7503 PIM_DO_DEBUG_ZEBRA;
7504 return CMD_SUCCESS;
7505 }
7506
7507 DEFUN (no_debug_pim_zebra,
7508 no_debug_pim_zebra_cmd,
7509 "no debug pim zebra",
7510 NO_STR
7511 DEBUG_STR
7512 DEBUG_PIM_STR
7513 DEBUG_PIM_ZEBRA_STR)
7514 {
7515 PIM_DONT_DEBUG_ZEBRA;
7516 return CMD_SUCCESS;
7517 }
7518
7519 DEFUN (debug_pim_vxlan,
7520 debug_pim_vxlan_cmd,
7521 "debug pim vxlan",
7522 DEBUG_STR
7523 DEBUG_PIM_STR
7524 DEBUG_PIM_VXLAN_STR)
7525 {
7526 PIM_DO_DEBUG_VXLAN;
7527 return CMD_SUCCESS;
7528 }
7529
7530 DEFUN (no_debug_pim_vxlan,
7531 no_debug_pim_vxlan_cmd,
7532 "no debug pim vxlan",
7533 NO_STR
7534 DEBUG_STR
7535 DEBUG_PIM_STR
7536 DEBUG_PIM_VXLAN_STR)
7537 {
7538 PIM_DONT_DEBUG_VXLAN;
7539 return CMD_SUCCESS;
7540 }
7541
7542 DEFUN (debug_msdp,
7543 debug_msdp_cmd,
7544 "debug msdp",
7545 DEBUG_STR
7546 DEBUG_MSDP_STR)
7547 {
7548 PIM_DO_DEBUG_MSDP_EVENTS;
7549 PIM_DO_DEBUG_MSDP_PACKETS;
7550 return CMD_SUCCESS;
7551 }
7552
7553 DEFUN (no_debug_msdp,
7554 no_debug_msdp_cmd,
7555 "no debug msdp",
7556 NO_STR
7557 DEBUG_STR
7558 DEBUG_MSDP_STR)
7559 {
7560 PIM_DONT_DEBUG_MSDP_EVENTS;
7561 PIM_DONT_DEBUG_MSDP_PACKETS;
7562 return CMD_SUCCESS;
7563 }
7564
7565 DEFUN (debug_msdp_events,
7566 debug_msdp_events_cmd,
7567 "debug msdp events",
7568 DEBUG_STR
7569 DEBUG_MSDP_STR
7570 DEBUG_MSDP_EVENTS_STR)
7571 {
7572 PIM_DO_DEBUG_MSDP_EVENTS;
7573 return CMD_SUCCESS;
7574 }
7575
7576 DEFUN (no_debug_msdp_events,
7577 no_debug_msdp_events_cmd,
7578 "no debug msdp events",
7579 NO_STR
7580 DEBUG_STR
7581 DEBUG_MSDP_STR
7582 DEBUG_MSDP_EVENTS_STR)
7583 {
7584 PIM_DONT_DEBUG_MSDP_EVENTS;
7585 return CMD_SUCCESS;
7586 }
7587
7588 DEFUN (debug_msdp_packets,
7589 debug_msdp_packets_cmd,
7590 "debug msdp packets",
7591 DEBUG_STR
7592 DEBUG_MSDP_STR
7593 DEBUG_MSDP_PACKETS_STR)
7594 {
7595 PIM_DO_DEBUG_MSDP_PACKETS;
7596 return CMD_SUCCESS;
7597 }
7598
7599 DEFUN (no_debug_msdp_packets,
7600 no_debug_msdp_packets_cmd,
7601 "no debug msdp packets",
7602 NO_STR
7603 DEBUG_STR
7604 DEBUG_MSDP_STR
7605 DEBUG_MSDP_PACKETS_STR)
7606 {
7607 PIM_DONT_DEBUG_MSDP_PACKETS;
7608 return CMD_SUCCESS;
7609 }
7610
7611 DEFUN (debug_mtrace,
7612 debug_mtrace_cmd,
7613 "debug mtrace",
7614 DEBUG_STR
7615 DEBUG_MTRACE_STR)
7616 {
7617 PIM_DO_DEBUG_MTRACE;
7618 return CMD_SUCCESS;
7619 }
7620
7621 DEFUN (no_debug_mtrace,
7622 no_debug_mtrace_cmd,
7623 "no debug mtrace",
7624 NO_STR
7625 DEBUG_STR
7626 DEBUG_MTRACE_STR)
7627 {
7628 PIM_DONT_DEBUG_MTRACE;
7629 return CMD_SUCCESS;
7630 }
7631
7632 DEFUN_NOSH (show_debugging_pim,
7633 show_debugging_pim_cmd,
7634 "show debugging [pim]",
7635 SHOW_STR
7636 DEBUG_STR
7637 PIM_STR)
7638 {
7639 vty_out(vty, "PIM debugging status\n");
7640
7641 pim_debug_config_write(vty);
7642
7643 return CMD_SUCCESS;
7644 }
7645
7646 static int interface_pim_use_src_cmd_worker(struct vty *vty, const char *source)
7647 {
7648 int result;
7649 struct in_addr source_addr;
7650 int ret = CMD_SUCCESS;
7651 VTY_DECLVAR_CONTEXT(interface, ifp);
7652
7653 result = inet_pton(AF_INET, source, &source_addr);
7654 if (result <= 0) {
7655 vty_out(vty, "%% Bad source address %s: errno=%d: %s\n", source,
7656 errno, safe_strerror(errno));
7657 return CMD_WARNING_CONFIG_FAILED;
7658 }
7659
7660 result = pim_update_source_set(ifp, source_addr);
7661 switch (result) {
7662 case PIM_SUCCESS:
7663 break;
7664 case PIM_IFACE_NOT_FOUND:
7665 ret = CMD_WARNING_CONFIG_FAILED;
7666 vty_out(vty, "Pim not enabled on this interface\n");
7667 break;
7668 case PIM_UPDATE_SOURCE_DUP:
7669 ret = CMD_WARNING;
7670 vty_out(vty, "%% Source already set to %s\n", source);
7671 break;
7672 default:
7673 ret = CMD_WARNING_CONFIG_FAILED;
7674 vty_out(vty, "%% Source set failed\n");
7675 }
7676
7677 return ret;
7678 }
7679
7680 DEFUN (interface_pim_use_source,
7681 interface_pim_use_source_cmd,
7682 "ip pim use-source A.B.C.D",
7683 IP_STR
7684 PIM_STR
7685 "Configure primary IP address\n"
7686 "source ip address\n")
7687 {
7688 return interface_pim_use_src_cmd_worker(vty, argv[3]->arg);
7689 }
7690
7691 DEFUN (interface_no_pim_use_source,
7692 interface_no_pim_use_source_cmd,
7693 "no ip pim use-source [A.B.C.D]",
7694 NO_STR
7695 IP_STR
7696 PIM_STR
7697 "Delete source IP address\n"
7698 "source ip address\n")
7699 {
7700 return interface_pim_use_src_cmd_worker(vty, "0.0.0.0");
7701 }
7702
7703 DEFUN (ip_pim_bfd,
7704 ip_pim_bfd_cmd,
7705 "ip pim bfd",
7706 IP_STR
7707 PIM_STR
7708 "Enables BFD support\n")
7709 {
7710 VTY_DECLVAR_CONTEXT(interface, ifp);
7711 struct pim_interface *pim_ifp = ifp->info;
7712 struct bfd_info *bfd_info = NULL;
7713
7714 if (!pim_ifp) {
7715 if (!pim_cmd_interface_add(ifp)) {
7716 vty_out(vty, "Could not enable PIM SM on interface\n");
7717 return CMD_WARNING;
7718 }
7719 }
7720 pim_ifp = ifp->info;
7721
7722 bfd_info = pim_ifp->bfd_info;
7723
7724 if (!bfd_info || !CHECK_FLAG(bfd_info->flags, BFD_FLAG_PARAM_CFG))
7725 pim_bfd_if_param_set(ifp, BFD_DEF_MIN_RX, BFD_DEF_MIN_TX,
7726 BFD_DEF_DETECT_MULT, 1);
7727
7728 return CMD_SUCCESS;
7729 }
7730
7731 DEFUN (no_ip_pim_bfd,
7732 no_ip_pim_bfd_cmd,
7733 "no ip pim bfd",
7734 NO_STR
7735 IP_STR
7736 PIM_STR
7737 "Disables BFD support\n")
7738 {
7739 VTY_DECLVAR_CONTEXT(interface, ifp);
7740 struct pim_interface *pim_ifp = ifp->info;
7741
7742 if (!pim_ifp) {
7743 vty_out(vty, "Pim not enabled on this interface\n");
7744 return CMD_WARNING;
7745 }
7746
7747 if (pim_ifp->bfd_info) {
7748 pim_bfd_reg_dereg_all_nbr(ifp, ZEBRA_BFD_DEST_DEREGISTER);
7749 bfd_info_free(&(pim_ifp->bfd_info));
7750 }
7751
7752 return CMD_SUCCESS;
7753 }
7754
7755 #if HAVE_BFDD > 0
7756 DEFUN_HIDDEN(
7757 #else
7758 DEFUN(
7759 #endif /* HAVE_BFDD */
7760 ip_pim_bfd_param,
7761 ip_pim_bfd_param_cmd,
7762 "ip pim bfd (2-255) (50-60000) (50-60000)",
7763 IP_STR
7764 PIM_STR
7765 "Enables BFD support\n"
7766 "Detect Multiplier\n"
7767 "Required min receive interval\n"
7768 "Desired min transmit interval\n")
7769 {
7770 VTY_DECLVAR_CONTEXT(interface, ifp);
7771 int idx_number = 3;
7772 int idx_number_2 = 4;
7773 int idx_number_3 = 5;
7774 uint32_t rx_val;
7775 uint32_t tx_val;
7776 uint8_t dm_val;
7777 int ret;
7778 struct pim_interface *pim_ifp = ifp->info;
7779
7780 if (!pim_ifp) {
7781 if (!pim_cmd_interface_add(ifp)) {
7782 vty_out(vty, "Could not enable PIM SM on interface\n");
7783 return CMD_WARNING;
7784 }
7785 }
7786
7787 if ((ret = bfd_validate_param(
7788 vty, argv[idx_number]->arg, argv[idx_number_2]->arg,
7789 argv[idx_number_3]->arg, &dm_val, &rx_val, &tx_val))
7790 != CMD_SUCCESS)
7791 return ret;
7792
7793 pim_bfd_if_param_set(ifp, rx_val, tx_val, dm_val, 0);
7794
7795 return CMD_SUCCESS;
7796 }
7797
7798 #if HAVE_BFDD == 0
7799 ALIAS(no_ip_pim_bfd, no_ip_pim_bfd_param_cmd,
7800 "no ip pim bfd (2-255) (50-60000) (50-60000)", NO_STR IP_STR PIM_STR
7801 "Enables BFD support\n"
7802 "Detect Multiplier\n"
7803 "Required min receive interval\n"
7804 "Desired min transmit interval\n")
7805 #endif /* !HAVE_BFDD */
7806
7807 static int ip_msdp_peer_cmd_worker(struct pim_instance *pim, struct vty *vty,
7808 const char *peer, const char *local)
7809 {
7810 enum pim_msdp_err result;
7811 struct in_addr peer_addr;
7812 struct in_addr local_addr;
7813 int ret = CMD_SUCCESS;
7814
7815 result = inet_pton(AF_INET, peer, &peer_addr);
7816 if (result <= 0) {
7817 vty_out(vty, "%% Bad peer address %s: errno=%d: %s\n", peer,
7818 errno, safe_strerror(errno));
7819 return CMD_WARNING_CONFIG_FAILED;
7820 }
7821
7822 result = inet_pton(AF_INET, local, &local_addr);
7823 if (result <= 0) {
7824 vty_out(vty, "%% Bad source address %s: errno=%d: %s\n", local,
7825 errno, safe_strerror(errno));
7826 return CMD_WARNING_CONFIG_FAILED;
7827 }
7828
7829 result = pim_msdp_peer_add(pim, peer_addr, local_addr, "default",
7830 NULL /* mp_p */);
7831 switch (result) {
7832 case PIM_MSDP_ERR_NONE:
7833 break;
7834 case PIM_MSDP_ERR_OOM:
7835 ret = CMD_WARNING_CONFIG_FAILED;
7836 vty_out(vty, "%% Out of memory\n");
7837 break;
7838 case PIM_MSDP_ERR_PEER_EXISTS:
7839 ret = CMD_WARNING;
7840 vty_out(vty, "%% Peer exists\n");
7841 break;
7842 case PIM_MSDP_ERR_MAX_MESH_GROUPS:
7843 ret = CMD_WARNING_CONFIG_FAILED;
7844 vty_out(vty, "%% Only one mesh-group allowed currently\n");
7845 break;
7846 default:
7847 ret = CMD_WARNING_CONFIG_FAILED;
7848 vty_out(vty, "%% peer add failed\n");
7849 }
7850
7851 return ret;
7852 }
7853
7854 DEFUN_HIDDEN (ip_msdp_peer,
7855 ip_msdp_peer_cmd,
7856 "ip msdp peer A.B.C.D source A.B.C.D",
7857 IP_STR
7858 CFG_MSDP_STR
7859 "Configure MSDP peer\n"
7860 "peer ip address\n"
7861 "Source address for TCP connection\n"
7862 "local ip address\n")
7863 {
7864 PIM_DECLVAR_CONTEXT(vrf, pim);
7865 return ip_msdp_peer_cmd_worker(pim, vty, argv[3]->arg, argv[5]->arg);
7866 }
7867
7868 static int ip_no_msdp_peer_cmd_worker(struct pim_instance *pim, struct vty *vty,
7869 const char *peer)
7870 {
7871 enum pim_msdp_err result;
7872 struct in_addr peer_addr;
7873
7874 result = inet_pton(AF_INET, peer, &peer_addr);
7875 if (result <= 0) {
7876 vty_out(vty, "%% Bad peer address %s: errno=%d: %s\n", peer,
7877 errno, safe_strerror(errno));
7878 return CMD_WARNING_CONFIG_FAILED;
7879 }
7880
7881 result = pim_msdp_peer_del(pim, peer_addr);
7882 switch (result) {
7883 case PIM_MSDP_ERR_NONE:
7884 break;
7885 case PIM_MSDP_ERR_NO_PEER:
7886 vty_out(vty, "%% Peer does not exist\n");
7887 break;
7888 default:
7889 vty_out(vty, "%% peer del failed\n");
7890 }
7891
7892 return result ? CMD_WARNING_CONFIG_FAILED : CMD_SUCCESS;
7893 }
7894
7895 DEFUN_HIDDEN (no_ip_msdp_peer,
7896 no_ip_msdp_peer_cmd,
7897 "no ip msdp peer A.B.C.D",
7898 NO_STR
7899 IP_STR
7900 CFG_MSDP_STR
7901 "Delete MSDP peer\n"
7902 "peer ip address\n")
7903 {
7904 PIM_DECLVAR_CONTEXT(vrf, pim);
7905 return ip_no_msdp_peer_cmd_worker(pim, vty, argv[4]->arg);
7906 }
7907
7908 static int ip_msdp_mesh_group_member_cmd_worker(struct pim_instance *pim,
7909 struct vty *vty, const char *mg,
7910 const char *mbr)
7911 {
7912 enum pim_msdp_err result;
7913 struct in_addr mbr_ip;
7914 int ret = CMD_SUCCESS;
7915
7916 result = inet_pton(AF_INET, mbr, &mbr_ip);
7917 if (result <= 0) {
7918 vty_out(vty, "%% Bad member address %s: errno=%d: %s\n", mbr,
7919 errno, safe_strerror(errno));
7920 return CMD_WARNING_CONFIG_FAILED;
7921 }
7922
7923 result = pim_msdp_mg_mbr_add(pim, mg, mbr_ip);
7924 switch (result) {
7925 case PIM_MSDP_ERR_NONE:
7926 break;
7927 case PIM_MSDP_ERR_OOM:
7928 ret = CMD_WARNING_CONFIG_FAILED;
7929 vty_out(vty, "%% Out of memory\n");
7930 break;
7931 case PIM_MSDP_ERR_MG_MBR_EXISTS:
7932 ret = CMD_WARNING;
7933 vty_out(vty, "%% mesh-group member exists\n");
7934 break;
7935 case PIM_MSDP_ERR_MAX_MESH_GROUPS:
7936 ret = CMD_WARNING_CONFIG_FAILED;
7937 vty_out(vty, "%% Only one mesh-group allowed currently\n");
7938 break;
7939 default:
7940 ret = CMD_WARNING_CONFIG_FAILED;
7941 vty_out(vty, "%% member add failed\n");
7942 }
7943
7944 return ret;
7945 }
7946
7947 DEFUN (ip_msdp_mesh_group_member,
7948 ip_msdp_mesh_group_member_cmd,
7949 "ip msdp mesh-group WORD member A.B.C.D",
7950 IP_STR
7951 CFG_MSDP_STR
7952 "Configure MSDP mesh-group\n"
7953 "mesh group name\n"
7954 "mesh group member\n"
7955 "peer ip address\n")
7956 {
7957 PIM_DECLVAR_CONTEXT(vrf, pim);
7958 return ip_msdp_mesh_group_member_cmd_worker(pim, vty, argv[3]->arg,
7959 argv[5]->arg);
7960 }
7961
7962 static int ip_no_msdp_mesh_group_member_cmd_worker(struct pim_instance *pim,
7963 struct vty *vty,
7964 const char *mg,
7965 const char *mbr)
7966 {
7967 enum pim_msdp_err result;
7968 struct in_addr mbr_ip;
7969
7970 result = inet_pton(AF_INET, mbr, &mbr_ip);
7971 if (result <= 0) {
7972 vty_out(vty, "%% Bad member address %s: errno=%d: %s\n", mbr,
7973 errno, safe_strerror(errno));
7974 return CMD_WARNING_CONFIG_FAILED;
7975 }
7976
7977 result = pim_msdp_mg_mbr_del(pim, mg, mbr_ip);
7978 switch (result) {
7979 case PIM_MSDP_ERR_NONE:
7980 break;
7981 case PIM_MSDP_ERR_NO_MG:
7982 vty_out(vty, "%% mesh-group does not exist\n");
7983 break;
7984 case PIM_MSDP_ERR_NO_MG_MBR:
7985 vty_out(vty, "%% mesh-group member does not exist\n");
7986 break;
7987 default:
7988 vty_out(vty, "%% mesh-group member del failed\n");
7989 }
7990
7991 return result ? CMD_WARNING_CONFIG_FAILED : CMD_SUCCESS;
7992 }
7993 DEFUN (no_ip_msdp_mesh_group_member,
7994 no_ip_msdp_mesh_group_member_cmd,
7995 "no ip msdp mesh-group WORD member A.B.C.D",
7996 NO_STR
7997 IP_STR
7998 CFG_MSDP_STR
7999 "Delete MSDP mesh-group member\n"
8000 "mesh group name\n"
8001 "mesh group member\n"
8002 "peer ip address\n")
8003 {
8004 PIM_DECLVAR_CONTEXT(vrf, pim);
8005 return ip_no_msdp_mesh_group_member_cmd_worker(pim, vty, argv[4]->arg,
8006 argv[6]->arg);
8007 }
8008
8009 static int ip_msdp_mesh_group_source_cmd_worker(struct pim_instance *pim,
8010 struct vty *vty, const char *mg,
8011 const char *src)
8012 {
8013 enum pim_msdp_err result;
8014 struct in_addr src_ip;
8015
8016 result = inet_pton(AF_INET, src, &src_ip);
8017 if (result <= 0) {
8018 vty_out(vty, "%% Bad source address %s: errno=%d: %s\n", src,
8019 errno, safe_strerror(errno));
8020 return CMD_WARNING_CONFIG_FAILED;
8021 }
8022
8023 result = pim_msdp_mg_src_add(pim, mg, src_ip);
8024 switch (result) {
8025 case PIM_MSDP_ERR_NONE:
8026 break;
8027 case PIM_MSDP_ERR_OOM:
8028 vty_out(vty, "%% Out of memory\n");
8029 break;
8030 case PIM_MSDP_ERR_MAX_MESH_GROUPS:
8031 vty_out(vty, "%% Only one mesh-group allowed currently\n");
8032 break;
8033 default:
8034 vty_out(vty, "%% source add failed\n");
8035 }
8036
8037 return result ? CMD_WARNING_CONFIG_FAILED : CMD_SUCCESS;
8038 }
8039
8040
8041 DEFUN (ip_msdp_mesh_group_source,
8042 ip_msdp_mesh_group_source_cmd,
8043 "ip msdp mesh-group WORD source A.B.C.D",
8044 IP_STR
8045 CFG_MSDP_STR
8046 "Configure MSDP mesh-group\n"
8047 "mesh group name\n"
8048 "mesh group local address\n"
8049 "source ip address for the TCP connection\n")
8050 {
8051 PIM_DECLVAR_CONTEXT(vrf, pim);
8052 return ip_msdp_mesh_group_source_cmd_worker(pim, vty, argv[3]->arg,
8053 argv[5]->arg);
8054 }
8055
8056 static int ip_no_msdp_mesh_group_source_cmd_worker(struct pim_instance *pim,
8057 struct vty *vty,
8058 const char *mg)
8059 {
8060 enum pim_msdp_err result;
8061
8062 result = pim_msdp_mg_src_del(pim, mg);
8063 switch (result) {
8064 case PIM_MSDP_ERR_NONE:
8065 break;
8066 case PIM_MSDP_ERR_NO_MG:
8067 vty_out(vty, "%% mesh-group does not exist\n");
8068 break;
8069 default:
8070 vty_out(vty, "%% mesh-group source del failed\n");
8071 }
8072
8073 return result ? CMD_WARNING_CONFIG_FAILED : CMD_SUCCESS;
8074 }
8075
8076 static int ip_no_msdp_mesh_group_cmd_worker(struct pim_instance *pim,
8077 struct vty *vty, const char *mg)
8078 {
8079 enum pim_msdp_err result;
8080
8081 result = pim_msdp_mg_del(pim, mg);
8082 switch (result) {
8083 case PIM_MSDP_ERR_NONE:
8084 break;
8085 case PIM_MSDP_ERR_NO_MG:
8086 vty_out(vty, "%% mesh-group does not exist\n");
8087 break;
8088 default:
8089 vty_out(vty, "%% mesh-group source del failed\n");
8090 }
8091
8092 return result ? CMD_WARNING_CONFIG_FAILED : CMD_SUCCESS;
8093 }
8094
8095 DEFUN (no_ip_msdp_mesh_group_source,
8096 no_ip_msdp_mesh_group_source_cmd,
8097 "no ip msdp mesh-group WORD source [A.B.C.D]",
8098 NO_STR
8099 IP_STR
8100 CFG_MSDP_STR
8101 "Delete MSDP mesh-group source\n"
8102 "mesh group name\n"
8103 "mesh group source\n"
8104 "mesh group local address\n")
8105 {
8106 PIM_DECLVAR_CONTEXT(vrf, pim);
8107 if (argc == 7)
8108 return ip_no_msdp_mesh_group_cmd_worker(pim, vty, argv[6]->arg);
8109 else
8110 return ip_no_msdp_mesh_group_source_cmd_worker(pim, vty,
8111 argv[4]->arg);
8112 }
8113
8114 static void print_empty_json_obj(struct vty *vty)
8115 {
8116 json_object *json;
8117 json = json_object_new_object();
8118 vty_out(vty, "%s\n",
8119 json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY));
8120 json_object_free(json);
8121 }
8122
8123 static void ip_msdp_show_mesh_group(struct pim_instance *pim, struct vty *vty,
8124 bool uj)
8125 {
8126 struct listnode *mbrnode;
8127 struct pim_msdp_mg_mbr *mbr;
8128 struct pim_msdp_mg *mg = pim->msdp.mg;
8129 char mbr_str[INET_ADDRSTRLEN];
8130 char src_str[INET_ADDRSTRLEN];
8131 char state_str[PIM_MSDP_STATE_STRLEN];
8132 enum pim_msdp_peer_state state;
8133 json_object *json = NULL;
8134 json_object *json_mg_row = NULL;
8135 json_object *json_members = NULL;
8136 json_object *json_row = NULL;
8137
8138 if (!mg) {
8139 if (uj)
8140 print_empty_json_obj(vty);
8141 return;
8142 }
8143
8144 pim_inet4_dump("<source?>", mg->src_ip, src_str, sizeof(src_str));
8145 if (uj) {
8146 json = json_object_new_object();
8147 /* currently there is only one mesh group but we should still
8148 * make
8149 * it a dict with mg-name as key */
8150 json_mg_row = json_object_new_object();
8151 json_object_string_add(json_mg_row, "name",
8152 mg->mesh_group_name);
8153 json_object_string_add(json_mg_row, "source", src_str);
8154 } else {
8155 vty_out(vty, "Mesh group : %s\n", mg->mesh_group_name);
8156 vty_out(vty, " Source : %s\n", src_str);
8157 vty_out(vty, " Member State\n");
8158 }
8159
8160 for (ALL_LIST_ELEMENTS_RO(mg->mbr_list, mbrnode, mbr)) {
8161 pim_inet4_dump("<mbr?>", mbr->mbr_ip, mbr_str, sizeof(mbr_str));
8162 if (mbr->mp) {
8163 state = mbr->mp->state;
8164 } else {
8165 state = PIM_MSDP_DISABLED;
8166 }
8167 pim_msdp_state_dump(state, state_str, sizeof(state_str));
8168 if (uj) {
8169 json_row = json_object_new_object();
8170 json_object_string_add(json_row, "member", mbr_str);
8171 json_object_string_add(json_row, "state", state_str);
8172 if (!json_members) {
8173 json_members = json_object_new_object();
8174 json_object_object_add(json_mg_row, "members",
8175 json_members);
8176 }
8177 json_object_object_add(json_members, mbr_str, json_row);
8178 } else {
8179 vty_out(vty, " %-15s %11s\n", mbr_str, state_str);
8180 }
8181 }
8182
8183 if (uj) {
8184 json_object_object_add(json, mg->mesh_group_name, json_mg_row);
8185 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8186 json, JSON_C_TO_STRING_PRETTY));
8187 json_object_free(json);
8188 }
8189 }
8190
8191 DEFUN (show_ip_msdp_mesh_group,
8192 show_ip_msdp_mesh_group_cmd,
8193 "show ip msdp [vrf NAME] mesh-group [json]",
8194 SHOW_STR
8195 IP_STR
8196 MSDP_STR
8197 VRF_CMD_HELP_STR
8198 "MSDP mesh-group information\n"
8199 JSON_STR)
8200 {
8201 bool uj = use_json(argc, argv);
8202 int idx = 2;
8203 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
8204
8205 if (!vrf)
8206 return CMD_WARNING;
8207
8208 ip_msdp_show_mesh_group(vrf->info, vty, uj);
8209
8210 return CMD_SUCCESS;
8211 }
8212
8213 DEFUN (show_ip_msdp_mesh_group_vrf_all,
8214 show_ip_msdp_mesh_group_vrf_all_cmd,
8215 "show ip msdp vrf all mesh-group [json]",
8216 SHOW_STR
8217 IP_STR
8218 MSDP_STR
8219 VRF_CMD_HELP_STR
8220 "MSDP mesh-group information\n"
8221 JSON_STR)
8222 {
8223 bool uj = use_json(argc, argv);
8224 struct vrf *vrf;
8225 bool first = true;
8226
8227 if (uj)
8228 vty_out(vty, "{ ");
8229 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
8230 if (uj) {
8231 if (!first)
8232 vty_out(vty, ", ");
8233 vty_out(vty, " \"%s\": ", vrf->name);
8234 first = false;
8235 } else
8236 vty_out(vty, "VRF: %s\n", vrf->name);
8237 ip_msdp_show_mesh_group(vrf->info, vty, uj);
8238 }
8239 if (uj)
8240 vty_out(vty, "}\n");
8241
8242 return CMD_SUCCESS;
8243 }
8244
8245 static void ip_msdp_show_peers(struct pim_instance *pim, struct vty *vty,
8246 bool uj)
8247 {
8248 struct listnode *mpnode;
8249 struct pim_msdp_peer *mp;
8250 char peer_str[INET_ADDRSTRLEN];
8251 char local_str[INET_ADDRSTRLEN];
8252 char state_str[PIM_MSDP_STATE_STRLEN];
8253 char timebuf[PIM_MSDP_UPTIME_STRLEN];
8254 int64_t now;
8255 json_object *json = NULL;
8256 json_object *json_row = NULL;
8257
8258
8259 if (uj) {
8260 json = json_object_new_object();
8261 } else {
8262 vty_out(vty,
8263 "Peer Local State Uptime SaCnt\n");
8264 }
8265
8266 for (ALL_LIST_ELEMENTS_RO(pim->msdp.peer_list, mpnode, mp)) {
8267 if (mp->state == PIM_MSDP_ESTABLISHED) {
8268 now = pim_time_monotonic_sec();
8269 pim_time_uptime(timebuf, sizeof(timebuf),
8270 now - mp->uptime);
8271 } else {
8272 strcpy(timebuf, "-");
8273 }
8274 pim_inet4_dump("<peer?>", mp->peer, peer_str, sizeof(peer_str));
8275 pim_inet4_dump("<local?>", mp->local, local_str,
8276 sizeof(local_str));
8277 pim_msdp_state_dump(mp->state, state_str, sizeof(state_str));
8278 if (uj) {
8279 json_row = json_object_new_object();
8280 json_object_string_add(json_row, "peer", peer_str);
8281 json_object_string_add(json_row, "local", local_str);
8282 json_object_string_add(json_row, "state", state_str);
8283 json_object_string_add(json_row, "upTime", timebuf);
8284 json_object_int_add(json_row, "saCount", mp->sa_cnt);
8285 json_object_object_add(json, peer_str, json_row);
8286 } else {
8287 vty_out(vty, "%-15s %15s %11s %8s %6d\n", peer_str,
8288 local_str, state_str, timebuf, mp->sa_cnt);
8289 }
8290 }
8291
8292 if (uj) {
8293 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8294 json, JSON_C_TO_STRING_PRETTY));
8295 json_object_free(json);
8296 }
8297 }
8298
8299 static void ip_msdp_show_peers_detail(struct pim_instance *pim, struct vty *vty,
8300 const char *peer, bool uj)
8301 {
8302 struct listnode *mpnode;
8303 struct pim_msdp_peer *mp;
8304 char peer_str[INET_ADDRSTRLEN];
8305 char local_str[INET_ADDRSTRLEN];
8306 char state_str[PIM_MSDP_STATE_STRLEN];
8307 char timebuf[PIM_MSDP_UPTIME_STRLEN];
8308 char katimer[PIM_MSDP_TIMER_STRLEN];
8309 char crtimer[PIM_MSDP_TIMER_STRLEN];
8310 char holdtimer[PIM_MSDP_TIMER_STRLEN];
8311 int64_t now;
8312 json_object *json = NULL;
8313 json_object *json_row = NULL;
8314
8315 if (uj) {
8316 json = json_object_new_object();
8317 }
8318
8319 for (ALL_LIST_ELEMENTS_RO(pim->msdp.peer_list, mpnode, mp)) {
8320 pim_inet4_dump("<peer?>", mp->peer, peer_str, sizeof(peer_str));
8321 if (strcmp(peer, "detail") && strcmp(peer, peer_str))
8322 continue;
8323
8324 if (mp->state == PIM_MSDP_ESTABLISHED) {
8325 now = pim_time_monotonic_sec();
8326 pim_time_uptime(timebuf, sizeof(timebuf),
8327 now - mp->uptime);
8328 } else {
8329 strcpy(timebuf, "-");
8330 }
8331 pim_inet4_dump("<local?>", mp->local, local_str,
8332 sizeof(local_str));
8333 pim_msdp_state_dump(mp->state, state_str, sizeof(state_str));
8334 pim_time_timer_to_hhmmss(katimer, sizeof(katimer),
8335 mp->ka_timer);
8336 pim_time_timer_to_hhmmss(crtimer, sizeof(crtimer),
8337 mp->cr_timer);
8338 pim_time_timer_to_hhmmss(holdtimer, sizeof(holdtimer),
8339 mp->hold_timer);
8340
8341 if (uj) {
8342 json_row = json_object_new_object();
8343 json_object_string_add(json_row, "peer", peer_str);
8344 json_object_string_add(json_row, "local", local_str);
8345 json_object_string_add(json_row, "meshGroupName",
8346 mp->mesh_group_name);
8347 json_object_string_add(json_row, "state", state_str);
8348 json_object_string_add(json_row, "upTime", timebuf);
8349 json_object_string_add(json_row, "keepAliveTimer",
8350 katimer);
8351 json_object_string_add(json_row, "connRetryTimer",
8352 crtimer);
8353 json_object_string_add(json_row, "holdTimer",
8354 holdtimer);
8355 json_object_string_add(json_row, "lastReset",
8356 mp->last_reset);
8357 json_object_int_add(json_row, "connAttempts",
8358 mp->conn_attempts);
8359 json_object_int_add(json_row, "establishedChanges",
8360 mp->est_flaps);
8361 json_object_int_add(json_row, "saCount", mp->sa_cnt);
8362 json_object_int_add(json_row, "kaSent", mp->ka_tx_cnt);
8363 json_object_int_add(json_row, "kaRcvd", mp->ka_rx_cnt);
8364 json_object_int_add(json_row, "saSent", mp->sa_tx_cnt);
8365 json_object_int_add(json_row, "saRcvd", mp->sa_rx_cnt);
8366 json_object_object_add(json, peer_str, json_row);
8367 } else {
8368 vty_out(vty, "Peer : %s\n", peer_str);
8369 vty_out(vty, " Local : %s\n", local_str);
8370 vty_out(vty, " Mesh Group : %s\n",
8371 mp->mesh_group_name);
8372 vty_out(vty, " State : %s\n", state_str);
8373 vty_out(vty, " Uptime : %s\n", timebuf);
8374
8375 vty_out(vty, " Keepalive Timer : %s\n", katimer);
8376 vty_out(vty, " Conn Retry Timer : %s\n", crtimer);
8377 vty_out(vty, " Hold Timer : %s\n", holdtimer);
8378 vty_out(vty, " Last Reset : %s\n",
8379 mp->last_reset);
8380 vty_out(vty, " Conn Attempts : %d\n",
8381 mp->conn_attempts);
8382 vty_out(vty, " Established Changes : %d\n",
8383 mp->est_flaps);
8384 vty_out(vty, " SA Count : %d\n",
8385 mp->sa_cnt);
8386 vty_out(vty, " Statistics :\n");
8387 vty_out(vty,
8388 " Sent Rcvd\n");
8389 vty_out(vty, " Keepalives : %10d %10d\n",
8390 mp->ka_tx_cnt, mp->ka_rx_cnt);
8391 vty_out(vty, " SAs : %10d %10d\n",
8392 mp->sa_tx_cnt, mp->sa_rx_cnt);
8393 vty_out(vty, "\n");
8394 }
8395 }
8396
8397 if (uj) {
8398 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8399 json, JSON_C_TO_STRING_PRETTY));
8400 json_object_free(json);
8401 }
8402 }
8403
8404 DEFUN (show_ip_msdp_peer_detail,
8405 show_ip_msdp_peer_detail_cmd,
8406 "show ip msdp [vrf NAME] peer [detail|A.B.C.D] [json]",
8407 SHOW_STR
8408 IP_STR
8409 MSDP_STR
8410 VRF_CMD_HELP_STR
8411 "MSDP peer information\n"
8412 "Detailed output\n"
8413 "peer ip address\n"
8414 JSON_STR)
8415 {
8416 bool uj = use_json(argc, argv);
8417 int idx = 2;
8418 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
8419
8420 if (!vrf)
8421 return CMD_WARNING;
8422
8423 char *arg = NULL;
8424
8425 if (argv_find(argv, argc, "detail", &idx))
8426 arg = argv[idx]->text;
8427 else if (argv_find(argv, argc, "A.B.C.D", &idx))
8428 arg = argv[idx]->arg;
8429
8430 if (arg)
8431 ip_msdp_show_peers_detail(vrf->info, vty, argv[idx]->arg, uj);
8432 else
8433 ip_msdp_show_peers(vrf->info, vty, uj);
8434
8435 return CMD_SUCCESS;
8436 }
8437
8438 DEFUN (show_ip_msdp_peer_detail_vrf_all,
8439 show_ip_msdp_peer_detail_vrf_all_cmd,
8440 "show ip msdp vrf all peer [detail|A.B.C.D] [json]",
8441 SHOW_STR
8442 IP_STR
8443 MSDP_STR
8444 VRF_CMD_HELP_STR
8445 "MSDP peer information\n"
8446 "Detailed output\n"
8447 "peer ip address\n"
8448 JSON_STR)
8449 {
8450 int idx = 2;
8451 bool uj = use_json(argc, argv);
8452 struct vrf *vrf;
8453 bool first = true;
8454
8455 if (uj)
8456 vty_out(vty, "{ ");
8457 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
8458 if (uj) {
8459 if (!first)
8460 vty_out(vty, ", ");
8461 vty_out(vty, " \"%s\": ", vrf->name);
8462 first = false;
8463 } else
8464 vty_out(vty, "VRF: %s\n", vrf->name);
8465 if (argv_find(argv, argc, "detail", &idx)
8466 || argv_find(argv, argc, "A.B.C.D", &idx))
8467 ip_msdp_show_peers_detail(vrf->info, vty,
8468 argv[idx]->arg, uj);
8469 else
8470 ip_msdp_show_peers(vrf->info, vty, uj);
8471 }
8472 if (uj)
8473 vty_out(vty, "}\n");
8474
8475 return CMD_SUCCESS;
8476 }
8477
8478 static void ip_msdp_show_sa(struct pim_instance *pim, struct vty *vty, bool uj)
8479 {
8480 struct listnode *sanode;
8481 struct pim_msdp_sa *sa;
8482 char src_str[INET_ADDRSTRLEN];
8483 char grp_str[INET_ADDRSTRLEN];
8484 char rp_str[INET_ADDRSTRLEN];
8485 char timebuf[PIM_MSDP_UPTIME_STRLEN];
8486 char spt_str[8];
8487 char local_str[8];
8488 int64_t now;
8489 json_object *json = NULL;
8490 json_object *json_group = NULL;
8491 json_object *json_row = NULL;
8492
8493 if (uj) {
8494 json = json_object_new_object();
8495 } else {
8496 vty_out(vty,
8497 "Source Group RP Local SPT Uptime\n");
8498 }
8499
8500 for (ALL_LIST_ELEMENTS_RO(pim->msdp.sa_list, sanode, sa)) {
8501 now = pim_time_monotonic_sec();
8502 pim_time_uptime(timebuf, sizeof(timebuf), now - sa->uptime);
8503 pim_inet4_dump("<src?>", sa->sg.src, src_str, sizeof(src_str));
8504 pim_inet4_dump("<grp?>", sa->sg.grp, grp_str, sizeof(grp_str));
8505 if (sa->flags & PIM_MSDP_SAF_PEER) {
8506 pim_inet4_dump("<rp?>", sa->rp, rp_str, sizeof(rp_str));
8507 if (sa->up) {
8508 strcpy(spt_str, "yes");
8509 } else {
8510 strcpy(spt_str, "no");
8511 }
8512 } else {
8513 strcpy(rp_str, "-");
8514 strcpy(spt_str, "-");
8515 }
8516 if (sa->flags & PIM_MSDP_SAF_LOCAL) {
8517 strcpy(local_str, "yes");
8518 } else {
8519 strcpy(local_str, "no");
8520 }
8521 if (uj) {
8522 json_object_object_get_ex(json, grp_str, &json_group);
8523
8524 if (!json_group) {
8525 json_group = json_object_new_object();
8526 json_object_object_add(json, grp_str,
8527 json_group);
8528 }
8529
8530 json_row = json_object_new_object();
8531 json_object_string_add(json_row, "source", src_str);
8532 json_object_string_add(json_row, "group", grp_str);
8533 json_object_string_add(json_row, "rp", rp_str);
8534 json_object_string_add(json_row, "local", local_str);
8535 json_object_string_add(json_row, "sptSetup", spt_str);
8536 json_object_string_add(json_row, "upTime", timebuf);
8537 json_object_object_add(json_group, src_str, json_row);
8538 } else {
8539 vty_out(vty, "%-15s %15s %15s %5c %3c %8s\n",
8540 src_str, grp_str, rp_str, local_str[0],
8541 spt_str[0], timebuf);
8542 }
8543 }
8544
8545 if (uj) {
8546 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8547 json, JSON_C_TO_STRING_PRETTY));
8548 json_object_free(json);
8549 }
8550 }
8551
8552 static void ip_msdp_show_sa_entry_detail(struct pim_msdp_sa *sa,
8553 const char *src_str,
8554 const char *grp_str, struct vty *vty,
8555 bool uj, json_object *json)
8556 {
8557 char rp_str[INET_ADDRSTRLEN];
8558 char peer_str[INET_ADDRSTRLEN];
8559 char timebuf[PIM_MSDP_UPTIME_STRLEN];
8560 char spt_str[8];
8561 char local_str[8];
8562 char statetimer[PIM_MSDP_TIMER_STRLEN];
8563 int64_t now;
8564 json_object *json_group = NULL;
8565 json_object *json_row = NULL;
8566
8567 now = pim_time_monotonic_sec();
8568 pim_time_uptime(timebuf, sizeof(timebuf), now - sa->uptime);
8569 if (sa->flags & PIM_MSDP_SAF_PEER) {
8570 pim_inet4_dump("<rp?>", sa->rp, rp_str, sizeof(rp_str));
8571 pim_inet4_dump("<peer?>", sa->peer, peer_str, sizeof(peer_str));
8572 if (sa->up) {
8573 strcpy(spt_str, "yes");
8574 } else {
8575 strcpy(spt_str, "no");
8576 }
8577 } else {
8578 strcpy(rp_str, "-");
8579 strcpy(peer_str, "-");
8580 strcpy(spt_str, "-");
8581 }
8582 if (sa->flags & PIM_MSDP_SAF_LOCAL) {
8583 strcpy(local_str, "yes");
8584 } else {
8585 strcpy(local_str, "no");
8586 }
8587 pim_time_timer_to_hhmmss(statetimer, sizeof(statetimer),
8588 sa->sa_state_timer);
8589 if (uj) {
8590 json_object_object_get_ex(json, grp_str, &json_group);
8591
8592 if (!json_group) {
8593 json_group = json_object_new_object();
8594 json_object_object_add(json, grp_str, json_group);
8595 }
8596
8597 json_row = json_object_new_object();
8598 json_object_string_add(json_row, "source", src_str);
8599 json_object_string_add(json_row, "group", grp_str);
8600 json_object_string_add(json_row, "rp", rp_str);
8601 json_object_string_add(json_row, "local", local_str);
8602 json_object_string_add(json_row, "sptSetup", spt_str);
8603 json_object_string_add(json_row, "upTime", timebuf);
8604 json_object_string_add(json_row, "stateTimer", statetimer);
8605 json_object_object_add(json_group, src_str, json_row);
8606 } else {
8607 vty_out(vty, "SA : %s\n", sa->sg_str);
8608 vty_out(vty, " RP : %s\n", rp_str);
8609 vty_out(vty, " Peer : %s\n", peer_str);
8610 vty_out(vty, " Local : %s\n", local_str);
8611 vty_out(vty, " SPT Setup : %s\n", spt_str);
8612 vty_out(vty, " Uptime : %s\n", timebuf);
8613 vty_out(vty, " State Timer : %s\n", statetimer);
8614 vty_out(vty, "\n");
8615 }
8616 }
8617
8618 static void ip_msdp_show_sa_detail(struct pim_instance *pim, struct vty *vty,
8619 bool uj)
8620 {
8621 struct listnode *sanode;
8622 struct pim_msdp_sa *sa;
8623 char src_str[INET_ADDRSTRLEN];
8624 char grp_str[INET_ADDRSTRLEN];
8625 json_object *json = NULL;
8626
8627 if (uj) {
8628 json = json_object_new_object();
8629 }
8630
8631 for (ALL_LIST_ELEMENTS_RO(pim->msdp.sa_list, sanode, sa)) {
8632 pim_inet4_dump("<src?>", sa->sg.src, src_str, sizeof(src_str));
8633 pim_inet4_dump("<grp?>", sa->sg.grp, grp_str, sizeof(grp_str));
8634 ip_msdp_show_sa_entry_detail(sa, src_str, grp_str, vty, uj,
8635 json);
8636 }
8637
8638 if (uj) {
8639 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8640 json, JSON_C_TO_STRING_PRETTY));
8641 json_object_free(json);
8642 }
8643 }
8644
8645 DEFUN (show_ip_msdp_sa_detail,
8646 show_ip_msdp_sa_detail_cmd,
8647 "show ip msdp [vrf NAME] sa detail [json]",
8648 SHOW_STR
8649 IP_STR
8650 MSDP_STR
8651 VRF_CMD_HELP_STR
8652 "MSDP active-source information\n"
8653 "Detailed output\n"
8654 JSON_STR)
8655 {
8656 bool uj = use_json(argc, argv);
8657 int idx = 2;
8658 struct vrf *vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
8659
8660 if (!vrf)
8661 return CMD_WARNING;
8662
8663 ip_msdp_show_sa_detail(vrf->info, vty, uj);
8664
8665 return CMD_SUCCESS;
8666 }
8667
8668 DEFUN (show_ip_msdp_sa_detail_vrf_all,
8669 show_ip_msdp_sa_detail_vrf_all_cmd,
8670 "show ip msdp vrf all sa detail [json]",
8671 SHOW_STR
8672 IP_STR
8673 MSDP_STR
8674 VRF_CMD_HELP_STR
8675 "MSDP active-source information\n"
8676 "Detailed output\n"
8677 JSON_STR)
8678 {
8679 bool uj = use_json(argc, argv);
8680 struct vrf *vrf;
8681 bool first = true;
8682
8683 if (uj)
8684 vty_out(vty, "{ ");
8685 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
8686 if (uj) {
8687 if (!first)
8688 vty_out(vty, ", ");
8689 vty_out(vty, " \"%s\": ", vrf->name);
8690 first = false;
8691 } else
8692 vty_out(vty, "VRF: %s\n", vrf->name);
8693 ip_msdp_show_sa_detail(vrf->info, vty, uj);
8694 }
8695 if (uj)
8696 vty_out(vty, "}\n");
8697
8698 return CMD_SUCCESS;
8699 }
8700
8701 static void ip_msdp_show_sa_addr(struct pim_instance *pim, struct vty *vty,
8702 const char *addr, bool uj)
8703 {
8704 struct listnode *sanode;
8705 struct pim_msdp_sa *sa;
8706 char src_str[INET_ADDRSTRLEN];
8707 char grp_str[INET_ADDRSTRLEN];
8708 json_object *json = NULL;
8709
8710 if (uj) {
8711 json = json_object_new_object();
8712 }
8713
8714 for (ALL_LIST_ELEMENTS_RO(pim->msdp.sa_list, sanode, sa)) {
8715 pim_inet4_dump("<src?>", sa->sg.src, src_str, sizeof(src_str));
8716 pim_inet4_dump("<grp?>", sa->sg.grp, grp_str, sizeof(grp_str));
8717 if (!strcmp(addr, src_str) || !strcmp(addr, grp_str)) {
8718 ip_msdp_show_sa_entry_detail(sa, src_str, grp_str, vty,
8719 uj, json);
8720 }
8721 }
8722
8723 if (uj) {
8724 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8725 json, JSON_C_TO_STRING_PRETTY));
8726 json_object_free(json);
8727 }
8728 }
8729
8730 static void ip_msdp_show_sa_sg(struct pim_instance *pim, struct vty *vty,
8731 const char *src, const char *grp, bool uj)
8732 {
8733 struct listnode *sanode;
8734 struct pim_msdp_sa *sa;
8735 char src_str[INET_ADDRSTRLEN];
8736 char grp_str[INET_ADDRSTRLEN];
8737 json_object *json = NULL;
8738
8739 if (uj) {
8740 json = json_object_new_object();
8741 }
8742
8743 for (ALL_LIST_ELEMENTS_RO(pim->msdp.sa_list, sanode, sa)) {
8744 pim_inet4_dump("<src?>", sa->sg.src, src_str, sizeof(src_str));
8745 pim_inet4_dump("<grp?>", sa->sg.grp, grp_str, sizeof(grp_str));
8746 if (!strcmp(src, src_str) && !strcmp(grp, grp_str)) {
8747 ip_msdp_show_sa_entry_detail(sa, src_str, grp_str, vty,
8748 uj, json);
8749 }
8750 }
8751
8752 if (uj) {
8753 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8754 json, JSON_C_TO_STRING_PRETTY));
8755 json_object_free(json);
8756 }
8757 }
8758
8759 DEFUN (show_ip_msdp_sa_sg,
8760 show_ip_msdp_sa_sg_cmd,
8761 "show ip msdp [vrf NAME] sa [A.B.C.D [A.B.C.D]] [json]",
8762 SHOW_STR
8763 IP_STR
8764 MSDP_STR
8765 VRF_CMD_HELP_STR
8766 "MSDP active-source information\n"
8767 "source or group ip\n"
8768 "group ip\n"
8769 JSON_STR)
8770 {
8771 bool uj = use_json(argc, argv);
8772 struct vrf *vrf;
8773 int idx = 2;
8774
8775 vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
8776
8777 if (!vrf)
8778 return CMD_WARNING;
8779
8780 char *src_ip = argv_find(argv, argc, "A.B.C.D", &idx) ? argv[idx++]->arg
8781 : NULL;
8782 char *grp_ip = idx < argc && argv_find(argv, argc, "A.B.C.D", &idx)
8783 ? argv[idx]->arg
8784 : NULL;
8785
8786 if (src_ip && grp_ip)
8787 ip_msdp_show_sa_sg(vrf->info, vty, src_ip, grp_ip, uj);
8788 else if (src_ip)
8789 ip_msdp_show_sa_addr(vrf->info, vty, src_ip, uj);
8790 else
8791 ip_msdp_show_sa(vrf->info, vty, uj);
8792
8793 return CMD_SUCCESS;
8794 }
8795
8796 DEFUN (show_ip_msdp_sa_sg_vrf_all,
8797 show_ip_msdp_sa_sg_vrf_all_cmd,
8798 "show ip msdp vrf all sa [A.B.C.D [A.B.C.D]] [json]",
8799 SHOW_STR
8800 IP_STR
8801 MSDP_STR
8802 VRF_CMD_HELP_STR
8803 "MSDP active-source information\n"
8804 "source or group ip\n"
8805 "group ip\n"
8806 JSON_STR)
8807 {
8808 bool uj = use_json(argc, argv);
8809 struct vrf *vrf;
8810 bool first = true;
8811 int idx = 2;
8812
8813 char *src_ip = argv_find(argv, argc, "A.B.C.D", &idx) ? argv[idx++]->arg
8814 : NULL;
8815 char *grp_ip = idx < argc && argv_find(argv, argc, "A.B.C.D", &idx)
8816 ? argv[idx]->arg
8817 : NULL;
8818
8819 if (uj)
8820 vty_out(vty, "{ ");
8821 RB_FOREACH (vrf, vrf_name_head, &vrfs_by_name) {
8822 if (uj) {
8823 if (!first)
8824 vty_out(vty, ", ");
8825 vty_out(vty, " \"%s\": ", vrf->name);
8826 first = false;
8827 } else
8828 vty_out(vty, "VRF: %s\n", vrf->name);
8829
8830 if (src_ip && grp_ip)
8831 ip_msdp_show_sa_sg(vrf->info, vty, src_ip, grp_ip, uj);
8832 else if (src_ip)
8833 ip_msdp_show_sa_addr(vrf->info, vty, src_ip, uj);
8834 else
8835 ip_msdp_show_sa(vrf->info, vty, uj);
8836 }
8837 if (uj)
8838 vty_out(vty, "}\n");
8839
8840 return CMD_SUCCESS;
8841 }
8842
8843 struct pim_sg_cache_walk_data {
8844 struct vty *vty;
8845 json_object *json;
8846 json_object *json_group;
8847 struct in_addr addr;
8848 bool addr_match;
8849 };
8850
8851 static void pim_show_vxlan_sg_entry(struct pim_vxlan_sg *vxlan_sg,
8852 struct pim_sg_cache_walk_data *cwd)
8853 {
8854 struct vty *vty = cwd->vty;
8855 json_object *json = cwd->json;
8856 char src_str[INET_ADDRSTRLEN];
8857 char grp_str[INET_ADDRSTRLEN];
8858 json_object *json_row;
8859 bool installed = (vxlan_sg->up)?TRUE:FALSE;
8860 const char *iif_name = vxlan_sg->iif?vxlan_sg->iif->name:"-";
8861 const char *oif_name;
8862
8863 if (pim_vxlan_is_orig_mroute(vxlan_sg))
8864 oif_name = vxlan_sg->orig_oif?vxlan_sg->orig_oif->name:"";
8865 else
8866 oif_name = vxlan_sg->term_oif?vxlan_sg->term_oif->name:"";
8867
8868 if (cwd->addr_match && (vxlan_sg->sg.src.s_addr != cwd->addr.s_addr) &&
8869 (vxlan_sg->sg.grp.s_addr != cwd->addr.s_addr)) {
8870 return;
8871 }
8872 pim_inet4_dump("<src?>", vxlan_sg->sg.src, src_str, sizeof(src_str));
8873 pim_inet4_dump("<grp?>", vxlan_sg->sg.grp, grp_str, sizeof(grp_str));
8874 if (json) {
8875 json_object_object_get_ex(json, grp_str, &cwd->json_group);
8876
8877 if (!cwd->json_group) {
8878 cwd->json_group = json_object_new_object();
8879 json_object_object_add(json, grp_str,
8880 cwd->json_group);
8881 }
8882
8883 json_row = json_object_new_object();
8884 json_object_string_add(json_row, "source", src_str);
8885 json_object_string_add(json_row, "group", grp_str);
8886 json_object_string_add(json_row, "input", iif_name);
8887 json_object_string_add(json_row, "output", oif_name);
8888 if (installed)
8889 json_object_boolean_true_add(json_row, "installed");
8890 else
8891 json_object_boolean_false_add(json_row, "installed");
8892 json_object_object_add(cwd->json_group, src_str, json_row);
8893 } else {
8894 vty_out(vty, "%-15s %-15s %-15s %-15s %-5s\n",
8895 src_str, grp_str, iif_name, oif_name,
8896 installed?"I":"");
8897 }
8898 }
8899
8900 static void pim_show_vxlan_sg_hash_entry(struct hash_backet *backet, void *arg)
8901 {
8902 pim_show_vxlan_sg_entry((struct pim_vxlan_sg *)backet->data,
8903 (struct pim_sg_cache_walk_data *)arg);
8904 }
8905
8906 static void pim_show_vxlan_sg(struct pim_instance *pim,
8907 struct vty *vty, bool uj)
8908 {
8909 json_object *json = NULL;
8910 struct pim_sg_cache_walk_data cwd;
8911
8912 if (uj) {
8913 json = json_object_new_object();
8914 } else {
8915 vty_out(vty, "Codes: I -> installed\n");
8916 vty_out(vty,
8917 "Source Group Input Output Flags\n");
8918 }
8919
8920 memset(&cwd, 0, sizeof(cwd));
8921 cwd.vty = vty;
8922 cwd.json = json;
8923 hash_iterate(pim->vxlan.sg_hash, pim_show_vxlan_sg_hash_entry, &cwd);
8924
8925 if (uj) {
8926 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8927 json, JSON_C_TO_STRING_PRETTY));
8928 json_object_free(json);
8929 }
8930 }
8931
8932 static void pim_show_vxlan_sg_match_addr(struct pim_instance *pim,
8933 struct vty *vty, char *addr_str, bool uj)
8934 {
8935 json_object *json = NULL;
8936 struct pim_sg_cache_walk_data cwd;
8937 int result = 0;
8938
8939 memset(&cwd, 0, sizeof(cwd));
8940 result = inet_pton(AF_INET, addr_str, &cwd.addr);
8941 if (result <= 0) {
8942 vty_out(vty, "Bad address %s: errno=%d: %s\n", addr_str,
8943 errno, safe_strerror(errno));
8944 return;
8945 }
8946
8947 if (uj) {
8948 json = json_object_new_object();
8949 } else {
8950 vty_out(vty, "Codes: I -> installed\n");
8951 vty_out(vty,
8952 "Source Group Input Output Flags\n");
8953 }
8954
8955 cwd.vty = vty;
8956 cwd.json = json;
8957 cwd.addr_match = TRUE;
8958 hash_iterate(pim->vxlan.sg_hash, pim_show_vxlan_sg_hash_entry, &cwd);
8959
8960 if (uj) {
8961 vty_out(vty, "%s\n", json_object_to_json_string_ext(
8962 json, JSON_C_TO_STRING_PRETTY));
8963 json_object_free(json);
8964 }
8965 }
8966
8967 static void pim_show_vxlan_sg_one(struct pim_instance *pim,
8968 struct vty *vty, char *src_str, char *grp_str, bool uj)
8969 {
8970 json_object *json = NULL;
8971 struct prefix_sg sg;
8972 int result = 0;
8973 struct pim_vxlan_sg *vxlan_sg;
8974 const char *iif_name;
8975 bool installed;
8976 const char *oif_name;
8977
8978 result = inet_pton(AF_INET, src_str, &sg.src);
8979 if (result <= 0) {
8980 vty_out(vty, "Bad src address %s: errno=%d: %s\n", src_str,
8981 errno, safe_strerror(errno));
8982 return;
8983 }
8984 result = inet_pton(AF_INET, grp_str, &sg.grp);
8985 if (result <= 0) {
8986 vty_out(vty, "Bad grp address %s: errno=%d: %s\n", grp_str,
8987 errno, safe_strerror(errno));
8988 return;
8989 }
8990
8991 sg.family = AF_INET;
8992 sg.prefixlen = IPV4_MAX_BITLEN;
8993 if (uj)
8994 json = json_object_new_object();
8995
8996 vxlan_sg = pim_vxlan_sg_find(pim, &sg);
8997 if (vxlan_sg) {
8998 installed = (vxlan_sg->up)?TRUE:FALSE;
8999 iif_name = vxlan_sg->iif?vxlan_sg->iif->name:"-";
9000
9001 if (pim_vxlan_is_orig_mroute(vxlan_sg))
9002 oif_name =
9003 vxlan_sg->orig_oif?vxlan_sg->orig_oif->name:"";
9004 else
9005 oif_name =
9006 vxlan_sg->term_oif?vxlan_sg->term_oif->name:"";
9007
9008 if (uj) {
9009 json_object_string_add(json, "source", src_str);
9010 json_object_string_add(json, "group", grp_str);
9011 json_object_string_add(json, "input", iif_name);
9012 json_object_string_add(json, "output", oif_name);
9013 if (installed)
9014 json_object_boolean_true_add(json, "installed");
9015 else
9016 json_object_boolean_false_add(json,
9017 "installed");
9018 } else {
9019 vty_out(vty, "SG : %s\n", vxlan_sg->sg_str);
9020 vty_out(vty, " Input : %s\n", iif_name);
9021 vty_out(vty, " Output : %s\n", oif_name);
9022 vty_out(vty, " installed : %s\n",
9023 installed?"yes":"no");
9024 }
9025 }
9026
9027 if (uj) {
9028 vty_out(vty, "%s\n", json_object_to_json_string_ext(
9029 json, JSON_C_TO_STRING_PRETTY));
9030 json_object_free(json);
9031 }
9032 }
9033
9034 DEFUN (show_ip_pim_vxlan_sg,
9035 show_ip_pim_vxlan_sg_cmd,
9036 "show ip pim [vrf NAME] vxlan-groups [A.B.C.D [A.B.C.D]] [json]",
9037 SHOW_STR
9038 IP_STR
9039 PIM_STR
9040 VRF_CMD_HELP_STR
9041 "VxLAN BUM groups\n"
9042 "source or group ip\n"
9043 "group ip\n"
9044 JSON_STR)
9045 {
9046 bool uj = use_json(argc, argv);
9047 struct vrf *vrf;
9048 int idx = 2;
9049
9050 vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
9051
9052 if (!vrf)
9053 return CMD_WARNING;
9054
9055 char *src_ip = argv_find(argv, argc, "A.B.C.D", &idx) ?
9056 argv[idx++]->arg:NULL;
9057 char *grp_ip = idx < argc && argv_find(argv, argc, "A.B.C.D", &idx) ?
9058 argv[idx]->arg:NULL;
9059
9060 if (src_ip && grp_ip)
9061 pim_show_vxlan_sg_one(vrf->info, vty, src_ip, grp_ip, uj);
9062 else if (src_ip)
9063 pim_show_vxlan_sg_match_addr(vrf->info, vty, src_ip, uj);
9064 else
9065 pim_show_vxlan_sg(vrf->info, vty, uj);
9066
9067 return CMD_SUCCESS;
9068 }
9069
9070 static void pim_show_vxlan_sg_work(struct pim_instance *pim,
9071 struct vty *vty, bool uj)
9072 {
9073 json_object *json = NULL;
9074 struct pim_sg_cache_walk_data cwd;
9075 struct listnode *node;
9076 struct pim_vxlan_sg *vxlan_sg;
9077
9078 if (uj) {
9079 json = json_object_new_object();
9080 } else {
9081 vty_out(vty, "Codes: I -> installed\n");
9082 vty_out(vty,
9083 "Source Group Input Flags\n");
9084 }
9085
9086 memset(&cwd, 0, sizeof(cwd));
9087 cwd.vty = vty;
9088 cwd.json = json;
9089 for (ALL_LIST_ELEMENTS_RO(pim_vxlan_p->work_list, node, vxlan_sg))
9090 pim_show_vxlan_sg_entry(vxlan_sg, &cwd);
9091
9092 if (uj) {
9093 vty_out(vty, "%s\n", json_object_to_json_string_ext(
9094 json, JSON_C_TO_STRING_PRETTY));
9095 json_object_free(json);
9096 }
9097 }
9098
9099 DEFUN_HIDDEN (show_ip_pim_vxlan_sg_work,
9100 show_ip_pim_vxlan_sg_work_cmd,
9101 "show ip pim [vrf NAME] vxlan-work [json]",
9102 SHOW_STR
9103 IP_STR
9104 PIM_STR
9105 VRF_CMD_HELP_STR
9106 "VxLAN work list\n"
9107 JSON_STR)
9108 {
9109 bool uj = use_json(argc, argv);
9110 struct vrf *vrf;
9111 int idx = 2;
9112
9113 vrf = pim_cmd_lookup_vrf(vty, argv, argc, &idx);
9114
9115 if (!vrf)
9116 return CMD_WARNING;
9117
9118 pim_show_vxlan_sg_work(vrf->info, vty, uj);
9119
9120 return CMD_SUCCESS;
9121 }
9122
9123 DEFUN_HIDDEN (no_ip_pim_mlag,
9124 no_ip_pim_mlag_cmd,
9125 "no ip pim mlag",
9126 NO_STR
9127 IP_STR
9128 PIM_STR
9129 "MLAG\n")
9130 {
9131 struct in_addr addr;
9132
9133 addr.s_addr = 0;
9134 pim_vxlan_mlag_update(TRUE /*mlag_enable*/,
9135 FALSE /*peer_state*/, PIM_VXLAN_MLAG_ROLE_SECONDARY,
9136 NULL/*peerlink*/, &addr);
9137
9138 return CMD_SUCCESS;
9139 }
9140
9141 DEFUN_HIDDEN (ip_pim_mlag,
9142 ip_pim_mlag_cmd,
9143 "ip pim mlag INTERFACE role [primary|secondary] state [up|down] addr A.B.C.D",
9144 IP_STR
9145 PIM_STR
9146 "MLAG\n"
9147 "peerlink sub interface\n"
9148 "MLAG role\n"
9149 "MLAG role primary\n"
9150 "MLAG role secondary\n"
9151 "peer session state\n"
9152 "peer session state up\n"
9153 "peer session state down\n"
9154 "configure PIP\n"
9155 "unique ip address\n")
9156 {
9157 struct interface *ifp;
9158 const char *peerlink;
9159 uint32_t role;
9160 int idx;
9161 bool peer_state;
9162 int result;
9163 struct in_addr reg_addr;
9164
9165 idx = 3;
9166 peerlink = argv[idx]->arg;
9167 ifp = if_lookup_by_name(peerlink, VRF_DEFAULT);
9168 if (!ifp) {
9169 vty_out(vty, "No such interface name %s\n", peerlink);
9170 return CMD_WARNING;
9171 }
9172
9173 idx += 2;
9174 if (!strcmp(argv[idx]->arg, "primary")) {
9175 role = PIM_VXLAN_MLAG_ROLE_PRIMARY;
9176 } else if (!strcmp(argv[idx]->arg, "secondary")) {
9177 role = PIM_VXLAN_MLAG_ROLE_SECONDARY;
9178 } else {
9179 vty_out(vty, "unknown MLAG role %s\n", argv[idx]->arg);
9180 return CMD_WARNING;
9181 }
9182
9183 idx += 2;
9184 if (!strcmp(argv[idx]->arg, "up")) {
9185 peer_state = TRUE;
9186 } else if (strcmp(argv[idx]->arg, "down")) {
9187 peer_state = FALSE;
9188 } else {
9189 vty_out(vty, "unknown MLAG state %s\n", argv[idx]->arg);
9190 return CMD_WARNING;
9191 }
9192
9193 idx += 2;
9194 result = inet_pton(AF_INET, argv[idx]->arg, &reg_addr);
9195 if (result <= 0) {
9196 vty_out(vty, "%% Bad reg address %s: errno=%d: %s\n",
9197 argv[idx]->arg,
9198 errno, safe_strerror(errno));
9199 return CMD_WARNING_CONFIG_FAILED;
9200 }
9201 pim_vxlan_mlag_update(TRUE, peer_state, role, ifp, &reg_addr);
9202
9203 return CMD_SUCCESS;
9204 }
9205
9206 void pim_cmd_init(void)
9207 {
9208 install_node(&interface_node,
9209 pim_interface_config_write); /* INTERFACE_NODE */
9210 if_cmd_init();
9211
9212 install_node(&debug_node, pim_debug_config_write);
9213
9214 install_element(ENABLE_NODE, &pim_test_sg_keepalive_cmd);
9215
9216 install_element(CONFIG_NODE, &ip_pim_rp_cmd);
9217 install_element(VRF_NODE, &ip_pim_rp_cmd);
9218 install_element(CONFIG_NODE, &no_ip_pim_rp_cmd);
9219 install_element(VRF_NODE, &no_ip_pim_rp_cmd);
9220 install_element(CONFIG_NODE, &ip_pim_rp_prefix_list_cmd);
9221 install_element(VRF_NODE, &ip_pim_rp_prefix_list_cmd);
9222 install_element(CONFIG_NODE, &no_ip_pim_rp_prefix_list_cmd);
9223 install_element(VRF_NODE, &no_ip_pim_rp_prefix_list_cmd);
9224 install_element(CONFIG_NODE, &no_ip_pim_ssm_prefix_list_cmd);
9225 install_element(VRF_NODE, &no_ip_pim_ssm_prefix_list_cmd);
9226 install_element(CONFIG_NODE, &no_ip_pim_ssm_prefix_list_name_cmd);
9227 install_element(VRF_NODE, &no_ip_pim_ssm_prefix_list_name_cmd);
9228 install_element(CONFIG_NODE, &ip_pim_ssm_prefix_list_cmd);
9229 install_element(VRF_NODE, &ip_pim_ssm_prefix_list_cmd);
9230 install_element(CONFIG_NODE, &ip_pim_register_suppress_cmd);
9231 install_element(VRF_NODE, &ip_pim_register_suppress_cmd);
9232 install_element(CONFIG_NODE, &no_ip_pim_register_suppress_cmd);
9233 install_element(VRF_NODE, &no_ip_pim_register_suppress_cmd);
9234 install_element(CONFIG_NODE, &ip_pim_spt_switchover_infinity_cmd);
9235 install_element(VRF_NODE, &ip_pim_spt_switchover_infinity_cmd);
9236 install_element(CONFIG_NODE, &ip_pim_spt_switchover_infinity_plist_cmd);
9237 install_element(VRF_NODE, &ip_pim_spt_switchover_infinity_plist_cmd);
9238 install_element(CONFIG_NODE, &no_ip_pim_spt_switchover_infinity_cmd);
9239 install_element(VRF_NODE, &no_ip_pim_spt_switchover_infinity_cmd);
9240 install_element(CONFIG_NODE,
9241 &no_ip_pim_spt_switchover_infinity_plist_cmd);
9242 install_element(VRF_NODE, &no_ip_pim_spt_switchover_infinity_plist_cmd);
9243 install_element(CONFIG_NODE, &ip_pim_joinprune_time_cmd);
9244 install_element(VRF_NODE, &ip_pim_joinprune_time_cmd);
9245 install_element(CONFIG_NODE, &no_ip_pim_joinprune_time_cmd);
9246 install_element(VRF_NODE, &no_ip_pim_joinprune_time_cmd);
9247 install_element(CONFIG_NODE, &ip_pim_keep_alive_cmd);
9248 install_element(VRF_NODE, &ip_pim_keep_alive_cmd);
9249 install_element(CONFIG_NODE, &ip_pim_rp_keep_alive_cmd);
9250 install_element(VRF_NODE, &ip_pim_rp_keep_alive_cmd);
9251 install_element(CONFIG_NODE, &no_ip_pim_keep_alive_cmd);
9252 install_element(VRF_NODE, &no_ip_pim_keep_alive_cmd);
9253 install_element(CONFIG_NODE, &no_ip_pim_rp_keep_alive_cmd);
9254 install_element(VRF_NODE, &no_ip_pim_rp_keep_alive_cmd);
9255 install_element(CONFIG_NODE, &ip_pim_packets_cmd);
9256 install_element(VRF_NODE, &ip_pim_packets_cmd);
9257 install_element(CONFIG_NODE, &no_ip_pim_packets_cmd);
9258 install_element(VRF_NODE, &no_ip_pim_packets_cmd);
9259 install_element(CONFIG_NODE, &ip_pim_v6_secondary_cmd);
9260 install_element(VRF_NODE, &ip_pim_v6_secondary_cmd);
9261 install_element(CONFIG_NODE, &no_ip_pim_v6_secondary_cmd);
9262 install_element(VRF_NODE, &no_ip_pim_v6_secondary_cmd);
9263 install_element(CONFIG_NODE, &ip_ssmpingd_cmd);
9264 install_element(VRF_NODE, &ip_ssmpingd_cmd);
9265 install_element(CONFIG_NODE, &no_ip_ssmpingd_cmd);
9266 install_element(VRF_NODE, &no_ip_ssmpingd_cmd);
9267 install_element(CONFIG_NODE, &ip_msdp_peer_cmd);
9268 install_element(VRF_NODE, &ip_msdp_peer_cmd);
9269 install_element(CONFIG_NODE, &no_ip_msdp_peer_cmd);
9270 install_element(VRF_NODE, &no_ip_msdp_peer_cmd);
9271 install_element(CONFIG_NODE, &ip_pim_ecmp_cmd);
9272 install_element(VRF_NODE, &ip_pim_ecmp_cmd);
9273 install_element(CONFIG_NODE, &no_ip_pim_ecmp_cmd);
9274 install_element(VRF_NODE, &no_ip_pim_ecmp_cmd);
9275 install_element(CONFIG_NODE, &ip_pim_ecmp_rebalance_cmd);
9276 install_element(VRF_NODE, &ip_pim_ecmp_rebalance_cmd);
9277 install_element(CONFIG_NODE, &no_ip_pim_ecmp_rebalance_cmd);
9278 install_element(VRF_NODE, &no_ip_pim_ecmp_rebalance_cmd);
9279 install_element(CONFIG_NODE, &ip_pim_mlag_cmd);
9280 install_element(CONFIG_NODE, &no_ip_pim_mlag_cmd);
9281
9282 install_element(INTERFACE_NODE, &interface_ip_igmp_cmd);
9283 install_element(INTERFACE_NODE, &interface_no_ip_igmp_cmd);
9284 install_element(INTERFACE_NODE, &interface_ip_igmp_join_cmd);
9285 install_element(INTERFACE_NODE, &interface_no_ip_igmp_join_cmd);
9286 install_element(INTERFACE_NODE, &interface_ip_igmp_version_cmd);
9287 install_element(INTERFACE_NODE, &interface_no_ip_igmp_version_cmd);
9288 install_element(INTERFACE_NODE, &interface_ip_igmp_query_interval_cmd);
9289 install_element(INTERFACE_NODE,
9290 &interface_no_ip_igmp_query_interval_cmd);
9291 install_element(INTERFACE_NODE,
9292 &interface_ip_igmp_query_max_response_time_cmd);
9293 install_element(INTERFACE_NODE,
9294 &interface_no_ip_igmp_query_max_response_time_cmd);
9295 install_element(INTERFACE_NODE,
9296 &interface_ip_igmp_query_max_response_time_dsec_cmd);
9297 install_element(INTERFACE_NODE,
9298 &interface_no_ip_igmp_query_max_response_time_dsec_cmd);
9299 install_element(INTERFACE_NODE, &interface_ip_pim_activeactive_cmd);
9300 install_element(INTERFACE_NODE, &interface_ip_pim_ssm_cmd);
9301 install_element(INTERFACE_NODE, &interface_no_ip_pim_ssm_cmd);
9302 install_element(INTERFACE_NODE, &interface_ip_pim_sm_cmd);
9303 install_element(INTERFACE_NODE, &interface_no_ip_pim_sm_cmd);
9304 install_element(INTERFACE_NODE, &interface_ip_pim_cmd);
9305 install_element(INTERFACE_NODE, &interface_no_ip_pim_cmd);
9306 install_element(INTERFACE_NODE, &interface_ip_pim_drprio_cmd);
9307 install_element(INTERFACE_NODE, &interface_no_ip_pim_drprio_cmd);
9308 install_element(INTERFACE_NODE, &interface_ip_pim_hello_cmd);
9309 install_element(INTERFACE_NODE, &interface_no_ip_pim_hello_cmd);
9310 install_element(INTERFACE_NODE, &interface_ip_pim_boundary_oil_cmd);
9311 install_element(INTERFACE_NODE, &interface_no_ip_pim_boundary_oil_cmd);
9312
9313 // Static mroutes NEB
9314 install_element(INTERFACE_NODE, &interface_ip_mroute_cmd);
9315 install_element(INTERFACE_NODE, &interface_ip_mroute_source_cmd);
9316 install_element(INTERFACE_NODE, &interface_no_ip_mroute_cmd);
9317 install_element(INTERFACE_NODE, &interface_no_ip_mroute_source_cmd);
9318
9319 install_element(VIEW_NODE, &show_ip_igmp_interface_cmd);
9320 install_element(VIEW_NODE, &show_ip_igmp_interface_vrf_all_cmd);
9321 install_element(VIEW_NODE, &show_ip_igmp_join_cmd);
9322 install_element(VIEW_NODE, &show_ip_igmp_join_vrf_all_cmd);
9323 install_element(VIEW_NODE, &show_ip_igmp_groups_cmd);
9324 install_element(VIEW_NODE, &show_ip_igmp_groups_vrf_all_cmd);
9325 install_element(VIEW_NODE, &show_ip_igmp_groups_retransmissions_cmd);
9326 install_element(VIEW_NODE, &show_ip_igmp_sources_cmd);
9327 install_element(VIEW_NODE, &show_ip_igmp_sources_retransmissions_cmd);
9328 install_element(VIEW_NODE, &show_ip_igmp_statistics_cmd);
9329 install_element(VIEW_NODE, &show_ip_pim_assert_cmd);
9330 install_element(VIEW_NODE, &show_ip_pim_assert_internal_cmd);
9331 install_element(VIEW_NODE, &show_ip_pim_assert_metric_cmd);
9332 install_element(VIEW_NODE, &show_ip_pim_assert_winner_metric_cmd);
9333 install_element(VIEW_NODE, &show_ip_pim_interface_traffic_cmd);
9334 install_element(VIEW_NODE, &show_ip_pim_interface_cmd);
9335 install_element(VIEW_NODE, &show_ip_pim_interface_vrf_all_cmd);
9336 install_element(VIEW_NODE, &show_ip_pim_join_cmd);
9337 install_element(VIEW_NODE, &show_ip_pim_join_vrf_all_cmd);
9338 install_element(VIEW_NODE, &show_ip_pim_local_membership_cmd);
9339 install_element(VIEW_NODE, &show_ip_pim_neighbor_cmd);
9340 install_element(VIEW_NODE, &show_ip_pim_neighbor_vrf_all_cmd);
9341 install_element(VIEW_NODE, &show_ip_pim_rpf_cmd);
9342 install_element(VIEW_NODE, &show_ip_pim_rpf_vrf_all_cmd);
9343 install_element(VIEW_NODE, &show_ip_pim_secondary_cmd);
9344 install_element(VIEW_NODE, &show_ip_pim_state_cmd);
9345 install_element(VIEW_NODE, &show_ip_pim_state_vrf_all_cmd);
9346 install_element(VIEW_NODE, &show_ip_pim_upstream_cmd);
9347 install_element(VIEW_NODE, &show_ip_pim_upstream_vrf_all_cmd);
9348 install_element(VIEW_NODE, &show_ip_pim_upstream_join_desired_cmd);
9349 install_element(VIEW_NODE, &show_ip_pim_upstream_rpf_cmd);
9350 install_element(VIEW_NODE, &show_ip_pim_rp_cmd);
9351 install_element(VIEW_NODE, &show_ip_pim_rp_vrf_all_cmd);
9352 install_element(VIEW_NODE, &show_ip_multicast_cmd);
9353 install_element(VIEW_NODE, &show_ip_multicast_vrf_all_cmd);
9354 install_element(VIEW_NODE, &show_ip_mroute_cmd);
9355 install_element(VIEW_NODE, &show_ip_mroute_vrf_all_cmd);
9356 install_element(VIEW_NODE, &show_ip_mroute_count_cmd);
9357 install_element(VIEW_NODE, &show_ip_mroute_count_vrf_all_cmd);
9358 install_element(VIEW_NODE, &show_ip_rib_cmd);
9359 install_element(VIEW_NODE, &show_ip_ssmpingd_cmd);
9360 install_element(VIEW_NODE, &show_debugging_pim_cmd);
9361 install_element(VIEW_NODE, &show_ip_pim_nexthop_cmd);
9362 install_element(VIEW_NODE, &show_ip_pim_nexthop_lookup_cmd);
9363
9364 install_element(ENABLE_NODE, &clear_ip_interfaces_cmd);
9365 install_element(ENABLE_NODE, &clear_ip_igmp_interfaces_cmd);
9366 install_element(ENABLE_NODE, &clear_ip_mroute_cmd);
9367 install_element(ENABLE_NODE, &clear_ip_pim_interfaces_cmd);
9368 install_element(ENABLE_NODE, &clear_ip_pim_interface_traffic_cmd);
9369 install_element(ENABLE_NODE, &clear_ip_pim_oil_cmd);
9370
9371 install_element(ENABLE_NODE, &debug_igmp_cmd);
9372 install_element(ENABLE_NODE, &no_debug_igmp_cmd);
9373 install_element(ENABLE_NODE, &debug_igmp_events_cmd);
9374 install_element(ENABLE_NODE, &no_debug_igmp_events_cmd);
9375 install_element(ENABLE_NODE, &debug_igmp_packets_cmd);
9376 install_element(ENABLE_NODE, &no_debug_igmp_packets_cmd);
9377 install_element(ENABLE_NODE, &debug_igmp_trace_cmd);
9378 install_element(ENABLE_NODE, &no_debug_igmp_trace_cmd);
9379 install_element(ENABLE_NODE, &debug_mroute_cmd);
9380 install_element(ENABLE_NODE, &debug_mroute_detail_cmd);
9381 install_element(ENABLE_NODE, &no_debug_mroute_cmd);
9382 install_element(ENABLE_NODE, &no_debug_mroute_detail_cmd);
9383 install_element(ENABLE_NODE, &debug_pim_static_cmd);
9384 install_element(ENABLE_NODE, &no_debug_pim_static_cmd);
9385 install_element(ENABLE_NODE, &debug_pim_cmd);
9386 install_element(ENABLE_NODE, &no_debug_pim_cmd);
9387 install_element(ENABLE_NODE, &debug_pim_nht_cmd);
9388 install_element(ENABLE_NODE, &no_debug_pim_nht_cmd);
9389 install_element(ENABLE_NODE, &debug_pim_nht_rp_cmd);
9390 install_element(ENABLE_NODE, &no_debug_pim_nht_rp_cmd);
9391 install_element(ENABLE_NODE, &debug_pim_events_cmd);
9392 install_element(ENABLE_NODE, &no_debug_pim_events_cmd);
9393 install_element(ENABLE_NODE, &debug_pim_packets_cmd);
9394 install_element(ENABLE_NODE, &no_debug_pim_packets_cmd);
9395 install_element(ENABLE_NODE, &debug_pim_packetdump_send_cmd);
9396 install_element(ENABLE_NODE, &no_debug_pim_packetdump_send_cmd);
9397 install_element(ENABLE_NODE, &debug_pim_packetdump_recv_cmd);
9398 install_element(ENABLE_NODE, &no_debug_pim_packetdump_recv_cmd);
9399 install_element(ENABLE_NODE, &debug_pim_trace_cmd);
9400 install_element(ENABLE_NODE, &no_debug_pim_trace_cmd);
9401 install_element(ENABLE_NODE, &debug_pim_trace_detail_cmd);
9402 install_element(ENABLE_NODE, &no_debug_pim_trace_detail_cmd);
9403 install_element(ENABLE_NODE, &debug_ssmpingd_cmd);
9404 install_element(ENABLE_NODE, &no_debug_ssmpingd_cmd);
9405 install_element(ENABLE_NODE, &debug_pim_zebra_cmd);
9406 install_element(ENABLE_NODE, &no_debug_pim_zebra_cmd);
9407 install_element(ENABLE_NODE, &debug_pim_vxlan_cmd);
9408 install_element(ENABLE_NODE, &no_debug_pim_vxlan_cmd);
9409 install_element(ENABLE_NODE, &debug_msdp_cmd);
9410 install_element(ENABLE_NODE, &no_debug_msdp_cmd);
9411 install_element(ENABLE_NODE, &debug_msdp_events_cmd);
9412 install_element(ENABLE_NODE, &no_debug_msdp_events_cmd);
9413 install_element(ENABLE_NODE, &debug_msdp_packets_cmd);
9414 install_element(ENABLE_NODE, &no_debug_msdp_packets_cmd);
9415 install_element(ENABLE_NODE, &debug_mtrace_cmd);
9416 install_element(ENABLE_NODE, &no_debug_mtrace_cmd);
9417
9418 install_element(CONFIG_NODE, &debug_igmp_cmd);
9419 install_element(CONFIG_NODE, &no_debug_igmp_cmd);
9420 install_element(CONFIG_NODE, &debug_igmp_events_cmd);
9421 install_element(CONFIG_NODE, &no_debug_igmp_events_cmd);
9422 install_element(CONFIG_NODE, &debug_igmp_packets_cmd);
9423 install_element(CONFIG_NODE, &no_debug_igmp_packets_cmd);
9424 install_element(CONFIG_NODE, &debug_igmp_trace_cmd);
9425 install_element(CONFIG_NODE, &no_debug_igmp_trace_cmd);
9426 install_element(CONFIG_NODE, &debug_mroute_cmd);
9427 install_element(CONFIG_NODE, &debug_mroute_detail_cmd);
9428 install_element(CONFIG_NODE, &no_debug_mroute_cmd);
9429 install_element(CONFIG_NODE, &no_debug_mroute_detail_cmd);
9430 install_element(CONFIG_NODE, &debug_pim_static_cmd);
9431 install_element(CONFIG_NODE, &no_debug_pim_static_cmd);
9432 install_element(CONFIG_NODE, &debug_pim_cmd);
9433 install_element(CONFIG_NODE, &no_debug_pim_cmd);
9434 install_element(CONFIG_NODE, &debug_pim_nht_cmd);
9435 install_element(CONFIG_NODE, &no_debug_pim_nht_cmd);
9436 install_element(CONFIG_NODE, &debug_pim_nht_rp_cmd);
9437 install_element(CONFIG_NODE, &no_debug_pim_nht_rp_cmd);
9438 install_element(CONFIG_NODE, &debug_pim_events_cmd);
9439 install_element(CONFIG_NODE, &no_debug_pim_events_cmd);
9440 install_element(CONFIG_NODE, &debug_pim_packets_cmd);
9441 install_element(CONFIG_NODE, &no_debug_pim_packets_cmd);
9442 install_element(CONFIG_NODE, &debug_pim_trace_cmd);
9443 install_element(CONFIG_NODE, &no_debug_pim_trace_cmd);
9444 install_element(CONFIG_NODE, &debug_pim_trace_detail_cmd);
9445 install_element(CONFIG_NODE, &no_debug_pim_trace_detail_cmd);
9446 install_element(CONFIG_NODE, &debug_ssmpingd_cmd);
9447 install_element(CONFIG_NODE, &no_debug_ssmpingd_cmd);
9448 install_element(CONFIG_NODE, &debug_pim_zebra_cmd);
9449 install_element(CONFIG_NODE, &no_debug_pim_zebra_cmd);
9450 install_element(CONFIG_NODE, &debug_pim_vxlan_cmd);
9451 install_element(CONFIG_NODE, &no_debug_pim_vxlan_cmd);
9452 install_element(CONFIG_NODE, &debug_msdp_cmd);
9453 install_element(CONFIG_NODE, &no_debug_msdp_cmd);
9454 install_element(CONFIG_NODE, &debug_msdp_events_cmd);
9455 install_element(CONFIG_NODE, &no_debug_msdp_events_cmd);
9456 install_element(CONFIG_NODE, &debug_msdp_packets_cmd);
9457 install_element(CONFIG_NODE, &no_debug_msdp_packets_cmd);
9458 install_element(CONFIG_NODE, &debug_mtrace_cmd);
9459 install_element(CONFIG_NODE, &no_debug_mtrace_cmd);
9460
9461 install_element(CONFIG_NODE, &ip_msdp_mesh_group_member_cmd);
9462 install_element(VRF_NODE, &ip_msdp_mesh_group_member_cmd);
9463 install_element(CONFIG_NODE, &no_ip_msdp_mesh_group_member_cmd);
9464 install_element(VRF_NODE, &no_ip_msdp_mesh_group_member_cmd);
9465 install_element(CONFIG_NODE, &ip_msdp_mesh_group_source_cmd);
9466 install_element(VRF_NODE, &ip_msdp_mesh_group_source_cmd);
9467 install_element(CONFIG_NODE, &no_ip_msdp_mesh_group_source_cmd);
9468 install_element(VRF_NODE, &no_ip_msdp_mesh_group_source_cmd);
9469 install_element(VIEW_NODE, &show_ip_msdp_peer_detail_cmd);
9470 install_element(VIEW_NODE, &show_ip_msdp_peer_detail_vrf_all_cmd);
9471 install_element(VIEW_NODE, &show_ip_msdp_sa_detail_cmd);
9472 install_element(VIEW_NODE, &show_ip_msdp_sa_detail_vrf_all_cmd);
9473 install_element(VIEW_NODE, &show_ip_msdp_sa_sg_cmd);
9474 install_element(VIEW_NODE, &show_ip_msdp_sa_sg_vrf_all_cmd);
9475 install_element(VIEW_NODE, &show_ip_msdp_mesh_group_cmd);
9476 install_element(VIEW_NODE, &show_ip_msdp_mesh_group_vrf_all_cmd);
9477 install_element(VIEW_NODE, &show_ip_pim_ssm_range_cmd);
9478 install_element(VIEW_NODE, &show_ip_pim_group_type_cmd);
9479 install_element(VIEW_NODE, &show_ip_pim_vxlan_sg_cmd);
9480 install_element(VIEW_NODE, &show_ip_pim_vxlan_sg_work_cmd);
9481 install_element(INTERFACE_NODE, &interface_pim_use_source_cmd);
9482 install_element(INTERFACE_NODE, &interface_no_pim_use_source_cmd);
9483 /* Install BFD command */
9484 install_element(INTERFACE_NODE, &ip_pim_bfd_cmd);
9485 install_element(INTERFACE_NODE, &ip_pim_bfd_param_cmd);
9486 install_element(INTERFACE_NODE, &no_ip_pim_bfd_cmd);
9487 #if HAVE_BFDD == 0
9488 install_element(INTERFACE_NODE, &no_ip_pim_bfd_param_cmd);
9489 #endif /* !HAVE_BFDD */
9490 }