3 * Copyright (C) 2008 Everton da Silva Marques
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.
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.
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
34 #include "pim_mroute.h"
36 #include "pim_iface.h"
38 #include "pim_mroute.h"
41 #include "pim_igmpv3.h"
46 #include "pim_neighbor.h"
48 #include "pim_ifchannel.h"
49 #include "pim_hello.h"
51 #include "pim_upstream.h"
53 #include "pim_macro.h"
54 #include "pim_ssmpingd.h"
55 #include "pim_zebra.h"
56 #include "pim_static.h"
58 #include "pim_zlookup.h"
65 static struct cmd_node interface_node
= {
66 INTERFACE_NODE
, "%s(config-if)# ", 1 /* vtysh ? yes */
69 static struct cmd_node debug_node
= {DEBUG_NODE
, "", 1};
71 static struct vrf
*pim_cmd_lookup_vrf(struct vty
*vty
, struct cmd_token
*argv
[],
72 const int argc
, int *idx
)
76 if (argv_find(argv
, argc
, "NAME", idx
))
77 vrf
= vrf_lookup_by_name(argv
[*idx
]->arg
);
79 vrf
= vrf_lookup_by_id(VRF_DEFAULT
);
82 vty_out(vty
, "Specified VRF: %s does not exist\n",
88 static void pim_if_membership_clear(struct interface
*ifp
)
90 struct pim_interface
*pim_ifp
;
95 if (PIM_IF_TEST_PIM(pim_ifp
->options
)
96 && PIM_IF_TEST_IGMP(pim_ifp
->options
)) {
100 pim_ifchannel_membership_clear(ifp
);
104 When PIM is disabled on interface, IGMPv3 local membership
105 information is not injected into PIM interface state.
107 The function pim_if_membership_refresh() fetches all IGMPv3 local
108 membership information into PIM. It is intented to be called
109 whenever PIM is enabled on the interface in order to collect missed
110 local membership information.
112 static void pim_if_membership_refresh(struct interface
*ifp
)
114 struct pim_interface
*pim_ifp
;
115 struct listnode
*sock_node
;
116 struct igmp_sock
*igmp
;
121 if (!PIM_IF_TEST_PIM(pim_ifp
->options
))
123 if (!PIM_IF_TEST_IGMP(pim_ifp
->options
))
127 First clear off membership from all PIM (S,G) entries on the
131 pim_ifchannel_membership_clear(ifp
);
134 Then restore PIM (S,G) membership from all IGMPv3 (S,G) entries on
138 /* scan igmp sockets */
139 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
, igmp
)) {
140 struct listnode
*grpnode
;
141 struct igmp_group
*grp
;
143 /* scan igmp groups */
144 for (ALL_LIST_ELEMENTS_RO(igmp
->igmp_group_list
, grpnode
,
146 struct listnode
*srcnode
;
147 struct igmp_source
*src
;
149 /* scan group sources */
150 for (ALL_LIST_ELEMENTS_RO(grp
->group_source_list
,
153 if (IGMP_SOURCE_TEST_FORWARDING(
154 src
->source_flags
)) {
158 sizeof(struct prefix_sg
));
159 sg
.src
= src
->source_addr
;
160 sg
.grp
= grp
->group_addr
;
161 pim_ifchannel_local_membership_add(ifp
,
165 } /* scan group sources */
166 } /* scan igmp groups */
167 } /* scan igmp sockets */
170 Finally delete every PIM (S,G) entry lacking all state info
173 pim_ifchannel_delete_on_noinfo(ifp
);
176 static void pim_show_assert_helper(struct vty
*vty
,
177 struct pim_interface
*pim_ifp
,
178 struct pim_ifchannel
*ch
, time_t now
)
180 char ch_src_str
[INET_ADDRSTRLEN
];
181 char ch_grp_str
[INET_ADDRSTRLEN
];
182 char winner_str
[INET_ADDRSTRLEN
];
183 struct in_addr ifaddr
;
187 ifaddr
= pim_ifp
->primary_address
;
189 pim_inet4_dump("<ch_src?>", ch
->sg
.src
, ch_src_str
, sizeof(ch_src_str
));
190 pim_inet4_dump("<ch_grp?>", ch
->sg
.grp
, ch_grp_str
, sizeof(ch_grp_str
));
191 pim_inet4_dump("<assrt_win?>", ch
->ifassert_winner
, winner_str
,
194 pim_time_uptime(uptime
, sizeof(uptime
), now
- ch
->ifassert_creation
);
195 pim_time_timer_to_mmss(timer
, sizeof(timer
), ch
->t_ifassert_timer
);
197 vty_out(vty
, "%-9s %-15s %-15s %-15s %-6s %-15s %-8s %-5s\n",
198 ch
->interface
->name
, inet_ntoa(ifaddr
), ch_src_str
, ch_grp_str
,
199 pim_ifchannel_ifassert_name(ch
->ifassert_state
), winner_str
,
203 static void pim_show_assert(struct pim_instance
*pim
, struct vty
*vty
)
205 struct pim_interface
*pim_ifp
;
206 struct pim_ifchannel
*ch
;
207 struct interface
*ifp
;
210 now
= pim_time_monotonic_sec();
213 "Interface Address Source Group State Winner Uptime Timer\n");
215 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
220 RB_FOREACH (ch
, pim_ifchannel_rb
, &pim_ifp
->ifchannel_rb
) {
221 pim_show_assert_helper(vty
, pim_ifp
, ch
, now
);
222 } /* scan interface channels */
226 static void pim_show_assert_internal_helper(struct vty
*vty
,
227 struct pim_interface
*pim_ifp
,
228 struct pim_ifchannel
*ch
)
230 char ch_src_str
[INET_ADDRSTRLEN
];
231 char ch_grp_str
[INET_ADDRSTRLEN
];
232 struct in_addr ifaddr
;
234 ifaddr
= pim_ifp
->primary_address
;
236 pim_inet4_dump("<ch_src?>", ch
->sg
.src
, ch_src_str
, sizeof(ch_src_str
));
237 pim_inet4_dump("<ch_grp?>", ch
->sg
.grp
, ch_grp_str
, sizeof(ch_grp_str
));
238 vty_out(vty
, "%-9s %-15s %-15s %-15s %-3s %-3s %-3s %-4s\n",
239 ch
->interface
->name
, inet_ntoa(ifaddr
), ch_src_str
, ch_grp_str
,
240 PIM_IF_FLAG_TEST_COULD_ASSERT(ch
->flags
) ? "yes" : "no",
241 pim_macro_ch_could_assert_eval(ch
) ? "yes" : "no",
242 PIM_IF_FLAG_TEST_ASSERT_TRACKING_DESIRED(ch
->flags
) ? "yes"
244 pim_macro_assert_tracking_desired_eval(ch
) ? "yes" : "no");
247 static void pim_show_assert_internal(struct pim_instance
*pim
, struct vty
*vty
)
249 struct pim_interface
*pim_ifp
;
250 struct pim_ifchannel
*ch
;
251 struct interface
*ifp
;
255 "ECA: Evaluate CouldAssert\n"
256 "ATD: AssertTrackingDesired\n"
257 "eATD: Evaluate AssertTrackingDesired\n\n");
260 "Interface Address Source Group CA eCA ATD eATD\n");
261 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
266 RB_FOREACH (ch
, pim_ifchannel_rb
, &pim_ifp
->ifchannel_rb
) {
267 pim_show_assert_internal_helper(vty
, pim_ifp
, ch
);
268 } /* scan interface channels */
272 static void pim_show_assert_metric_helper(struct vty
*vty
,
273 struct pim_interface
*pim_ifp
,
274 struct pim_ifchannel
*ch
)
276 char ch_src_str
[INET_ADDRSTRLEN
];
277 char ch_grp_str
[INET_ADDRSTRLEN
];
278 char addr_str
[INET_ADDRSTRLEN
];
279 struct pim_assert_metric am
;
280 struct in_addr ifaddr
;
282 ifaddr
= pim_ifp
->primary_address
;
284 am
= pim_macro_spt_assert_metric(&ch
->upstream
->rpf
,
285 pim_ifp
->primary_address
);
287 pim_inet4_dump("<ch_src?>", ch
->sg
.src
, ch_src_str
, sizeof(ch_src_str
));
288 pim_inet4_dump("<ch_grp?>", ch
->sg
.grp
, ch_grp_str
, sizeof(ch_grp_str
));
289 pim_inet4_dump("<addr?>", am
.ip_address
, addr_str
, sizeof(addr_str
));
291 vty_out(vty
, "%-9s %-15s %-15s %-15s %-3s %4u %6u %-15s\n",
292 ch
->interface
->name
, inet_ntoa(ifaddr
), ch_src_str
, ch_grp_str
,
293 am
.rpt_bit_flag
? "yes" : "no", am
.metric_preference
,
294 am
.route_metric
, addr_str
);
297 static void pim_show_assert_metric(struct pim_instance
*pim
, struct vty
*vty
)
299 struct pim_interface
*pim_ifp
;
300 struct pim_ifchannel
*ch
;
301 struct interface
*ifp
;
304 "Interface Address Source Group RPT Pref Metric Address \n");
306 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
311 RB_FOREACH (ch
, pim_ifchannel_rb
, &pim_ifp
->ifchannel_rb
) {
312 pim_show_assert_metric_helper(vty
, pim_ifp
, ch
);
313 } /* scan interface channels */
317 static void pim_show_assert_winner_metric_helper(struct vty
*vty
,
318 struct pim_interface
*pim_ifp
,
319 struct pim_ifchannel
*ch
)
321 char ch_src_str
[INET_ADDRSTRLEN
];
322 char ch_grp_str
[INET_ADDRSTRLEN
];
323 char addr_str
[INET_ADDRSTRLEN
];
324 struct pim_assert_metric
*am
;
325 struct in_addr ifaddr
;
329 ifaddr
= pim_ifp
->primary_address
;
331 am
= &ch
->ifassert_winner_metric
;
333 pim_inet4_dump("<ch_src?>", ch
->sg
.src
, ch_src_str
, sizeof(ch_src_str
));
334 pim_inet4_dump("<ch_grp?>", ch
->sg
.grp
, ch_grp_str
, sizeof(ch_grp_str
));
335 pim_inet4_dump("<addr?>", am
->ip_address
, addr_str
, sizeof(addr_str
));
337 if (am
->metric_preference
== PIM_ASSERT_METRIC_PREFERENCE_MAX
)
338 snprintf(pref_str
, sizeof(pref_str
), "INFI");
340 snprintf(pref_str
, sizeof(pref_str
), "%4u",
341 am
->metric_preference
);
343 if (am
->route_metric
== PIM_ASSERT_ROUTE_METRIC_MAX
)
344 snprintf(metr_str
, sizeof(metr_str
), "INFI");
346 snprintf(metr_str
, sizeof(metr_str
), "%6u", am
->route_metric
);
348 vty_out(vty
, "%-9s %-15s %-15s %-15s %-3s %-4s %-6s %-15s\n",
349 ch
->interface
->name
, inet_ntoa(ifaddr
), ch_src_str
, ch_grp_str
,
350 am
->rpt_bit_flag
? "yes" : "no", pref_str
, metr_str
, addr_str
);
353 static void pim_show_assert_winner_metric(struct pim_instance
*pim
,
356 struct pim_interface
*pim_ifp
;
357 struct pim_ifchannel
*ch
;
358 struct interface
*ifp
;
361 "Interface Address Source Group RPT Pref Metric Address \n");
363 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
368 RB_FOREACH (ch
, pim_ifchannel_rb
, &pim_ifp
->ifchannel_rb
) {
369 pim_show_assert_winner_metric_helper(vty
, pim_ifp
, ch
);
370 } /* scan interface channels */
374 static void json_object_pim_ifp_add(struct json_object
*json
,
375 struct interface
*ifp
)
377 struct pim_interface
*pim_ifp
;
380 json_object_string_add(json
, "name", ifp
->name
);
381 json_object_string_add(json
, "state", if_is_up(ifp
) ? "up" : "down");
382 json_object_string_add(json
, "address",
383 inet_ntoa(pim_ifp
->primary_address
));
384 json_object_int_add(json
, "index", ifp
->ifindex
);
386 if (if_is_multicast(ifp
))
387 json_object_boolean_true_add(json
, "flagMulticast");
389 if (if_is_broadcast(ifp
))
390 json_object_boolean_true_add(json
, "flagBroadcast");
392 if (ifp
->flags
& IFF_ALLMULTI
)
393 json_object_boolean_true_add(json
, "flagAllMulticast");
395 if (ifp
->flags
& IFF_PROMISC
)
396 json_object_boolean_true_add(json
, "flagPromiscuous");
398 if (PIM_IF_IS_DELETED(ifp
))
399 json_object_boolean_true_add(json
, "flagDeleted");
401 if (pim_if_lan_delay_enabled(ifp
))
402 json_object_boolean_true_add(json
, "lanDelayEnabled");
405 static void pim_show_membership_helper(struct vty
*vty
,
406 struct pim_interface
*pim_ifp
,
407 struct pim_ifchannel
*ch
,
408 struct json_object
*json
)
410 char ch_src_str
[INET_ADDRSTRLEN
];
411 char ch_grp_str
[INET_ADDRSTRLEN
];
412 json_object
*json_iface
= NULL
;
413 json_object
*json_row
= NULL
;
415 pim_inet4_dump("<ch_src?>", ch
->sg
.src
, ch_src_str
, sizeof(ch_src_str
));
416 pim_inet4_dump("<ch_grp?>", ch
->sg
.grp
, ch_grp_str
, sizeof(ch_grp_str
));
418 json_object_object_get_ex(json
, ch
->interface
->name
, &json_iface
);
420 json_iface
= json_object_new_object();
421 json_object_pim_ifp_add(json_iface
, ch
->interface
);
422 json_object_object_add(json
, ch
->interface
->name
, json_iface
);
425 json_row
= json_object_new_object();
426 json_object_string_add(json_row
, "source", ch_src_str
);
427 json_object_string_add(json_row
, "group", ch_grp_str
);
428 json_object_string_add(json_row
, "localMembership",
429 ch
->local_ifmembership
== PIM_IFMEMBERSHIP_NOINFO
432 json_object_object_add(json_iface
, ch_grp_str
, json_row
);
434 static void pim_show_membership(struct pim_instance
*pim
, struct vty
*vty
,
437 struct pim_interface
*pim_ifp
;
438 struct pim_ifchannel
*ch
;
439 struct interface
*ifp
;
441 json_object
*json
= NULL
;
442 json_object
*json_tmp
= NULL
;
444 json
= json_object_new_object();
446 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
451 RB_FOREACH (ch
, pim_ifchannel_rb
, &pim_ifp
->ifchannel_rb
) {
452 pim_show_membership_helper(vty
, pim_ifp
, ch
, json
);
453 } /* scan interface channels */
457 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
458 json
, JSON_C_TO_STRING_PRETTY
));
461 "Interface Address Source Group Membership\n");
464 * Example of the json data we are traversing
470 * "address":"10.1.20.1",
472 * "flagMulticast":true,
473 * "flagBroadcast":true,
474 * "lanDelayEnabled":true,
477 * "group":"226.10.10.10",
478 * "localMembership":"INCLUDE"
484 /* foreach interface */
485 json_object_object_foreach(json
, key
, val
)
488 /* Find all of the keys where the val is an object. In
490 * above the only one is 226.10.10.10
492 json_object_object_foreach(val
, if_field_key
,
495 type
= json_object_get_type(if_field_val
);
497 if (type
== json_type_object
) {
498 vty_out(vty
, "%-9s ", key
);
500 json_object_object_get_ex(
501 val
, "address", &json_tmp
);
502 vty_out(vty
, "%-15s ",
503 json_object_get_string(
506 json_object_object_get_ex(if_field_val
,
509 vty_out(vty
, "%-15s ",
510 json_object_get_string(
514 vty_out(vty
, "%-15s ", if_field_key
);
516 json_object_object_get_ex(
517 if_field_val
, "localMembership",
519 vty_out(vty
, "%-10s\n",
520 json_object_get_string(
527 json_object_free(json
);
530 static void pim_print_ifp_flags(struct vty
*vty
, struct interface
*ifp
,
533 vty_out(vty
, "Flags\n");
534 vty_out(vty
, "-----\n");
535 vty_out(vty
, "All Multicast : %s\n",
536 (ifp
->flags
& IFF_ALLMULTI
) ? "yes" : "no");
537 vty_out(vty
, "Broadcast : %s\n",
538 if_is_broadcast(ifp
) ? "yes" : "no");
539 vty_out(vty
, "Deleted : %s\n",
540 PIM_IF_IS_DELETED(ifp
) ? "yes" : "no");
541 vty_out(vty
, "Interface Index : %d\n", ifp
->ifindex
);
542 vty_out(vty
, "Multicast : %s\n",
543 if_is_multicast(ifp
) ? "yes" : "no");
544 vty_out(vty
, "Multicast Loop : %d\n", mloop
);
545 vty_out(vty
, "Promiscuous : %s\n",
546 (ifp
->flags
& IFF_PROMISC
) ? "yes" : "no");
551 static void igmp_show_interfaces(struct pim_instance
*pim
, struct vty
*vty
,
554 struct interface
*ifp
;
556 json_object
*json
= NULL
;
557 json_object
*json_row
= NULL
;
559 now
= pim_time_monotonic_sec();
562 json
= json_object_new_object();
565 "Interface State Address V Querier Query Timer Uptime\n");
567 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
568 struct pim_interface
*pim_ifp
;
569 struct listnode
*sock_node
;
570 struct igmp_sock
*igmp
;
577 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
,
580 char query_hhmmss
[10];
582 pim_time_uptime(uptime
, sizeof(uptime
),
583 now
- igmp
->sock_creation
);
584 pim_time_timer_to_hhmmss(query_hhmmss
,
585 sizeof(query_hhmmss
),
586 igmp
->t_igmp_query_timer
);
589 json_row
= json_object_new_object();
590 json_object_pim_ifp_add(json_row
, ifp
);
591 json_object_string_add(json_row
, "upTime",
593 json_object_int_add(json_row
, "version",
594 pim_ifp
->igmp_version
);
596 if (igmp
->t_igmp_query_timer
) {
597 json_object_boolean_true_add(json_row
,
599 json_object_string_add(json_row
,
604 json_object_object_add(json
, ifp
->name
,
607 if (igmp
->mtrace_only
) {
608 json_object_boolean_true_add(
609 json_row
, "mtraceOnly");
613 "%-9s %5s %15s %d %7s %11s %8s\n",
616 ? (igmp
->mtrace_only
? "mtrc"
619 inet_ntoa(igmp
->ifaddr
),
620 pim_ifp
->igmp_version
,
621 igmp
->t_igmp_query_timer
? "local"
623 query_hhmmss
, uptime
);
629 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
630 json
, JSON_C_TO_STRING_PRETTY
));
631 json_object_free(json
);
635 static void igmp_show_interfaces_single(struct pim_instance
*pim
,
636 struct vty
*vty
, const char *ifname
,
639 struct igmp_sock
*igmp
;
640 struct interface
*ifp
;
641 struct listnode
*sock_node
;
642 struct pim_interface
*pim_ifp
;
644 char query_hhmmss
[10];
645 char other_hhmmss
[10];
646 int found_ifname
= 0;
649 long gmi_msec
; /* Group Membership Interval */
652 long oqpi_msec
; /* Other Querier Present Interval */
656 json_object
*json
= NULL
;
657 json_object
*json_row
= NULL
;
660 json
= json_object_new_object();
662 now
= pim_time_monotonic_sec();
664 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
670 if (strcmp(ifname
, "detail") && strcmp(ifname
, ifp
->name
))
673 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
,
676 pim_time_uptime(uptime
, sizeof(uptime
),
677 now
- igmp
->sock_creation
);
678 pim_time_timer_to_hhmmss(query_hhmmss
,
679 sizeof(query_hhmmss
),
680 igmp
->t_igmp_query_timer
);
681 pim_time_timer_to_hhmmss(other_hhmmss
,
682 sizeof(other_hhmmss
),
683 igmp
->t_other_querier_timer
);
685 gmi_msec
= PIM_IGMP_GMI_MSEC(
686 igmp
->querier_robustness_variable
,
687 igmp
->querier_query_interval
,
688 pim_ifp
->igmp_query_max_response_time_dsec
);
691 pim_ifp
->igmp_default_query_interval
);
693 oqpi_msec
= PIM_IGMP_OQPI_MSEC(
694 igmp
->querier_robustness_variable
,
695 igmp
->querier_query_interval
,
696 pim_ifp
->igmp_query_max_response_time_dsec
);
698 lmqt_msec
= PIM_IGMP_LMQT_MSEC(
699 pim_ifp
->igmp_query_max_response_time_dsec
,
700 igmp
->querier_robustness_variable
);
704 igmp
->querier_robustness_variable
,
705 igmp
->querier_query_interval
,
706 pim_ifp
->igmp_query_max_response_time_dsec
)
709 qri_msec
= pim_ifp
->igmp_query_max_response_time_dsec
711 if (pim_ifp
->pim_sock_fd
>= 0)
712 mloop
= pim_socket_mcastloop_get(
713 pim_ifp
->pim_sock_fd
);
718 json_row
= json_object_new_object();
719 json_object_pim_ifp_add(json_row
, ifp
);
720 json_object_string_add(json_row
, "upTime",
722 json_object_string_add(json_row
, "querier",
723 igmp
->t_igmp_query_timer
726 json_object_int_add(json_row
, "queryStartCount",
727 igmp
->startup_query_count
);
728 json_object_string_add(json_row
,
731 json_object_string_add(json_row
,
734 json_object_int_add(json_row
, "version",
735 pim_ifp
->igmp_version
);
738 "timerGroupMembershipIntervalMsec",
740 json_object_int_add(json_row
,
741 "timerLastMemberQueryMsec",
745 "timerOlderHostPresentIntervalMsec",
749 "timerOtherQuerierPresentIntervalMsec",
752 json_row
, "timerQueryInterval",
753 igmp
->querier_query_interval
);
756 "timerQueryResponseIntervalMsec",
759 json_row
, "timerRobustnessVariable",
760 igmp
->querier_robustness_variable
);
761 json_object_int_add(json_row
,
762 "timerStartupQueryInterval",
765 json_object_object_add(json
, ifp
->name
,
768 if (igmp
->mtrace_only
) {
769 json_object_boolean_true_add(
770 json_row
, "mtraceOnly");
773 vty_out(vty
, "Interface : %s\n", ifp
->name
);
774 vty_out(vty
, "State : %s\n",
776 ? (igmp
->mtrace_only
? "mtrace"
779 vty_out(vty
, "Address : %s\n",
780 inet_ntoa(pim_ifp
->primary_address
));
781 vty_out(vty
, "Uptime : %s\n", uptime
);
782 vty_out(vty
, "Version : %d\n",
783 pim_ifp
->igmp_version
);
787 vty_out(vty
, "Querier\n");
788 vty_out(vty
, "-------\n");
789 vty_out(vty
, "Querier : %s\n",
790 igmp
->t_igmp_query_timer
? "local"
792 vty_out(vty
, "Start Count : %d\n",
793 igmp
->startup_query_count
);
794 vty_out(vty
, "Query Timer : %s\n",
796 vty_out(vty
, "Other Timer : %s\n",
801 vty_out(vty
, "Timers\n");
802 vty_out(vty
, "------\n");
804 "Group Membership Interval : %lis\n",
807 "Last Member Query Time : %lis\n",
810 "Older Host Present Interval : %lis\n",
813 "Other Querier Present Interval : %lis\n",
816 "Query Interval : %ds\n",
817 igmp
->querier_query_interval
);
819 "Query Response Interval : %lis\n",
822 "Robustness Variable : %d\n",
823 igmp
->querier_robustness_variable
);
825 "Startup Query Interval : %ds\n",
830 pim_print_ifp_flags(vty
, ifp
, mloop
);
836 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
837 json
, JSON_C_TO_STRING_PRETTY
));
838 json_object_free(json
);
841 vty_out(vty
, "%% No such interface\n");
845 static void igmp_show_interface_join(struct pim_instance
*pim
, struct vty
*vty
)
847 struct interface
*ifp
;
850 now
= pim_time_monotonic_sec();
853 "Interface Address Source Group Socket Uptime \n");
855 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
856 struct pim_interface
*pim_ifp
;
857 struct listnode
*join_node
;
858 struct igmp_join
*ij
;
859 struct in_addr pri_addr
;
860 char pri_addr_str
[INET_ADDRSTRLEN
];
867 if (!pim_ifp
->igmp_join_list
)
870 pri_addr
= pim_find_primary_addr(ifp
);
871 pim_inet4_dump("<pri?>", pri_addr
, pri_addr_str
,
872 sizeof(pri_addr_str
));
874 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_join_list
, join_node
,
876 char group_str
[INET_ADDRSTRLEN
];
877 char source_str
[INET_ADDRSTRLEN
];
880 pim_time_uptime(uptime
, sizeof(uptime
),
881 now
- ij
->sock_creation
);
882 pim_inet4_dump("<grp?>", ij
->group_addr
, group_str
,
884 pim_inet4_dump("<src?>", ij
->source_addr
, source_str
,
887 vty_out(vty
, "%-9s %-15s %-15s %-15s %6d %8s\n",
888 ifp
->name
, pri_addr_str
, source_str
, group_str
,
889 ij
->sock_fd
, uptime
);
890 } /* for (pim_ifp->igmp_join_list) */
895 static void pim_show_interfaces_single(struct pim_instance
*pim
,
896 struct vty
*vty
, const char *ifname
,
899 struct in_addr ifaddr
;
900 struct interface
*ifp
;
901 struct listnode
*neighnode
;
902 struct listnode
*upnode
;
903 struct pim_interface
*pim_ifp
;
904 struct pim_neighbor
*neigh
;
905 struct pim_upstream
*up
;
907 char dr_str
[INET_ADDRSTRLEN
];
910 char grp_str
[INET_ADDRSTRLEN
];
911 char hello_period
[10];
912 char hello_timer
[10];
913 char neigh_src_str
[INET_ADDRSTRLEN
];
914 char src_str
[INET_ADDRSTRLEN
];
915 char stat_uptime
[10];
918 int found_ifname
= 0;
920 json_object
*json
= NULL
;
921 json_object
*json_row
= NULL
;
922 json_object
*json_pim_neighbor
= NULL
;
923 json_object
*json_pim_neighbors
= NULL
;
924 json_object
*json_group
= NULL
;
925 json_object
*json_group_source
= NULL
;
926 json_object
*json_fhr_sources
= NULL
;
927 struct pim_secondary_addr
*sec_addr
;
928 struct listnode
*sec_node
;
930 now
= pim_time_monotonic_sec();
933 json
= json_object_new_object();
935 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
941 if (strcmp(ifname
, "detail") && strcmp(ifname
, ifp
->name
))
945 ifaddr
= pim_ifp
->primary_address
;
946 pim_inet4_dump("<dr?>", pim_ifp
->pim_dr_addr
, dr_str
,
948 pim_time_uptime_begin(dr_uptime
, sizeof(dr_uptime
), now
,
949 pim_ifp
->pim_dr_election_last
);
950 pim_time_timer_to_hhmmss(hello_timer
, sizeof(hello_timer
),
951 pim_ifp
->t_pim_hello_timer
);
952 pim_time_mmss(hello_period
, sizeof(hello_period
),
953 pim_ifp
->pim_hello_period
);
954 pim_time_uptime(stat_uptime
, sizeof(stat_uptime
),
955 now
- pim_ifp
->pim_ifstat_start
);
956 if (pim_ifp
->pim_sock_fd
>= 0)
957 mloop
= pim_socket_mcastloop_get(pim_ifp
->pim_sock_fd
);
962 char pbuf
[PREFIX2STR_BUFFER
];
963 json_row
= json_object_new_object();
964 json_object_pim_ifp_add(json_row
, ifp
);
966 if (pim_ifp
->update_source
.s_addr
!= INADDR_ANY
) {
967 json_object_string_add(
968 json_row
, "useSource",
969 inet_ntoa(pim_ifp
->update_source
));
971 if (pim_ifp
->sec_addr_list
) {
972 json_object
*sec_list
= NULL
;
974 sec_list
= json_object_new_array();
975 for (ALL_LIST_ELEMENTS_RO(
976 pim_ifp
->sec_addr_list
, sec_node
,
978 json_object_array_add(
980 json_object_new_string(
986 json_object_object_add(json_row
,
987 "secondaryAddressList",
992 if (pim_ifp
->pim_neighbor_list
->count
) {
993 json_pim_neighbors
= json_object_new_object();
995 for (ALL_LIST_ELEMENTS_RO(
996 pim_ifp
->pim_neighbor_list
,
999 json_object_new_object();
1000 pim_inet4_dump("<src?>",
1003 sizeof(neigh_src_str
));
1004 pim_time_uptime(uptime
, sizeof(uptime
),
1005 now
- neigh
->creation
);
1006 pim_time_timer_to_hhmmss(
1007 expire
, sizeof(expire
),
1008 neigh
->t_expire_timer
);
1010 json_object_string_add(
1011 json_pim_neighbor
, "address",
1013 json_object_string_add(
1014 json_pim_neighbor
, "upTime",
1016 json_object_string_add(
1017 json_pim_neighbor
, "holdtime",
1020 json_object_object_add(
1026 json_object_object_add(json_row
, "neighbors",
1027 json_pim_neighbors
);
1030 json_object_string_add(json_row
, "drAddress", dr_str
);
1031 json_object_int_add(json_row
, "drPriority",
1032 pim_ifp
->pim_dr_priority
);
1033 json_object_string_add(json_row
, "drUptime", dr_uptime
);
1034 json_object_int_add(json_row
, "drElections",
1035 pim_ifp
->pim_dr_election_count
);
1036 json_object_int_add(json_row
, "drChanges",
1037 pim_ifp
->pim_dr_election_changes
);
1040 for (ALL_LIST_ELEMENTS_RO(pim
->upstream_list
, upnode
,
1042 if (ifp
!= up
->rpf
.source_nexthop
.interface
)
1045 if (!(up
->flags
& PIM_UPSTREAM_FLAG_MASK_FHR
))
1048 if (!json_fhr_sources
)
1050 json_object_new_object();
1052 pim_inet4_dump("<src?>", up
->sg
.src
, src_str
,
1054 pim_inet4_dump("<grp?>", up
->sg
.grp
, grp_str
,
1056 pim_time_uptime(uptime
, sizeof(uptime
),
1057 now
- up
->state_transition
);
1060 * Does this group live in json_fhr_sources?
1063 json_object_object_get_ex(json_fhr_sources
,
1064 grp_str
, &json_group
);
1067 json_group
= json_object_new_object();
1068 json_object_object_add(json_fhr_sources
,
1073 json_group_source
= json_object_new_object();
1074 json_object_string_add(json_group_source
,
1076 json_object_string_add(json_group_source
,
1078 json_object_string_add(json_group_source
,
1080 json_object_object_add(json_group
, src_str
,
1084 if (json_fhr_sources
) {
1085 json_object_object_add(json_row
,
1090 json_object_int_add(json_row
, "helloPeriod",
1091 pim_ifp
->pim_hello_period
);
1092 json_object_string_add(json_row
, "helloTimer",
1094 json_object_string_add(json_row
, "helloStatStart",
1096 json_object_int_add(json_row
, "helloReceived",
1097 pim_ifp
->pim_ifstat_hello_recv
);
1098 json_object_int_add(json_row
, "helloReceivedFailed",
1099 pim_ifp
->pim_ifstat_hello_recvfail
);
1100 json_object_int_add(json_row
, "helloSend",
1101 pim_ifp
->pim_ifstat_hello_sent
);
1102 json_object_int_add(json_row
, "hellosendFailed",
1103 pim_ifp
->pim_ifstat_hello_sendfail
);
1104 json_object_int_add(json_row
, "helloGenerationId",
1105 pim_ifp
->pim_generation_id
);
1106 json_object_int_add(json_row
, "flagMulticastLoop",
1109 json_object_int_add(
1110 json_row
, "effectivePropagationDelay",
1111 pim_if_effective_propagation_delay_msec(ifp
));
1112 json_object_int_add(
1113 json_row
, "effectiveOverrideInterval",
1114 pim_if_effective_override_interval_msec(ifp
));
1115 json_object_int_add(
1116 json_row
, "joinPruneOverrideInterval",
1117 pim_if_jp_override_interval_msec(ifp
));
1119 json_object_int_add(
1120 json_row
, "propagationDelay",
1121 pim_ifp
->pim_propagation_delay_msec
);
1122 json_object_int_add(
1123 json_row
, "propagationDelayHighest",
1124 pim_ifp
->pim_neighbors_highest_propagation_delay_msec
);
1125 json_object_int_add(
1126 json_row
, "overrideInterval",
1127 pim_ifp
->pim_override_interval_msec
);
1128 json_object_int_add(
1129 json_row
, "overrideIntervalHighest",
1130 pim_ifp
->pim_neighbors_highest_override_interval_msec
);
1131 json_object_object_add(json
, ifp
->name
, json_row
);
1134 vty_out(vty
, "Interface : %s\n", ifp
->name
);
1135 vty_out(vty
, "State : %s\n",
1136 if_is_up(ifp
) ? "up" : "down");
1137 if (pim_ifp
->update_source
.s_addr
!= INADDR_ANY
) {
1138 vty_out(vty
, "Use Source : %s\n",
1139 inet_ntoa(pim_ifp
->update_source
));
1141 if (pim_ifp
->sec_addr_list
) {
1142 char pbuf
[PREFIX2STR_BUFFER
];
1143 vty_out(vty
, "Address : %s (primary)\n",
1145 for (ALL_LIST_ELEMENTS_RO(
1146 pim_ifp
->sec_addr_list
, sec_node
,
1148 vty_out(vty
, " %s\n",
1149 prefix2str(&sec_addr
->addr
,
1150 pbuf
, sizeof(pbuf
)));
1153 vty_out(vty
, "Address : %s\n",
1161 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->pim_neighbor_list
,
1162 neighnode
, neigh
)) {
1165 vty_out(vty
, "PIM Neighbors\n");
1166 vty_out(vty
, "-------------\n");
1170 pim_inet4_dump("<src?>", neigh
->source_addr
,
1172 sizeof(neigh_src_str
));
1173 pim_time_uptime(uptime
, sizeof(uptime
),
1174 now
- neigh
->creation
);
1175 pim_time_timer_to_hhmmss(expire
, sizeof(expire
),
1176 neigh
->t_expire_timer
);
1178 "%-15s : up for %s, holdtime expires in %s\n",
1179 neigh_src_str
, uptime
, expire
);
1182 if (!print_header
) {
1187 vty_out(vty
, "Designated Router\n");
1188 vty_out(vty
, "-----------------\n");
1189 vty_out(vty
, "Address : %s\n", dr_str
);
1190 vty_out(vty
, "Priority : %d\n",
1191 pim_ifp
->pim_dr_priority
);
1192 vty_out(vty
, "Uptime : %s\n", dr_uptime
);
1193 vty_out(vty
, "Elections : %d\n",
1194 pim_ifp
->pim_dr_election_count
);
1195 vty_out(vty
, "Changes : %d\n",
1196 pim_ifp
->pim_dr_election_changes
);
1202 for (ALL_LIST_ELEMENTS_RO(pim
->upstream_list
, upnode
,
1205 if (strcmp(ifp
->name
,
1206 up
->rpf
.source_nexthop
1211 if (!(up
->flags
& PIM_UPSTREAM_FLAG_MASK_FHR
))
1216 "FHR - First Hop Router\n");
1218 "----------------------\n");
1222 pim_inet4_dump("<src?>", up
->sg
.src
, src_str
,
1224 pim_inet4_dump("<grp?>", up
->sg
.grp
, grp_str
,
1226 pim_time_uptime(uptime
, sizeof(uptime
),
1227 now
- up
->state_transition
);
1229 "%s : %s is a source, uptime is %s\n",
1230 grp_str
, src_str
, uptime
);
1233 if (!print_header
) {
1238 vty_out(vty
, "Hellos\n");
1239 vty_out(vty
, "------\n");
1240 vty_out(vty
, "Period : %d\n",
1241 pim_ifp
->pim_hello_period
);
1242 vty_out(vty
, "Timer : %s\n", hello_timer
);
1243 vty_out(vty
, "StatStart : %s\n", stat_uptime
);
1244 vty_out(vty
, "Receive : %d\n",
1245 pim_ifp
->pim_ifstat_hello_recv
);
1246 vty_out(vty
, "Receive Failed : %d\n",
1247 pim_ifp
->pim_ifstat_hello_recvfail
);
1248 vty_out(vty
, "Send : %d\n",
1249 pim_ifp
->pim_ifstat_hello_sent
);
1250 vty_out(vty
, "Send Failed : %d\n",
1251 pim_ifp
->pim_ifstat_hello_sendfail
);
1252 vty_out(vty
, "Generation ID : %08x\n",
1253 pim_ifp
->pim_generation_id
);
1257 pim_print_ifp_flags(vty
, ifp
, mloop
);
1259 vty_out(vty
, "Join Prune Interval\n");
1260 vty_out(vty
, "-------------------\n");
1261 vty_out(vty
, "LAN Delay : %s\n",
1262 pim_if_lan_delay_enabled(ifp
) ? "yes" : "no");
1263 vty_out(vty
, "Effective Propagation Delay : %d msec\n",
1264 pim_if_effective_propagation_delay_msec(ifp
));
1265 vty_out(vty
, "Effective Override Interval : %d msec\n",
1266 pim_if_effective_override_interval_msec(ifp
));
1267 vty_out(vty
, "Join Prune Override Interval : %d msec\n",
1268 pim_if_jp_override_interval_msec(ifp
));
1272 vty_out(vty
, "LAN Prune Delay\n");
1273 vty_out(vty
, "---------------\n");
1274 vty_out(vty
, "Propagation Delay : %d msec\n",
1275 pim_ifp
->pim_propagation_delay_msec
);
1276 vty_out(vty
, "Propagation Delay (Highest) : %d msec\n",
1277 pim_ifp
->pim_neighbors_highest_propagation_delay_msec
);
1278 vty_out(vty
, "Override Interval : %d msec\n",
1279 pim_ifp
->pim_override_interval_msec
);
1280 vty_out(vty
, "Override Interval (Highest) : %d msec\n",
1281 pim_ifp
->pim_neighbors_highest_override_interval_msec
);
1288 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
1289 json
, JSON_C_TO_STRING_PRETTY
));
1290 json_object_free(json
);
1293 vty_out(vty
, "%% No such interface\n");
1297 static void igmp_show_statistics(struct pim_instance
*pim
, struct vty
*vty
,
1298 const char *ifname
, uint8_t uj
)
1300 struct interface
*ifp
;
1301 struct igmp_stats rx_stats
;
1303 igmp_stats_init(&rx_stats
);
1305 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
1306 struct pim_interface
*pim_ifp
;
1307 struct listnode
*sock_node
;
1308 struct igmp_sock
*igmp
;
1310 pim_ifp
= ifp
->info
;
1315 if (ifname
&& strcmp(ifname
, ifp
->name
))
1318 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
,
1320 igmp_stats_add(&rx_stats
, &igmp
->rx_stats
);
1324 json_object
*json
= NULL
;
1325 json_object
*json_row
= NULL
;
1327 json
= json_object_new_object();
1328 json_row
= json_object_new_object();
1330 json_object_string_add(json_row
, "name", ifname
? ifname
:
1332 json_object_int_add(json_row
, "queryV1", rx_stats
.query_v1
);
1333 json_object_int_add(json_row
, "queryV2", rx_stats
.query_v2
);
1334 json_object_int_add(json_row
, "queryV3", rx_stats
.query_v3
);
1335 json_object_int_add(json_row
, "leaveV3", rx_stats
.leave_v2
);
1336 json_object_int_add(json_row
, "reportV1", rx_stats
.report_v1
);
1337 json_object_int_add(json_row
, "reportV2", rx_stats
.report_v2
);
1338 json_object_int_add(json_row
, "reportV3", rx_stats
.report_v3
);
1339 json_object_int_add(json_row
, "mtraceResponse",
1340 rx_stats
.mtrace_rsp
);
1341 json_object_int_add(json_row
, "mtraceRequest",
1342 rx_stats
.mtrace_req
);
1343 json_object_int_add(json_row
, "unsupported",
1344 rx_stats
.unsupported
);
1345 json_object_object_add(json
, ifname
? ifname
: "global",
1347 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
1348 json
, JSON_C_TO_STRING_PRETTY
));
1349 json_object_free(json
);
1351 vty_out(vty
, "IGMP RX statistics\n");
1352 vty_out(vty
, "Interface : %s\n",
1353 ifname
? ifname
: "global");
1354 vty_out(vty
, "V1 query : %u\n", rx_stats
.query_v1
);
1355 vty_out(vty
, "V2 query : %u\n", rx_stats
.query_v2
);
1356 vty_out(vty
, "V3 query : %u\n", rx_stats
.query_v3
);
1357 vty_out(vty
, "V2 leave : %u\n", rx_stats
.leave_v2
);
1358 vty_out(vty
, "V1 report : %u\n", rx_stats
.report_v1
);
1359 vty_out(vty
, "V2 report : %u\n", rx_stats
.report_v2
);
1360 vty_out(vty
, "V3 report : %u\n", rx_stats
.report_v3
);
1361 vty_out(vty
, "mtrace response : %u\n", rx_stats
.mtrace_rsp
);
1362 vty_out(vty
, "mtrace request : %u\n", rx_stats
.mtrace_req
);
1363 vty_out(vty
, "unsupported : %u\n", rx_stats
.unsupported
);
1367 static void pim_show_interfaces(struct pim_instance
*pim
, struct vty
*vty
,
1370 struct interface
*ifp
;
1371 struct listnode
*upnode
;
1372 struct pim_interface
*pim_ifp
;
1373 struct pim_upstream
*up
;
1376 int pim_ifchannels
= 0;
1377 json_object
*json
= NULL
;
1378 json_object
*json_row
= NULL
;
1379 json_object
*json_tmp
;
1381 json
= json_object_new_object();
1383 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
1384 pim_ifp
= ifp
->info
;
1389 pim_nbrs
= pim_ifp
->pim_neighbor_list
->count
;
1390 pim_ifchannels
= pim_if_ifchannel_count(pim_ifp
);
1393 for (ALL_LIST_ELEMENTS_RO(pim
->upstream_list
, upnode
, up
))
1394 if (ifp
== up
->rpf
.source_nexthop
.interface
)
1395 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_FHR
)
1398 json_row
= json_object_new_object();
1399 json_object_pim_ifp_add(json_row
, ifp
);
1400 json_object_int_add(json_row
, "pimNeighbors", pim_nbrs
);
1401 json_object_int_add(json_row
, "pimIfChannels", pim_ifchannels
);
1402 json_object_int_add(json_row
, "firstHopRouterCount", fhr
);
1403 json_object_string_add(json_row
, "pimDesignatedRouter",
1404 inet_ntoa(pim_ifp
->pim_dr_addr
));
1406 if (pim_ifp
->pim_dr_addr
.s_addr
1407 == pim_ifp
->primary_address
.s_addr
)
1408 json_object_boolean_true_add(
1409 json_row
, "pimDesignatedRouterLocal");
1411 json_object_object_add(json
, ifp
->name
, json_row
);
1415 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
1416 json
, JSON_C_TO_STRING_PRETTY
));
1419 "Interface State Address PIM Nbrs PIM DR FHR IfChannels\n");
1421 json_object_object_foreach(json
, key
, val
)
1423 vty_out(vty
, "%-9s ", key
);
1425 json_object_object_get_ex(val
, "state", &json_tmp
);
1426 vty_out(vty
, "%5s ", json_object_get_string(json_tmp
));
1428 json_object_object_get_ex(val
, "address", &json_tmp
);
1429 vty_out(vty
, "%15s ",
1430 json_object_get_string(json_tmp
));
1432 json_object_object_get_ex(val
, "pimNeighbors",
1434 vty_out(vty
, "%8d ", json_object_get_int(json_tmp
));
1436 if (json_object_object_get_ex(
1437 val
, "pimDesignatedRouterLocal",
1439 vty_out(vty
, "%15s ", "local");
1441 json_object_object_get_ex(
1442 val
, "pimDesignatedRouter", &json_tmp
);
1443 vty_out(vty
, "%15s ",
1444 json_object_get_string(json_tmp
));
1447 json_object_object_get_ex(val
, "firstHopRouter",
1449 vty_out(vty
, "%3d ", json_object_get_int(json_tmp
));
1451 json_object_object_get_ex(val
, "pimIfChannels",
1453 vty_out(vty
, "%9d\n", json_object_get_int(json_tmp
));
1457 json_object_free(json
);
1460 static void pim_show_interface_traffic(struct pim_instance
*pim
,
1461 struct vty
*vty
, uint8_t uj
)
1463 struct interface
*ifp
= NULL
;
1464 struct pim_interface
*pim_ifp
= NULL
;
1465 json_object
*json
= NULL
;
1466 json_object
*json_row
= NULL
;
1469 json
= json_object_new_object();
1472 vty_out(vty
, "%-12s%-17s%-17s%-17s%-17s%-17s%-17s\n",
1473 "Interface", " HELLO", " JOIN", " PRUNE",
1474 " REGISTER", " REGISTER-STOP", " ASSERT");
1475 vty_out(vty
, "%-10s%-18s%-17s%-17s%-17s%-17s%-17s\n", "",
1476 " Rx/Tx", " Rx/Tx", " Rx/Tx", " Rx/Tx",
1477 " Rx/Tx", " Rx/Tx");
1479 "---------------------------------------------------------------------------------------------------------------\n");
1482 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
1483 pim_ifp
= ifp
->info
;
1488 if (pim_ifp
->pim_sock_fd
< 0)
1491 json_row
= json_object_new_object();
1492 json_object_pim_ifp_add(json_row
, ifp
);
1493 json_object_int_add(json_row
, "helloRx",
1494 pim_ifp
->pim_ifstat_hello_recv
);
1495 json_object_int_add(json_row
, "helloTx",
1496 pim_ifp
->pim_ifstat_hello_sent
);
1497 json_object_int_add(json_row
, "joinRx",
1498 pim_ifp
->pim_ifstat_join_recv
);
1499 json_object_int_add(json_row
, "joinTx",
1500 pim_ifp
->pim_ifstat_join_send
);
1501 json_object_int_add(json_row
, "registerRx",
1502 pim_ifp
->pim_ifstat_reg_recv
);
1503 json_object_int_add(json_row
, "registerTx",
1504 pim_ifp
->pim_ifstat_reg_recv
);
1505 json_object_int_add(json_row
, "registerStopRx",
1506 pim_ifp
->pim_ifstat_reg_stop_recv
);
1507 json_object_int_add(json_row
, "registerStopTx",
1508 pim_ifp
->pim_ifstat_reg_stop_send
);
1509 json_object_int_add(json_row
, "assertRx",
1510 pim_ifp
->pim_ifstat_assert_recv
);
1511 json_object_int_add(json_row
, "assertTx",
1512 pim_ifp
->pim_ifstat_assert_send
);
1514 json_object_object_add(json
, ifp
->name
, json_row
);
1517 "%-10s %8u/%-8u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u \n",
1518 ifp
->name
, pim_ifp
->pim_ifstat_hello_recv
,
1519 pim_ifp
->pim_ifstat_hello_sent
,
1520 pim_ifp
->pim_ifstat_join_recv
,
1521 pim_ifp
->pim_ifstat_join_send
,
1522 pim_ifp
->pim_ifstat_prune_recv
,
1523 pim_ifp
->pim_ifstat_prune_send
,
1524 pim_ifp
->pim_ifstat_reg_recv
,
1525 pim_ifp
->pim_ifstat_reg_send
,
1526 pim_ifp
->pim_ifstat_reg_stop_recv
,
1527 pim_ifp
->pim_ifstat_reg_stop_send
,
1528 pim_ifp
->pim_ifstat_assert_recv
,
1529 pim_ifp
->pim_ifstat_assert_send
);
1533 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
1534 json
, JSON_C_TO_STRING_PRETTY
));
1535 json_object_free(json
);
1539 static void pim_show_interface_traffic_single(struct pim_instance
*pim
,
1541 const char *ifname
, uint8_t uj
)
1543 struct interface
*ifp
= NULL
;
1544 struct pim_interface
*pim_ifp
= NULL
;
1545 json_object
*json
= NULL
;
1546 json_object
*json_row
= NULL
;
1547 uint8_t found_ifname
= 0;
1550 json
= json_object_new_object();
1553 vty_out(vty
, "%-12s%-17s%-17s%-17s%-17s%-17s%-17s\n",
1554 "Interface", " HELLO", " JOIN", " PRUNE",
1555 " REGISTER", " REGISTER-STOP", " ASSERT");
1556 vty_out(vty
, "%-10s%-18s%-17s%-17s%-17s%-17s%-17s\n", "",
1557 " Rx/Tx", " Rx/Tx", " Rx/Tx", " Rx/Tx",
1558 " Rx/Tx", " Rx/Tx");
1560 "---------------------------------------------------------------------------------------------------------------\n");
1563 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
1564 if (strcmp(ifname
, ifp
->name
))
1567 pim_ifp
= ifp
->info
;
1572 if (pim_ifp
->pim_sock_fd
< 0)
1577 json_row
= json_object_new_object();
1578 json_object_pim_ifp_add(json_row
, ifp
);
1579 json_object_int_add(json_row
, "helloRx",
1580 pim_ifp
->pim_ifstat_hello_recv
);
1581 json_object_int_add(json_row
, "helloTx",
1582 pim_ifp
->pim_ifstat_hello_sent
);
1583 json_object_int_add(json_row
, "joinRx",
1584 pim_ifp
->pim_ifstat_join_recv
);
1585 json_object_int_add(json_row
, "joinTx",
1586 pim_ifp
->pim_ifstat_join_send
);
1587 json_object_int_add(json_row
, "registerRx",
1588 pim_ifp
->pim_ifstat_reg_recv
);
1589 json_object_int_add(json_row
, "registerTx",
1590 pim_ifp
->pim_ifstat_reg_recv
);
1591 json_object_int_add(json_row
, "registerStopRx",
1592 pim_ifp
->pim_ifstat_reg_stop_recv
);
1593 json_object_int_add(json_row
, "registerStopTx",
1594 pim_ifp
->pim_ifstat_reg_stop_send
);
1595 json_object_int_add(json_row
, "assertRx",
1596 pim_ifp
->pim_ifstat_assert_recv
);
1597 json_object_int_add(json_row
, "assertTx",
1598 pim_ifp
->pim_ifstat_assert_send
);
1600 json_object_object_add(json
, ifp
->name
, json_row
);
1603 "%-10s %8u/%-8u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u \n",
1604 ifp
->name
, pim_ifp
->pim_ifstat_hello_recv
,
1605 pim_ifp
->pim_ifstat_hello_sent
,
1606 pim_ifp
->pim_ifstat_join_recv
,
1607 pim_ifp
->pim_ifstat_join_send
,
1608 pim_ifp
->pim_ifstat_prune_recv
,
1609 pim_ifp
->pim_ifstat_prune_send
,
1610 pim_ifp
->pim_ifstat_reg_recv
,
1611 pim_ifp
->pim_ifstat_reg_send
,
1612 pim_ifp
->pim_ifstat_reg_stop_recv
,
1613 pim_ifp
->pim_ifstat_reg_stop_send
,
1614 pim_ifp
->pim_ifstat_assert_recv
,
1615 pim_ifp
->pim_ifstat_assert_send
);
1619 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
1620 json
, JSON_C_TO_STRING_PRETTY
));
1621 json_object_free(json
);
1624 vty_out(vty
, "%% No such interface\n");
1628 static void pim_show_join_helper(struct vty
*vty
, struct pim_interface
*pim_ifp
,
1629 struct pim_ifchannel
*ch
, json_object
*json
,
1630 time_t now
, uint8_t uj
)
1632 char ch_src_str
[INET_ADDRSTRLEN
];
1633 char ch_grp_str
[INET_ADDRSTRLEN
];
1634 json_object
*json_iface
= NULL
;
1635 json_object
*json_row
= NULL
;
1636 json_object
*json_grp
= NULL
;
1637 struct in_addr ifaddr
;
1642 ifaddr
= pim_ifp
->primary_address
;
1644 pim_inet4_dump("<ch_src?>", ch
->sg
.src
, ch_src_str
, sizeof(ch_src_str
));
1645 pim_inet4_dump("<ch_grp?>", ch
->sg
.grp
, ch_grp_str
, sizeof(ch_grp_str
));
1647 pim_time_uptime_begin(uptime
, sizeof(uptime
), now
, ch
->ifjoin_creation
);
1648 pim_time_timer_to_mmss(expire
, sizeof(expire
),
1649 ch
->t_ifjoin_expiry_timer
);
1650 pim_time_timer_to_mmss(prune
, sizeof(prune
),
1651 ch
->t_ifjoin_prune_pending_timer
);
1654 json_object_object_get_ex(json
, ch
->interface
->name
,
1658 json_iface
= json_object_new_object();
1659 json_object_pim_ifp_add(json_iface
, ch
->interface
);
1660 json_object_object_add(json
, ch
->interface
->name
,
1664 json_row
= json_object_new_object();
1665 json_object_string_add(json_row
, "source", ch_src_str
);
1666 json_object_string_add(json_row
, "group", ch_grp_str
);
1667 json_object_string_add(json_row
, "upTime", uptime
);
1668 json_object_string_add(json_row
, "expire", expire
);
1669 json_object_string_add(json_row
, "prune", prune
);
1670 json_object_string_add(
1671 json_row
, "channelJoinName",
1672 pim_ifchannel_ifjoin_name(ch
->ifjoin_state
, ch
->flags
));
1673 if (PIM_IF_FLAG_TEST_S_G_RPT(ch
->flags
))
1674 json_object_int_add(json_row
, "SGRpt", 1);
1676 json_object_object_get_ex(json_iface
, ch_grp_str
, &json_grp
);
1678 json_grp
= json_object_new_object();
1679 json_object_object_add(json_grp
, ch_src_str
, json_row
);
1680 json_object_object_add(json_iface
, ch_grp_str
,
1683 json_object_object_add(json_grp
, ch_src_str
, json_row
);
1685 vty_out(vty
, "%-9s %-15s %-15s %-15s %-10s %8s %-6s %5s\n",
1686 ch
->interface
->name
, inet_ntoa(ifaddr
), ch_src_str
,
1688 pim_ifchannel_ifjoin_name(ch
->ifjoin_state
, ch
->flags
),
1689 uptime
, expire
, prune
);
1693 static void pim_show_join(struct pim_instance
*pim
, struct vty
*vty
, uint8_t uj
)
1695 struct pim_interface
*pim_ifp
;
1696 struct pim_ifchannel
*ch
;
1697 struct interface
*ifp
;
1699 json_object
*json
= NULL
;
1701 now
= pim_time_monotonic_sec();
1704 json
= json_object_new_object();
1707 "Interface Address Source Group State Uptime Expire Prune\n");
1709 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
1710 pim_ifp
= ifp
->info
;
1714 RB_FOREACH (ch
, pim_ifchannel_rb
, &pim_ifp
->ifchannel_rb
) {
1715 pim_show_join_helper(vty
, pim_ifp
, ch
, json
, now
, uj
);
1716 } /* scan interface channels */
1720 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
1721 json
, JSON_C_TO_STRING_PRETTY
));
1722 json_object_free(json
);
1726 static void pim_show_neighbors_single(struct pim_instance
*pim
, struct vty
*vty
,
1727 const char *neighbor
, uint8_t uj
)
1729 struct listnode
*neighnode
;
1730 struct interface
*ifp
;
1731 struct pim_interface
*pim_ifp
;
1732 struct pim_neighbor
*neigh
;
1734 int found_neighbor
= 0;
1735 int option_address_list
;
1736 int option_dr_priority
;
1737 int option_generation_id
;
1738 int option_holdtime
;
1739 int option_lan_prune_delay
;
1743 char neigh_src_str
[INET_ADDRSTRLEN
];
1745 json_object
*json
= NULL
;
1746 json_object
*json_ifp
= NULL
;
1747 json_object
*json_row
= NULL
;
1749 now
= pim_time_monotonic_sec();
1752 json
= json_object_new_object();
1754 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
1755 pim_ifp
= ifp
->info
;
1760 if (pim_ifp
->pim_sock_fd
< 0)
1763 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->pim_neighbor_list
, neighnode
,
1765 pim_inet4_dump("<src?>", neigh
->source_addr
,
1766 neigh_src_str
, sizeof(neigh_src_str
));
1769 * The user can specify either the interface name or the
1771 * If this pim_ifp matches neither then skip.
1773 if (strcmp(neighbor
, "detail")
1774 && strcmp(neighbor
, ifp
->name
)
1775 && strcmp(neighbor
, neigh_src_str
))
1779 pim_time_uptime(uptime
, sizeof(uptime
),
1780 now
- neigh
->creation
);
1781 pim_time_timer_to_hhmmss(expire
, sizeof(expire
),
1782 neigh
->t_expire_timer
);
1784 option_address_list
= 0;
1785 option_dr_priority
= 0;
1786 option_generation_id
= 0;
1787 option_holdtime
= 0;
1788 option_lan_prune_delay
= 0;
1791 if (PIM_OPTION_IS_SET(neigh
->hello_options
,
1792 PIM_OPTION_MASK_ADDRESS_LIST
))
1793 option_address_list
= 1;
1795 if (PIM_OPTION_IS_SET(neigh
->hello_options
,
1796 PIM_OPTION_MASK_DR_PRIORITY
))
1797 option_dr_priority
= 1;
1799 if (PIM_OPTION_IS_SET(neigh
->hello_options
,
1800 PIM_OPTION_MASK_GENERATION_ID
))
1801 option_generation_id
= 1;
1803 if (PIM_OPTION_IS_SET(neigh
->hello_options
,
1804 PIM_OPTION_MASK_HOLDTIME
))
1805 option_holdtime
= 1;
1807 if (PIM_OPTION_IS_SET(neigh
->hello_options
,
1808 PIM_OPTION_MASK_LAN_PRUNE_DELAY
))
1809 option_lan_prune_delay
= 1;
1811 if (PIM_OPTION_IS_SET(
1812 neigh
->hello_options
,
1813 PIM_OPTION_MASK_CAN_DISABLE_JOIN_SUPPRESSION
))
1818 /* Does this ifp live in json? If not create
1820 json_object_object_get_ex(json
, ifp
->name
,
1824 json_ifp
= json_object_new_object();
1825 json_object_pim_ifp_add(json_ifp
, ifp
);
1826 json_object_object_add(json
, ifp
->name
,
1830 json_row
= json_object_new_object();
1831 json_object_string_add(json_row
, "interface",
1833 json_object_string_add(json_row
, "address",
1835 json_object_string_add(json_row
, "upTime",
1837 json_object_string_add(json_row
, "holdtime",
1839 json_object_int_add(json_row
, "drPriority",
1840 neigh
->dr_priority
);
1841 json_object_int_add(json_row
, "generationId",
1842 neigh
->generation_id
);
1844 if (option_address_list
)
1845 json_object_boolean_true_add(
1847 "helloOptionAddressList");
1849 if (option_dr_priority
)
1850 json_object_boolean_true_add(
1852 "helloOptionDrPriority");
1854 if (option_generation_id
)
1855 json_object_boolean_true_add(
1857 "helloOptionGenerationId");
1859 if (option_holdtime
)
1860 json_object_boolean_true_add(
1862 "helloOptionHoldtime");
1864 if (option_lan_prune_delay
)
1865 json_object_boolean_true_add(
1867 "helloOptionLanPruneDelay");
1870 json_object_boolean_true_add(
1871 json_row
, "helloOptionTBit");
1873 json_object_object_add(json_ifp
, neigh_src_str
,
1877 vty_out(vty
, "Interface : %s\n", ifp
->name
);
1878 vty_out(vty
, "Neighbor : %s\n", neigh_src_str
);
1886 " DR Priority : %d\n",
1887 neigh
->dr_priority
);
1889 " Generation ID : %08x\n",
1890 neigh
->generation_id
);
1892 " Override Interval (msec) : %d\n",
1893 neigh
->override_interval_msec
);
1895 " Propagation Delay (msec) : %d\n",
1896 neigh
->propagation_delay_msec
);
1898 " Hello Option - Address List : %s\n",
1899 option_address_list
? "yes" : "no");
1901 " Hello Option - DR Priority : %s\n",
1902 option_dr_priority
? "yes" : "no");
1904 " Hello Option - Generation ID : %s\n",
1905 option_generation_id
? "yes" : "no");
1907 " Hello Option - Holdtime : %s\n",
1908 option_holdtime
? "yes" : "no");
1910 " Hello Option - LAN Prune Delay : %s\n",
1911 option_lan_prune_delay
? "yes" : "no");
1913 " Hello Option - T-bit : %s\n",
1914 option_t_bit
? "yes" : "no");
1915 pim_bfd_show_info(vty
, neigh
->bfd_info
,
1923 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
1924 json
, JSON_C_TO_STRING_PRETTY
));
1925 json_object_free(json
);
1928 if (!found_neighbor
)
1930 "%% No such interface or neighbor\n");
1935 static void pim_show_state(struct pim_instance
*pim
, struct vty
*vty
,
1936 const char *src_or_group
, const char *group
,
1939 struct channel_oil
*c_oil
;
1940 struct listnode
*node
;
1941 json_object
*json
= NULL
;
1942 json_object
*json_group
= NULL
;
1943 json_object
*json_ifp_in
= NULL
;
1944 json_object
*json_ifp_out
= NULL
;
1945 json_object
*json_source
= NULL
;
1948 now
= pim_time_monotonic_sec();
1951 json
= json_object_new_object();
1954 "Codes: J -> Pim Join, I -> IGMP Report, S -> Source, * -> Inherited from (*,G)");
1956 "\nInstalled Source Group IIF OIL\n");
1959 for (ALL_LIST_ELEMENTS_RO(pim
->channel_oil_list
, node
, c_oil
)) {
1960 char grp_str
[INET_ADDRSTRLEN
];
1961 char src_str
[INET_ADDRSTRLEN
];
1962 char in_ifname
[INTERFACE_NAMSIZ
+ 1];
1963 char out_ifname
[INTERFACE_NAMSIZ
+ 1];
1965 struct interface
*ifp_in
;
1968 pim_inet4_dump("<group?>", c_oil
->oil
.mfcc_mcastgrp
, grp_str
,
1970 pim_inet4_dump("<source?>", c_oil
->oil
.mfcc_origin
, src_str
,
1972 ifp_in
= pim_if_find_by_vif_index(pim
, c_oil
->oil
.mfcc_parent
);
1975 strcpy(in_ifname
, ifp_in
->name
);
1977 strcpy(in_ifname
, "<iif?>");
1980 if (strcmp(src_or_group
, src_str
)
1981 && strcmp(src_or_group
, grp_str
))
1984 if (group
&& strcmp(group
, grp_str
))
1990 /* Find the group, create it if it doesn't exist */
1991 json_object_object_get_ex(json
, grp_str
, &json_group
);
1994 json_group
= json_object_new_object();
1995 json_object_object_add(json
, grp_str
,
1999 /* Find the source nested under the group, create it if
2000 * it doesn't exist */
2001 json_object_object_get_ex(json_group
, src_str
,
2005 json_source
= json_object_new_object();
2006 json_object_object_add(json_group
, src_str
,
2010 /* Find the inbound interface nested under the source,
2011 * create it if it doesn't exist */
2012 json_object_object_get_ex(json_source
, in_ifname
,
2016 json_ifp_in
= json_object_new_object();
2017 json_object_object_add(json_source
, in_ifname
,
2019 json_object_int_add(json_source
, "Installed",
2021 json_object_int_add(json_source
, "RefCount",
2022 c_oil
->oil_ref_count
);
2023 json_object_int_add(json_source
, "OilListSize",
2025 json_object_int_add(
2026 json_source
, "OilRescan",
2027 c_oil
->oil_inherited_rescan
);
2028 json_object_int_add(json_source
, "LastUsed",
2029 c_oil
->cc
.lastused
);
2030 json_object_int_add(json_source
, "PacketCount",
2032 json_object_int_add(json_source
, "ByteCount",
2034 json_object_int_add(json_source
,
2036 c_oil
->cc
.wrong_if
);
2039 vty_out(vty
, "%-9d %-15s %-15s %-7s ",
2040 c_oil
->installed
, src_str
, grp_str
,
2044 for (oif_vif_index
= 0; oif_vif_index
< MAXVIFS
;
2046 struct interface
*ifp_out
;
2047 char oif_uptime
[10];
2050 ttl
= c_oil
->oil
.mfcc_ttls
[oif_vif_index
];
2054 ifp_out
= pim_if_find_by_vif_index(pim
, oif_vif_index
);
2056 oif_uptime
, sizeof(oif_uptime
),
2057 now
- c_oil
->oif_creation
[oif_vif_index
]);
2060 strcpy(out_ifname
, ifp_out
->name
);
2062 strcpy(out_ifname
, "<oif?>");
2065 json_ifp_out
= json_object_new_object();
2066 json_object_string_add(json_ifp_out
, "source",
2068 json_object_string_add(json_ifp_out
, "group",
2070 json_object_string_add(json_ifp_out
,
2073 json_object_string_add(json_ifp_out
,
2074 "outboundInterface",
2076 json_object_int_add(json_ifp_out
, "installed",
2079 json_object_object_add(json_ifp_in
, out_ifname
,
2084 vty_out(vty
, "%s(%c%c%c%c)", out_ifname
,
2085 (c_oil
->oif_flags
[oif_vif_index
]
2086 & PIM_OIF_FLAG_PROTO_IGMP
)
2089 (c_oil
->oif_flags
[oif_vif_index
]
2090 & PIM_OIF_FLAG_PROTO_PIM
)
2093 (c_oil
->oif_flags
[oif_vif_index
]
2094 & PIM_OIF_FLAG_PROTO_SOURCE
)
2097 (c_oil
->oif_flags
[oif_vif_index
]
2098 & PIM_OIF_FLAG_PROTO_STAR
)
2102 vty_out(vty
, ", %s(%c%c%c%c)",
2104 (c_oil
->oif_flags
[oif_vif_index
]
2105 & PIM_OIF_FLAG_PROTO_IGMP
)
2108 (c_oil
->oif_flags
[oif_vif_index
]
2109 & PIM_OIF_FLAG_PROTO_PIM
)
2112 (c_oil
->oif_flags
[oif_vif_index
]
2113 & PIM_OIF_FLAG_PROTO_SOURCE
)
2116 (c_oil
->oif_flags
[oif_vif_index
]
2117 & PIM_OIF_FLAG_PROTO_STAR
)
2129 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
2130 json
, JSON_C_TO_STRING_PRETTY
));
2131 json_object_free(json
);
2137 static void pim_show_neighbors(struct pim_instance
*pim
, struct vty
*vty
,
2140 struct listnode
*neighnode
;
2141 struct interface
*ifp
;
2142 struct pim_interface
*pim_ifp
;
2143 struct pim_neighbor
*neigh
;
2147 char neigh_src_str
[INET_ADDRSTRLEN
];
2148 json_object
*json
= NULL
;
2149 json_object
*json_ifp_rows
= NULL
;
2150 json_object
*json_row
= NULL
;
2152 now
= pim_time_monotonic_sec();
2155 json
= json_object_new_object();
2158 "Interface Neighbor Uptime Holdtime DR Pri\n");
2161 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
2162 pim_ifp
= ifp
->info
;
2167 if (pim_ifp
->pim_sock_fd
< 0)
2171 json_ifp_rows
= json_object_new_object();
2173 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->pim_neighbor_list
, neighnode
,
2175 pim_inet4_dump("<src?>", neigh
->source_addr
,
2176 neigh_src_str
, sizeof(neigh_src_str
));
2177 pim_time_uptime(uptime
, sizeof(uptime
),
2178 now
- neigh
->creation
);
2179 pim_time_timer_to_hhmmss(expire
, sizeof(expire
),
2180 neigh
->t_expire_timer
);
2183 json_row
= json_object_new_object();
2184 json_object_string_add(json_row
, "interface",
2186 json_object_string_add(json_row
, "neighbor",
2188 json_object_string_add(json_row
, "upTime",
2190 json_object_string_add(json_row
, "holdTime",
2192 json_object_int_add(json_row
, "holdTimeMax",
2194 json_object_int_add(json_row
, "drPriority",
2195 neigh
->dr_priority
);
2196 json_object_object_add(json_ifp_rows
,
2197 neigh_src_str
, json_row
);
2200 vty_out(vty
, "%-9s %15s %8s %8s %6d\n",
2201 ifp
->name
, neigh_src_str
, uptime
,
2202 expire
, neigh
->dr_priority
);
2207 json_object_object_add(json
, ifp
->name
, json_ifp_rows
);
2208 json_ifp_rows
= NULL
;
2213 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
2214 json
, JSON_C_TO_STRING_PRETTY
));
2215 json_object_free(json
);
2219 static void pim_show_neighbors_secondary(struct pim_instance
*pim
,
2222 struct interface
*ifp
;
2225 "Interface Address Neighbor Secondary \n");
2227 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
2228 struct pim_interface
*pim_ifp
;
2229 struct in_addr ifaddr
;
2230 struct listnode
*neighnode
;
2231 struct pim_neighbor
*neigh
;
2233 pim_ifp
= ifp
->info
;
2238 if (pim_ifp
->pim_sock_fd
< 0)
2241 ifaddr
= pim_ifp
->primary_address
;
2243 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->pim_neighbor_list
, neighnode
,
2245 char neigh_src_str
[INET_ADDRSTRLEN
];
2246 struct listnode
*prefix_node
;
2249 if (!neigh
->prefix_list
)
2252 pim_inet4_dump("<src?>", neigh
->source_addr
,
2253 neigh_src_str
, sizeof(neigh_src_str
));
2255 for (ALL_LIST_ELEMENTS_RO(neigh
->prefix_list
,
2257 char neigh_sec_str
[PREFIX2STR_BUFFER
];
2259 prefix2str(p
, neigh_sec_str
,
2260 sizeof(neigh_sec_str
));
2262 vty_out(vty
, "%-9s %-15s %-15s %-15s\n",
2263 ifp
->name
, inet_ntoa(ifaddr
),
2264 neigh_src_str
, neigh_sec_str
);
2270 static void json_object_pim_upstream_add(json_object
*json
,
2271 struct pim_upstream
*up
)
2273 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_DR_JOIN_DESIRED
)
2274 json_object_boolean_true_add(json
, "drJoinDesired");
2276 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_DR_JOIN_DESIRED_UPDATED
)
2277 json_object_boolean_true_add(json
, "drJoinDesiredUpdated");
2279 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_FHR
)
2280 json_object_boolean_true_add(json
, "firstHopRouter");
2282 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_SRC_IGMP
)
2283 json_object_boolean_true_add(json
, "sourceIgmp");
2285 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_SRC_PIM
)
2286 json_object_boolean_true_add(json
, "sourcePim");
2288 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_SRC_STREAM
)
2289 json_object_boolean_true_add(json
, "sourceStream");
2291 /* XXX: need to print ths flag in the plain text display as well */
2292 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_SRC_MSDP
)
2293 json_object_boolean_true_add(json
, "sourceMsdp");
2297 pim_upstream_state2brief_str(enum pim_upstream_state join_state
,
2300 switch (join_state
) {
2301 case PIM_UPSTREAM_NOTJOINED
:
2302 strcpy(state_str
, "NotJ");
2304 case PIM_UPSTREAM_JOINED
:
2305 strcpy(state_str
, "J");
2308 strcpy(state_str
, "Unk");
2313 static const char *pim_reg_state2brief_str(enum pim_reg_state reg_state
,
2316 switch (reg_state
) {
2317 case PIM_REG_NOINFO
:
2318 strcpy(state_str
, "RegNI");
2321 strcpy(state_str
, "RegJ");
2323 case PIM_REG_JOIN_PENDING
:
2325 strcpy(state_str
, "RegP");
2328 strcpy(state_str
, "Unk");
2333 static void pim_show_upstream(struct pim_instance
*pim
, struct vty
*vty
,
2336 struct listnode
*upnode
;
2337 struct pim_upstream
*up
;
2339 json_object
*json
= NULL
;
2340 json_object
*json_group
= NULL
;
2341 json_object
*json_row
= NULL
;
2343 now
= pim_time_monotonic_sec();
2346 json
= json_object_new_object();
2349 "Iif Source Group State Uptime JoinTimer RSTimer KATimer RefCnt\n");
2351 for (ALL_LIST_ELEMENTS_RO(pim
->upstream_list
, upnode
, up
)) {
2352 char src_str
[INET_ADDRSTRLEN
];
2353 char grp_str
[INET_ADDRSTRLEN
];
2355 char join_timer
[10];
2358 char msdp_reg_timer
[10];
2359 char state_str
[PIM_REG_STATE_STR_LEN
];
2361 pim_inet4_dump("<src?>", up
->sg
.src
, src_str
, sizeof(src_str
));
2362 pim_inet4_dump("<grp?>", up
->sg
.grp
, grp_str
, sizeof(grp_str
));
2363 pim_time_uptime(uptime
, sizeof(uptime
),
2364 now
- up
->state_transition
);
2365 pim_time_timer_to_hhmmss(join_timer
, sizeof(join_timer
),
2369 * If we have a J/P timer for the neighbor display that
2371 if (!up
->t_join_timer
) {
2372 struct pim_neighbor
*nbr
;
2374 nbr
= pim_neighbor_find(
2375 up
->rpf
.source_nexthop
.interface
,
2376 up
->rpf
.rpf_addr
.u
.prefix4
);
2378 pim_time_timer_to_hhmmss(join_timer
,
2383 pim_time_timer_to_hhmmss(rs_timer
, sizeof(rs_timer
),
2385 pim_time_timer_to_hhmmss(ka_timer
, sizeof(ka_timer
),
2387 pim_time_timer_to_hhmmss(msdp_reg_timer
, sizeof(msdp_reg_timer
),
2388 up
->t_msdp_reg_timer
);
2390 pim_upstream_state2brief_str(up
->join_state
, state_str
);
2391 if (up
->reg_state
!= PIM_REG_NOINFO
) {
2392 char tmp_str
[PIM_REG_STATE_STR_LEN
];
2394 sprintf(state_str
+ strlen(state_str
), ",%s",
2395 pim_reg_state2brief_str(up
->reg_state
,
2400 json_object_object_get_ex(json
, grp_str
, &json_group
);
2403 json_group
= json_object_new_object();
2404 json_object_object_add(json
, grp_str
,
2408 json_row
= json_object_new_object();
2409 json_object_pim_upstream_add(json_row
, up
);
2410 json_object_string_add(
2411 json_row
, "inboundInterface",
2412 up
->rpf
.source_nexthop
.interface
->name
);
2415 * The RPF address we use is slightly different
2416 * based upon what we are looking up.
2417 * If we have a S, list that unless
2418 * we are the FHR, else we just put
2419 * the RP as the rpfAddress
2421 if (up
->flags
& PIM_UPSTREAM_FLAG_MASK_FHR
2422 || up
->sg
.src
.s_addr
== INADDR_ANY
) {
2423 char rpf
[PREFIX_STRLEN
];
2424 struct pim_rpf
*rpg
;
2426 rpg
= RP(pim
, up
->sg
.grp
);
2427 pim_inet4_dump("<rpf?>",
2428 rpg
->rpf_addr
.u
.prefix4
, rpf
,
2430 json_object_string_add(json_row
, "rpfAddress",
2433 json_object_string_add(json_row
, "rpfAddress",
2437 json_object_string_add(json_row
, "source", src_str
);
2438 json_object_string_add(json_row
, "group", grp_str
);
2439 json_object_string_add(json_row
, "state", state_str
);
2440 json_object_string_add(
2441 json_row
, "joinState",
2442 pim_upstream_state2str(up
->join_state
));
2443 json_object_string_add(
2444 json_row
, "regState",
2445 pim_reg_state2str(up
->reg_state
, state_str
));
2446 json_object_string_add(json_row
, "upTime", uptime
);
2447 json_object_string_add(json_row
, "joinTimer",
2449 json_object_string_add(json_row
, "resetTimer",
2451 json_object_string_add(json_row
, "keepaliveTimer",
2453 json_object_string_add(json_row
, "msdpRegTimer",
2455 json_object_int_add(json_row
, "refCount",
2457 json_object_int_add(json_row
, "sptBit", up
->sptbit
);
2458 json_object_object_add(json_group
, src_str
, json_row
);
2461 "%-10s%-15s %-15s %-11s %-8s %-9s %-9s %-9s %6d\n",
2462 up
->rpf
.source_nexthop
.interface
->name
, src_str
,
2463 grp_str
, state_str
, uptime
, join_timer
,
2464 rs_timer
, ka_timer
, up
->ref_count
);
2469 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
2470 json
, JSON_C_TO_STRING_PRETTY
));
2471 json_object_free(json
);
2475 static void pim_show_join_desired_helper(struct pim_instance
*pim
,
2477 struct pim_interface
*pim_ifp
,
2478 struct pim_ifchannel
*ch
,
2479 json_object
*json
, uint8_t uj
)
2481 struct pim_upstream
*up
= ch
->upstream
;
2482 json_object
*json_group
= NULL
;
2483 char src_str
[INET_ADDRSTRLEN
];
2484 char grp_str
[INET_ADDRSTRLEN
];
2485 json_object
*json_row
= NULL
;
2487 pim_inet4_dump("<src?>", up
->sg
.src
, src_str
, sizeof(src_str
));
2488 pim_inet4_dump("<grp?>", up
->sg
.grp
, grp_str
, sizeof(grp_str
));
2491 json_object_object_get_ex(json
, grp_str
, &json_group
);
2494 json_group
= json_object_new_object();
2495 json_object_object_add(json
, grp_str
, json_group
);
2498 json_row
= json_object_new_object();
2499 json_object_pim_upstream_add(json_row
, up
);
2500 json_object_string_add(json_row
, "interface",
2501 ch
->interface
->name
);
2502 json_object_string_add(json_row
, "source", src_str
);
2503 json_object_string_add(json_row
, "group", grp_str
);
2505 if (pim_macro_ch_lost_assert(ch
))
2506 json_object_boolean_true_add(json_row
, "lostAssert");
2508 if (pim_macro_chisin_joins(ch
))
2509 json_object_boolean_true_add(json_row
, "joins");
2511 if (pim_macro_chisin_pim_include(ch
))
2512 json_object_boolean_true_add(json_row
, "pimInclude");
2514 if (pim_upstream_evaluate_join_desired(pim
, up
))
2515 json_object_boolean_true_add(json_row
,
2516 "evaluateJoinDesired");
2518 json_object_object_add(json_group
, src_str
, json_row
);
2521 vty_out(vty
, "%-9s %-15s %-15s %-10s %-5s %-10s %-11s %-6s\n",
2522 ch
->interface
->name
, src_str
, grp_str
,
2523 pim_macro_ch_lost_assert(ch
) ? "yes" : "no",
2524 pim_macro_chisin_joins(ch
) ? "yes" : "no",
2525 pim_macro_chisin_pim_include(ch
) ? "yes" : "no",
2526 PIM_UPSTREAM_FLAG_TEST_DR_JOIN_DESIRED(up
->flags
)
2529 pim_upstream_evaluate_join_desired(pim
, up
) ? "yes"
2534 static void pim_show_join_desired(struct pim_instance
*pim
, struct vty
*vty
,
2537 struct pim_interface
*pim_ifp
;
2538 struct pim_ifchannel
*ch
;
2539 struct interface
*ifp
;
2541 json_object
*json
= NULL
;
2544 json
= json_object_new_object();
2547 "Interface Source Group LostAssert Joins PimInclude JoinDesired EvalJD\n");
2549 /* scan per-interface (S,G) state */
2550 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
2551 pim_ifp
= ifp
->info
;
2556 RB_FOREACH (ch
, pim_ifchannel_rb
, &pim_ifp
->ifchannel_rb
) {
2557 /* scan all interfaces */
2558 pim_show_join_desired_helper(pim
, vty
, pim_ifp
, ch
,
2564 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
2565 json
, JSON_C_TO_STRING_PRETTY
));
2566 json_object_free(json
);
2570 static void pim_show_upstream_rpf(struct pim_instance
*pim
, struct vty
*vty
,
2573 struct listnode
*upnode
;
2574 struct pim_upstream
*up
;
2575 json_object
*json
= NULL
;
2576 json_object
*json_group
= NULL
;
2577 json_object
*json_row
= NULL
;
2580 json
= json_object_new_object();
2583 "Source Group RpfIface RibNextHop RpfAddress \n");
2585 for (ALL_LIST_ELEMENTS_RO(pim
->upstream_list
, upnode
, up
)) {
2586 char src_str
[INET_ADDRSTRLEN
];
2587 char grp_str
[INET_ADDRSTRLEN
];
2588 char rpf_nexthop_str
[PREFIX_STRLEN
];
2589 char rpf_addr_str
[PREFIX_STRLEN
];
2590 struct pim_rpf
*rpf
;
2591 const char *rpf_ifname
;
2595 pim_inet4_dump("<src?>", up
->sg
.src
, src_str
, sizeof(src_str
));
2596 pim_inet4_dump("<grp?>", up
->sg
.grp
, grp_str
, sizeof(grp_str
));
2597 pim_addr_dump("<nexthop?>",
2598 &rpf
->source_nexthop
.mrib_nexthop_addr
,
2599 rpf_nexthop_str
, sizeof(rpf_nexthop_str
));
2600 pim_addr_dump("<rpf?>", &rpf
->rpf_addr
, rpf_addr_str
,
2601 sizeof(rpf_addr_str
));
2603 rpf_ifname
= rpf
->source_nexthop
.interface
? rpf
->source_nexthop
.interface
->name
: "<ifname?>";
2606 json_object_object_get_ex(json
, grp_str
, &json_group
);
2609 json_group
= json_object_new_object();
2610 json_object_object_add(json
, grp_str
,
2614 json_row
= json_object_new_object();
2615 json_object_pim_upstream_add(json_row
, up
);
2616 json_object_string_add(json_row
, "source", src_str
);
2617 json_object_string_add(json_row
, "group", grp_str
);
2618 json_object_string_add(json_row
, "rpfInterface",
2620 json_object_string_add(json_row
, "ribNexthop",
2622 json_object_string_add(json_row
, "rpfAddress",
2624 json_object_object_add(json_group
, src_str
, json_row
);
2626 vty_out(vty
, "%-15s %-15s %-8s %-15s %-15s\n", src_str
,
2627 grp_str
, rpf_ifname
, rpf_nexthop_str
,
2633 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
2634 json
, JSON_C_TO_STRING_PRETTY
));
2635 json_object_free(json
);
2639 static void show_rpf_refresh_stats(struct vty
*vty
, struct pim_instance
*pim
,
2640 time_t now
, json_object
*json
)
2642 char refresh_uptime
[10];
2644 pim_time_uptime_begin(refresh_uptime
, sizeof(refresh_uptime
), now
,
2645 pim
->rpf_cache_refresh_last
);
2648 json_object_int_add(json
, "rpfCacheRefreshDelayMsecs",
2649 qpim_rpf_cache_refresh_delay_msec
);
2650 json_object_int_add(
2651 json
, "rpfCacheRefreshTimer",
2652 pim_time_timer_remain_msec(pim
->rpf_cache_refresher
));
2653 json_object_int_add(json
, "rpfCacheRefreshRequests",
2654 pim
->rpf_cache_refresh_requests
);
2655 json_object_int_add(json
, "rpfCacheRefreshEvents",
2656 pim
->rpf_cache_refresh_events
);
2657 json_object_string_add(json
, "rpfCacheRefreshLast",
2659 json_object_int_add(json
, "nexthopLookups",
2660 pim
->nexthop_lookups
);
2661 json_object_int_add(json
, "nexthopLookupsAvoided",
2662 pim
->nexthop_lookups_avoided
);
2665 "RPF Cache Refresh Delay: %ld msecs\n"
2666 "RPF Cache Refresh Timer: %ld msecs\n"
2667 "RPF Cache Refresh Requests: %lld\n"
2668 "RPF Cache Refresh Events: %lld\n"
2669 "RPF Cache Refresh Last: %s\n"
2670 "Nexthop Lookups: %lld\n"
2671 "Nexthop Lookups Avoided: %lld\n",
2672 qpim_rpf_cache_refresh_delay_msec
,
2673 pim_time_timer_remain_msec(pim
->rpf_cache_refresher
),
2674 (long long)pim
->rpf_cache_refresh_requests
,
2675 (long long)pim
->rpf_cache_refresh_events
,
2676 refresh_uptime
, (long long)pim
->nexthop_lookups
,
2677 (long long)pim
->nexthop_lookups_avoided
);
2681 static void show_scan_oil_stats(struct pim_instance
*pim
, struct vty
*vty
,
2684 char uptime_scan_oil
[10];
2685 char uptime_mroute_add
[10];
2686 char uptime_mroute_del
[10];
2688 pim_time_uptime_begin(uptime_scan_oil
, sizeof(uptime_scan_oil
), now
,
2689 pim
->scan_oil_last
);
2690 pim_time_uptime_begin(uptime_mroute_add
, sizeof(uptime_mroute_add
), now
,
2691 pim
->mroute_add_last
);
2692 pim_time_uptime_begin(uptime_mroute_del
, sizeof(uptime_mroute_del
), now
,
2693 pim
->mroute_del_last
);
2696 "Scan OIL - Last: %s Events: %lld\n"
2697 "MFC Add - Last: %s Events: %lld\n"
2698 "MFC Del - Last: %s Events: %lld\n",
2699 uptime_scan_oil
, (long long)pim
->scan_oil_events
,
2700 uptime_mroute_add
, (long long)pim
->mroute_add_events
,
2701 uptime_mroute_del
, (long long)pim
->mroute_del_events
);
2704 static void pim_show_rpf(struct pim_instance
*pim
, struct vty
*vty
, uint8_t uj
)
2706 struct listnode
*up_node
;
2707 struct pim_upstream
*up
;
2708 time_t now
= pim_time_monotonic_sec();
2709 json_object
*json
= NULL
;
2710 json_object
*json_group
= NULL
;
2711 json_object
*json_row
= NULL
;
2714 json
= json_object_new_object();
2715 show_rpf_refresh_stats(vty
, pim
, now
, json
);
2717 show_rpf_refresh_stats(vty
, pim
, now
, json
);
2720 "Source Group RpfIface RpfAddress RibNextHop Metric Pref\n");
2723 for (ALL_LIST_ELEMENTS_RO(pim
->upstream_list
, up_node
, up
)) {
2724 char src_str
[INET_ADDRSTRLEN
];
2725 char grp_str
[INET_ADDRSTRLEN
];
2726 char rpf_addr_str
[PREFIX_STRLEN
];
2727 char rib_nexthop_str
[PREFIX_STRLEN
];
2728 const char *rpf_ifname
;
2729 struct pim_rpf
*rpf
= &up
->rpf
;
2731 pim_inet4_dump("<src?>", up
->sg
.src
, src_str
, sizeof(src_str
));
2732 pim_inet4_dump("<grp?>", up
->sg
.grp
, grp_str
, sizeof(grp_str
));
2733 pim_addr_dump("<rpf?>", &rpf
->rpf_addr
, rpf_addr_str
,
2734 sizeof(rpf_addr_str
));
2735 pim_addr_dump("<nexthop?>",
2736 &rpf
->source_nexthop
.mrib_nexthop_addr
,
2737 rib_nexthop_str
, sizeof(rib_nexthop_str
));
2739 rpf_ifname
= rpf
->source_nexthop
.interface
? rpf
->source_nexthop
.interface
->name
: "<ifname?>";
2742 json_object_object_get_ex(json
, grp_str
, &json_group
);
2745 json_group
= json_object_new_object();
2746 json_object_object_add(json
, grp_str
,
2750 json_row
= json_object_new_object();
2751 json_object_string_add(json_row
, "source", src_str
);
2752 json_object_string_add(json_row
, "group", grp_str
);
2753 json_object_string_add(json_row
, "rpfInterface",
2755 json_object_string_add(json_row
, "rpfAddress",
2757 json_object_string_add(json_row
, "ribNexthop",
2759 json_object_int_add(
2760 json_row
, "routeMetric",
2761 rpf
->source_nexthop
.mrib_route_metric
);
2762 json_object_int_add(
2763 json_row
, "routePreference",
2764 rpf
->source_nexthop
.mrib_metric_preference
);
2765 json_object_object_add(json_group
, src_str
, json_row
);
2768 vty_out(vty
, "%-15s %-15s %-8s %-15s %-15s %6d %4d\n",
2769 src_str
, grp_str
, rpf_ifname
, rpf_addr_str
,
2771 rpf
->source_nexthop
.mrib_route_metric
,
2772 rpf
->source_nexthop
.mrib_metric_preference
);
2777 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
2778 json
, JSON_C_TO_STRING_PRETTY
));
2779 json_object_free(json
);
2783 struct pnc_cache_walk_data
{
2785 struct pim_instance
*pim
;
2788 static int pim_print_pnc_cache_walkcb(struct hash_backet
*backet
, void *arg
)
2790 struct pim_nexthop_cache
*pnc
= backet
->data
;
2791 struct pnc_cache_walk_data
*cwd
= arg
;
2792 struct vty
*vty
= cwd
->vty
;
2793 struct pim_instance
*pim
= cwd
->pim
;
2794 struct nexthop
*nh_node
= NULL
;
2795 ifindex_t first_ifindex
;
2796 struct interface
*ifp
= NULL
;
2801 for (nh_node
= pnc
->nexthop
; nh_node
; nh_node
= nh_node
->next
) {
2802 first_ifindex
= nh_node
->ifindex
;
2803 ifp
= if_lookup_by_index(first_ifindex
, pim
->vrf_id
);
2805 vty_out(vty
, "%-15s ", inet_ntoa(pnc
->rpf
.rpf_addr
.u
.prefix4
));
2806 vty_out(vty
, "%-14s ", ifp
? ifp
->name
: "NULL");
2807 vty_out(vty
, "%s ", inet_ntoa(nh_node
->gate
.ipv4
));
2813 static void pim_show_nexthop(struct pim_instance
*pim
, struct vty
*vty
)
2815 struct pnc_cache_walk_data cwd
;
2819 vty_out(vty
, "Number of registered addresses: %lu\n",
2820 pim
->rpf_hash
->count
);
2821 vty_out(vty
, "Address Interface Nexthop\n");
2822 vty_out(vty
, "-------------------------------------------\n");
2824 hash_walk(pim
->rpf_hash
, pim_print_pnc_cache_walkcb
, &cwd
);
2827 static void igmp_show_groups(struct pim_instance
*pim
, struct vty
*vty
,
2830 struct interface
*ifp
;
2832 json_object
*json
= NULL
;
2833 json_object
*json_iface
= NULL
;
2834 json_object
*json_row
= NULL
;
2836 now
= pim_time_monotonic_sec();
2839 json
= json_object_new_object();
2842 "Interface Address Group Mode Timer Srcs V Uptime \n");
2844 /* scan interfaces */
2845 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
2846 struct pim_interface
*pim_ifp
= ifp
->info
;
2847 struct listnode
*sock_node
;
2848 struct igmp_sock
*igmp
;
2853 /* scan igmp sockets */
2854 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
,
2856 char ifaddr_str
[INET_ADDRSTRLEN
];
2857 struct listnode
*grpnode
;
2858 struct igmp_group
*grp
;
2860 pim_inet4_dump("<ifaddr?>", igmp
->ifaddr
, ifaddr_str
,
2861 sizeof(ifaddr_str
));
2863 /* scan igmp groups */
2864 for (ALL_LIST_ELEMENTS_RO(igmp
->igmp_group_list
,
2866 char group_str
[INET_ADDRSTRLEN
];
2870 pim_inet4_dump("<group?>", grp
->group_addr
,
2871 group_str
, sizeof(group_str
));
2872 pim_time_timer_to_hhmmss(hhmmss
, sizeof(hhmmss
),
2873 grp
->t_group_timer
);
2874 pim_time_uptime(uptime
, sizeof(uptime
),
2875 now
- grp
->group_creation
);
2878 json_object_object_get_ex(
2879 json
, ifp
->name
, &json_iface
);
2883 json_object_new_object();
2884 json_object_pim_ifp_add(
2886 json_object_object_add(
2891 json_row
= json_object_new_object();
2892 json_object_string_add(
2893 json_row
, "source", ifaddr_str
);
2894 json_object_string_add(
2895 json_row
, "group", group_str
);
2897 if (grp
->igmp_version
== 3)
2898 json_object_string_add(
2900 grp
->group_filtermode_isexcl
2904 json_object_string_add(json_row
,
2906 json_object_int_add(
2907 json_row
, "sourcesCount",
2908 grp
->group_source_list
2910 grp
->group_source_list
)
2912 json_object_int_add(json_row
, "version",
2914 json_object_string_add(
2915 json_row
, "uptime", uptime
);
2916 json_object_object_add(json_iface
,
2922 "%-9s %-15s %-15s %4s %8s %4d %d %8s\n",
2923 ifp
->name
, ifaddr_str
,
2925 grp
->igmp_version
== 3
2926 ? (grp
->group_filtermode_isexcl
2931 grp
->group_source_list
2933 grp
->group_source_list
)
2935 grp
->igmp_version
, uptime
);
2937 } /* scan igmp groups */
2938 } /* scan igmp sockets */
2939 } /* scan interfaces */
2942 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
2943 json
, JSON_C_TO_STRING_PRETTY
));
2944 json_object_free(json
);
2948 static void igmp_show_group_retransmission(struct pim_instance
*pim
,
2951 struct interface
*ifp
;
2954 "Interface Address Group RetTimer Counter RetSrcs\n");
2956 /* scan interfaces */
2957 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
2958 struct pim_interface
*pim_ifp
= ifp
->info
;
2959 struct listnode
*sock_node
;
2960 struct igmp_sock
*igmp
;
2965 /* scan igmp sockets */
2966 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
,
2968 char ifaddr_str
[INET_ADDRSTRLEN
];
2969 struct listnode
*grpnode
;
2970 struct igmp_group
*grp
;
2972 pim_inet4_dump("<ifaddr?>", igmp
->ifaddr
, ifaddr_str
,
2973 sizeof(ifaddr_str
));
2975 /* scan igmp groups */
2976 for (ALL_LIST_ELEMENTS_RO(igmp
->igmp_group_list
,
2978 char group_str
[INET_ADDRSTRLEN
];
2979 char grp_retr_mmss
[10];
2980 struct listnode
*src_node
;
2981 struct igmp_source
*src
;
2982 int grp_retr_sources
= 0;
2984 pim_inet4_dump("<group?>", grp
->group_addr
,
2985 group_str
, sizeof(group_str
));
2986 pim_time_timer_to_mmss(
2987 grp_retr_mmss
, sizeof(grp_retr_mmss
),
2988 grp
->t_group_query_retransmit_timer
);
2991 /* count group sources with retransmission state
2993 for (ALL_LIST_ELEMENTS_RO(
2994 grp
->group_source_list
, src_node
,
2996 if (src
->source_query_retransmit_count
3002 vty_out(vty
, "%-9s %-15s %-15s %-8s %7d %7d\n",
3003 ifp
->name
, ifaddr_str
, group_str
,
3005 grp
->group_specific_query_retransmit_count
,
3008 } /* scan igmp groups */
3009 } /* scan igmp sockets */
3010 } /* scan interfaces */
3013 static void igmp_show_sources(struct pim_instance
*pim
, struct vty
*vty
)
3015 struct interface
*ifp
;
3018 now
= pim_time_monotonic_sec();
3021 "Interface Address Group Source Timer Fwd Uptime \n");
3023 /* scan interfaces */
3024 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
3025 struct pim_interface
*pim_ifp
= ifp
->info
;
3026 struct listnode
*sock_node
;
3027 struct igmp_sock
*igmp
;
3032 /* scan igmp sockets */
3033 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
,
3035 char ifaddr_str
[INET_ADDRSTRLEN
];
3036 struct listnode
*grpnode
;
3037 struct igmp_group
*grp
;
3039 pim_inet4_dump("<ifaddr?>", igmp
->ifaddr
, ifaddr_str
,
3040 sizeof(ifaddr_str
));
3042 /* scan igmp groups */
3043 for (ALL_LIST_ELEMENTS_RO(igmp
->igmp_group_list
,
3045 char group_str
[INET_ADDRSTRLEN
];
3046 struct listnode
*srcnode
;
3047 struct igmp_source
*src
;
3049 pim_inet4_dump("<group?>", grp
->group_addr
,
3050 group_str
, sizeof(group_str
));
3052 /* scan group sources */
3053 for (ALL_LIST_ELEMENTS_RO(
3054 grp
->group_source_list
, srcnode
,
3056 char source_str
[INET_ADDRSTRLEN
];
3061 "<source?>", src
->source_addr
,
3062 source_str
, sizeof(source_str
));
3064 pim_time_timer_to_mmss(
3066 src
->t_source_timer
);
3069 uptime
, sizeof(uptime
),
3070 now
- src
->source_creation
);
3073 "%-9s %-15s %-15s %-15s %5s %3s %8s\n",
3074 ifp
->name
, ifaddr_str
,
3075 group_str
, source_str
, mmss
,
3076 IGMP_SOURCE_TEST_FORWARDING(
3082 } /* scan group sources */
3083 } /* scan igmp groups */
3084 } /* scan igmp sockets */
3085 } /* scan interfaces */
3088 static void igmp_show_source_retransmission(struct pim_instance
*pim
,
3091 struct interface
*ifp
;
3094 "Interface Address Group Source Counter\n");
3096 /* scan interfaces */
3097 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
3098 struct pim_interface
*pim_ifp
= ifp
->info
;
3099 struct listnode
*sock_node
;
3100 struct igmp_sock
*igmp
;
3105 /* scan igmp sockets */
3106 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
,
3108 char ifaddr_str
[INET_ADDRSTRLEN
];
3109 struct listnode
*grpnode
;
3110 struct igmp_group
*grp
;
3112 pim_inet4_dump("<ifaddr?>", igmp
->ifaddr
, ifaddr_str
,
3113 sizeof(ifaddr_str
));
3115 /* scan igmp groups */
3116 for (ALL_LIST_ELEMENTS_RO(igmp
->igmp_group_list
,
3118 char group_str
[INET_ADDRSTRLEN
];
3119 struct listnode
*srcnode
;
3120 struct igmp_source
*src
;
3122 pim_inet4_dump("<group?>", grp
->group_addr
,
3123 group_str
, sizeof(group_str
));
3125 /* scan group sources */
3126 for (ALL_LIST_ELEMENTS_RO(
3127 grp
->group_source_list
, srcnode
,
3129 char source_str
[INET_ADDRSTRLEN
];
3132 "<source?>", src
->source_addr
,
3133 source_str
, sizeof(source_str
));
3136 "%-9s %-15s %-15s %-15s %7d\n",
3137 ifp
->name
, ifaddr_str
,
3138 group_str
, source_str
,
3139 src
->source_query_retransmit_count
);
3141 } /* scan group sources */
3142 } /* scan igmp groups */
3143 } /* scan igmp sockets */
3144 } /* scan interfaces */
3147 static void clear_igmp_interfaces(struct pim_instance
*pim
)
3149 struct interface
*ifp
;
3151 FOR_ALL_INTERFACES (pim
->vrf
, ifp
)
3152 pim_if_addr_del_all_igmp(ifp
);
3154 FOR_ALL_INTERFACES (pim
->vrf
, ifp
)
3155 pim_if_addr_add_all(ifp
);
3158 static void clear_pim_interfaces(struct pim_instance
*pim
)
3160 struct interface
*ifp
;
3162 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
3164 pim_neighbor_delete_all(ifp
, "interface cleared");
3169 static void clear_interfaces(struct pim_instance
*pim
)
3171 clear_igmp_interfaces(pim
);
3172 clear_pim_interfaces(pim
);
3175 #define PIM_GET_PIM_INTERFACE(pim_ifp, ifp) \
3176 pim_ifp = ifp->info; \
3179 "%% Enable PIM and/or IGMP on this interface first\n"); \
3180 return CMD_WARNING_CONFIG_FAILED; \
3183 DEFUN (clear_ip_interfaces
,
3184 clear_ip_interfaces_cmd
,
3185 "clear ip interfaces [vrf NAME]",
3188 "Reset interfaces\n"
3192 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3197 clear_interfaces(vrf
->info
);
3202 DEFUN (clear_ip_igmp_interfaces
,
3203 clear_ip_igmp_interfaces_cmd
,
3204 "clear ip igmp [vrf NAME] interfaces",
3209 "Reset IGMP interfaces\n")
3212 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3217 clear_igmp_interfaces(vrf
->info
);
3222 static void mroute_add_all(struct pim_instance
*pim
)
3224 struct listnode
*node
;
3225 struct channel_oil
*c_oil
;
3227 for (ALL_LIST_ELEMENTS_RO(pim
->channel_oil_list
, node
, c_oil
)) {
3228 if (pim_mroute_add(c_oil
, __PRETTY_FUNCTION__
)) {
3229 /* just log warning */
3230 char source_str
[INET_ADDRSTRLEN
];
3231 char group_str
[INET_ADDRSTRLEN
];
3232 pim_inet4_dump("<source?>", c_oil
->oil
.mfcc_origin
,
3233 source_str
, sizeof(source_str
));
3234 pim_inet4_dump("<group?>", c_oil
->oil
.mfcc_mcastgrp
,
3235 group_str
, sizeof(group_str
));
3236 zlog_warn("%s %s: (S,G)=(%s,%s) failure writing MFC",
3237 __FILE__
, __PRETTY_FUNCTION__
, source_str
,
3243 static void mroute_del_all(struct pim_instance
*pim
)
3245 struct listnode
*node
;
3246 struct channel_oil
*c_oil
;
3248 for (ALL_LIST_ELEMENTS_RO(pim
->channel_oil_list
, node
, c_oil
)) {
3249 if (pim_mroute_del(c_oil
, __PRETTY_FUNCTION__
)) {
3250 /* just log warning */
3251 char source_str
[INET_ADDRSTRLEN
];
3252 char group_str
[INET_ADDRSTRLEN
];
3253 pim_inet4_dump("<source?>", c_oil
->oil
.mfcc_origin
,
3254 source_str
, sizeof(source_str
));
3255 pim_inet4_dump("<group?>", c_oil
->oil
.mfcc_mcastgrp
,
3256 group_str
, sizeof(group_str
));
3257 zlog_warn("%s %s: (S,G)=(%s,%s) failure clearing MFC",
3258 __FILE__
, __PRETTY_FUNCTION__
, source_str
,
3264 DEFUN (clear_ip_mroute
,
3265 clear_ip_mroute_cmd
,
3266 "clear ip mroute [vrf NAME]",
3269 "Reset multicast routes\n"
3273 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3278 mroute_del_all(vrf
->info
);
3279 mroute_add_all(vrf
->info
);
3284 DEFUN (clear_ip_pim_interfaces
,
3285 clear_ip_pim_interfaces_cmd
,
3286 "clear ip pim [vrf NAME] interfaces",
3291 "Reset PIM interfaces\n")
3294 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3299 clear_pim_interfaces(vrf
->info
);
3304 DEFUN (clear_ip_pim_interface_traffic
,
3305 clear_ip_pim_interface_traffic_cmd
,
3306 "clear ip pim [vrf NAME] interface traffic",
3309 "PIM clear commands\n"
3311 "Reset PIM interfaces\n"
3312 "Reset Protocol Packet counters\n")
3315 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3316 struct interface
*ifp
= NULL
;
3317 struct pim_interface
*pim_ifp
= NULL
;
3322 FOR_ALL_INTERFACES (vrf
, ifp
) {
3323 pim_ifp
= ifp
->info
;
3328 pim_ifp
->pim_ifstat_hello_recv
= 0;
3329 pim_ifp
->pim_ifstat_hello_sent
= 0;
3330 pim_ifp
->pim_ifstat_join_recv
= 0;
3331 pim_ifp
->pim_ifstat_join_send
= 0;
3332 pim_ifp
->pim_ifstat_prune_recv
= 0;
3333 pim_ifp
->pim_ifstat_prune_send
= 0;
3334 pim_ifp
->pim_ifstat_reg_recv
= 0;
3335 pim_ifp
->pim_ifstat_reg_send
= 0;
3336 pim_ifp
->pim_ifstat_reg_stop_recv
= 0;
3337 pim_ifp
->pim_ifstat_reg_stop_send
= 0;
3338 pim_ifp
->pim_ifstat_assert_recv
= 0;
3339 pim_ifp
->pim_ifstat_assert_send
= 0;
3345 DEFUN (clear_ip_pim_oil
,
3346 clear_ip_pim_oil_cmd
,
3347 "clear ip pim [vrf NAME] oil",
3352 "Rescan PIM OIL (output interface list)\n")
3355 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3360 pim_scan_oil(vrf
->info
);
3365 DEFUN (show_ip_igmp_interface
,
3366 show_ip_igmp_interface_cmd
,
3367 "show ip igmp [vrf NAME] interface [detail|WORD] [json]",
3372 "IGMP interface information\n"
3378 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3379 uint8_t uj
= use_json(argc
, argv
);
3384 if (argv_find(argv
, argc
, "detail", &idx
)
3385 || argv_find(argv
, argc
, "WORD", &idx
))
3386 igmp_show_interfaces_single(vrf
->info
, vty
, argv
[idx
]->arg
, uj
);
3388 igmp_show_interfaces(vrf
->info
, vty
, uj
);
3393 DEFUN (show_ip_igmp_interface_vrf_all
,
3394 show_ip_igmp_interface_vrf_all_cmd
,
3395 "show ip igmp vrf all interface [detail|WORD] [json]",
3400 "IGMP interface information\n"
3406 uint8_t uj
= use_json(argc
, argv
);
3412 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
3416 vty_out(vty
, " \"%s\": ", vrf
->name
);
3419 vty_out(vty
, "VRF: %s\n", vrf
->name
);
3420 if (argv_find(argv
, argc
, "detail", &idx
)
3421 || argv_find(argv
, argc
, "WORD", &idx
))
3422 igmp_show_interfaces_single(vrf
->info
, vty
,
3423 argv
[idx
]->arg
, uj
);
3425 igmp_show_interfaces(vrf
->info
, vty
, uj
);
3428 vty_out(vty
, "}\n");
3433 DEFUN (show_ip_igmp_join
,
3434 show_ip_igmp_join_cmd
,
3435 "show ip igmp [vrf NAME] join",
3440 "IGMP static join information\n")
3443 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3448 igmp_show_interface_join(vrf
->info
, vty
);
3453 DEFUN (show_ip_igmp_join_vrf_all
,
3454 show_ip_igmp_join_vrf_all_cmd
,
3455 "show ip igmp vrf all join",
3460 "IGMP static join information\n")
3462 uint8_t uj
= use_json(argc
, argv
);
3468 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
3472 vty_out(vty
, " \"%s\": ", vrf
->name
);
3475 vty_out(vty
, "VRF: %s\n", vrf
->name
);
3476 igmp_show_interface_join(vrf
->info
, vty
);
3479 vty_out(vty
, "}\n");
3484 DEFUN (show_ip_igmp_groups
,
3485 show_ip_igmp_groups_cmd
,
3486 "show ip igmp [vrf NAME] groups [json]",
3495 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3496 uint8_t uj
= use_json(argc
, argv
);
3501 igmp_show_groups(vrf
->info
, vty
, uj
);
3506 DEFUN (show_ip_igmp_groups_vrf_all
,
3507 show_ip_igmp_groups_vrf_all_cmd
,
3508 "show ip igmp vrf all groups [json]",
3516 uint8_t uj
= use_json(argc
, argv
);
3522 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
3526 vty_out(vty
, " \"%s\": ", vrf
->name
);
3529 vty_out(vty
, "VRF: %s\n", vrf
->name
);
3530 igmp_show_groups(vrf
->info
, vty
, uj
);
3533 vty_out(vty
, "}\n");
3538 DEFUN (show_ip_igmp_groups_retransmissions
,
3539 show_ip_igmp_groups_retransmissions_cmd
,
3540 "show ip igmp [vrf NAME] groups retransmissions",
3546 "IGMP group retransmissions\n")
3549 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3554 igmp_show_group_retransmission(vrf
->info
, vty
);
3559 DEFUN (show_ip_igmp_sources
,
3560 show_ip_igmp_sources_cmd
,
3561 "show ip igmp [vrf NAME] sources",
3569 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3574 igmp_show_sources(vrf
->info
, vty
);
3579 DEFUN (show_ip_igmp_sources_retransmissions
,
3580 show_ip_igmp_sources_retransmissions_cmd
,
3581 "show ip igmp [vrf NAME] sources retransmissions",
3587 "IGMP source retransmissions\n")
3590 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3595 igmp_show_source_retransmission(vrf
->info
, vty
);
3600 DEFUN (show_ip_igmp_statistics
,
3601 show_ip_igmp_statistics_cmd
,
3602 "show ip igmp [vrf NAME] statistics [interface WORD] [json]",
3613 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3614 uint8_t uj
= use_json(argc
, argv
);
3619 if (argv_find(argv
, argc
, "WORD", &idx
))
3620 igmp_show_statistics(vrf
->info
, vty
, argv
[idx
]->arg
, uj
);
3622 igmp_show_statistics(vrf
->info
, vty
, NULL
, uj
);
3627 DEFUN (show_ip_pim_assert
,
3628 show_ip_pim_assert_cmd
,
3629 "show ip pim [vrf NAME] assert",
3634 "PIM interface assert\n")
3637 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3642 pim_show_assert(vrf
->info
, vty
);
3647 DEFUN (show_ip_pim_assert_internal
,
3648 show_ip_pim_assert_internal_cmd
,
3649 "show ip pim [vrf NAME] assert-internal",
3654 "PIM interface internal assert state\n")
3657 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3662 pim_show_assert_internal(vrf
->info
, vty
);
3667 DEFUN (show_ip_pim_assert_metric
,
3668 show_ip_pim_assert_metric_cmd
,
3669 "show ip pim [vrf NAME] assert-metric",
3674 "PIM interface assert metric\n")
3677 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3682 pim_show_assert_metric(vrf
->info
, vty
);
3687 DEFUN (show_ip_pim_assert_winner_metric
,
3688 show_ip_pim_assert_winner_metric_cmd
,
3689 "show ip pim [vrf NAME] assert-winner-metric",
3694 "PIM interface assert winner metric\n")
3697 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3702 pim_show_assert_winner_metric(vrf
->info
, vty
);
3707 DEFUN (show_ip_pim_interface
,
3708 show_ip_pim_interface_cmd
,
3709 "show ip pim [vrf NAME] interface [detail|WORD] [json]",
3714 "PIM interface information\n"
3720 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3721 uint8_t uj
= use_json(argc
, argv
);
3726 if (argv_find(argv
, argc
, "WORD", &idx
)
3727 || argv_find(argv
, argc
, "detail", &idx
))
3728 pim_show_interfaces_single(vrf
->info
, vty
, argv
[idx
]->arg
, uj
);
3730 pim_show_interfaces(vrf
->info
, vty
, uj
);
3735 DEFUN (show_ip_pim_interface_vrf_all
,
3736 show_ip_pim_interface_vrf_all_cmd
,
3737 "show ip pim vrf all interface [detail|WORD] [json]",
3742 "PIM interface information\n"
3748 uint8_t uj
= use_json(argc
, argv
);
3754 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
3758 vty_out(vty
, " \"%s\": ", vrf
->name
);
3761 vty_out(vty
, "VRF: %s\n", vrf
->name
);
3762 if (argv_find(argv
, argc
, "WORD", &idx
)
3763 || argv_find(argv
, argc
, "detail", &idx
))
3764 pim_show_interfaces_single(vrf
->info
, vty
,
3765 argv
[idx
]->arg
, uj
);
3767 pim_show_interfaces(vrf
->info
, vty
, uj
);
3770 vty_out(vty
, "}\n");
3775 DEFUN (show_ip_pim_join
,
3776 show_ip_pim_join_cmd
,
3777 "show ip pim [vrf NAME] join [json]",
3782 "PIM interface join information\n"
3786 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3787 uint8_t uj
= use_json(argc
, argv
);
3792 pim_show_join(vrf
->info
, vty
, uj
);
3797 DEFUN (show_ip_pim_join_vrf_all
,
3798 show_ip_pim_join_vrf_all_cmd
,
3799 "show ip pim vrf all join [json]",
3804 "PIM interface join information\n"
3807 uint8_t uj
= use_json(argc
, argv
);
3813 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
3817 vty_out(vty
, " \"%s\": ", vrf
->name
);
3820 vty_out(vty
, "VRF: %s\n", vrf
->name
);
3821 pim_show_join(vrf
->info
, vty
, uj
);
3824 vty_out(vty
, "}\n");
3829 DEFUN (show_ip_pim_local_membership
,
3830 show_ip_pim_local_membership_cmd
,
3831 "show ip pim [vrf NAME] local-membership [json]",
3836 "PIM interface local-membership\n"
3840 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3841 uint8_t uj
= use_json(argc
, argv
);
3846 pim_show_membership(vrf
->info
, vty
, uj
);
3851 DEFUN (show_ip_pim_neighbor
,
3852 show_ip_pim_neighbor_cmd
,
3853 "show ip pim [vrf NAME] neighbor [detail|WORD] [json]",
3858 "PIM neighbor information\n"
3860 "Name of interface or neighbor\n"
3864 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3865 uint8_t uj
= use_json(argc
, argv
);
3870 if (argv_find(argv
, argc
, "detail", &idx
)
3871 || argv_find(argv
, argc
, "WORD", &idx
))
3872 pim_show_neighbors_single(vrf
->info
, vty
, argv
[idx
]->arg
, uj
);
3874 pim_show_neighbors(vrf
->info
, vty
, uj
);
3879 DEFUN (show_ip_pim_neighbor_vrf_all
,
3880 show_ip_pim_neighbor_vrf_all_cmd
,
3881 "show ip pim vrf all neighbor [detail|WORD] [json]",
3886 "PIM neighbor information\n"
3888 "Name of interface or neighbor\n"
3892 uint8_t uj
= use_json(argc
, argv
);
3898 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
3902 vty_out(vty
, " \"%s\": ", vrf
->name
);
3905 vty_out(vty
, "VRF: %s\n", vrf
->name
);
3906 if (argv_find(argv
, argc
, "detail", &idx
)
3907 || argv_find(argv
, argc
, "WORD", &idx
))
3908 pim_show_neighbors_single(vrf
->info
, vty
,
3909 argv
[idx
]->arg
, uj
);
3911 pim_show_neighbors(vrf
->info
, vty
, uj
);
3914 vty_out(vty
, "}\n");
3919 DEFUN (show_ip_pim_secondary
,
3920 show_ip_pim_secondary_cmd
,
3921 "show ip pim [vrf NAME] secondary",
3926 "PIM neighbor addresses\n")
3929 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3934 pim_show_neighbors_secondary(vrf
->info
, vty
);
3939 DEFUN (show_ip_pim_state
,
3940 show_ip_pim_state_cmd
,
3941 "show ip pim [vrf NAME] state [A.B.C.D [A.B.C.D]] [json]",
3946 "PIM state information\n"
3947 "Unicast or Multicast address\n"
3948 "Multicast address\n"
3951 const char *src_or_group
= NULL
;
3952 const char *group
= NULL
;
3954 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
3955 uint8_t uj
= use_json(argc
, argv
);
3963 if (argv_find(argv
, argc
, "A.B.C.D", &idx
)) {
3964 src_or_group
= argv
[idx
]->arg
;
3966 group
= argv
[idx
+ 1]->arg
;
3969 pim_show_state(vrf
->info
, vty
, src_or_group
, group
, uj
);
3974 DEFUN (show_ip_pim_state_vrf_all
,
3975 show_ip_pim_state_vrf_all_cmd
,
3976 "show ip pim vrf all state [A.B.C.D [A.B.C.D]] [json]",
3981 "PIM state information\n"
3982 "Unicast or Multicast address\n"
3983 "Multicast address\n"
3986 const char *src_or_group
= NULL
;
3987 const char *group
= NULL
;
3989 uint8_t uj
= use_json(argc
, argv
);
3998 if (argv_find(argv
, argc
, "A.B.C.D", &idx
)) {
3999 src_or_group
= argv
[idx
]->arg
;
4001 group
= argv
[idx
+ 1]->arg
;
4004 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
4008 vty_out(vty
, " \"%s\": ", vrf
->name
);
4011 vty_out(vty
, "VRF: %s\n", vrf
->name
);
4012 pim_show_state(vrf
->info
, vty
, src_or_group
, group
, uj
);
4015 vty_out(vty
, "}\n");
4020 DEFUN (show_ip_pim_upstream
,
4021 show_ip_pim_upstream_cmd
,
4022 "show ip pim [vrf NAME] upstream [json]",
4027 "PIM upstream information\n"
4031 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4032 uint8_t uj
= use_json(argc
, argv
);
4037 pim_show_upstream(vrf
->info
, vty
, uj
);
4042 DEFUN (show_ip_pim_upstream_vrf_all
,
4043 show_ip_pim_upstream_vrf_all_cmd
,
4044 "show ip pim vrf all upstream [json]",
4049 "PIM upstream information\n"
4052 uint8_t uj
= use_json(argc
, argv
);
4058 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
4062 vty_out(vty
, " \"%s\": ", vrf
->name
);
4065 vty_out(vty
, "VRF: %s\n", vrf
->name
);
4066 pim_show_upstream(vrf
->info
, vty
, uj
);
4072 DEFUN (show_ip_pim_upstream_join_desired
,
4073 show_ip_pim_upstream_join_desired_cmd
,
4074 "show ip pim [vrf NAME] upstream-join-desired [json]",
4079 "PIM upstream join-desired\n"
4083 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4084 uint8_t uj
= use_json(argc
, argv
);
4089 pim_show_join_desired(vrf
->info
, vty
, uj
);
4094 DEFUN (show_ip_pim_upstream_rpf
,
4095 show_ip_pim_upstream_rpf_cmd
,
4096 "show ip pim [vrf NAME] upstream-rpf [json]",
4101 "PIM upstream source rpf\n"
4105 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4106 uint8_t uj
= use_json(argc
, argv
);
4111 pim_show_upstream_rpf(vrf
->info
, vty
, uj
);
4116 DEFUN (show_ip_pim_rp
,
4118 "show ip pim [vrf NAME] rp-info [json]",
4123 "PIM RP information\n"
4127 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4128 uint8_t uj
= use_json(argc
, argv
);
4133 pim_rp_show_information(vrf
->info
, vty
, uj
);
4138 DEFUN (show_ip_pim_rp_vrf_all
,
4139 show_ip_pim_rp_vrf_all_cmd
,
4140 "show ip pim vrf all rp-info [json]",
4145 "PIM RP information\n"
4148 uint8_t uj
= use_json(argc
, argv
);
4154 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
4158 vty_out(vty
, " \"%s\": ", vrf
->name
);
4161 vty_out(vty
, "VRF: %s\n", vrf
->name
);
4162 pim_rp_show_information(vrf
->info
, vty
, uj
);
4165 vty_out(vty
, "}\n");
4170 DEFUN (show_ip_pim_rpf
,
4171 show_ip_pim_rpf_cmd
,
4172 "show ip pim [vrf NAME] rpf [json]",
4177 "PIM cached source rpf information\n"
4181 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4182 uint8_t uj
= use_json(argc
, argv
);
4187 pim_show_rpf(vrf
->info
, vty
, uj
);
4192 DEFUN (show_ip_pim_rpf_vrf_all
,
4193 show_ip_pim_rpf_vrf_all_cmd
,
4194 "show ip pim vrf all rpf [json]",
4199 "PIM cached source rpf information\n"
4202 uint8_t uj
= use_json(argc
, argv
);
4208 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
4212 vty_out(vty
, " \"%s\": ", vrf
->name
);
4215 vty_out(vty
, "VRF: %s\n", vrf
->name
);
4216 pim_show_rpf(vrf
->info
, vty
, uj
);
4219 vty_out(vty
, "}\n");
4224 DEFUN (show_ip_pim_nexthop
,
4225 show_ip_pim_nexthop_cmd
,
4226 "show ip pim [vrf NAME] nexthop",
4231 "PIM cached nexthop rpf information\n")
4234 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4239 pim_show_nexthop(vrf
->info
, vty
);
4244 DEFUN (show_ip_pim_nexthop_lookup
,
4245 show_ip_pim_nexthop_lookup_cmd
,
4246 "show ip pim [vrf NAME] nexthop-lookup A.B.C.D A.B.C.D",
4251 "PIM cached nexthop rpf lookup\n"
4252 "Source/RP address\n"
4253 "Multicast Group address\n")
4255 struct pim_nexthop_cache pnc
;
4256 struct prefix nht_p
;
4258 struct in_addr src_addr
, grp_addr
;
4259 struct in_addr vif_source
;
4260 const char *addr_str
, *addr_str1
;
4262 struct pim_nexthop nexthop
;
4263 char nexthop_addr_str
[PREFIX_STRLEN
];
4264 char grp_str
[PREFIX_STRLEN
];
4266 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4271 argv_find(argv
, argc
, "A.B.C.D", &idx
);
4272 addr_str
= argv
[idx
]->arg
;
4273 result
= inet_pton(AF_INET
, addr_str
, &src_addr
);
4275 vty_out(vty
, "Bad unicast address %s: errno=%d: %s\n", addr_str
,
4276 errno
, safe_strerror(errno
));
4280 if (pim_is_group_224_4(src_addr
)) {
4282 "Invalid argument. Expected Valid Source Address.\n");
4286 addr_str1
= argv
[idx
+ 1]->arg
;
4287 result
= inet_pton(AF_INET
, addr_str1
, &grp_addr
);
4289 vty_out(vty
, "Bad unicast address %s: errno=%d: %s\n", addr_str
,
4290 errno
, safe_strerror(errno
));
4294 if (!pim_is_group_224_4(grp_addr
)) {
4296 "Invalid argument. Expected Valid Multicast Group Address.\n");
4300 if (!pim_rp_set_upstream_addr(vrf
->info
, &vif_source
, src_addr
,
4304 memset(&pnc
, 0, sizeof(struct pim_nexthop_cache
));
4305 nht_p
.family
= AF_INET
;
4306 nht_p
.prefixlen
= IPV4_MAX_BITLEN
;
4307 nht_p
.u
.prefix4
= vif_source
;
4308 grp
.family
= AF_INET
;
4309 grp
.prefixlen
= IPV4_MAX_BITLEN
;
4310 grp
.u
.prefix4
= grp_addr
;
4311 memset(&nexthop
, 0, sizeof(nexthop
));
4313 if (pim_find_or_track_nexthop(vrf
->info
, &nht_p
, NULL
, NULL
, &pnc
))
4314 result
= pim_ecmp_nexthop_search(vrf
->info
, &pnc
, &nexthop
,
4317 result
= pim_ecmp_nexthop_lookup(vrf
->info
, &nexthop
,
4318 vif_source
, &nht_p
, &grp
, 0);
4322 "Nexthop Lookup failed, no usable routes returned.\n");
4326 pim_addr_dump("<grp?>", &grp
, grp_str
, sizeof(grp_str
));
4327 pim_addr_dump("<nexthop?>", &nexthop
.mrib_nexthop_addr
,
4328 nexthop_addr_str
, sizeof(nexthop_addr_str
));
4329 vty_out(vty
, "Group %s --- Nexthop %s Interface %s \n", grp_str
,
4330 nexthop_addr_str
, nexthop
.interface
->name
);
4335 DEFUN (show_ip_pim_interface_traffic
,
4336 show_ip_pim_interface_traffic_cmd
,
4337 "show ip pim [vrf NAME] interface traffic [WORD] [json]",
4342 "PIM interface information\n"
4343 "Protocol Packet counters\n"
4348 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4349 uint8_t uj
= use_json(argc
, argv
);
4354 if (argv_find(argv
, argc
, "WORD", &idx
))
4355 pim_show_interface_traffic_single(vrf
->info
, vty
,
4356 argv
[idx
]->arg
, uj
);
4358 pim_show_interface_traffic(vrf
->info
, vty
, uj
);
4363 static void show_multicast_interfaces(struct pim_instance
*pim
, struct vty
*vty
)
4365 struct interface
*ifp
;
4370 "Interface Address ifi Vif PktsIn PktsOut BytesIn BytesOut\n");
4372 FOR_ALL_INTERFACES (pim
->vrf
, ifp
) {
4373 struct pim_interface
*pim_ifp
;
4374 struct in_addr ifaddr
;
4375 struct sioc_vif_req vreq
;
4377 pim_ifp
= ifp
->info
;
4382 memset(&vreq
, 0, sizeof(vreq
));
4383 vreq
.vifi
= pim_ifp
->mroute_vif_index
;
4385 if (ioctl(pim
->mroute_socket
, SIOCGETVIFCNT
, &vreq
)) {
4387 "ioctl(SIOCGETVIFCNT=%lu) failure for interface %s vif_index=%d: errno=%d: %s",
4388 (unsigned long)SIOCGETVIFCNT
, ifp
->name
,
4389 pim_ifp
->mroute_vif_index
, errno
,
4390 safe_strerror(errno
));
4393 ifaddr
= pim_ifp
->primary_address
;
4395 vty_out(vty
, "%-12s %-15s %3d %3d %7lu %7lu %10lu %10lu\n",
4396 ifp
->name
, inet_ntoa(ifaddr
), ifp
->ifindex
,
4397 pim_ifp
->mroute_vif_index
, (unsigned long)vreq
.icount
,
4398 (unsigned long)vreq
.ocount
, (unsigned long)vreq
.ibytes
,
4399 (unsigned long)vreq
.obytes
);
4403 static void pim_cmd_show_ip_multicast_helper(struct pim_instance
*pim
,
4406 struct vrf
*vrf
= pim
->vrf
;
4407 time_t now
= pim_time_monotonic_sec();
4412 vty_out(vty
, "Mroute socket descriptor:");
4414 vty_out(vty
, " %d(%s)\n", pim
->mroute_socket
, vrf
->name
);
4416 pim_time_uptime(uptime
, sizeof(uptime
),
4417 now
- pim
->mroute_socket_creation
);
4418 vty_out(vty
, "Mroute socket uptime: %s\n", uptime
);
4422 pim_zebra_zclient_update(vty
);
4423 pim_zlookup_show_ip_multicast(vty
);
4426 vty_out(vty
, "Maximum highest VifIndex: %d\n", PIM_MAX_USABLE_VIFS
);
4429 vty_out(vty
, "Upstream Join Timer: %d secs\n", qpim_t_periodic
);
4430 vty_out(vty
, "Join/Prune Holdtime: %d secs\n", PIM_JP_HOLDTIME
);
4431 vty_out(vty
, "PIM ECMP: %s\n", pim
->ecmp_enable
? "Enable" : "Disable");
4432 vty_out(vty
, "PIM ECMP Rebalance: %s\n",
4433 pim
->ecmp_rebalance_enable
? "Enable" : "Disable");
4437 show_rpf_refresh_stats(vty
, pim
, now
, NULL
);
4441 show_scan_oil_stats(pim
, vty
, now
);
4443 show_multicast_interfaces(pim
, vty
);
4446 DEFUN (show_ip_multicast
,
4447 show_ip_multicast_cmd
,
4448 "show ip multicast [vrf NAME]",
4452 "Multicast global information\n")
4455 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4460 pim_cmd_show_ip_multicast_helper(vrf
->info
, vty
);
4465 DEFUN (show_ip_multicast_vrf_all
,
4466 show_ip_multicast_vrf_all_cmd
,
4467 "show ip multicast vrf all",
4471 "Multicast global information\n")
4473 uint8_t uj
= use_json(argc
, argv
);
4479 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
4483 vty_out(vty
, " \"%s\": ", vrf
->name
);
4486 vty_out(vty
, "VRF: %s\n", vrf
->name
);
4487 pim_cmd_show_ip_multicast_helper(vrf
->info
, vty
);
4490 vty_out(vty
, "}\n");
4495 static void show_mroute(struct pim_instance
*pim
, struct vty
*vty
, bool fill
,
4498 struct listnode
*node
;
4499 struct channel_oil
*c_oil
;
4500 struct static_route
*s_route
;
4502 json_object
*json
= NULL
;
4503 json_object
*json_group
= NULL
;
4504 json_object
*json_source
= NULL
;
4505 json_object
*json_oil
= NULL
;
4506 json_object
*json_ifp_out
= NULL
;
4509 char grp_str
[INET_ADDRSTRLEN
];
4510 char src_str
[INET_ADDRSTRLEN
];
4511 char in_ifname
[INTERFACE_NAMSIZ
+ 1];
4512 char out_ifname
[INTERFACE_NAMSIZ
+ 1];
4514 struct interface
*ifp_in
;
4518 json
= json_object_new_object();
4521 "Source Group Proto Input Output TTL Uptime\n");
4524 now
= pim_time_monotonic_sec();
4526 /* print list of PIM and IGMP routes */
4527 for (ALL_LIST_ELEMENTS_RO(pim
->channel_oil_list
, node
, c_oil
)) {
4530 if (!c_oil
->installed
&& !uj
)
4533 pim_inet4_dump("<group?>", c_oil
->oil
.mfcc_mcastgrp
, grp_str
,
4535 pim_inet4_dump("<source?>", c_oil
->oil
.mfcc_origin
, src_str
,
4537 ifp_in
= pim_if_find_by_vif_index(pim
, c_oil
->oil
.mfcc_parent
);
4540 strcpy(in_ifname
, ifp_in
->name
);
4542 strcpy(in_ifname
, "<iif?>");
4546 /* Find the group, create it if it doesn't exist */
4547 json_object_object_get_ex(json
, grp_str
, &json_group
);
4550 json_group
= json_object_new_object();
4551 json_object_object_add(json
, grp_str
,
4555 /* Find the source nested under the group, create it if
4556 * it doesn't exist */
4557 json_object_object_get_ex(json_group
, src_str
,
4561 json_source
= json_object_new_object();
4562 json_object_object_add(json_group
, src_str
,
4566 /* Find the inbound interface nested under the source,
4567 * create it if it doesn't exist */
4568 json_object_int_add(json_source
, "installed",
4570 json_object_int_add(json_source
, "refCount",
4571 c_oil
->oil_ref_count
);
4572 json_object_int_add(json_source
, "oilSize",
4574 json_object_int_add(json_source
, "OilInheritedRescan",
4575 c_oil
->oil_inherited_rescan
);
4576 json_object_string_add(json_source
, "iif", in_ifname
);
4580 for (oif_vif_index
= 0; oif_vif_index
< MAXVIFS
;
4582 struct interface
*ifp_out
;
4583 char oif_uptime
[10];
4586 ttl
= c_oil
->oil
.mfcc_ttls
[oif_vif_index
];
4590 ifp_out
= pim_if_find_by_vif_index(pim
, oif_vif_index
);
4592 oif_uptime
, sizeof(oif_uptime
),
4593 now
- c_oil
->oif_creation
[oif_vif_index
]);
4597 strcpy(out_ifname
, ifp_out
->name
);
4599 strcpy(out_ifname
, "<oif?>");
4602 json_ifp_out
= json_object_new_object();
4603 json_object_string_add(json_ifp_out
, "source",
4605 json_object_string_add(json_ifp_out
, "group",
4608 if (c_oil
->oif_flags
[oif_vif_index
]
4609 & PIM_OIF_FLAG_PROTO_PIM
)
4610 json_object_boolean_true_add(
4611 json_ifp_out
, "protocolPim");
4613 if (c_oil
->oif_flags
[oif_vif_index
]
4614 & PIM_OIF_FLAG_PROTO_IGMP
)
4615 json_object_boolean_true_add(
4616 json_ifp_out
, "protocolIgmp");
4618 if (c_oil
->oif_flags
[oif_vif_index
]
4619 & PIM_OIF_FLAG_PROTO_SOURCE
)
4620 json_object_boolean_true_add(
4621 json_ifp_out
, "protocolSource");
4623 if (c_oil
->oif_flags
[oif_vif_index
]
4624 & PIM_OIF_FLAG_PROTO_STAR
)
4625 json_object_boolean_true_add(
4627 "protocolInherited");
4629 json_object_string_add(json_ifp_out
,
4632 json_object_int_add(json_ifp_out
, "iVifI",
4633 c_oil
->oil
.mfcc_parent
);
4634 json_object_string_add(json_ifp_out
,
4635 "outboundInterface",
4637 json_object_int_add(json_ifp_out
, "oVifI",
4639 json_object_int_add(json_ifp_out
, "ttl", ttl
);
4640 json_object_string_add(json_ifp_out
, "upTime",
4643 json_oil
= json_object_new_object();
4644 json_object_object_add(json_source
,
4647 json_object_object_add(json_oil
, out_ifname
,
4650 if (c_oil
->oif_flags
[oif_vif_index
]
4651 & PIM_OIF_FLAG_PROTO_PIM
) {
4652 strcpy(proto
, "PIM");
4655 if (c_oil
->oif_flags
[oif_vif_index
]
4656 & PIM_OIF_FLAG_PROTO_IGMP
) {
4657 strcpy(proto
, "IGMP");
4660 if (c_oil
->oif_flags
[oif_vif_index
]
4661 & PIM_OIF_FLAG_PROTO_SOURCE
) {
4662 strcpy(proto
, "SRC");
4665 if (c_oil
->oif_flags
[oif_vif_index
]
4666 & PIM_OIF_FLAG_PROTO_STAR
) {
4667 strcpy(proto
, "STAR");
4671 "%-15s %-15s %-6s %-10s %-10s %-3d %8s\n",
4672 src_str
, grp_str
, proto
, in_ifname
,
4673 out_ifname
, ttl
, oif_uptime
);
4678 in_ifname
[0] = '\0';
4684 if (!uj
&& !found_oif
) {
4685 vty_out(vty
, "%-15s %-15s %-6s %-10s %-10s %-3d %8s\n",
4686 src_str
, grp_str
, "none", in_ifname
, "none", 0,
4691 /* Print list of static routes */
4692 for (ALL_LIST_ELEMENTS_RO(pim
->static_routes
, node
, s_route
)) {
4695 if (!s_route
->c_oil
.installed
)
4698 pim_inet4_dump("<group?>", s_route
->group
, grp_str
,
4700 pim_inet4_dump("<source?>", s_route
->source
, src_str
,
4702 ifp_in
= pim_if_find_by_vif_index(pim
, s_route
->iif
);
4706 strcpy(in_ifname
, ifp_in
->name
);
4708 strcpy(in_ifname
, "<iif?>");
4712 /* Find the group, create it if it doesn't exist */
4713 json_object_object_get_ex(json
, grp_str
, &json_group
);
4716 json_group
= json_object_new_object();
4717 json_object_object_add(json
, grp_str
,
4721 /* Find the source nested under the group, create it if
4722 * it doesn't exist */
4723 json_object_object_get_ex(json_group
, src_str
,
4727 json_source
= json_object_new_object();
4728 json_object_object_add(json_group
, src_str
,
4732 json_object_string_add(json_source
, "iif", in_ifname
);
4735 strcpy(proto
, "STATIC");
4738 for (oif_vif_index
= 0; oif_vif_index
< MAXVIFS
;
4740 struct interface
*ifp_out
;
4741 char oif_uptime
[10];
4744 ttl
= s_route
->oif_ttls
[oif_vif_index
];
4748 ifp_out
= pim_if_find_by_vif_index(pim
, oif_vif_index
);
4750 oif_uptime
, sizeof(oif_uptime
),
4753 .oif_creation
[oif_vif_index
]);
4757 strcpy(out_ifname
, ifp_out
->name
);
4759 strcpy(out_ifname
, "<oif?>");
4762 json_ifp_out
= json_object_new_object();
4763 json_object_string_add(json_ifp_out
, "source",
4765 json_object_string_add(json_ifp_out
, "group",
4767 json_object_boolean_true_add(json_ifp_out
,
4769 json_object_string_add(json_ifp_out
,
4772 json_object_int_add(
4773 json_ifp_out
, "iVifI",
4774 s_route
->c_oil
.oil
.mfcc_parent
);
4775 json_object_string_add(json_ifp_out
,
4776 "outboundInterface",
4778 json_object_int_add(json_ifp_out
, "oVifI",
4780 json_object_int_add(json_ifp_out
, "ttl", ttl
);
4781 json_object_string_add(json_ifp_out
, "upTime",
4784 json_oil
= json_object_new_object();
4785 json_object_object_add(json_source
,
4788 json_object_object_add(json_oil
, out_ifname
,
4792 "%-15s %-15s %-6s %-10s %-10s %-3d %8s %s\n",
4793 src_str
, grp_str
, proto
, in_ifname
,
4794 out_ifname
, ttl
, oif_uptime
,
4796 if (first
&& !fill
) {
4799 in_ifname
[0] = '\0';
4805 if (!uj
&& !found_oif
) {
4807 "%-15s %-15s %-6s %-10s %-10s %-3d %8s %s\n",
4808 src_str
, grp_str
, proto
, in_ifname
, "none", 0,
4809 "--:--:--", pim
->vrf
->name
);
4814 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
4815 json
, JSON_C_TO_STRING_PRETTY
));
4816 json_object_free(json
);
4820 DEFUN (show_ip_mroute
,
4822 "show ip mroute [vrf NAME] [fill] [json]",
4827 "Fill in Assumed data\n"
4830 uint8_t uj
= use_json(argc
, argv
);
4833 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4838 if (argv_find(argv
, argc
, "fill", &idx
))
4841 show_mroute(vrf
->info
, vty
, fill
, uj
);
4845 DEFUN (show_ip_mroute_vrf_all
,
4846 show_ip_mroute_vrf_all_cmd
,
4847 "show ip mroute vrf all [fill] [json]",
4852 "Fill in Assumed data\n"
4855 uint8_t uj
= use_json(argc
, argv
);
4861 if (argv_find(argv
, argc
, "fill", &idx
))
4866 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
4870 vty_out(vty
, " \"%s\": ", vrf
->name
);
4873 vty_out(vty
, "VRF: %s\n", vrf
->name
);
4874 show_mroute(vrf
->info
, vty
, fill
, uj
);
4877 vty_out(vty
, "}\n");
4882 static void show_mroute_count(struct pim_instance
*pim
, struct vty
*vty
)
4884 struct listnode
*node
;
4885 struct channel_oil
*c_oil
;
4886 struct static_route
*s_route
;
4891 "Source Group LastUsed Packets Bytes WrongIf \n");
4893 /* Print PIM and IGMP route counts */
4894 for (ALL_LIST_ELEMENTS_RO(pim
->channel_oil_list
, node
, c_oil
)) {
4895 char group_str
[INET_ADDRSTRLEN
];
4896 char source_str
[INET_ADDRSTRLEN
];
4898 if (!c_oil
->installed
)
4901 pim_mroute_update_counters(c_oil
);
4903 pim_inet4_dump("<group?>", c_oil
->oil
.mfcc_mcastgrp
, group_str
,
4905 pim_inet4_dump("<source?>", c_oil
->oil
.mfcc_origin
, source_str
,
4906 sizeof(source_str
));
4908 vty_out(vty
, "%-15s %-15s %-8llu %-7ld %-10ld %-7ld\n",
4909 source_str
, group_str
, c_oil
->cc
.lastused
/ 100,
4910 c_oil
->cc
.pktcnt
, c_oil
->cc
.bytecnt
,
4911 c_oil
->cc
.wrong_if
);
4914 for (ALL_LIST_ELEMENTS_RO(pim
->static_routes
, node
, s_route
)) {
4915 char group_str
[INET_ADDRSTRLEN
];
4916 char source_str
[INET_ADDRSTRLEN
];
4918 if (!s_route
->c_oil
.installed
)
4921 pim_mroute_update_counters(&s_route
->c_oil
);
4923 pim_inet4_dump("<group?>", s_route
->c_oil
.oil
.mfcc_mcastgrp
,
4924 group_str
, sizeof(group_str
));
4925 pim_inet4_dump("<source?>", s_route
->c_oil
.oil
.mfcc_origin
,
4926 source_str
, sizeof(source_str
));
4928 vty_out(vty
, "%-15s %-15s %-8llu %-7ld %-10ld %-7ld\n",
4929 source_str
, group_str
, s_route
->c_oil
.cc
.lastused
,
4930 s_route
->c_oil
.cc
.pktcnt
, s_route
->c_oil
.cc
.bytecnt
,
4931 s_route
->c_oil
.cc
.wrong_if
);
4935 DEFUN (show_ip_mroute_count
,
4936 show_ip_mroute_count_cmd
,
4937 "show ip mroute [vrf NAME] count",
4942 "Route and packet count data\n")
4945 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4950 show_mroute_count(vrf
->info
, vty
);
4954 DEFUN (show_ip_mroute_count_vrf_all
,
4955 show_ip_mroute_count_vrf_all_cmd
,
4956 "show ip mroute vrf all count",
4961 "Route and packet count data\n")
4963 uint8_t uj
= use_json(argc
, argv
);
4969 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
4973 vty_out(vty
, " \"%s\": ", vrf
->name
);
4976 vty_out(vty
, "VRF: %s\n", vrf
->name
);
4977 show_mroute_count(vrf
->info
, vty
);
4980 vty_out(vty
, "}\n");
4987 "show ip rib [vrf NAME] A.B.C.D",
4992 "Unicast address\n")
4995 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
4996 struct in_addr addr
;
4997 const char *addr_str
;
4998 struct pim_nexthop nexthop
;
4999 char nexthop_addr_str
[PREFIX_STRLEN
];
5005 memset(&nexthop
, 0, sizeof(nexthop
));
5006 argv_find(argv
, argc
, "A.B.C.D", &idx
);
5007 addr_str
= argv
[idx
]->arg
;
5008 result
= inet_pton(AF_INET
, addr_str
, &addr
);
5010 vty_out(vty
, "Bad unicast address %s: errno=%d: %s\n", addr_str
,
5011 errno
, safe_strerror(errno
));
5015 if (pim_nexthop_lookup(vrf
->info
, &nexthop
, addr
, 0)) {
5017 "Failure querying RIB nexthop for unicast address %s\n",
5023 "Address NextHop Interface Metric Preference\n");
5025 pim_addr_dump("<nexthop?>", &nexthop
.mrib_nexthop_addr
,
5026 nexthop_addr_str
, sizeof(nexthop_addr_str
));
5028 vty_out(vty
, "%-15s %-15s %-9s %6d %10d\n", addr_str
, nexthop_addr_str
,
5029 nexthop
.interface
? nexthop
.interface
->name
: "<ifname?>",
5030 nexthop
.mrib_route_metric
, nexthop
.mrib_metric_preference
);
5035 static void show_ssmpingd(struct pim_instance
*pim
, struct vty
*vty
)
5037 struct listnode
*node
;
5038 struct ssmpingd_sock
*ss
;
5042 "Source Socket Address Port Uptime Requests\n");
5044 if (!pim
->ssmpingd_list
)
5047 now
= pim_time_monotonic_sec();
5049 for (ALL_LIST_ELEMENTS_RO(pim
->ssmpingd_list
, node
, ss
)) {
5050 char source_str
[INET_ADDRSTRLEN
];
5052 struct sockaddr_in bind_addr
;
5053 socklen_t len
= sizeof(bind_addr
);
5054 char bind_addr_str
[INET_ADDRSTRLEN
];
5056 pim_inet4_dump("<src?>", ss
->source_addr
, source_str
,
5057 sizeof(source_str
));
5059 if (pim_socket_getsockname(
5060 ss
->sock_fd
, (struct sockaddr
*)&bind_addr
, &len
)) {
5062 "%% Failure reading socket name for ssmpingd source %s on fd=%d\n",
5063 source_str
, ss
->sock_fd
);
5066 pim_inet4_dump("<addr?>", bind_addr
.sin_addr
, bind_addr_str
,
5067 sizeof(bind_addr_str
));
5068 pim_time_uptime(ss_uptime
, sizeof(ss_uptime
),
5069 now
- ss
->creation
);
5071 vty_out(vty
, "%-15s %6d %-15s %5d %8s %8lld\n", source_str
,
5072 ss
->sock_fd
, bind_addr_str
, ntohs(bind_addr
.sin_port
),
5073 ss_uptime
, (long long)ss
->requests
);
5077 DEFUN (show_ip_ssmpingd
,
5078 show_ip_ssmpingd_cmd
,
5079 "show ip ssmpingd [vrf NAME]",
5086 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
5091 show_ssmpingd(vrf
->info
, vty
);
5095 static int pim_rp_cmd_worker(struct pim_instance
*pim
, struct vty
*vty
,
5096 const char *rp
, const char *group
,
5101 result
= pim_rp_new(pim
, rp
, group
, plist
);
5103 if (result
== PIM_MALLOC_FAIL
) {
5104 vty_out(vty
, "%% Out of memory\n");
5105 return CMD_WARNING_CONFIG_FAILED
;
5108 if (result
== PIM_GROUP_BAD_ADDRESS
) {
5109 vty_out(vty
, "%% Bad group address specified: %s\n", group
);
5110 return CMD_WARNING_CONFIG_FAILED
;
5113 if (result
== PIM_RP_BAD_ADDRESS
) {
5114 vty_out(vty
, "%% Bad RP address specified: %s\n", rp
);
5115 return CMD_WARNING_CONFIG_FAILED
;
5118 if (result
== PIM_RP_NO_PATH
) {
5119 vty_out(vty
, "%% No Path to RP address specified: %s\n", rp
);
5123 if (result
== PIM_GROUP_OVERLAP
) {
5125 "%% Group range specified cannot exact match another\n");
5126 return CMD_WARNING_CONFIG_FAILED
;
5129 if (result
== PIM_GROUP_PFXLIST_OVERLAP
) {
5131 "%% This group is already covered by a RP prefix-list\n");
5132 return CMD_WARNING_CONFIG_FAILED
;
5135 if (result
== PIM_RP_PFXLIST_IN_USE
) {
5137 "%% The same prefix-list cannot be applied to multiple RPs\n");
5138 return CMD_WARNING_CONFIG_FAILED
;
5144 static int pim_cmd_spt_switchover(struct pim_instance
*pim
,
5145 enum pim_spt_switchover spt
,
5148 pim
->spt
.switchover
= spt
;
5150 switch (pim
->spt
.switchover
) {
5151 case PIM_SPT_IMMEDIATE
:
5153 XFREE(MTYPE_PIM_SPT_PLIST_NAME
, pim
->spt
.plist
);
5155 pim_upstream_add_lhr_star_pimreg(pim
);
5157 case PIM_SPT_INFINITY
:
5158 pim_upstream_remove_lhr_star_pimreg(pim
, plist
);
5161 XFREE(MTYPE_PIM_SPT_PLIST_NAME
, pim
->spt
.plist
);
5165 XSTRDUP(MTYPE_PIM_SPT_PLIST_NAME
, plist
);
5172 DEFUN (ip_pim_spt_switchover_infinity
,
5173 ip_pim_spt_switchover_infinity_cmd
,
5174 "ip pim spt-switchover infinity-and-beyond",
5178 "Never switch to SPT Tree\n")
5180 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5181 return pim_cmd_spt_switchover(pim
, PIM_SPT_INFINITY
, NULL
);
5184 DEFUN (ip_pim_spt_switchover_infinity_plist
,
5185 ip_pim_spt_switchover_infinity_plist_cmd
,
5186 "ip pim spt-switchover infinity-and-beyond prefix-list WORD",
5190 "Never switch to SPT Tree\n"
5191 "Prefix-List to control which groups to switch\n"
5192 "Prefix-List name\n")
5194 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5195 return pim_cmd_spt_switchover(pim
, PIM_SPT_INFINITY
, argv
[5]->arg
);
5198 DEFUN (no_ip_pim_spt_switchover_infinity
,
5199 no_ip_pim_spt_switchover_infinity_cmd
,
5200 "no ip pim spt-switchover infinity-and-beyond",
5205 "Never switch to SPT Tree\n")
5207 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5208 return pim_cmd_spt_switchover(pim
, PIM_SPT_IMMEDIATE
, NULL
);
5211 DEFUN (no_ip_pim_spt_switchover_infinity_plist
,
5212 no_ip_pim_spt_switchover_infinity_plist_cmd
,
5213 "no ip pim spt-switchover infinity-and-beyond prefix-list WORD",
5218 "Never switch to SPT Tree\n"
5219 "Prefix-List to control which groups to switch\n"
5220 "Prefix-List name\n")
5222 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5223 return pim_cmd_spt_switchover(pim
, PIM_SPT_IMMEDIATE
, NULL
);
5226 DEFUN (ip_pim_joinprune_time
,
5227 ip_pim_joinprune_time_cmd
,
5228 "ip pim join-prune-interval (60-600)",
5230 "pim multicast routing\n"
5231 "Join Prune Send Interval\n"
5234 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5235 qpim_t_periodic
= atoi(argv
[3]->arg
);
5239 DEFUN (no_ip_pim_joinprune_time
,
5240 no_ip_pim_joinprune_time_cmd
,
5241 "no ip pim join-prune-interval (60-600)",
5244 "pim multicast routing\n"
5245 "Join Prune Send Interval\n"
5248 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5249 qpim_t_periodic
= PIM_DEFAULT_T_PERIODIC
;
5253 DEFUN (ip_pim_register_suppress
,
5254 ip_pim_register_suppress_cmd
,
5255 "ip pim register-suppress-time (5-60000)",
5257 "pim multicast routing\n"
5258 "Register Suppress Timer\n"
5261 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5262 qpim_register_suppress_time
= atoi(argv
[3]->arg
);
5266 DEFUN (no_ip_pim_register_suppress
,
5267 no_ip_pim_register_suppress_cmd
,
5268 "no ip pim register-suppress-time (5-60000)",
5271 "pim multicast routing\n"
5272 "Register Suppress Timer\n"
5275 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5276 qpim_register_suppress_time
= PIM_REGISTER_SUPPRESSION_TIME_DEFAULT
;
5280 DEFUN (ip_pim_rp_keep_alive
,
5281 ip_pim_rp_keep_alive_cmd
,
5282 "ip pim rp keep-alive-timer (31-60000)",
5284 "pim multicast routing\n"
5286 "Keep alive Timer\n"
5289 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5290 pim
->rp_keep_alive_time
= atoi(argv
[4]->arg
);
5294 DEFUN (no_ip_pim_rp_keep_alive
,
5295 no_ip_pim_rp_keep_alive_cmd
,
5296 "no ip pim rp keep-alive-timer (31-60000)",
5299 "pim multicast routing\n"
5301 "Keep alive Timer\n"
5304 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5305 pim
->rp_keep_alive_time
= PIM_KEEPALIVE_PERIOD
;
5309 DEFUN (ip_pim_keep_alive
,
5310 ip_pim_keep_alive_cmd
,
5311 "ip pim keep-alive-timer (31-60000)",
5313 "pim multicast routing\n"
5314 "Keep alive Timer\n"
5317 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5318 pim
->keep_alive_time
= atoi(argv
[3]->arg
);
5322 DEFUN (no_ip_pim_keep_alive
,
5323 no_ip_pim_keep_alive_cmd
,
5324 "no ip pim keep-alive-timer (31-60000)",
5327 "pim multicast routing\n"
5328 "Keep alive Timer\n"
5331 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5332 pim
->keep_alive_time
= PIM_KEEPALIVE_PERIOD
;
5336 DEFUN (ip_pim_packets
,
5338 "ip pim packets (1-100)",
5340 "pim multicast routing\n"
5341 "packets to process at one time per fd\n"
5342 "Number of packets\n")
5344 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5345 qpim_packet_process
= atoi(argv
[3]->arg
);
5349 DEFUN (no_ip_pim_packets
,
5350 no_ip_pim_packets_cmd
,
5351 "no ip pim packets (1-100)",
5354 "pim multicast routing\n"
5355 "packets to process at one time per fd\n"
5356 "Number of packets\n")
5358 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5359 qpim_packet_process
= PIM_DEFAULT_PACKET_PROCESS
;
5363 DEFUN (ip_pim_v6_secondary
,
5364 ip_pim_v6_secondary_cmd
,
5365 "ip pim send-v6-secondary",
5367 "pim multicast routing\n"
5368 "Send v6 secondary addresses\n")
5370 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5371 pim
->send_v6_secondary
= 1;
5376 DEFUN (no_ip_pim_v6_secondary
,
5377 no_ip_pim_v6_secondary_cmd
,
5378 "no ip pim send-v6-secondary",
5381 "pim multicast routing\n"
5382 "Send v6 secondary addresses\n")
5384 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5385 pim
->send_v6_secondary
= 0;
5392 "ip pim rp A.B.C.D [A.B.C.D/M]",
5394 "pim multicast routing\n"
5396 "ip address of RP\n"
5397 "Group Address range to cover\n")
5399 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5402 if (argc
== (idx_ipv4
+ 1))
5403 return pim_rp_cmd_worker(pim
, vty
, argv
[idx_ipv4
]->arg
, NULL
,
5406 return pim_rp_cmd_worker(pim
, vty
, argv
[idx_ipv4
]->arg
,
5407 argv
[idx_ipv4
+ 1]->arg
, NULL
);
5410 DEFUN (ip_pim_rp_prefix_list
,
5411 ip_pim_rp_prefix_list_cmd
,
5412 "ip pim rp A.B.C.D prefix-list WORD",
5414 "pim multicast routing\n"
5416 "ip address of RP\n"
5417 "group prefix-list filter\n"
5418 "Name of a prefix-list\n")
5420 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5421 return pim_rp_cmd_worker(pim
, vty
, argv
[3]->arg
, NULL
, argv
[5]->arg
);
5424 static int pim_no_rp_cmd_worker(struct pim_instance
*pim
, struct vty
*vty
,
5425 const char *rp
, const char *group
,
5428 int result
= pim_rp_del(pim
, rp
, group
, plist
);
5430 if (result
== PIM_GROUP_BAD_ADDRESS
) {
5431 vty_out(vty
, "%% Bad group address specified: %s\n", group
);
5432 return CMD_WARNING_CONFIG_FAILED
;
5435 if (result
== PIM_RP_BAD_ADDRESS
) {
5436 vty_out(vty
, "%% Bad RP address specified: %s\n", rp
);
5437 return CMD_WARNING_CONFIG_FAILED
;
5440 if (result
== PIM_RP_NOT_FOUND
) {
5441 vty_out(vty
, "%% Unable to find specified RP\n");
5442 return CMD_WARNING_CONFIG_FAILED
;
5448 DEFUN (no_ip_pim_rp
,
5450 "no ip pim rp A.B.C.D [A.B.C.D/M]",
5453 "pim multicast routing\n"
5455 "ip address of RP\n"
5456 "Group Address range to cover\n")
5458 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5459 int idx_ipv4
= 4, idx_group
= 0;
5461 if (argv_find(argv
, argc
, "A.B.C.D/M", &idx_group
))
5462 return pim_no_rp_cmd_worker(pim
, vty
, argv
[idx_ipv4
]->arg
,
5463 argv
[idx_group
]->arg
, NULL
);
5465 return pim_no_rp_cmd_worker(pim
, vty
, argv
[idx_ipv4
]->arg
, NULL
,
5469 DEFUN (no_ip_pim_rp_prefix_list
,
5470 no_ip_pim_rp_prefix_list_cmd
,
5471 "no ip pim rp A.B.C.D prefix-list WORD",
5474 "pim multicast routing\n"
5476 "ip address of RP\n"
5477 "group prefix-list filter\n"
5478 "Name of a prefix-list\n")
5480 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5481 return pim_no_rp_cmd_worker(pim
, vty
, argv
[4]->arg
, NULL
, argv
[6]->arg
);
5484 static int pim_ssm_cmd_worker(struct pim_instance
*pim
, struct vty
*vty
,
5487 int result
= pim_ssm_range_set(pim
, pim
->vrf_id
, plist
);
5489 if (result
== PIM_SSM_ERR_NONE
)
5493 case PIM_SSM_ERR_NO_VRF
:
5494 vty_out(vty
, "%% VRF doesn't exist\n");
5496 case PIM_SSM_ERR_DUP
:
5497 vty_out(vty
, "%% duplicate config\n");
5500 vty_out(vty
, "%% ssm range config failed\n");
5503 return CMD_WARNING_CONFIG_FAILED
;
5506 DEFUN (ip_pim_ssm_prefix_list
,
5507 ip_pim_ssm_prefix_list_cmd
,
5508 "ip pim ssm prefix-list WORD",
5510 "pim multicast routing\n"
5511 "Source Specific Multicast\n"
5512 "group range prefix-list filter\n"
5513 "Name of a prefix-list\n")
5515 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5516 return pim_ssm_cmd_worker(pim
, vty
, argv
[4]->arg
);
5519 DEFUN (no_ip_pim_ssm_prefix_list
,
5520 no_ip_pim_ssm_prefix_list_cmd
,
5521 "no ip pim ssm prefix-list",
5524 "pim multicast routing\n"
5525 "Source Specific Multicast\n"
5526 "group range prefix-list filter\n")
5528 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5529 return pim_ssm_cmd_worker(pim
, vty
, NULL
);
5532 DEFUN (no_ip_pim_ssm_prefix_list_name
,
5533 no_ip_pim_ssm_prefix_list_name_cmd
,
5534 "no ip pim ssm prefix-list WORD",
5537 "pim multicast routing\n"
5538 "Source Specific Multicast\n"
5539 "group range prefix-list filter\n"
5540 "Name of a prefix-list\n")
5542 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5543 struct pim_ssm
*ssm
= pim
->ssm_info
;
5545 if (ssm
->plist_name
&& !strcmp(ssm
->plist_name
, argv
[5]->arg
))
5546 return pim_ssm_cmd_worker(pim
, vty
, NULL
);
5548 vty_out(vty
, "%% pim ssm prefix-list %s doesn't exist\n", argv
[5]->arg
);
5550 return CMD_WARNING_CONFIG_FAILED
;
5553 static void ip_pim_ssm_show_group_range(struct pim_instance
*pim
,
5554 struct vty
*vty
, uint8_t uj
)
5556 struct pim_ssm
*ssm
= pim
->ssm_info
;
5557 const char *range_str
=
5558 ssm
->plist_name
? ssm
->plist_name
: PIM_SSM_STANDARD_RANGE
;
5562 json
= json_object_new_object();
5563 json_object_string_add(json
, "ssmGroups", range_str
);
5564 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
5565 json
, JSON_C_TO_STRING_PRETTY
));
5566 json_object_free(json
);
5568 vty_out(vty
, "SSM group range : %s\n", range_str
);
5571 DEFUN (show_ip_pim_ssm_range
,
5572 show_ip_pim_ssm_range_cmd
,
5573 "show ip pim [vrf NAME] group-type [json]",
5582 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
5583 uint8_t uj
= use_json(argc
, argv
);
5588 ip_pim_ssm_show_group_range(vrf
->info
, vty
, uj
);
5593 static void ip_pim_ssm_show_group_type(struct pim_instance
*pim
,
5594 struct vty
*vty
, uint8_t uj
,
5597 struct in_addr group_addr
;
5598 const char *type_str
;
5601 result
= inet_pton(AF_INET
, group
, &group_addr
);
5603 type_str
= "invalid";
5605 if (pim_is_group_224_4(group_addr
))
5607 pim_is_grp_ssm(pim
, group_addr
) ? "SSM" : "ASM";
5609 type_str
= "not-multicast";
5614 json
= json_object_new_object();
5615 json_object_string_add(json
, "groupType", type_str
);
5616 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
5617 json
, JSON_C_TO_STRING_PRETTY
));
5618 json_object_free(json
);
5620 vty_out(vty
, "Group type : %s\n", type_str
);
5623 DEFUN (show_ip_pim_group_type
,
5624 show_ip_pim_group_type_cmd
,
5625 "show ip pim [vrf NAME] group-type A.B.C.D [json]",
5630 "multicast group type\n"
5635 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
5636 uint8_t uj
= use_json(argc
, argv
);
5641 argv_find(argv
, argc
, "A.B.C.D", &idx
);
5642 ip_pim_ssm_show_group_type(vrf
->info
, vty
, uj
, argv
[idx
]->arg
);
5647 DEFUN_HIDDEN (ip_multicast_routing
,
5648 ip_multicast_routing_cmd
,
5649 "ip multicast-routing",
5651 "Enable IP multicast forwarding\n")
5656 DEFUN_HIDDEN (no_ip_multicast_routing
,
5657 no_ip_multicast_routing_cmd
,
5658 "no ip multicast-routing",
5661 "Enable IP multicast forwarding\n")
5664 "Command is Disabled and will be removed in a future version\n");
5670 "ip ssmpingd [A.B.C.D]",
5675 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5678 struct in_addr source_addr
;
5679 const char *source_str
= (argc
== 3) ? argv
[idx_ipv4
]->arg
: "0.0.0.0";
5681 result
= inet_pton(AF_INET
, source_str
, &source_addr
);
5683 vty_out(vty
, "%% Bad source address %s: errno=%d: %s\n",
5684 source_str
, errno
, safe_strerror(errno
));
5685 return CMD_WARNING_CONFIG_FAILED
;
5688 result
= pim_ssmpingd_start(pim
, source_addr
);
5690 vty_out(vty
, "%% Failure starting ssmpingd for source %s: %d\n",
5691 source_str
, result
);
5692 return CMD_WARNING_CONFIG_FAILED
;
5698 DEFUN (no_ip_ssmpingd
,
5700 "no ip ssmpingd [A.B.C.D]",
5706 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5709 struct in_addr source_addr
;
5710 const char *source_str
= (argc
== 4) ? argv
[idx_ipv4
]->arg
: "0.0.0.0";
5712 result
= inet_pton(AF_INET
, source_str
, &source_addr
);
5714 vty_out(vty
, "%% Bad source address %s: errno=%d: %s\n",
5715 source_str
, errno
, safe_strerror(errno
));
5716 return CMD_WARNING_CONFIG_FAILED
;
5719 result
= pim_ssmpingd_stop(pim
, source_addr
);
5721 vty_out(vty
, "%% Failure stopping ssmpingd for source %s: %d\n",
5722 source_str
, result
);
5723 return CMD_WARNING_CONFIG_FAILED
;
5733 "pim multicast routing\n"
5734 "Enable PIM ECMP \n")
5736 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5737 pim
->ecmp_enable
= true;
5742 DEFUN (no_ip_pim_ecmp
,
5747 "pim multicast routing\n"
5748 "Disable PIM ECMP \n")
5750 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5751 pim
->ecmp_enable
= false;
5756 DEFUN (ip_pim_ecmp_rebalance
,
5757 ip_pim_ecmp_rebalance_cmd
,
5758 "ip pim ecmp rebalance",
5760 "pim multicast routing\n"
5761 "Enable PIM ECMP \n"
5762 "Enable PIM ECMP Rebalance\n")
5764 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5765 pim
->ecmp_enable
= true;
5766 pim
->ecmp_rebalance_enable
= true;
5771 DEFUN (no_ip_pim_ecmp_rebalance
,
5772 no_ip_pim_ecmp_rebalance_cmd
,
5773 "no ip pim ecmp rebalance",
5776 "pim multicast routing\n"
5777 "Disable PIM ECMP \n"
5778 "Disable PIM ECMP Rebalance\n")
5780 PIM_DECLVAR_CONTEXT(vrf
, pim
);
5781 pim
->ecmp_rebalance_enable
= false;
5786 static int pim_cmd_igmp_start(struct vty
*vty
, struct interface
*ifp
)
5788 struct pim_interface
*pim_ifp
;
5789 uint8_t need_startup
= 0;
5791 pim_ifp
= ifp
->info
;
5794 pim_ifp
= pim_if_new(ifp
, 1 /* igmp=true */, 0 /* pim=false */);
5796 vty_out(vty
, "Could not enable IGMP on interface %s\n",
5798 return CMD_WARNING_CONFIG_FAILED
;
5802 if (!PIM_IF_TEST_IGMP(pim_ifp
->options
)) {
5803 PIM_IF_DO_IGMP(pim_ifp
->options
);
5808 /* 'ip igmp' executed multiple times, with need_startup
5809 avoid multiple if add all and membership refresh */
5811 pim_if_addr_add_all(ifp
);
5812 pim_if_membership_refresh(ifp
);
5818 DEFUN (interface_ip_igmp
,
5819 interface_ip_igmp_cmd
,
5824 VTY_DECLVAR_CONTEXT(interface
, ifp
);
5826 return pim_cmd_igmp_start(vty
, ifp
);
5829 DEFUN (interface_no_ip_igmp
,
5830 interface_no_ip_igmp_cmd
,
5836 VTY_DECLVAR_CONTEXT(interface
, ifp
);
5837 struct pim_interface
*pim_ifp
= ifp
->info
;
5842 PIM_IF_DONT_IGMP(pim_ifp
->options
);
5844 pim_if_membership_clear(ifp
);
5846 pim_if_addr_del_all_igmp(ifp
);
5848 if (!PIM_IF_TEST_PIM(pim_ifp
->options
)) {
5855 DEFUN (interface_ip_igmp_join
,
5856 interface_ip_igmp_join_cmd
,
5857 "ip igmp join A.B.C.D A.B.C.D",
5860 "IGMP join multicast group\n"
5861 "Multicast group address\n"
5864 VTY_DECLVAR_CONTEXT(interface
, ifp
);
5867 const char *group_str
;
5868 const char *source_str
;
5869 struct in_addr group_addr
;
5870 struct in_addr source_addr
;
5874 group_str
= argv
[idx_ipv4
]->arg
;
5875 result
= inet_pton(AF_INET
, group_str
, &group_addr
);
5877 vty_out(vty
, "Bad group address %s: errno=%d: %s\n", group_str
,
5878 errno
, safe_strerror(errno
));
5879 return CMD_WARNING_CONFIG_FAILED
;
5882 /* Source address */
5883 source_str
= argv
[idx_ipv4_2
]->arg
;
5884 result
= inet_pton(AF_INET
, source_str
, &source_addr
);
5886 vty_out(vty
, "Bad source address %s: errno=%d: %s\n",
5887 source_str
, errno
, safe_strerror(errno
));
5888 return CMD_WARNING_CONFIG_FAILED
;
5891 CMD_FERR_RETURN(pim_if_igmp_join_add(ifp
, group_addr
, source_addr
),
5892 "Failure joining IGMP group: $ERR");
5897 DEFUN (interface_no_ip_igmp_join
,
5898 interface_no_ip_igmp_join_cmd
,
5899 "no ip igmp join A.B.C.D A.B.C.D",
5903 "IGMP join multicast group\n"
5904 "Multicast group address\n"
5907 VTY_DECLVAR_CONTEXT(interface
, ifp
);
5910 const char *group_str
;
5911 const char *source_str
;
5912 struct in_addr group_addr
;
5913 struct in_addr source_addr
;
5917 group_str
= argv
[idx_ipv4
]->arg
;
5918 result
= inet_pton(AF_INET
, group_str
, &group_addr
);
5920 vty_out(vty
, "Bad group address %s: errno=%d: %s\n", group_str
,
5921 errno
, safe_strerror(errno
));
5922 return CMD_WARNING_CONFIG_FAILED
;
5925 /* Source address */
5926 source_str
= argv
[idx_ipv4_2
]->arg
;
5927 result
= inet_pton(AF_INET
, source_str
, &source_addr
);
5929 vty_out(vty
, "Bad source address %s: errno=%d: %s\n",
5930 source_str
, errno
, safe_strerror(errno
));
5931 return CMD_WARNING_CONFIG_FAILED
;
5934 result
= pim_if_igmp_join_del(ifp
, group_addr
, source_addr
);
5937 "%% Failure leaving IGMP group %s source %s on interface %s: %d\n",
5938 group_str
, source_str
, ifp
->name
, result
);
5939 return CMD_WARNING_CONFIG_FAILED
;
5946 CLI reconfiguration affects the interface level (struct pim_interface).
5947 This function propagates the reconfiguration to every active socket
5950 static void igmp_sock_query_interval_reconfig(struct igmp_sock
*igmp
)
5952 struct interface
*ifp
;
5953 struct pim_interface
*pim_ifp
;
5957 /* other querier present? */
5959 if (igmp
->t_other_querier_timer
)
5962 /* this is the querier */
5964 zassert(igmp
->interface
);
5965 zassert(igmp
->interface
->info
);
5967 ifp
= igmp
->interface
;
5968 pim_ifp
= ifp
->info
;
5970 if (PIM_DEBUG_IGMP_TRACE
) {
5971 char ifaddr_str
[INET_ADDRSTRLEN
];
5972 pim_inet4_dump("<ifaddr?>", igmp
->ifaddr
, ifaddr_str
,
5973 sizeof(ifaddr_str
));
5974 zlog_debug("%s: Querier %s on %s reconfig query_interval=%d",
5975 __PRETTY_FUNCTION__
, ifaddr_str
, ifp
->name
,
5976 pim_ifp
->igmp_default_query_interval
);
5980 igmp_startup_mode_on() will reset QQI:
5982 igmp->querier_query_interval = pim_ifp->igmp_default_query_interval;
5984 igmp_startup_mode_on(igmp
);
5987 static void igmp_sock_query_reschedule(struct igmp_sock
*igmp
)
5989 if (igmp
->t_igmp_query_timer
) {
5990 /* other querier present */
5991 zassert(igmp
->t_igmp_query_timer
);
5992 zassert(!igmp
->t_other_querier_timer
);
5994 pim_igmp_general_query_off(igmp
);
5995 pim_igmp_general_query_on(igmp
);
5997 zassert(igmp
->t_igmp_query_timer
);
5998 zassert(!igmp
->t_other_querier_timer
);
6000 /* this is the querier */
6002 zassert(!igmp
->t_igmp_query_timer
);
6003 zassert(igmp
->t_other_querier_timer
);
6005 pim_igmp_other_querier_timer_off(igmp
);
6006 pim_igmp_other_querier_timer_on(igmp
);
6008 zassert(!igmp
->t_igmp_query_timer
);
6009 zassert(igmp
->t_other_querier_timer
);
6013 static void change_query_interval(struct pim_interface
*pim_ifp
,
6016 struct listnode
*sock_node
;
6017 struct igmp_sock
*igmp
;
6019 pim_ifp
->igmp_default_query_interval
= query_interval
;
6021 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
, igmp
)) {
6022 igmp_sock_query_interval_reconfig(igmp
);
6023 igmp_sock_query_reschedule(igmp
);
6027 static void change_query_max_response_time(struct pim_interface
*pim_ifp
,
6028 int query_max_response_time_dsec
)
6030 struct listnode
*sock_node
;
6031 struct igmp_sock
*igmp
;
6033 pim_ifp
->igmp_query_max_response_time_dsec
=
6034 query_max_response_time_dsec
;
6037 Below we modify socket/group/source timers in order to quickly
6038 reflect the change. Otherwise, those timers would eventually catch
6042 /* scan all sockets */
6043 for (ALL_LIST_ELEMENTS_RO(pim_ifp
->igmp_socket_list
, sock_node
, igmp
)) {
6044 struct listnode
*grp_node
;
6045 struct igmp_group
*grp
;
6047 /* reschedule socket general query */
6048 igmp_sock_query_reschedule(igmp
);
6050 /* scan socket groups */
6051 for (ALL_LIST_ELEMENTS_RO(igmp
->igmp_group_list
, grp_node
,
6053 struct listnode
*src_node
;
6054 struct igmp_source
*src
;
6056 /* reset group timers for groups in EXCLUDE mode */
6057 if (grp
->group_filtermode_isexcl
) {
6058 igmp_group_reset_gmi(grp
);
6061 /* scan group sources */
6062 for (ALL_LIST_ELEMENTS_RO(grp
->group_source_list
,
6065 /* reset source timers for sources with running
6067 if (src
->t_source_timer
) {
6068 igmp_source_reset_gmi(igmp
, grp
, src
);
6075 #define IGMP_QUERY_INTERVAL_MIN (1)
6076 #define IGMP_QUERY_INTERVAL_MAX (1800)
6078 DEFUN (interface_ip_igmp_query_interval
,
6079 interface_ip_igmp_query_interval_cmd
,
6080 "ip igmp query-interval (1-1800)",
6083 IFACE_IGMP_QUERY_INTERVAL_STR
6084 "Query interval in seconds\n")
6086 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6087 struct pim_interface
*pim_ifp
= ifp
->info
;
6089 int query_interval_dsec
;
6093 ret
= pim_cmd_igmp_start(vty
, ifp
);
6094 if (ret
!= CMD_SUCCESS
)
6096 pim_ifp
= ifp
->info
;
6099 query_interval
= atoi(argv
[3]->arg
);
6100 query_interval_dsec
= 10 * query_interval
;
6103 It seems we don't need to check bounds since command.c does it
6104 already, but we verify them anyway for extra safety.
6106 if (query_interval
< IGMP_QUERY_INTERVAL_MIN
) {
6108 "General query interval %d lower than minimum %d\n",
6109 query_interval
, IGMP_QUERY_INTERVAL_MIN
);
6110 return CMD_WARNING_CONFIG_FAILED
;
6112 if (query_interval
> IGMP_QUERY_INTERVAL_MAX
) {
6114 "General query interval %d higher than maximum %d\n",
6115 query_interval
, IGMP_QUERY_INTERVAL_MAX
);
6116 return CMD_WARNING_CONFIG_FAILED
;
6119 if (query_interval_dsec
<= pim_ifp
->igmp_query_max_response_time_dsec
) {
6121 "Can't set general query interval %d dsec <= query max response time %d dsec.\n",
6122 query_interval_dsec
,
6123 pim_ifp
->igmp_query_max_response_time_dsec
);
6124 return CMD_WARNING_CONFIG_FAILED
;
6127 change_query_interval(pim_ifp
, query_interval
);
6132 DEFUN (interface_no_ip_igmp_query_interval
,
6133 interface_no_ip_igmp_query_interval_cmd
,
6134 "no ip igmp query-interval",
6138 IFACE_IGMP_QUERY_INTERVAL_STR
)
6140 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6141 struct pim_interface
*pim_ifp
= ifp
->info
;
6142 int default_query_interval_dsec
;
6147 default_query_interval_dsec
= IGMP_GENERAL_QUERY_INTERVAL
* 10;
6149 if (default_query_interval_dsec
6150 <= pim_ifp
->igmp_query_max_response_time_dsec
) {
6152 "Can't set default general query interval %d dsec <= query max response time %d dsec.\n",
6153 default_query_interval_dsec
,
6154 pim_ifp
->igmp_query_max_response_time_dsec
);
6155 return CMD_WARNING_CONFIG_FAILED
;
6158 change_query_interval(pim_ifp
, IGMP_GENERAL_QUERY_INTERVAL
);
6163 DEFUN (interface_ip_igmp_version
,
6164 interface_ip_igmp_version_cmd
,
6165 "ip igmp version (2-3)",
6169 "IGMP version number\n")
6171 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6172 struct pim_interface
*pim_ifp
= ifp
->info
;
6173 int igmp_version
, old_version
= 0;
6177 ret
= pim_cmd_igmp_start(vty
, ifp
);
6178 if (ret
!= CMD_SUCCESS
)
6180 pim_ifp
= ifp
->info
;
6183 igmp_version
= atoi(argv
[3]->arg
);
6184 old_version
= pim_ifp
->igmp_version
;
6185 pim_ifp
->igmp_version
= igmp_version
;
6187 // Check if IGMP is Enabled otherwise, enable on interface
6188 if (!PIM_IF_TEST_IGMP(pim_ifp
->options
)) {
6189 PIM_IF_DO_IGMP(pim_ifp
->options
);
6190 pim_if_addr_add_all(ifp
);
6191 pim_if_membership_refresh(ifp
);
6192 old_version
= igmp_version
;
6193 // avoid refreshing membership again.
6195 /* Current and new version is different refresh existing
6196 membership. Going from 3 -> 2 or 2 -> 3. */
6197 if (old_version
!= igmp_version
)
6198 pim_if_membership_refresh(ifp
);
6203 DEFUN (interface_no_ip_igmp_version
,
6204 interface_no_ip_igmp_version_cmd
,
6205 "no ip igmp version (2-3)",
6210 "IGMP version number\n")
6212 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6213 struct pim_interface
*pim_ifp
= ifp
->info
;
6218 pim_ifp
->igmp_version
= IGMP_DEFAULT_VERSION
;
6223 #define IGMP_QUERY_MAX_RESPONSE_TIME_MIN_DSEC (10)
6224 #define IGMP_QUERY_MAX_RESPONSE_TIME_MAX_DSEC (250)
6226 DEFUN (interface_ip_igmp_query_max_response_time
,
6227 interface_ip_igmp_query_max_response_time_cmd
,
6228 "ip igmp query-max-response-time (10-250)",
6231 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_STR
6232 "Query response value in deci-seconds\n")
6234 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6235 struct pim_interface
*pim_ifp
= ifp
->info
;
6236 int query_max_response_time
;
6240 ret
= pim_cmd_igmp_start(vty
, ifp
);
6241 if (ret
!= CMD_SUCCESS
)
6243 pim_ifp
= ifp
->info
;
6246 query_max_response_time
= atoi(argv
[3]->arg
);
6248 if (query_max_response_time
6249 >= pim_ifp
->igmp_default_query_interval
* 10) {
6251 "Can't set query max response time %d sec >= general query interval %d sec\n",
6252 query_max_response_time
,
6253 pim_ifp
->igmp_default_query_interval
);
6254 return CMD_WARNING_CONFIG_FAILED
;
6257 change_query_max_response_time(pim_ifp
, query_max_response_time
);
6262 DEFUN (interface_no_ip_igmp_query_max_response_time
,
6263 interface_no_ip_igmp_query_max_response_time_cmd
,
6264 "no ip igmp query-max-response-time (10-250)",
6268 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_STR
6269 "Time for response in deci-seconds\n")
6271 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6272 struct pim_interface
*pim_ifp
= ifp
->info
;
6277 change_query_max_response_time(pim_ifp
,
6278 IGMP_QUERY_MAX_RESPONSE_TIME_DSEC
);
6283 #define IGMP_QUERY_MAX_RESPONSE_TIME_MIN_DSEC (10)
6284 #define IGMP_QUERY_MAX_RESPONSE_TIME_MAX_DSEC (250)
6286 DEFUN_HIDDEN (interface_ip_igmp_query_max_response_time_dsec
,
6287 interface_ip_igmp_query_max_response_time_dsec_cmd
,
6288 "ip igmp query-max-response-time-dsec (10-250)",
6291 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_DSEC_STR
6292 "Query response value in deciseconds\n")
6294 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6295 struct pim_interface
*pim_ifp
= ifp
->info
;
6296 int query_max_response_time_dsec
;
6297 int default_query_interval_dsec
;
6301 ret
= pim_cmd_igmp_start(vty
, ifp
);
6302 if (ret
!= CMD_SUCCESS
)
6304 pim_ifp
= ifp
->info
;
6307 query_max_response_time_dsec
= atoi(argv
[4]->arg
);
6309 default_query_interval_dsec
= 10 * pim_ifp
->igmp_default_query_interval
;
6311 if (query_max_response_time_dsec
>= default_query_interval_dsec
) {
6313 "Can't set query max response time %d dsec >= general query interval %d dsec\n",
6314 query_max_response_time_dsec
,
6315 default_query_interval_dsec
);
6316 return CMD_WARNING_CONFIG_FAILED
;
6319 change_query_max_response_time(pim_ifp
, query_max_response_time_dsec
);
6324 DEFUN_HIDDEN (interface_no_ip_igmp_query_max_response_time_dsec
,
6325 interface_no_ip_igmp_query_max_response_time_dsec_cmd
,
6326 "no ip igmp query-max-response-time-dsec",
6330 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_DSEC_STR
)
6332 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6333 struct pim_interface
*pim_ifp
= ifp
->info
;
6338 change_query_max_response_time(pim_ifp
,
6339 IGMP_QUERY_MAX_RESPONSE_TIME_DSEC
);
6344 DEFUN (interface_ip_pim_drprio
,
6345 interface_ip_pim_drprio_cmd
,
6346 "ip pim drpriority (1-4294967295)",
6349 "Set the Designated Router Election Priority\n"
6350 "Value of the new DR Priority\n")
6352 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6354 struct pim_interface
*pim_ifp
= ifp
->info
;
6355 uint32_t old_dr_prio
;
6358 vty_out(vty
, "Please enable PIM on interface, first\n");
6359 return CMD_WARNING_CONFIG_FAILED
;
6362 old_dr_prio
= pim_ifp
->pim_dr_priority
;
6364 pim_ifp
->pim_dr_priority
= strtol(argv
[idx_number
]->arg
, NULL
, 10);
6366 if (old_dr_prio
!= pim_ifp
->pim_dr_priority
) {
6367 if (pim_if_dr_election(ifp
))
6368 pim_hello_restart_now(ifp
);
6374 DEFUN (interface_no_ip_pim_drprio
,
6375 interface_no_ip_pim_drprio_cmd
,
6376 "no ip pim drpriority [(1-4294967295)]",
6380 "Revert the Designated Router Priority to default\n"
6381 "Old Value of the Priority\n")
6383 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6384 struct pim_interface
*pim_ifp
= ifp
->info
;
6387 vty_out(vty
, "Pim not enabled on this interface\n");
6388 return CMD_WARNING_CONFIG_FAILED
;
6391 if (pim_ifp
->pim_dr_priority
!= PIM_DEFAULT_DR_PRIORITY
) {
6392 pim_ifp
->pim_dr_priority
= PIM_DEFAULT_DR_PRIORITY
;
6393 if (pim_if_dr_election(ifp
))
6394 pim_hello_restart_now(ifp
);
6400 static int pim_cmd_interface_add(struct interface
*ifp
)
6402 struct pim_interface
*pim_ifp
= ifp
->info
;
6405 pim_ifp
= pim_if_new(ifp
, 0 /* igmp=false */, 1 /* pim=true */);
6410 PIM_IF_DO_PIM(pim_ifp
->options
);
6413 pim_if_addr_add_all(ifp
);
6414 pim_if_membership_refresh(ifp
);
6418 DEFUN_HIDDEN (interface_ip_pim_ssm
,
6419 interface_ip_pim_ssm_cmd
,
6425 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6427 if (!pim_cmd_interface_add(ifp
)) {
6428 vty_out(vty
, "Could not enable PIM SM on interface\n");
6429 return CMD_WARNING_CONFIG_FAILED
;
6433 "WARN: Enabled PIM SM on interface; configure PIM SSM "
6434 "range if needed\n");
6438 DEFUN (interface_ip_pim_sm
,
6439 interface_ip_pim_sm_cmd
,
6445 struct pim_interface
*pim_ifp
;
6447 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6448 if (!pim_cmd_interface_add(ifp
)) {
6449 vty_out(vty
, "Could not enable PIM SM on interface\n");
6450 return CMD_WARNING_CONFIG_FAILED
;
6453 pim_ifp
= ifp
->info
;
6455 pim_if_create_pimreg(pim_ifp
->pim
);
6460 static int pim_cmd_interface_delete(struct interface
*ifp
)
6462 struct pim_interface
*pim_ifp
= ifp
->info
;
6467 PIM_IF_DONT_PIM(pim_ifp
->options
);
6469 pim_if_membership_clear(ifp
);
6472 pim_sock_delete() removes all neighbors from
6473 pim_ifp->pim_neighbor_list.
6475 pim_sock_delete(ifp
, "pim unconfigured on interface");
6477 if (!PIM_IF_TEST_IGMP(pim_ifp
->options
)) {
6478 pim_if_addr_del_all(ifp
);
6485 DEFUN_HIDDEN (interface_no_ip_pim_ssm
,
6486 interface_no_ip_pim_ssm_cmd
,
6493 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6494 if (!pim_cmd_interface_delete(ifp
)) {
6495 vty_out(vty
, "Unable to delete interface information\n");
6496 return CMD_WARNING_CONFIG_FAILED
;
6502 DEFUN (interface_no_ip_pim_sm
,
6503 interface_no_ip_pim_sm_cmd
,
6510 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6511 if (!pim_cmd_interface_delete(ifp
)) {
6512 vty_out(vty
, "Unable to delete interface information\n");
6513 return CMD_WARNING_CONFIG_FAILED
;
6520 DEFUN(interface_ip_pim_boundary_oil
,
6521 interface_ip_pim_boundary_oil_cmd
,
6522 "ip multicast boundary oil WORD",
6524 "Generic multicast configuration options\n"
6525 "Define multicast boundary\n"
6526 "Filter OIL by group using prefix list\n"
6527 "Prefix list to filter OIL with\n")
6529 VTY_DECLVAR_CONTEXT(interface
, iif
);
6530 struct pim_interface
*pim_ifp
;
6533 argv_find(argv
, argc
, "WORD", &idx
);
6535 PIM_GET_PIM_INTERFACE(pim_ifp
, iif
);
6537 if (pim_ifp
->boundary_oil_plist
)
6538 XFREE(MTYPE_PIM_INTERFACE
, pim_ifp
->boundary_oil_plist
);
6540 pim_ifp
->boundary_oil_plist
=
6541 XSTRDUP(MTYPE_PIM_INTERFACE
, argv
[idx
]->arg
);
6543 /* Interface will be pruned from OIL on next Join */
6547 DEFUN(interface_no_ip_pim_boundary_oil
,
6548 interface_no_ip_pim_boundary_oil_cmd
,
6549 "no ip multicast boundary oil [WORD]",
6552 "Generic multicast configuration options\n"
6553 "Define multicast boundary\n"
6554 "Filter OIL by group using prefix list\n"
6555 "Prefix list to filter OIL with\n")
6557 VTY_DECLVAR_CONTEXT(interface
, iif
);
6558 struct pim_interface
*pim_ifp
;
6561 argv_find(argv
, argc
, "WORD", &idx
);
6563 PIM_GET_PIM_INTERFACE(pim_ifp
, iif
);
6565 if (pim_ifp
->boundary_oil_plist
)
6566 XFREE(MTYPE_PIM_INTERFACE
, pim_ifp
->boundary_oil_plist
);
6571 DEFUN (interface_ip_mroute
,
6572 interface_ip_mroute_cmd
,
6573 "ip mroute INTERFACE A.B.C.D",
6575 "Add multicast route\n"
6576 "Outgoing interface name\n"
6579 VTY_DECLVAR_CONTEXT(interface
, iif
);
6580 struct pim_interface
*pim_ifp
;
6581 struct pim_instance
*pim
;
6582 int idx_interface
= 2;
6584 struct interface
*oif
;
6585 const char *oifname
;
6586 const char *grp_str
;
6587 struct in_addr grp_addr
;
6588 struct in_addr src_addr
;
6591 PIM_GET_PIM_INTERFACE(pim_ifp
, iif
);
6594 oifname
= argv
[idx_interface
]->arg
;
6595 oif
= if_lookup_by_name(oifname
, pim
->vrf_id
);
6597 vty_out(vty
, "No such interface name %s\n", oifname
);
6601 grp_str
= argv
[idx_ipv4
]->arg
;
6602 result
= inet_pton(AF_INET
, grp_str
, &grp_addr
);
6604 vty_out(vty
, "Bad group address %s: errno=%d: %s\n", grp_str
,
6605 errno
, safe_strerror(errno
));
6609 src_addr
.s_addr
= INADDR_ANY
;
6611 if (pim_static_add(pim
, iif
, oif
, grp_addr
, src_addr
)) {
6612 vty_out(vty
, "Failed to add route\n");
6619 DEFUN (interface_ip_mroute_source
,
6620 interface_ip_mroute_source_cmd
,
6621 "ip mroute INTERFACE A.B.C.D A.B.C.D",
6623 "Add multicast route\n"
6624 "Outgoing interface name\n"
6628 VTY_DECLVAR_CONTEXT(interface
, iif
);
6629 struct pim_interface
*pim_ifp
;
6630 struct pim_instance
*pim
;
6631 int idx_interface
= 2;
6634 struct interface
*oif
;
6635 const char *oifname
;
6636 const char *grp_str
;
6637 struct in_addr grp_addr
;
6638 const char *src_str
;
6639 struct in_addr src_addr
;
6642 PIM_GET_PIM_INTERFACE(pim_ifp
, iif
);
6645 oifname
= argv
[idx_interface
]->arg
;
6646 oif
= if_lookup_by_name(oifname
, pim
->vrf_id
);
6648 vty_out(vty
, "No such interface name %s\n", oifname
);
6652 grp_str
= argv
[idx_ipv4
]->arg
;
6653 result
= inet_pton(AF_INET
, grp_str
, &grp_addr
);
6655 vty_out(vty
, "Bad group address %s: errno=%d: %s\n", grp_str
,
6656 errno
, safe_strerror(errno
));
6660 src_str
= argv
[idx_ipv4_2
]->arg
;
6661 result
= inet_pton(AF_INET
, src_str
, &src_addr
);
6663 vty_out(vty
, "Bad source address %s: errno=%d: %s\n", src_str
,
6664 errno
, safe_strerror(errno
));
6668 if (pim_static_add(pim
, iif
, oif
, grp_addr
, src_addr
)) {
6669 vty_out(vty
, "Failed to add route\n");
6676 DEFUN (interface_no_ip_mroute
,
6677 interface_no_ip_mroute_cmd
,
6678 "no ip mroute INTERFACE A.B.C.D",
6681 "Add multicast route\n"
6682 "Outgoing interface name\n"
6685 VTY_DECLVAR_CONTEXT(interface
, iif
);
6686 struct pim_interface
*pim_ifp
;
6687 struct pim_instance
*pim
;
6688 int idx_interface
= 3;
6690 struct interface
*oif
;
6691 const char *oifname
;
6692 const char *grp_str
;
6693 struct in_addr grp_addr
;
6694 struct in_addr src_addr
;
6697 PIM_GET_PIM_INTERFACE(pim_ifp
, iif
);
6700 oifname
= argv
[idx_interface
]->arg
;
6701 oif
= if_lookup_by_name(oifname
, pim
->vrf_id
);
6703 vty_out(vty
, "No such interface name %s\n", oifname
);
6707 grp_str
= argv
[idx_ipv4
]->arg
;
6708 result
= inet_pton(AF_INET
, grp_str
, &grp_addr
);
6710 vty_out(vty
, "Bad group address %s: errno=%d: %s\n", grp_str
,
6711 errno
, safe_strerror(errno
));
6715 src_addr
.s_addr
= INADDR_ANY
;
6717 if (pim_static_del(pim
, iif
, oif
, grp_addr
, src_addr
)) {
6718 vty_out(vty
, "Failed to remove route\n");
6725 DEFUN (interface_no_ip_mroute_source
,
6726 interface_no_ip_mroute_source_cmd
,
6727 "no ip mroute INTERFACE A.B.C.D A.B.C.D",
6730 "Add multicast route\n"
6731 "Outgoing interface name\n"
6735 VTY_DECLVAR_CONTEXT(interface
, iif
);
6736 struct pim_interface
*pim_ifp
;
6737 struct pim_instance
*pim
;
6738 int idx_interface
= 3;
6741 struct interface
*oif
;
6742 const char *oifname
;
6743 const char *grp_str
;
6744 struct in_addr grp_addr
;
6745 const char *src_str
;
6746 struct in_addr src_addr
;
6749 PIM_GET_PIM_INTERFACE(pim_ifp
, iif
);
6752 oifname
= argv
[idx_interface
]->arg
;
6753 oif
= if_lookup_by_name(oifname
, pim
->vrf_id
);
6755 vty_out(vty
, "No such interface name %s\n", oifname
);
6759 grp_str
= argv
[idx_ipv4
]->arg
;
6760 result
= inet_pton(AF_INET
, grp_str
, &grp_addr
);
6762 vty_out(vty
, "Bad group address %s: errno=%d: %s\n", grp_str
,
6763 errno
, safe_strerror(errno
));
6767 src_str
= argv
[idx_ipv4_2
]->arg
;
6768 result
= inet_pton(AF_INET
, src_str
, &src_addr
);
6770 vty_out(vty
, "Bad source address %s: errno=%d: %s\n", src_str
,
6771 errno
, safe_strerror(errno
));
6775 if (pim_static_del(pim
, iif
, oif
, grp_addr
, src_addr
)) {
6776 vty_out(vty
, "Failed to remove route\n");
6783 DEFUN (interface_ip_pim_hello
,
6784 interface_ip_pim_hello_cmd
,
6785 "ip pim hello (1-180) [(1-180)]",
6789 IFACE_PIM_HELLO_TIME_STR
6790 IFACE_PIM_HELLO_HOLD_STR
)
6792 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6795 struct pim_interface
*pim_ifp
= ifp
->info
;
6798 if (!pim_cmd_interface_add(ifp
)) {
6799 vty_out(vty
, "Could not enable PIM SM on interface\n");
6800 return CMD_WARNING_CONFIG_FAILED
;
6804 pim_ifp
= ifp
->info
;
6805 pim_ifp
->pim_hello_period
= strtol(argv
[idx_time
]->arg
, NULL
, 10);
6807 if (argc
== idx_hold
+ 1)
6808 pim_ifp
->pim_default_holdtime
=
6809 strtol(argv
[idx_hold
]->arg
, NULL
, 10);
6814 DEFUN (interface_no_ip_pim_hello
,
6815 interface_no_ip_pim_hello_cmd
,
6816 "no ip pim hello [(1-180) (1-180)]",
6821 IFACE_PIM_HELLO_TIME_STR
6822 IFACE_PIM_HELLO_HOLD_STR
)
6824 VTY_DECLVAR_CONTEXT(interface
, ifp
);
6825 struct pim_interface
*pim_ifp
= ifp
->info
;
6828 vty_out(vty
, "Pim not enabled on this interface\n");
6829 return CMD_WARNING_CONFIG_FAILED
;
6832 pim_ifp
->pim_hello_period
= PIM_DEFAULT_HELLO_PERIOD
;
6833 pim_ifp
->pim_default_holdtime
= -1;
6844 PIM_DO_DEBUG_IGMP_EVENTS
;
6845 PIM_DO_DEBUG_IGMP_PACKETS
;
6846 PIM_DO_DEBUG_IGMP_TRACE
;
6850 DEFUN (no_debug_igmp
,
6857 PIM_DONT_DEBUG_IGMP_EVENTS
;
6858 PIM_DONT_DEBUG_IGMP_PACKETS
;
6859 PIM_DONT_DEBUG_IGMP_TRACE
;
6864 DEFUN (debug_igmp_events
,
6865 debug_igmp_events_cmd
,
6866 "debug igmp events",
6869 DEBUG_IGMP_EVENTS_STR
)
6871 PIM_DO_DEBUG_IGMP_EVENTS
;
6875 DEFUN (no_debug_igmp_events
,
6876 no_debug_igmp_events_cmd
,
6877 "no debug igmp events",
6881 DEBUG_IGMP_EVENTS_STR
)
6883 PIM_DONT_DEBUG_IGMP_EVENTS
;
6888 DEFUN (debug_igmp_packets
,
6889 debug_igmp_packets_cmd
,
6890 "debug igmp packets",
6893 DEBUG_IGMP_PACKETS_STR
)
6895 PIM_DO_DEBUG_IGMP_PACKETS
;
6899 DEFUN (no_debug_igmp_packets
,
6900 no_debug_igmp_packets_cmd
,
6901 "no debug igmp packets",
6905 DEBUG_IGMP_PACKETS_STR
)
6907 PIM_DONT_DEBUG_IGMP_PACKETS
;
6912 DEFUN (debug_igmp_trace
,
6913 debug_igmp_trace_cmd
,
6917 DEBUG_IGMP_TRACE_STR
)
6919 PIM_DO_DEBUG_IGMP_TRACE
;
6923 DEFUN (no_debug_igmp_trace
,
6924 no_debug_igmp_trace_cmd
,
6925 "no debug igmp trace",
6929 DEBUG_IGMP_TRACE_STR
)
6931 PIM_DONT_DEBUG_IGMP_TRACE
;
6936 DEFUN (debug_mroute
,
6942 PIM_DO_DEBUG_MROUTE
;
6946 DEFUN (debug_mroute_detail
,
6947 debug_mroute_detail_cmd
,
6948 "debug mroute detail",
6953 PIM_DO_DEBUG_MROUTE_DETAIL
;
6957 DEFUN (no_debug_mroute
,
6958 no_debug_mroute_cmd
,
6964 PIM_DONT_DEBUG_MROUTE
;
6968 DEFUN (no_debug_mroute_detail
,
6969 no_debug_mroute_detail_cmd
,
6970 "no debug mroute detail",
6976 PIM_DONT_DEBUG_MROUTE_DETAIL
;
6980 DEFUN (debug_static
,
6986 PIM_DO_DEBUG_STATIC
;
6990 DEFUN (no_debug_static
,
6991 no_debug_static_cmd
,
6997 PIM_DONT_DEBUG_STATIC
;
7008 PIM_DO_DEBUG_PIM_EVENTS
;
7009 PIM_DO_DEBUG_PIM_PACKETS
;
7010 PIM_DO_DEBUG_PIM_TRACE
;
7011 PIM_DO_DEBUG_MSDP_EVENTS
;
7012 PIM_DO_DEBUG_MSDP_PACKETS
;
7016 DEFUN (no_debug_pim
,
7023 PIM_DONT_DEBUG_PIM_EVENTS
;
7024 PIM_DONT_DEBUG_PIM_PACKETS
;
7025 PIM_DONT_DEBUG_PIM_TRACE
;
7026 PIM_DONT_DEBUG_MSDP_EVENTS
;
7027 PIM_DONT_DEBUG_MSDP_PACKETS
;
7029 PIM_DONT_DEBUG_PIM_PACKETDUMP_SEND
;
7030 PIM_DONT_DEBUG_PIM_PACKETDUMP_RECV
;
7035 DEFUN (debug_pim_nht
,
7040 "Nexthop Tracking\n")
7042 PIM_DO_DEBUG_PIM_NHT
;
7046 DEFUN (no_debug_pim_nht
,
7047 no_debug_pim_nht_cmd
,
7052 "Nexthop Tracking\n")
7054 PIM_DONT_DEBUG_PIM_NHT
;
7058 DEFUN (debug_pim_nht_rp
,
7059 debug_pim_nht_rp_cmd
,
7063 "Nexthop Tracking\n"
7064 "RP Nexthop Tracking\n")
7066 PIM_DO_DEBUG_PIM_NHT_RP
;
7070 DEFUN (no_debug_pim_nht_rp
,
7071 no_debug_pim_nht_rp_cmd
,
7072 "no debug pim nht rp",
7076 "Nexthop Tracking\n"
7077 "RP Nexthop Tracking\n")
7079 PIM_DONT_DEBUG_PIM_NHT_RP
;
7083 DEFUN (debug_pim_events
,
7084 debug_pim_events_cmd
,
7088 DEBUG_PIM_EVENTS_STR
)
7090 PIM_DO_DEBUG_PIM_EVENTS
;
7094 DEFUN (no_debug_pim_events
,
7095 no_debug_pim_events_cmd
,
7096 "no debug pim events",
7100 DEBUG_PIM_EVENTS_STR
)
7102 PIM_DONT_DEBUG_PIM_EVENTS
;
7106 DEFUN (debug_pim_packets
,
7107 debug_pim_packets_cmd
,
7108 "debug pim packets [<hello|joins|register>]",
7111 DEBUG_PIM_PACKETS_STR
7112 DEBUG_PIM_HELLO_PACKETS_STR
7113 DEBUG_PIM_J_P_PACKETS_STR
7114 DEBUG_PIM_PIM_REG_PACKETS_STR
)
7117 if (argv_find(argv
, argc
, "hello", &idx
)) {
7118 PIM_DO_DEBUG_PIM_HELLO
;
7119 vty_out(vty
, "PIM Hello debugging is on\n");
7120 } else if (argv_find(argv
, argc
, "joins", &idx
)) {
7121 PIM_DO_DEBUG_PIM_J_P
;
7122 vty_out(vty
, "PIM Join/Prune debugging is on\n");
7123 } else if (argv_find(argv
, argc
, "register", &idx
)) {
7124 PIM_DO_DEBUG_PIM_REG
;
7125 vty_out(vty
, "PIM Register debugging is on\n");
7127 PIM_DO_DEBUG_PIM_PACKETS
;
7128 vty_out(vty
, "PIM Packet debugging is on \n");
7133 DEFUN (no_debug_pim_packets
,
7134 no_debug_pim_packets_cmd
,
7135 "no debug pim packets [<hello|joins|register>]",
7139 DEBUG_PIM_PACKETS_STR
7140 DEBUG_PIM_HELLO_PACKETS_STR
7141 DEBUG_PIM_J_P_PACKETS_STR
7142 DEBUG_PIM_PIM_REG_PACKETS_STR
)
7145 if (argv_find(argv
, argc
, "hello", &idx
)) {
7146 PIM_DONT_DEBUG_PIM_HELLO
;
7147 vty_out(vty
, "PIM Hello debugging is off \n");
7148 } else if (argv_find(argv
, argc
, "joins", &idx
)) {
7149 PIM_DONT_DEBUG_PIM_J_P
;
7150 vty_out(vty
, "PIM Join/Prune debugging is off \n");
7151 } else if (argv_find(argv
, argc
, "register", &idx
)) {
7152 PIM_DONT_DEBUG_PIM_REG
;
7153 vty_out(vty
, "PIM Register debugging is off\n");
7155 PIM_DONT_DEBUG_PIM_PACKETS
;
7161 DEFUN (debug_pim_packetdump_send
,
7162 debug_pim_packetdump_send_cmd
,
7163 "debug pim packet-dump send",
7166 DEBUG_PIM_PACKETDUMP_STR
7167 DEBUG_PIM_PACKETDUMP_SEND_STR
)
7169 PIM_DO_DEBUG_PIM_PACKETDUMP_SEND
;
7173 DEFUN (no_debug_pim_packetdump_send
,
7174 no_debug_pim_packetdump_send_cmd
,
7175 "no debug pim packet-dump send",
7179 DEBUG_PIM_PACKETDUMP_STR
7180 DEBUG_PIM_PACKETDUMP_SEND_STR
)
7182 PIM_DONT_DEBUG_PIM_PACKETDUMP_SEND
;
7186 DEFUN (debug_pim_packetdump_recv
,
7187 debug_pim_packetdump_recv_cmd
,
7188 "debug pim packet-dump receive",
7191 DEBUG_PIM_PACKETDUMP_STR
7192 DEBUG_PIM_PACKETDUMP_RECV_STR
)
7194 PIM_DO_DEBUG_PIM_PACKETDUMP_RECV
;
7198 DEFUN (no_debug_pim_packetdump_recv
,
7199 no_debug_pim_packetdump_recv_cmd
,
7200 "no debug pim packet-dump receive",
7204 DEBUG_PIM_PACKETDUMP_STR
7205 DEBUG_PIM_PACKETDUMP_RECV_STR
)
7207 PIM_DONT_DEBUG_PIM_PACKETDUMP_RECV
;
7211 DEFUN (debug_pim_trace
,
7212 debug_pim_trace_cmd
,
7216 DEBUG_PIM_TRACE_STR
)
7218 PIM_DO_DEBUG_PIM_TRACE
;
7222 DEFUN (debug_pim_trace_detail
,
7223 debug_pim_trace_detail_cmd
,
7224 "debug pim trace detail",
7228 "Detailed Information\n")
7230 PIM_DO_DEBUG_PIM_TRACE_DETAIL
;
7234 DEFUN (no_debug_pim_trace
,
7235 no_debug_pim_trace_cmd
,
7236 "no debug pim trace",
7240 DEBUG_PIM_TRACE_STR
)
7242 PIM_DONT_DEBUG_PIM_TRACE
;
7246 DEFUN (no_debug_pim_trace_detail
,
7247 no_debug_pim_trace_detail_cmd
,
7248 "no debug pim trace detail",
7253 "Detailed Information\n")
7255 PIM_DONT_DEBUG_PIM_TRACE_DETAIL
;
7259 DEFUN (debug_ssmpingd
,
7265 PIM_DO_DEBUG_SSMPINGD
;
7269 DEFUN (no_debug_ssmpingd
,
7270 no_debug_ssmpingd_cmd
,
7271 "no debug ssmpingd",
7276 PIM_DONT_DEBUG_SSMPINGD
;
7280 DEFUN (debug_pim_zebra
,
7281 debug_pim_zebra_cmd
,
7285 DEBUG_PIM_ZEBRA_STR
)
7291 DEFUN (no_debug_pim_zebra
,
7292 no_debug_pim_zebra_cmd
,
7293 "no debug pim zebra",
7297 DEBUG_PIM_ZEBRA_STR
)
7299 PIM_DONT_DEBUG_ZEBRA
;
7309 PIM_DO_DEBUG_MSDP_EVENTS
;
7310 PIM_DO_DEBUG_MSDP_PACKETS
;
7314 DEFUN (no_debug_msdp
,
7321 PIM_DONT_DEBUG_MSDP_EVENTS
;
7322 PIM_DONT_DEBUG_MSDP_PACKETS
;
7326 #if CONFDATE > 20190402
7327 CPP_NOTICE("bgpd: time to remove undebug commands")
7329 ALIAS_HIDDEN (no_debug_msdp
,
7332 UNDEBUG_STR DEBUG_MSDP_STR
)
7334 DEFUN (debug_msdp_events
,
7335 debug_msdp_events_cmd
,
7336 "debug msdp events",
7339 DEBUG_MSDP_EVENTS_STR
)
7341 PIM_DO_DEBUG_MSDP_EVENTS
;
7345 DEFUN (no_debug_msdp_events
,
7346 no_debug_msdp_events_cmd
,
7347 "no debug msdp events",
7351 DEBUG_MSDP_EVENTS_STR
)
7353 PIM_DONT_DEBUG_MSDP_EVENTS
;
7357 #if CONFDATE > 20190402
7358 CPP_NOTICE("bgpd: time to remove undebug commands")
7360 ALIAS_HIDDEN (no_debug_msdp_events
,
7361 undebug_msdp_events_cmd
,
7362 "undebug msdp events",
7365 DEBUG_MSDP_EVENTS_STR
)
7367 DEFUN (debug_msdp_packets
,
7368 debug_msdp_packets_cmd
,
7369 "debug msdp packets",
7372 DEBUG_MSDP_PACKETS_STR
)
7374 PIM_DO_DEBUG_MSDP_PACKETS
;
7378 DEFUN (no_debug_msdp_packets
,
7379 no_debug_msdp_packets_cmd
,
7380 "no debug msdp packets",
7384 DEBUG_MSDP_PACKETS_STR
)
7386 PIM_DONT_DEBUG_MSDP_PACKETS
;
7390 #if CONFDATE > 20190402
7391 CPP_NOTICE("bgpd: time to remove undebug commands")
7393 ALIAS_HIDDEN (no_debug_msdp_packets
,
7394 undebug_msdp_packets_cmd
,
7395 "undebug msdp packets",
7398 DEBUG_MSDP_PACKETS_STR
)
7400 DEFUN (debug_mtrace
,
7406 PIM_DO_DEBUG_MTRACE
;
7410 DEFUN (no_debug_mtrace
,
7411 no_debug_mtrace_cmd
,
7417 PIM_DONT_DEBUG_MTRACE
;
7421 DEFUN_NOSH (show_debugging_pim
,
7422 show_debugging_pim_cmd
,
7423 "show debugging [pim]",
7428 vty_out(vty
, "PIM debugging status\n");
7430 pim_debug_config_write(vty
);
7435 static int interface_pim_use_src_cmd_worker(struct vty
*vty
, const char *source
)
7438 struct in_addr source_addr
;
7439 int ret
= CMD_SUCCESS
;
7440 VTY_DECLVAR_CONTEXT(interface
, ifp
);
7442 result
= inet_pton(AF_INET
, source
, &source_addr
);
7444 vty_out(vty
, "%% Bad source address %s: errno=%d: %s\n", source
,
7445 errno
, safe_strerror(errno
));
7446 return CMD_WARNING_CONFIG_FAILED
;
7449 result
= pim_update_source_set(ifp
, source_addr
);
7453 case PIM_IFACE_NOT_FOUND
:
7454 ret
= CMD_WARNING_CONFIG_FAILED
;
7455 vty_out(vty
, "Pim not enabled on this interface\n");
7457 case PIM_UPDATE_SOURCE_DUP
:
7459 vty_out(vty
, "%% Source already set to %s\n", source
);
7462 ret
= CMD_WARNING_CONFIG_FAILED
;
7463 vty_out(vty
, "%% Source set failed\n");
7469 DEFUN (interface_pim_use_source
,
7470 interface_pim_use_source_cmd
,
7471 "ip pim use-source A.B.C.D",
7473 "pim multicast routing\n"
7474 "Configure primary IP address\n"
7475 "source ip address\n")
7477 return interface_pim_use_src_cmd_worker(vty
, argv
[3]->arg
);
7480 DEFUN (interface_no_pim_use_source
,
7481 interface_no_pim_use_source_cmd
,
7482 "no ip pim use-source [A.B.C.D]",
7485 "pim multicast routing\n"
7486 "Delete source IP address\n"
7487 "source ip address\n")
7489 return interface_pim_use_src_cmd_worker(vty
, "0.0.0.0");
7497 "Enables BFD support\n")
7499 VTY_DECLVAR_CONTEXT(interface
, ifp
);
7500 struct pim_interface
*pim_ifp
= ifp
->info
;
7501 struct bfd_info
*bfd_info
= NULL
;
7504 if (!pim_cmd_interface_add(ifp
)) {
7505 vty_out(vty
, "Could not enable PIM SM on interface\n");
7509 pim_ifp
= ifp
->info
;
7511 bfd_info
= pim_ifp
->bfd_info
;
7513 if (!bfd_info
|| !CHECK_FLAG(bfd_info
->flags
, BFD_FLAG_PARAM_CFG
))
7514 pim_bfd_if_param_set(ifp
, BFD_DEF_MIN_RX
, BFD_DEF_MIN_TX
,
7515 BFD_DEF_DETECT_MULT
, 1);
7520 DEFUN (no_ip_pim_bfd
,
7526 "Disables BFD support\n")
7528 VTY_DECLVAR_CONTEXT(interface
, ifp
);
7529 struct pim_interface
*pim_ifp
= ifp
->info
;
7532 vty_out(vty
, "Pim not enabled on this interface\n");
7536 if (pim_ifp
->bfd_info
) {
7537 pim_bfd_reg_dereg_all_nbr(ifp
, ZEBRA_BFD_DEST_DEREGISTER
);
7538 bfd_info_free(&(pim_ifp
->bfd_info
));
7544 DEFUN (ip_pim_bfd_param
,
7545 ip_pim_bfd_param_cmd
,
7546 "ip pim bfd (2-255) (50-60000) (50-60000)",
7549 "Enables BFD support\n"
7550 "Detect Multiplier\n"
7551 "Required min receive interval\n"
7552 "Desired min transmit interval\n")
7554 VTY_DECLVAR_CONTEXT(interface
, ifp
);
7556 int idx_number_2
= 4;
7557 int idx_number_3
= 5;
7562 struct pim_interface
*pim_ifp
= ifp
->info
;
7565 if (!pim_cmd_interface_add(ifp
)) {
7566 vty_out(vty
, "Could not enable PIM SM on interface\n");
7571 if ((ret
= bfd_validate_param(
7572 vty
, argv
[idx_number
]->arg
, argv
[idx_number_2
]->arg
,
7573 argv
[idx_number_3
]->arg
, &dm_val
, &rx_val
, &tx_val
))
7577 pim_bfd_if_param_set(ifp
, rx_val
, tx_val
, dm_val
, 0);
7582 ALIAS(no_ip_pim_bfd
, no_ip_pim_bfd_param_cmd
,
7583 "no ip pim bfd (2-255) (50-60000) (50-60000)", NO_STR IP_STR PIM_STR
7584 "Enables BFD support\n"
7585 "Detect Multiplier\n"
7586 "Required min receive interval\n"
7587 "Desired min transmit interval\n")
7589 static int ip_msdp_peer_cmd_worker(struct pim_instance
*pim
, struct vty
*vty
,
7590 const char *peer
, const char *local
)
7592 enum pim_msdp_err result
;
7593 struct in_addr peer_addr
;
7594 struct in_addr local_addr
;
7595 int ret
= CMD_SUCCESS
;
7597 result
= inet_pton(AF_INET
, peer
, &peer_addr
);
7599 vty_out(vty
, "%% Bad peer address %s: errno=%d: %s\n", peer
,
7600 errno
, safe_strerror(errno
));
7601 return CMD_WARNING_CONFIG_FAILED
;
7604 result
= inet_pton(AF_INET
, local
, &local_addr
);
7606 vty_out(vty
, "%% Bad source address %s: errno=%d: %s\n", local
,
7607 errno
, safe_strerror(errno
));
7608 return CMD_WARNING_CONFIG_FAILED
;
7611 result
= pim_msdp_peer_add(pim
, peer_addr
, local_addr
, "default",
7614 case PIM_MSDP_ERR_NONE
:
7616 case PIM_MSDP_ERR_OOM
:
7617 ret
= CMD_WARNING_CONFIG_FAILED
;
7618 vty_out(vty
, "%% Out of memory\n");
7620 case PIM_MSDP_ERR_PEER_EXISTS
:
7622 vty_out(vty
, "%% Peer exists\n");
7624 case PIM_MSDP_ERR_MAX_MESH_GROUPS
:
7625 ret
= CMD_WARNING_CONFIG_FAILED
;
7626 vty_out(vty
, "%% Only one mesh-group allowed currently\n");
7629 ret
= CMD_WARNING_CONFIG_FAILED
;
7630 vty_out(vty
, "%% peer add failed\n");
7636 DEFUN_HIDDEN (ip_msdp_peer
,
7638 "ip msdp peer A.B.C.D source A.B.C.D",
7641 "Configure MSDP peer\n"
7643 "Source address for TCP connection\n"
7644 "local ip address\n")
7646 PIM_DECLVAR_CONTEXT(vrf
, pim
);
7647 return ip_msdp_peer_cmd_worker(pim
, vty
, argv
[3]->arg
, argv
[5]->arg
);
7650 static int ip_no_msdp_peer_cmd_worker(struct pim_instance
*pim
, struct vty
*vty
,
7653 enum pim_msdp_err result
;
7654 struct in_addr peer_addr
;
7656 result
= inet_pton(AF_INET
, peer
, &peer_addr
);
7658 vty_out(vty
, "%% Bad peer address %s: errno=%d: %s\n", peer
,
7659 errno
, safe_strerror(errno
));
7660 return CMD_WARNING_CONFIG_FAILED
;
7663 result
= pim_msdp_peer_del(pim
, peer_addr
);
7665 case PIM_MSDP_ERR_NONE
:
7667 case PIM_MSDP_ERR_NO_PEER
:
7668 vty_out(vty
, "%% Peer does not exist\n");
7671 vty_out(vty
, "%% peer del failed\n");
7674 return result
? CMD_WARNING_CONFIG_FAILED
: CMD_SUCCESS
;
7677 DEFUN_HIDDEN (no_ip_msdp_peer
,
7678 no_ip_msdp_peer_cmd
,
7679 "no ip msdp peer A.B.C.D",
7683 "Delete MSDP peer\n"
7684 "peer ip address\n")
7686 PIM_DECLVAR_CONTEXT(vrf
, pim
);
7687 return ip_no_msdp_peer_cmd_worker(pim
, vty
, argv
[4]->arg
);
7690 static int ip_msdp_mesh_group_member_cmd_worker(struct pim_instance
*pim
,
7691 struct vty
*vty
, const char *mg
,
7694 enum pim_msdp_err result
;
7695 struct in_addr mbr_ip
;
7696 int ret
= CMD_SUCCESS
;
7698 result
= inet_pton(AF_INET
, mbr
, &mbr_ip
);
7700 vty_out(vty
, "%% Bad member address %s: errno=%d: %s\n", mbr
,
7701 errno
, safe_strerror(errno
));
7702 return CMD_WARNING_CONFIG_FAILED
;
7705 result
= pim_msdp_mg_mbr_add(pim
, mg
, mbr_ip
);
7707 case PIM_MSDP_ERR_NONE
:
7709 case PIM_MSDP_ERR_OOM
:
7710 ret
= CMD_WARNING_CONFIG_FAILED
;
7711 vty_out(vty
, "%% Out of memory\n");
7713 case PIM_MSDP_ERR_MG_MBR_EXISTS
:
7715 vty_out(vty
, "%% mesh-group member exists\n");
7717 case PIM_MSDP_ERR_MAX_MESH_GROUPS
:
7718 ret
= CMD_WARNING_CONFIG_FAILED
;
7719 vty_out(vty
, "%% Only one mesh-group allowed currently\n");
7722 ret
= CMD_WARNING_CONFIG_FAILED
;
7723 vty_out(vty
, "%% member add failed\n");
7729 DEFUN (ip_msdp_mesh_group_member
,
7730 ip_msdp_mesh_group_member_cmd
,
7731 "ip msdp mesh-group WORD member A.B.C.D",
7734 "Configure MSDP mesh-group\n"
7736 "mesh group member\n"
7737 "peer ip address\n")
7739 PIM_DECLVAR_CONTEXT(vrf
, pim
);
7740 return ip_msdp_mesh_group_member_cmd_worker(pim
, vty
, argv
[3]->arg
,
7744 static int ip_no_msdp_mesh_group_member_cmd_worker(struct pim_instance
*pim
,
7749 enum pim_msdp_err result
;
7750 struct in_addr mbr_ip
;
7752 result
= inet_pton(AF_INET
, mbr
, &mbr_ip
);
7754 vty_out(vty
, "%% Bad member address %s: errno=%d: %s\n", mbr
,
7755 errno
, safe_strerror(errno
));
7756 return CMD_WARNING_CONFIG_FAILED
;
7759 result
= pim_msdp_mg_mbr_del(pim
, mg
, mbr_ip
);
7761 case PIM_MSDP_ERR_NONE
:
7763 case PIM_MSDP_ERR_NO_MG
:
7764 vty_out(vty
, "%% mesh-group does not exist\n");
7766 case PIM_MSDP_ERR_NO_MG_MBR
:
7767 vty_out(vty
, "%% mesh-group member does not exist\n");
7770 vty_out(vty
, "%% mesh-group member del failed\n");
7773 return result
? CMD_WARNING_CONFIG_FAILED
: CMD_SUCCESS
;
7775 DEFUN (no_ip_msdp_mesh_group_member
,
7776 no_ip_msdp_mesh_group_member_cmd
,
7777 "no ip msdp mesh-group WORD member A.B.C.D",
7781 "Delete MSDP mesh-group member\n"
7783 "mesh group member\n"
7784 "peer ip address\n")
7786 PIM_DECLVAR_CONTEXT(vrf
, pim
);
7787 return ip_no_msdp_mesh_group_member_cmd_worker(pim
, vty
, argv
[4]->arg
,
7791 static int ip_msdp_mesh_group_source_cmd_worker(struct pim_instance
*pim
,
7792 struct vty
*vty
, const char *mg
,
7795 enum pim_msdp_err result
;
7796 struct in_addr src_ip
;
7798 result
= inet_pton(AF_INET
, src
, &src_ip
);
7800 vty_out(vty
, "%% Bad source address %s: errno=%d: %s\n", src
,
7801 errno
, safe_strerror(errno
));
7802 return CMD_WARNING_CONFIG_FAILED
;
7805 result
= pim_msdp_mg_src_add(pim
, mg
, src_ip
);
7807 case PIM_MSDP_ERR_NONE
:
7809 case PIM_MSDP_ERR_OOM
:
7810 vty_out(vty
, "%% Out of memory\n");
7812 case PIM_MSDP_ERR_MAX_MESH_GROUPS
:
7813 vty_out(vty
, "%% Only one mesh-group allowed currently\n");
7816 vty_out(vty
, "%% source add failed\n");
7819 return result
? CMD_WARNING_CONFIG_FAILED
: CMD_SUCCESS
;
7823 DEFUN (ip_msdp_mesh_group_source
,
7824 ip_msdp_mesh_group_source_cmd
,
7825 "ip msdp mesh-group WORD source A.B.C.D",
7828 "Configure MSDP mesh-group\n"
7830 "mesh group local address\n"
7831 "source ip address for the TCP connection\n")
7833 PIM_DECLVAR_CONTEXT(vrf
, pim
);
7834 return ip_msdp_mesh_group_source_cmd_worker(pim
, vty
, argv
[3]->arg
,
7838 static int ip_no_msdp_mesh_group_source_cmd_worker(struct pim_instance
*pim
,
7842 enum pim_msdp_err result
;
7844 result
= pim_msdp_mg_src_del(pim
, mg
);
7846 case PIM_MSDP_ERR_NONE
:
7848 case PIM_MSDP_ERR_NO_MG
:
7849 vty_out(vty
, "%% mesh-group does not exist\n");
7852 vty_out(vty
, "%% mesh-group source del failed\n");
7855 return result
? CMD_WARNING_CONFIG_FAILED
: CMD_SUCCESS
;
7858 static int ip_no_msdp_mesh_group_cmd_worker(struct pim_instance
*pim
,
7859 struct vty
*vty
, const char *mg
)
7861 enum pim_msdp_err result
;
7863 result
= pim_msdp_mg_del(pim
, mg
);
7865 case PIM_MSDP_ERR_NONE
:
7867 case PIM_MSDP_ERR_NO_MG
:
7868 vty_out(vty
, "%% mesh-group does not exist\n");
7871 vty_out(vty
, "%% mesh-group source del failed\n");
7874 return result
? CMD_WARNING_CONFIG_FAILED
: CMD_SUCCESS
;
7877 DEFUN (no_ip_msdp_mesh_group_source
,
7878 no_ip_msdp_mesh_group_source_cmd
,
7879 "no ip msdp mesh-group WORD source [A.B.C.D]",
7883 "Delete MSDP mesh-group source\n"
7885 "mesh group source\n"
7886 "mesh group local address\n")
7888 PIM_DECLVAR_CONTEXT(vrf
, pim
);
7890 return ip_no_msdp_mesh_group_cmd_worker(pim
, vty
, argv
[6]->arg
);
7892 return ip_no_msdp_mesh_group_source_cmd_worker(pim
, vty
,
7896 static void print_empty_json_obj(struct vty
*vty
)
7899 json
= json_object_new_object();
7900 vty_out(vty
, "%s\n",
7901 json_object_to_json_string_ext(json
, JSON_C_TO_STRING_PRETTY
));
7902 json_object_free(json
);
7905 static void ip_msdp_show_mesh_group(struct pim_instance
*pim
, struct vty
*vty
,
7908 struct listnode
*mbrnode
;
7909 struct pim_msdp_mg_mbr
*mbr
;
7910 struct pim_msdp_mg
*mg
= pim
->msdp
.mg
;
7911 char mbr_str
[INET_ADDRSTRLEN
];
7912 char src_str
[INET_ADDRSTRLEN
];
7913 char state_str
[PIM_MSDP_STATE_STRLEN
];
7914 enum pim_msdp_peer_state state
;
7915 json_object
*json
= NULL
;
7916 json_object
*json_mg_row
= NULL
;
7917 json_object
*json_members
= NULL
;
7918 json_object
*json_row
= NULL
;
7922 print_empty_json_obj(vty
);
7926 pim_inet4_dump("<source?>", mg
->src_ip
, src_str
, sizeof(src_str
));
7928 json
= json_object_new_object();
7929 /* currently there is only one mesh group but we should still
7931 * it a dict with mg-name as key */
7932 json_mg_row
= json_object_new_object();
7933 json_object_string_add(json_mg_row
, "name",
7934 mg
->mesh_group_name
);
7935 json_object_string_add(json_mg_row
, "source", src_str
);
7937 vty_out(vty
, "Mesh group : %s\n", mg
->mesh_group_name
);
7938 vty_out(vty
, " Source : %s\n", src_str
);
7939 vty_out(vty
, " Member State\n");
7942 for (ALL_LIST_ELEMENTS_RO(mg
->mbr_list
, mbrnode
, mbr
)) {
7943 pim_inet4_dump("<mbr?>", mbr
->mbr_ip
, mbr_str
, sizeof(mbr_str
));
7945 state
= mbr
->mp
->state
;
7947 state
= PIM_MSDP_DISABLED
;
7949 pim_msdp_state_dump(state
, state_str
, sizeof(state_str
));
7951 json_row
= json_object_new_object();
7952 json_object_string_add(json_row
, "member", mbr_str
);
7953 json_object_string_add(json_row
, "state", state_str
);
7954 if (!json_members
) {
7955 json_members
= json_object_new_object();
7956 json_object_object_add(json_mg_row
, "members",
7959 json_object_object_add(json_members
, mbr_str
, json_row
);
7961 vty_out(vty
, " %-15s %11s\n", mbr_str
, state_str
);
7966 json_object_object_add(json
, mg
->mesh_group_name
, json_mg_row
);
7967 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
7968 json
, JSON_C_TO_STRING_PRETTY
));
7969 json_object_free(json
);
7973 DEFUN (show_ip_msdp_mesh_group
,
7974 show_ip_msdp_mesh_group_cmd
,
7975 "show ip msdp [vrf NAME] mesh-group [json]",
7980 "MSDP mesh-group information\n"
7983 uint8_t uj
= use_json(argc
, argv
);
7985 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
7990 ip_msdp_show_mesh_group(vrf
->info
, vty
, uj
);
7995 DEFUN (show_ip_msdp_mesh_group_vrf_all
,
7996 show_ip_msdp_mesh_group_vrf_all_cmd
,
7997 "show ip msdp vrf all mesh-group [json]",
8002 "MSDP mesh-group information\n"
8005 uint8_t uj
= use_json(argc
, argv
);
8011 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
8015 vty_out(vty
, " \"%s\": ", vrf
->name
);
8018 vty_out(vty
, "VRF: %s\n", vrf
->name
);
8019 ip_msdp_show_mesh_group(vrf
->info
, vty
, uj
);
8022 vty_out(vty
, "}\n");
8027 static void ip_msdp_show_peers(struct pim_instance
*pim
, struct vty
*vty
,
8030 struct listnode
*mpnode
;
8031 struct pim_msdp_peer
*mp
;
8032 char peer_str
[INET_ADDRSTRLEN
];
8033 char local_str
[INET_ADDRSTRLEN
];
8034 char state_str
[PIM_MSDP_STATE_STRLEN
];
8035 char timebuf
[PIM_MSDP_UPTIME_STRLEN
];
8037 json_object
*json
= NULL
;
8038 json_object
*json_row
= NULL
;
8042 json
= json_object_new_object();
8045 "Peer Local State Uptime SaCnt\n");
8048 for (ALL_LIST_ELEMENTS_RO(pim
->msdp
.peer_list
, mpnode
, mp
)) {
8049 if (mp
->state
== PIM_MSDP_ESTABLISHED
) {
8050 now
= pim_time_monotonic_sec();
8051 pim_time_uptime(timebuf
, sizeof(timebuf
),
8054 strcpy(timebuf
, "-");
8056 pim_inet4_dump("<peer?>", mp
->peer
, peer_str
, sizeof(peer_str
));
8057 pim_inet4_dump("<local?>", mp
->local
, local_str
,
8059 pim_msdp_state_dump(mp
->state
, state_str
, sizeof(state_str
));
8061 json_row
= json_object_new_object();
8062 json_object_string_add(json_row
, "peer", peer_str
);
8063 json_object_string_add(json_row
, "local", local_str
);
8064 json_object_string_add(json_row
, "state", state_str
);
8065 json_object_string_add(json_row
, "upTime", timebuf
);
8066 json_object_int_add(json_row
, "saCount", mp
->sa_cnt
);
8067 json_object_object_add(json
, peer_str
, json_row
);
8069 vty_out(vty
, "%-15s %15s %11s %8s %6d\n", peer_str
,
8070 local_str
, state_str
, timebuf
, mp
->sa_cnt
);
8075 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
8076 json
, JSON_C_TO_STRING_PRETTY
));
8077 json_object_free(json
);
8081 static void ip_msdp_show_peers_detail(struct pim_instance
*pim
, struct vty
*vty
,
8082 const char *peer
, uint8_t uj
)
8084 struct listnode
*mpnode
;
8085 struct pim_msdp_peer
*mp
;
8086 char peer_str
[INET_ADDRSTRLEN
];
8087 char local_str
[INET_ADDRSTRLEN
];
8088 char state_str
[PIM_MSDP_STATE_STRLEN
];
8089 char timebuf
[PIM_MSDP_UPTIME_STRLEN
];
8090 char katimer
[PIM_MSDP_TIMER_STRLEN
];
8091 char crtimer
[PIM_MSDP_TIMER_STRLEN
];
8092 char holdtimer
[PIM_MSDP_TIMER_STRLEN
];
8094 json_object
*json
= NULL
;
8095 json_object
*json_row
= NULL
;
8098 json
= json_object_new_object();
8101 for (ALL_LIST_ELEMENTS_RO(pim
->msdp
.peer_list
, mpnode
, mp
)) {
8102 pim_inet4_dump("<peer?>", mp
->peer
, peer_str
, sizeof(peer_str
));
8103 if (strcmp(peer
, "detail") && strcmp(peer
, peer_str
))
8106 if (mp
->state
== PIM_MSDP_ESTABLISHED
) {
8107 now
= pim_time_monotonic_sec();
8108 pim_time_uptime(timebuf
, sizeof(timebuf
),
8111 strcpy(timebuf
, "-");
8113 pim_inet4_dump("<local?>", mp
->local
, local_str
,
8115 pim_msdp_state_dump(mp
->state
, state_str
, sizeof(state_str
));
8116 pim_time_timer_to_hhmmss(katimer
, sizeof(katimer
),
8118 pim_time_timer_to_hhmmss(crtimer
, sizeof(crtimer
),
8120 pim_time_timer_to_hhmmss(holdtimer
, sizeof(holdtimer
),
8124 json_row
= json_object_new_object();
8125 json_object_string_add(json_row
, "peer", peer_str
);
8126 json_object_string_add(json_row
, "local", local_str
);
8127 json_object_string_add(json_row
, "meshGroupName",
8128 mp
->mesh_group_name
);
8129 json_object_string_add(json_row
, "state", state_str
);
8130 json_object_string_add(json_row
, "upTime", timebuf
);
8131 json_object_string_add(json_row
, "keepAliveTimer",
8133 json_object_string_add(json_row
, "connRetryTimer",
8135 json_object_string_add(json_row
, "holdTimer",
8137 json_object_string_add(json_row
, "lastReset",
8139 json_object_int_add(json_row
, "connAttempts",
8141 json_object_int_add(json_row
, "establishedChanges",
8143 json_object_int_add(json_row
, "saCount", mp
->sa_cnt
);
8144 json_object_int_add(json_row
, "kaSent", mp
->ka_tx_cnt
);
8145 json_object_int_add(json_row
, "kaRcvd", mp
->ka_rx_cnt
);
8146 json_object_int_add(json_row
, "saSent", mp
->sa_tx_cnt
);
8147 json_object_int_add(json_row
, "saRcvd", mp
->sa_rx_cnt
);
8148 json_object_object_add(json
, peer_str
, json_row
);
8150 vty_out(vty
, "Peer : %s\n", peer_str
);
8151 vty_out(vty
, " Local : %s\n", local_str
);
8152 vty_out(vty
, " Mesh Group : %s\n",
8153 mp
->mesh_group_name
);
8154 vty_out(vty
, " State : %s\n", state_str
);
8155 vty_out(vty
, " Uptime : %s\n", timebuf
);
8157 vty_out(vty
, " Keepalive Timer : %s\n", katimer
);
8158 vty_out(vty
, " Conn Retry Timer : %s\n", crtimer
);
8159 vty_out(vty
, " Hold Timer : %s\n", holdtimer
);
8160 vty_out(vty
, " Last Reset : %s\n",
8162 vty_out(vty
, " Conn Attempts : %d\n",
8164 vty_out(vty
, " Established Changes : %d\n",
8166 vty_out(vty
, " SA Count : %d\n",
8168 vty_out(vty
, " Statistics :\n");
8171 vty_out(vty
, " Keepalives : %10d %10d\n",
8172 mp
->ka_tx_cnt
, mp
->ka_rx_cnt
);
8173 vty_out(vty
, " SAs : %10d %10d\n",
8174 mp
->sa_tx_cnt
, mp
->sa_rx_cnt
);
8180 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
8181 json
, JSON_C_TO_STRING_PRETTY
));
8182 json_object_free(json
);
8186 DEFUN (show_ip_msdp_peer_detail
,
8187 show_ip_msdp_peer_detail_cmd
,
8188 "show ip msdp [vrf NAME] peer [detail|A.B.C.D] [json]",
8193 "MSDP peer information\n"
8198 uint8_t uj
= use_json(argc
, argv
);
8200 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
8207 if (argv_find(argv
, argc
, "detail", &idx
))
8208 arg
= argv
[idx
]->text
;
8209 else if (argv_find(argv
, argc
, "A.B.C.D", &idx
))
8210 arg
= argv
[idx
]->arg
;
8213 ip_msdp_show_peers_detail(vrf
->info
, vty
, argv
[idx
]->arg
, uj
);
8215 ip_msdp_show_peers(vrf
->info
, vty
, uj
);
8220 DEFUN (show_ip_msdp_peer_detail_vrf_all
,
8221 show_ip_msdp_peer_detail_vrf_all_cmd
,
8222 "show ip msdp vrf all peer [detail|A.B.C.D] [json]",
8227 "MSDP peer information\n"
8233 uint8_t uj
= use_json(argc
, argv
);
8239 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
8243 vty_out(vty
, " \"%s\": ", vrf
->name
);
8246 vty_out(vty
, "VRF: %s\n", vrf
->name
);
8247 if (argv_find(argv
, argc
, "detail", &idx
)
8248 || argv_find(argv
, argc
, "A.B.C.D", &idx
))
8249 ip_msdp_show_peers_detail(vrf
->info
, vty
,
8250 argv
[idx
]->arg
, uj
);
8252 ip_msdp_show_peers(vrf
->info
, vty
, uj
);
8255 vty_out(vty
, "}\n");
8260 static void ip_msdp_show_sa(struct pim_instance
*pim
, struct vty
*vty
,
8263 struct listnode
*sanode
;
8264 struct pim_msdp_sa
*sa
;
8265 char src_str
[INET_ADDRSTRLEN
];
8266 char grp_str
[INET_ADDRSTRLEN
];
8267 char rp_str
[INET_ADDRSTRLEN
];
8268 char timebuf
[PIM_MSDP_UPTIME_STRLEN
];
8272 json_object
*json
= NULL
;
8273 json_object
*json_group
= NULL
;
8274 json_object
*json_row
= NULL
;
8277 json
= json_object_new_object();
8280 "Source Group RP Local SPT Uptime\n");
8283 for (ALL_LIST_ELEMENTS_RO(pim
->msdp
.sa_list
, sanode
, sa
)) {
8284 now
= pim_time_monotonic_sec();
8285 pim_time_uptime(timebuf
, sizeof(timebuf
), now
- sa
->uptime
);
8286 pim_inet4_dump("<src?>", sa
->sg
.src
, src_str
, sizeof(src_str
));
8287 pim_inet4_dump("<grp?>", sa
->sg
.grp
, grp_str
, sizeof(grp_str
));
8288 if (sa
->flags
& PIM_MSDP_SAF_PEER
) {
8289 pim_inet4_dump("<rp?>", sa
->rp
, rp_str
, sizeof(rp_str
));
8291 strcpy(spt_str
, "yes");
8293 strcpy(spt_str
, "no");
8296 strcpy(rp_str
, "-");
8297 strcpy(spt_str
, "-");
8299 if (sa
->flags
& PIM_MSDP_SAF_LOCAL
) {
8300 strcpy(local_str
, "yes");
8302 strcpy(local_str
, "no");
8305 json_object_object_get_ex(json
, grp_str
, &json_group
);
8308 json_group
= json_object_new_object();
8309 json_object_object_add(json
, grp_str
,
8313 json_row
= json_object_new_object();
8314 json_object_string_add(json_row
, "source", src_str
);
8315 json_object_string_add(json_row
, "group", grp_str
);
8316 json_object_string_add(json_row
, "rp", rp_str
);
8317 json_object_string_add(json_row
, "local", local_str
);
8318 json_object_string_add(json_row
, "sptSetup", spt_str
);
8319 json_object_string_add(json_row
, "upTime", timebuf
);
8320 json_object_object_add(json_group
, src_str
, json_row
);
8322 vty_out(vty
, "%-15s %15s %15s %5c %3c %8s\n",
8323 src_str
, grp_str
, rp_str
, local_str
[0],
8324 spt_str
[0], timebuf
);
8329 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
8330 json
, JSON_C_TO_STRING_PRETTY
));
8331 json_object_free(json
);
8335 static void ip_msdp_show_sa_entry_detail(struct pim_msdp_sa
*sa
,
8336 const char *src_str
,
8337 const char *grp_str
, struct vty
*vty
,
8338 uint8_t uj
, json_object
*json
)
8340 char rp_str
[INET_ADDRSTRLEN
];
8341 char peer_str
[INET_ADDRSTRLEN
];
8342 char timebuf
[PIM_MSDP_UPTIME_STRLEN
];
8345 char statetimer
[PIM_MSDP_TIMER_STRLEN
];
8347 json_object
*json_group
= NULL
;
8348 json_object
*json_row
= NULL
;
8350 now
= pim_time_monotonic_sec();
8351 pim_time_uptime(timebuf
, sizeof(timebuf
), now
- sa
->uptime
);
8352 if (sa
->flags
& PIM_MSDP_SAF_PEER
) {
8353 pim_inet4_dump("<rp?>", sa
->rp
, rp_str
, sizeof(rp_str
));
8354 pim_inet4_dump("<peer?>", sa
->peer
, peer_str
, sizeof(peer_str
));
8356 strcpy(spt_str
, "yes");
8358 strcpy(spt_str
, "no");
8361 strcpy(rp_str
, "-");
8362 strcpy(peer_str
, "-");
8363 strcpy(spt_str
, "-");
8365 if (sa
->flags
& PIM_MSDP_SAF_LOCAL
) {
8366 strcpy(local_str
, "yes");
8368 strcpy(local_str
, "no");
8370 pim_time_timer_to_hhmmss(statetimer
, sizeof(statetimer
),
8371 sa
->sa_state_timer
);
8373 json_object_object_get_ex(json
, grp_str
, &json_group
);
8376 json_group
= json_object_new_object();
8377 json_object_object_add(json
, grp_str
, json_group
);
8380 json_row
= json_object_new_object();
8381 json_object_string_add(json_row
, "source", src_str
);
8382 json_object_string_add(json_row
, "group", grp_str
);
8383 json_object_string_add(json_row
, "rp", rp_str
);
8384 json_object_string_add(json_row
, "local", local_str
);
8385 json_object_string_add(json_row
, "sptSetup", spt_str
);
8386 json_object_string_add(json_row
, "upTime", timebuf
);
8387 json_object_string_add(json_row
, "stateTimer", statetimer
);
8388 json_object_object_add(json_group
, src_str
, json_row
);
8390 vty_out(vty
, "SA : %s\n", sa
->sg_str
);
8391 vty_out(vty
, " RP : %s\n", rp_str
);
8392 vty_out(vty
, " Peer : %s\n", peer_str
);
8393 vty_out(vty
, " Local : %s\n", local_str
);
8394 vty_out(vty
, " SPT Setup : %s\n", spt_str
);
8395 vty_out(vty
, " Uptime : %s\n", timebuf
);
8396 vty_out(vty
, " State Timer : %s\n", statetimer
);
8401 static void ip_msdp_show_sa_detail(struct pim_instance
*pim
, struct vty
*vty
,
8404 struct listnode
*sanode
;
8405 struct pim_msdp_sa
*sa
;
8406 char src_str
[INET_ADDRSTRLEN
];
8407 char grp_str
[INET_ADDRSTRLEN
];
8408 json_object
*json
= NULL
;
8411 json
= json_object_new_object();
8414 for (ALL_LIST_ELEMENTS_RO(pim
->msdp
.sa_list
, sanode
, sa
)) {
8415 pim_inet4_dump("<src?>", sa
->sg
.src
, src_str
, sizeof(src_str
));
8416 pim_inet4_dump("<grp?>", sa
->sg
.grp
, grp_str
, sizeof(grp_str
));
8417 ip_msdp_show_sa_entry_detail(sa
, src_str
, grp_str
, vty
, uj
,
8422 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
8423 json
, JSON_C_TO_STRING_PRETTY
));
8424 json_object_free(json
);
8428 DEFUN (show_ip_msdp_sa_detail
,
8429 show_ip_msdp_sa_detail_cmd
,
8430 "show ip msdp [vrf NAME] sa detail [json]",
8435 "MSDP active-source information\n"
8439 uint8_t uj
= use_json(argc
, argv
);
8441 struct vrf
*vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
8446 ip_msdp_show_sa_detail(vrf
->info
, vty
, uj
);
8451 DEFUN (show_ip_msdp_sa_detail_vrf_all
,
8452 show_ip_msdp_sa_detail_vrf_all_cmd
,
8453 "show ip msdp vrf all sa detail [json]",
8458 "MSDP active-source information\n"
8462 uint8_t uj
= use_json(argc
, argv
);
8468 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
8472 vty_out(vty
, " \"%s\": ", vrf
->name
);
8475 vty_out(vty
, "VRF: %s\n", vrf
->name
);
8476 ip_msdp_show_sa_detail(vrf
->info
, vty
, uj
);
8479 vty_out(vty
, "}\n");
8484 static void ip_msdp_show_sa_addr(struct pim_instance
*pim
, struct vty
*vty
,
8485 const char *addr
, uint8_t uj
)
8487 struct listnode
*sanode
;
8488 struct pim_msdp_sa
*sa
;
8489 char src_str
[INET_ADDRSTRLEN
];
8490 char grp_str
[INET_ADDRSTRLEN
];
8491 json_object
*json
= NULL
;
8494 json
= json_object_new_object();
8497 for (ALL_LIST_ELEMENTS_RO(pim
->msdp
.sa_list
, sanode
, sa
)) {
8498 pim_inet4_dump("<src?>", sa
->sg
.src
, src_str
, sizeof(src_str
));
8499 pim_inet4_dump("<grp?>", sa
->sg
.grp
, grp_str
, sizeof(grp_str
));
8500 if (!strcmp(addr
, src_str
) || !strcmp(addr
, grp_str
)) {
8501 ip_msdp_show_sa_entry_detail(sa
, src_str
, grp_str
, vty
,
8507 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
8508 json
, JSON_C_TO_STRING_PRETTY
));
8509 json_object_free(json
);
8513 static void ip_msdp_show_sa_sg(struct pim_instance
*pim
, struct vty
*vty
,
8514 const char *src
, const char *grp
, uint8_t uj
)
8516 struct listnode
*sanode
;
8517 struct pim_msdp_sa
*sa
;
8518 char src_str
[INET_ADDRSTRLEN
];
8519 char grp_str
[INET_ADDRSTRLEN
];
8520 json_object
*json
= NULL
;
8523 json
= json_object_new_object();
8526 for (ALL_LIST_ELEMENTS_RO(pim
->msdp
.sa_list
, sanode
, sa
)) {
8527 pim_inet4_dump("<src?>", sa
->sg
.src
, src_str
, sizeof(src_str
));
8528 pim_inet4_dump("<grp?>", sa
->sg
.grp
, grp_str
, sizeof(grp_str
));
8529 if (!strcmp(src
, src_str
) && !strcmp(grp
, grp_str
)) {
8530 ip_msdp_show_sa_entry_detail(sa
, src_str
, grp_str
, vty
,
8536 vty_out(vty
, "%s\n", json_object_to_json_string_ext(
8537 json
, JSON_C_TO_STRING_PRETTY
));
8538 json_object_free(json
);
8542 DEFUN (show_ip_msdp_sa_sg
,
8543 show_ip_msdp_sa_sg_cmd
,
8544 "show ip msdp [vrf NAME] sa [A.B.C.D [A.B.C.D]] [json]",
8549 "MSDP active-source information\n"
8550 "source or group ip\n"
8554 uint8_t uj
= use_json(argc
, argv
);
8558 vrf
= pim_cmd_lookup_vrf(vty
, argv
, argc
, &idx
);
8563 char *src_ip
= argv_find(argv
, argc
, "A.B.C.D", &idx
) ? argv
[idx
++]->arg
8565 char *grp_ip
= idx
< argc
&& argv_find(argv
, argc
, "A.B.C.D", &idx
)
8569 if (src_ip
&& grp_ip
)
8570 ip_msdp_show_sa_sg(vrf
->info
, vty
, src_ip
, grp_ip
, uj
);
8572 ip_msdp_show_sa_addr(vrf
->info
, vty
, src_ip
, uj
);
8574 ip_msdp_show_sa(vrf
->info
, vty
, uj
);
8579 DEFUN (show_ip_msdp_sa_sg_vrf_all
,
8580 show_ip_msdp_sa_sg_vrf_all_cmd
,
8581 "show ip msdp vrf all sa [A.B.C.D [A.B.C.D]] [json]",
8586 "MSDP active-source information\n"
8587 "source or group ip\n"
8591 uint8_t uj
= use_json(argc
, argv
);
8596 char *src_ip
= argv_find(argv
, argc
, "A.B.C.D", &idx
) ? argv
[idx
++]->arg
8598 char *grp_ip
= idx
< argc
&& argv_find(argv
, argc
, "A.B.C.D", &idx
)
8604 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
8608 vty_out(vty
, " \"%s\": ", vrf
->name
);
8611 vty_out(vty
, "VRF: %s\n", vrf
->name
);
8613 if (src_ip
&& grp_ip
)
8614 ip_msdp_show_sa_sg(vrf
->info
, vty
, src_ip
, grp_ip
, uj
);
8616 ip_msdp_show_sa_addr(vrf
->info
, vty
, src_ip
, uj
);
8618 ip_msdp_show_sa(vrf
->info
, vty
, uj
);
8621 vty_out(vty
, "}\n");
8627 void pim_cmd_init(void)
8629 install_node(&interface_node
,
8630 pim_interface_config_write
); /* INTERFACE_NODE */
8633 install_node(&debug_node
, pim_debug_config_write
);
8635 install_element(CONFIG_NODE
, &ip_multicast_routing_cmd
);
8636 install_element(CONFIG_NODE
, &no_ip_multicast_routing_cmd
);
8637 install_element(CONFIG_NODE
, &ip_pim_rp_cmd
);
8638 install_element(VRF_NODE
, &ip_pim_rp_cmd
);
8639 install_element(CONFIG_NODE
, &no_ip_pim_rp_cmd
);
8640 install_element(VRF_NODE
, &no_ip_pim_rp_cmd
);
8641 install_element(CONFIG_NODE
, &ip_pim_rp_prefix_list_cmd
);
8642 install_element(VRF_NODE
, &ip_pim_rp_prefix_list_cmd
);
8643 install_element(CONFIG_NODE
, &no_ip_pim_rp_prefix_list_cmd
);
8644 install_element(VRF_NODE
, &no_ip_pim_rp_prefix_list_cmd
);
8645 install_element(CONFIG_NODE
, &no_ip_pim_ssm_prefix_list_cmd
);
8646 install_element(VRF_NODE
, &no_ip_pim_ssm_prefix_list_cmd
);
8647 install_element(CONFIG_NODE
, &no_ip_pim_ssm_prefix_list_name_cmd
);
8648 install_element(VRF_NODE
, &no_ip_pim_ssm_prefix_list_name_cmd
);
8649 install_element(CONFIG_NODE
, &ip_pim_ssm_prefix_list_cmd
);
8650 install_element(VRF_NODE
, &ip_pim_ssm_prefix_list_cmd
);
8651 install_element(CONFIG_NODE
, &ip_pim_register_suppress_cmd
);
8652 install_element(VRF_NODE
, &ip_pim_register_suppress_cmd
);
8653 install_element(CONFIG_NODE
, &no_ip_pim_register_suppress_cmd
);
8654 install_element(VRF_NODE
, &no_ip_pim_register_suppress_cmd
);
8655 install_element(CONFIG_NODE
, &ip_pim_spt_switchover_infinity_cmd
);
8656 install_element(VRF_NODE
, &ip_pim_spt_switchover_infinity_cmd
);
8657 install_element(CONFIG_NODE
, &ip_pim_spt_switchover_infinity_plist_cmd
);
8658 install_element(VRF_NODE
, &ip_pim_spt_switchover_infinity_plist_cmd
);
8659 install_element(CONFIG_NODE
, &no_ip_pim_spt_switchover_infinity_cmd
);
8660 install_element(VRF_NODE
, &no_ip_pim_spt_switchover_infinity_cmd
);
8661 install_element(CONFIG_NODE
,
8662 &no_ip_pim_spt_switchover_infinity_plist_cmd
);
8663 install_element(VRF_NODE
, &no_ip_pim_spt_switchover_infinity_plist_cmd
);
8664 install_element(CONFIG_NODE
, &ip_pim_joinprune_time_cmd
);
8665 install_element(VRF_NODE
, &ip_pim_joinprune_time_cmd
);
8666 install_element(CONFIG_NODE
, &no_ip_pim_joinprune_time_cmd
);
8667 install_element(VRF_NODE
, &no_ip_pim_joinprune_time_cmd
);
8668 install_element(CONFIG_NODE
, &ip_pim_keep_alive_cmd
);
8669 install_element(VRF_NODE
, &ip_pim_keep_alive_cmd
);
8670 install_element(CONFIG_NODE
, &ip_pim_rp_keep_alive_cmd
);
8671 install_element(VRF_NODE
, &ip_pim_rp_keep_alive_cmd
);
8672 install_element(CONFIG_NODE
, &no_ip_pim_keep_alive_cmd
);
8673 install_element(VRF_NODE
, &no_ip_pim_keep_alive_cmd
);
8674 install_element(CONFIG_NODE
, &no_ip_pim_rp_keep_alive_cmd
);
8675 install_element(VRF_NODE
, &no_ip_pim_rp_keep_alive_cmd
);
8676 install_element(CONFIG_NODE
, &ip_pim_packets_cmd
);
8677 install_element(VRF_NODE
, &ip_pim_packets_cmd
);
8678 install_element(CONFIG_NODE
, &no_ip_pim_packets_cmd
);
8679 install_element(VRF_NODE
, &no_ip_pim_packets_cmd
);
8680 install_element(CONFIG_NODE
, &ip_pim_v6_secondary_cmd
);
8681 install_element(VRF_NODE
, &ip_pim_v6_secondary_cmd
);
8682 install_element(CONFIG_NODE
, &no_ip_pim_v6_secondary_cmd
);
8683 install_element(VRF_NODE
, &no_ip_pim_v6_secondary_cmd
);
8684 install_element(CONFIG_NODE
, &ip_ssmpingd_cmd
);
8685 install_element(VRF_NODE
, &ip_ssmpingd_cmd
);
8686 install_element(CONFIG_NODE
, &no_ip_ssmpingd_cmd
);
8687 install_element(VRF_NODE
, &no_ip_ssmpingd_cmd
);
8688 install_element(CONFIG_NODE
, &ip_msdp_peer_cmd
);
8689 install_element(VRF_NODE
, &ip_msdp_peer_cmd
);
8690 install_element(CONFIG_NODE
, &no_ip_msdp_peer_cmd
);
8691 install_element(VRF_NODE
, &no_ip_msdp_peer_cmd
);
8692 install_element(CONFIG_NODE
, &ip_pim_ecmp_cmd
);
8693 install_element(VRF_NODE
, &ip_pim_ecmp_cmd
);
8694 install_element(CONFIG_NODE
, &no_ip_pim_ecmp_cmd
);
8695 install_element(VRF_NODE
, &no_ip_pim_ecmp_cmd
);
8696 install_element(CONFIG_NODE
, &ip_pim_ecmp_rebalance_cmd
);
8697 install_element(VRF_NODE
, &ip_pim_ecmp_rebalance_cmd
);
8698 install_element(CONFIG_NODE
, &no_ip_pim_ecmp_rebalance_cmd
);
8699 install_element(VRF_NODE
, &no_ip_pim_ecmp_rebalance_cmd
);
8701 install_element(INTERFACE_NODE
, &interface_ip_igmp_cmd
);
8702 install_element(INTERFACE_NODE
, &interface_no_ip_igmp_cmd
);
8703 install_element(INTERFACE_NODE
, &interface_ip_igmp_join_cmd
);
8704 install_element(INTERFACE_NODE
, &interface_no_ip_igmp_join_cmd
);
8705 install_element(INTERFACE_NODE
, &interface_ip_igmp_version_cmd
);
8706 install_element(INTERFACE_NODE
, &interface_no_ip_igmp_version_cmd
);
8707 install_element(INTERFACE_NODE
, &interface_ip_igmp_query_interval_cmd
);
8708 install_element(INTERFACE_NODE
,
8709 &interface_no_ip_igmp_query_interval_cmd
);
8710 install_element(INTERFACE_NODE
,
8711 &interface_ip_igmp_query_max_response_time_cmd
);
8712 install_element(INTERFACE_NODE
,
8713 &interface_no_ip_igmp_query_max_response_time_cmd
);
8714 install_element(INTERFACE_NODE
,
8715 &interface_ip_igmp_query_max_response_time_dsec_cmd
);
8716 install_element(INTERFACE_NODE
,
8717 &interface_no_ip_igmp_query_max_response_time_dsec_cmd
);
8718 install_element(INTERFACE_NODE
, &interface_ip_pim_ssm_cmd
);
8719 install_element(INTERFACE_NODE
, &interface_no_ip_pim_ssm_cmd
);
8720 install_element(INTERFACE_NODE
, &interface_ip_pim_sm_cmd
);
8721 install_element(INTERFACE_NODE
, &interface_no_ip_pim_sm_cmd
);
8722 install_element(INTERFACE_NODE
, &interface_ip_pim_drprio_cmd
);
8723 install_element(INTERFACE_NODE
, &interface_no_ip_pim_drprio_cmd
);
8724 install_element(INTERFACE_NODE
, &interface_ip_pim_hello_cmd
);
8725 install_element(INTERFACE_NODE
, &interface_no_ip_pim_hello_cmd
);
8726 install_element(INTERFACE_NODE
, &interface_ip_pim_boundary_oil_cmd
);
8727 install_element(INTERFACE_NODE
, &interface_no_ip_pim_boundary_oil_cmd
);
8729 // Static mroutes NEB
8730 install_element(INTERFACE_NODE
, &interface_ip_mroute_cmd
);
8731 install_element(INTERFACE_NODE
, &interface_ip_mroute_source_cmd
);
8732 install_element(INTERFACE_NODE
, &interface_no_ip_mroute_cmd
);
8733 install_element(INTERFACE_NODE
, &interface_no_ip_mroute_source_cmd
);
8735 install_element(VIEW_NODE
, &show_ip_igmp_interface_cmd
);
8736 install_element(VIEW_NODE
, &show_ip_igmp_interface_vrf_all_cmd
);
8737 install_element(VIEW_NODE
, &show_ip_igmp_join_cmd
);
8738 install_element(VIEW_NODE
, &show_ip_igmp_join_vrf_all_cmd
);
8739 install_element(VIEW_NODE
, &show_ip_igmp_groups_cmd
);
8740 install_element(VIEW_NODE
, &show_ip_igmp_groups_vrf_all_cmd
);
8741 install_element(VIEW_NODE
, &show_ip_igmp_groups_retransmissions_cmd
);
8742 install_element(VIEW_NODE
, &show_ip_igmp_sources_cmd
);
8743 install_element(VIEW_NODE
, &show_ip_igmp_sources_retransmissions_cmd
);
8744 install_element(VIEW_NODE
, &show_ip_igmp_statistics_cmd
);
8745 install_element(VIEW_NODE
, &show_ip_pim_assert_cmd
);
8746 install_element(VIEW_NODE
, &show_ip_pim_assert_internal_cmd
);
8747 install_element(VIEW_NODE
, &show_ip_pim_assert_metric_cmd
);
8748 install_element(VIEW_NODE
, &show_ip_pim_assert_winner_metric_cmd
);
8749 install_element(VIEW_NODE
, &show_ip_pim_interface_traffic_cmd
);
8750 install_element(VIEW_NODE
, &show_ip_pim_interface_cmd
);
8751 install_element(VIEW_NODE
, &show_ip_pim_interface_vrf_all_cmd
);
8752 install_element(VIEW_NODE
, &show_ip_pim_join_cmd
);
8753 install_element(VIEW_NODE
, &show_ip_pim_join_vrf_all_cmd
);
8754 install_element(VIEW_NODE
, &show_ip_pim_local_membership_cmd
);
8755 install_element(VIEW_NODE
, &show_ip_pim_neighbor_cmd
);
8756 install_element(VIEW_NODE
, &show_ip_pim_neighbor_vrf_all_cmd
);
8757 install_element(VIEW_NODE
, &show_ip_pim_rpf_cmd
);
8758 install_element(VIEW_NODE
, &show_ip_pim_rpf_vrf_all_cmd
);
8759 install_element(VIEW_NODE
, &show_ip_pim_secondary_cmd
);
8760 install_element(VIEW_NODE
, &show_ip_pim_state_cmd
);
8761 install_element(VIEW_NODE
, &show_ip_pim_state_vrf_all_cmd
);
8762 install_element(VIEW_NODE
, &show_ip_pim_upstream_cmd
);
8763 install_element(VIEW_NODE
, &show_ip_pim_upstream_vrf_all_cmd
);
8764 install_element(VIEW_NODE
, &show_ip_pim_upstream_join_desired_cmd
);
8765 install_element(VIEW_NODE
, &show_ip_pim_upstream_rpf_cmd
);
8766 install_element(VIEW_NODE
, &show_ip_pim_rp_cmd
);
8767 install_element(VIEW_NODE
, &show_ip_pim_rp_vrf_all_cmd
);
8768 install_element(VIEW_NODE
, &show_ip_multicast_cmd
);
8769 install_element(VIEW_NODE
, &show_ip_multicast_vrf_all_cmd
);
8770 install_element(VIEW_NODE
, &show_ip_mroute_cmd
);
8771 install_element(VIEW_NODE
, &show_ip_mroute_vrf_all_cmd
);
8772 install_element(VIEW_NODE
, &show_ip_mroute_count_cmd
);
8773 install_element(VIEW_NODE
, &show_ip_mroute_count_vrf_all_cmd
);
8774 install_element(VIEW_NODE
, &show_ip_rib_cmd
);
8775 install_element(VIEW_NODE
, &show_ip_ssmpingd_cmd
);
8776 install_element(VIEW_NODE
, &show_debugging_pim_cmd
);
8777 install_element(VIEW_NODE
, &show_ip_pim_nexthop_cmd
);
8778 install_element(VIEW_NODE
, &show_ip_pim_nexthop_lookup_cmd
);
8780 install_element(ENABLE_NODE
, &clear_ip_interfaces_cmd
);
8781 install_element(ENABLE_NODE
, &clear_ip_igmp_interfaces_cmd
);
8782 install_element(ENABLE_NODE
, &clear_ip_mroute_cmd
);
8783 install_element(ENABLE_NODE
, &clear_ip_pim_interfaces_cmd
);
8784 install_element(ENABLE_NODE
, &clear_ip_pim_interface_traffic_cmd
);
8785 install_element(ENABLE_NODE
, &clear_ip_pim_oil_cmd
);
8787 install_element(ENABLE_NODE
, &debug_igmp_cmd
);
8788 install_element(ENABLE_NODE
, &no_debug_igmp_cmd
);
8789 install_element(ENABLE_NODE
, &debug_igmp_events_cmd
);
8790 install_element(ENABLE_NODE
, &no_debug_igmp_events_cmd
);
8791 install_element(ENABLE_NODE
, &debug_igmp_packets_cmd
);
8792 install_element(ENABLE_NODE
, &no_debug_igmp_packets_cmd
);
8793 install_element(ENABLE_NODE
, &debug_igmp_trace_cmd
);
8794 install_element(ENABLE_NODE
, &no_debug_igmp_trace_cmd
);
8795 install_element(ENABLE_NODE
, &debug_mroute_cmd
);
8796 install_element(ENABLE_NODE
, &debug_mroute_detail_cmd
);
8797 install_element(ENABLE_NODE
, &no_debug_mroute_cmd
);
8798 install_element(ENABLE_NODE
, &no_debug_mroute_detail_cmd
);
8799 install_element(ENABLE_NODE
, &debug_static_cmd
);
8800 install_element(ENABLE_NODE
, &no_debug_static_cmd
);
8801 install_element(ENABLE_NODE
, &debug_pim_cmd
);
8802 install_element(ENABLE_NODE
, &no_debug_pim_cmd
);
8803 install_element(ENABLE_NODE
, &debug_pim_nht_cmd
);
8804 install_element(ENABLE_NODE
, &no_debug_pim_nht_cmd
);
8805 install_element(ENABLE_NODE
, &debug_pim_nht_rp_cmd
);
8806 install_element(ENABLE_NODE
, &no_debug_pim_nht_rp_cmd
);
8807 install_element(ENABLE_NODE
, &debug_pim_events_cmd
);
8808 install_element(ENABLE_NODE
, &no_debug_pim_events_cmd
);
8809 install_element(ENABLE_NODE
, &debug_pim_packets_cmd
);
8810 install_element(ENABLE_NODE
, &no_debug_pim_packets_cmd
);
8811 install_element(ENABLE_NODE
, &debug_pim_packetdump_send_cmd
);
8812 install_element(ENABLE_NODE
, &no_debug_pim_packetdump_send_cmd
);
8813 install_element(ENABLE_NODE
, &debug_pim_packetdump_recv_cmd
);
8814 install_element(ENABLE_NODE
, &no_debug_pim_packetdump_recv_cmd
);
8815 install_element(ENABLE_NODE
, &debug_pim_trace_cmd
);
8816 install_element(ENABLE_NODE
, &no_debug_pim_trace_cmd
);
8817 install_element(ENABLE_NODE
, &debug_pim_trace_detail_cmd
);
8818 install_element(ENABLE_NODE
, &no_debug_pim_trace_detail_cmd
);
8819 install_element(ENABLE_NODE
, &debug_ssmpingd_cmd
);
8820 install_element(ENABLE_NODE
, &no_debug_ssmpingd_cmd
);
8821 install_element(ENABLE_NODE
, &debug_pim_zebra_cmd
);
8822 install_element(ENABLE_NODE
, &no_debug_pim_zebra_cmd
);
8823 install_element(ENABLE_NODE
, &debug_msdp_cmd
);
8824 install_element(ENABLE_NODE
, &no_debug_msdp_cmd
);
8825 install_element(ENABLE_NODE
, &undebug_msdp_cmd
);
8826 install_element(ENABLE_NODE
, &debug_msdp_events_cmd
);
8827 install_element(ENABLE_NODE
, &no_debug_msdp_events_cmd
);
8828 install_element(ENABLE_NODE
, &undebug_msdp_events_cmd
);
8829 install_element(ENABLE_NODE
, &debug_msdp_packets_cmd
);
8830 install_element(ENABLE_NODE
, &no_debug_msdp_packets_cmd
);
8831 install_element(ENABLE_NODE
, &undebug_msdp_packets_cmd
);
8832 install_element(ENABLE_NODE
, &debug_mtrace_cmd
);
8833 install_element(ENABLE_NODE
, &no_debug_mtrace_cmd
);
8835 install_element(CONFIG_NODE
, &debug_igmp_cmd
);
8836 install_element(CONFIG_NODE
, &no_debug_igmp_cmd
);
8837 install_element(CONFIG_NODE
, &debug_igmp_events_cmd
);
8838 install_element(CONFIG_NODE
, &no_debug_igmp_events_cmd
);
8839 install_element(CONFIG_NODE
, &debug_igmp_packets_cmd
);
8840 install_element(CONFIG_NODE
, &no_debug_igmp_packets_cmd
);
8841 install_element(CONFIG_NODE
, &debug_igmp_trace_cmd
);
8842 install_element(CONFIG_NODE
, &no_debug_igmp_trace_cmd
);
8843 install_element(CONFIG_NODE
, &debug_mroute_cmd
);
8844 install_element(CONFIG_NODE
, &debug_mroute_detail_cmd
);
8845 install_element(CONFIG_NODE
, &no_debug_mroute_cmd
);
8846 install_element(CONFIG_NODE
, &no_debug_mroute_detail_cmd
);
8847 install_element(CONFIG_NODE
, &debug_static_cmd
);
8848 install_element(CONFIG_NODE
, &no_debug_static_cmd
);
8849 install_element(CONFIG_NODE
, &debug_pim_cmd
);
8850 install_element(CONFIG_NODE
, &no_debug_pim_cmd
);
8851 install_element(CONFIG_NODE
, &debug_pim_nht_cmd
);
8852 install_element(CONFIG_NODE
, &no_debug_pim_nht_cmd
);
8853 install_element(CONFIG_NODE
, &debug_pim_nht_rp_cmd
);
8854 install_element(CONFIG_NODE
, &no_debug_pim_nht_rp_cmd
);
8855 install_element(CONFIG_NODE
, &debug_pim_events_cmd
);
8856 install_element(CONFIG_NODE
, &no_debug_pim_events_cmd
);
8857 install_element(CONFIG_NODE
, &debug_pim_packets_cmd
);
8858 install_element(CONFIG_NODE
, &no_debug_pim_packets_cmd
);
8859 install_element(CONFIG_NODE
, &debug_pim_trace_cmd
);
8860 install_element(CONFIG_NODE
, &no_debug_pim_trace_cmd
);
8861 install_element(CONFIG_NODE
, &debug_pim_trace_detail_cmd
);
8862 install_element(CONFIG_NODE
, &no_debug_pim_trace_detail_cmd
);
8863 install_element(CONFIG_NODE
, &debug_ssmpingd_cmd
);
8864 install_element(CONFIG_NODE
, &no_debug_ssmpingd_cmd
);
8865 install_element(CONFIG_NODE
, &debug_pim_zebra_cmd
);
8866 install_element(CONFIG_NODE
, &no_debug_pim_zebra_cmd
);
8867 install_element(CONFIG_NODE
, &debug_msdp_cmd
);
8868 install_element(CONFIG_NODE
, &no_debug_msdp_cmd
);
8869 install_element(CONFIG_NODE
, &undebug_msdp_cmd
);
8870 install_element(CONFIG_NODE
, &debug_msdp_events_cmd
);
8871 install_element(CONFIG_NODE
, &no_debug_msdp_events_cmd
);
8872 install_element(CONFIG_NODE
, &undebug_msdp_events_cmd
);
8873 install_element(CONFIG_NODE
, &debug_msdp_packets_cmd
);
8874 install_element(CONFIG_NODE
, &no_debug_msdp_packets_cmd
);
8875 install_element(CONFIG_NODE
, &undebug_msdp_packets_cmd
);
8876 install_element(CONFIG_NODE
, &debug_mtrace_cmd
);
8877 install_element(CONFIG_NODE
, &no_debug_mtrace_cmd
);
8879 install_element(CONFIG_NODE
, &ip_msdp_mesh_group_member_cmd
);
8880 install_element(VRF_NODE
, &ip_msdp_mesh_group_member_cmd
);
8881 install_element(CONFIG_NODE
, &no_ip_msdp_mesh_group_member_cmd
);
8882 install_element(VRF_NODE
, &no_ip_msdp_mesh_group_member_cmd
);
8883 install_element(CONFIG_NODE
, &ip_msdp_mesh_group_source_cmd
);
8884 install_element(VRF_NODE
, &ip_msdp_mesh_group_source_cmd
);
8885 install_element(CONFIG_NODE
, &no_ip_msdp_mesh_group_source_cmd
);
8886 install_element(VRF_NODE
, &no_ip_msdp_mesh_group_source_cmd
);
8887 install_element(VIEW_NODE
, &show_ip_msdp_peer_detail_cmd
);
8888 install_element(VIEW_NODE
, &show_ip_msdp_peer_detail_vrf_all_cmd
);
8889 install_element(VIEW_NODE
, &show_ip_msdp_sa_detail_cmd
);
8890 install_element(VIEW_NODE
, &show_ip_msdp_sa_detail_vrf_all_cmd
);
8891 install_element(VIEW_NODE
, &show_ip_msdp_sa_sg_cmd
);
8892 install_element(VIEW_NODE
, &show_ip_msdp_sa_sg_vrf_all_cmd
);
8893 install_element(VIEW_NODE
, &show_ip_msdp_mesh_group_cmd
);
8894 install_element(VIEW_NODE
, &show_ip_msdp_mesh_group_vrf_all_cmd
);
8895 install_element(VIEW_NODE
, &show_ip_pim_ssm_range_cmd
);
8896 install_element(VIEW_NODE
, &show_ip_pim_group_type_cmd
);
8897 install_element(INTERFACE_NODE
, &interface_pim_use_source_cmd
);
8898 install_element(INTERFACE_NODE
, &interface_no_pim_use_source_cmd
);
8899 /* Install BFD command */
8900 install_element(INTERFACE_NODE
, &ip_pim_bfd_cmd
);
8901 install_element(INTERFACE_NODE
, &ip_pim_bfd_param_cmd
);
8902 install_element(INTERFACE_NODE
, &no_ip_pim_bfd_cmd
);
8903 install_element(INTERFACE_NODE
, &no_ip_pim_bfd_param_cmd
);