]> git.proxmox.com Git - mirror_frr.git/blob - pimd/pim_cmd.c
Merge remote-tracking branch 'origin/stable/2.0'
[mirror_frr.git] / pimd / pim_cmd.c
1 /*
2 * PIM for Quagga
3 * Copyright (C) 2008 Everton da Silva Marques
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License along
16 * with this program; see the file COPYING; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18 */
19
20 #include <zebra.h>
21
22 #include "lib/json.h"
23 #include "command.h"
24 #include "if.h"
25 #include "prefix.h"
26 #include "zclient.h"
27 #include "plist.h"
28 #include "hash.h"
29 #include "nexthop.h"
30
31 #include "pimd.h"
32 #include "pim_mroute.h"
33 #include "pim_cmd.h"
34 #include "pim_iface.h"
35 #include "pim_vty.h"
36 #include "pim_mroute.h"
37 #include "pim_str.h"
38 #include "pim_igmp.h"
39 #include "pim_igmpv3.h"
40 #include "pim_sock.h"
41 #include "pim_time.h"
42 #include "pim_util.h"
43 #include "pim_oil.h"
44 #include "pim_neighbor.h"
45 #include "pim_pim.h"
46 #include "pim_ifchannel.h"
47 #include "pim_hello.h"
48 #include "pim_msg.h"
49 #include "pim_upstream.h"
50 #include "pim_rpf.h"
51 #include "pim_macro.h"
52 #include "pim_ssmpingd.h"
53 #include "pim_zebra.h"
54 #include "pim_static.h"
55 #include "pim_rp.h"
56 #include "pim_zlookup.h"
57 #include "pim_msdp.h"
58 #include "pim_ssm.h"
59 #include "pim_nht.h"
60
61 static struct cmd_node pim_global_node = {
62 PIM_NODE,
63 "",
64 1 /* vtysh ? yes */
65 };
66
67 static struct cmd_node interface_node = {
68 INTERFACE_NODE,
69 "%s(config-if)# ",
70 1 /* vtysh ? yes */
71 };
72
73 static struct cmd_node debug_node =
74 {
75 DEBUG_NODE,
76 "",
77 1
78 };
79
80 static void pim_if_membership_clear(struct interface *ifp)
81 {
82 struct pim_interface *pim_ifp;
83
84 pim_ifp = ifp->info;
85 zassert(pim_ifp);
86
87 if (PIM_IF_TEST_PIM(pim_ifp->options) &&
88 PIM_IF_TEST_IGMP(pim_ifp->options)) {
89 return;
90 }
91
92 pim_ifchannel_membership_clear(ifp);
93 }
94
95 /*
96 When PIM is disabled on interface, IGMPv3 local membership
97 information is not injected into PIM interface state.
98
99 The function pim_if_membership_refresh() fetches all IGMPv3 local
100 membership information into PIM. It is intented to be called
101 whenever PIM is enabled on the interface in order to collect missed
102 local membership information.
103 */
104 static void pim_if_membership_refresh(struct interface *ifp)
105 {
106 struct pim_interface *pim_ifp;
107 struct listnode *sock_node;
108 struct igmp_sock *igmp;
109
110 pim_ifp = ifp->info;
111 zassert(pim_ifp);
112
113 if (!PIM_IF_TEST_PIM(pim_ifp->options))
114 return;
115 if (!PIM_IF_TEST_IGMP(pim_ifp->options))
116 return;
117
118 /*
119 First clear off membership from all PIM (S,G) entries on the
120 interface
121 */
122
123 pim_ifchannel_membership_clear(ifp);
124
125 /*
126 Then restore PIM (S,G) membership from all IGMPv3 (S,G) entries on
127 the interface
128 */
129
130 /* scan igmp sockets */
131 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
132 struct listnode *grpnode;
133 struct igmp_group *grp;
134
135 /* scan igmp groups */
136 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list, grpnode, grp)) {
137 struct listnode *srcnode;
138 struct igmp_source *src;
139
140 /* scan group sources */
141 for (ALL_LIST_ELEMENTS_RO(grp->group_source_list, srcnode, src)) {
142
143 if (IGMP_SOURCE_TEST_FORWARDING(src->source_flags)) {
144 struct prefix_sg sg;
145
146 memset (&sg, 0, sizeof (struct prefix_sg));
147 sg.src = src->source_addr;
148 sg.grp = grp->group_addr;
149 pim_ifchannel_local_membership_add(ifp, &sg);
150 }
151
152 } /* scan group sources */
153 } /* scan igmp groups */
154 } /* scan igmp sockets */
155
156 /*
157 Finally delete every PIM (S,G) entry lacking all state info
158 */
159
160 pim_ifchannel_delete_on_noinfo(ifp);
161
162 }
163
164 static void pim_show_assert(struct vty *vty)
165 {
166 struct pim_interface *pim_ifp;
167 struct pim_ifchannel *ch;
168 struct listnode *ch_node;
169 struct in_addr ifaddr;
170 time_t now;
171
172 now = pim_time_monotonic_sec();
173
174 vty_out(vty,
175 "Interface Address Source Group State Winner Uptime Timer%s",
176 VTY_NEWLINE);
177
178 for (ALL_LIST_ELEMENTS_RO(pim_ifchannel_list, ch_node, ch)) {
179 char ch_src_str[INET_ADDRSTRLEN];
180 char ch_grp_str[INET_ADDRSTRLEN];
181 char winner_str[INET_ADDRSTRLEN];
182 char uptime[10];
183 char timer[10];
184
185 pim_ifp = ch->interface->info;
186
187 if (!pim_ifp)
188 continue;
189
190 ifaddr = pim_ifp->primary_address;
191
192 pim_inet4_dump("<ch_src?>", ch->sg.src,
193 ch_src_str, sizeof(ch_src_str));
194 pim_inet4_dump("<ch_grp?>", ch->sg.grp,
195 ch_grp_str, sizeof(ch_grp_str));
196 pim_inet4_dump("<assrt_win?>", ch->ifassert_winner,
197 winner_str, sizeof(winner_str));
198
199 pim_time_uptime(uptime, sizeof(uptime), now - ch->ifassert_creation);
200 pim_time_timer_to_mmss(timer, sizeof(timer),
201 ch->t_ifassert_timer);
202
203 vty_out(vty, "%-9s %-15s %-15s %-15s %-6s %-15s %-8s %-5s%s",
204 ch->interface->name,
205 inet_ntoa(ifaddr),
206 ch_src_str,
207 ch_grp_str,
208 pim_ifchannel_ifassert_name(ch->ifassert_state),
209 winner_str,
210 uptime,
211 timer,
212 VTY_NEWLINE);
213 } /* scan interface channels */
214 }
215
216 static void pim_show_assert_internal(struct vty *vty)
217 {
218 struct pim_interface *pim_ifp;
219 struct listnode *ch_node;
220 struct pim_ifchannel *ch;
221 struct in_addr ifaddr;
222
223 vty_out(vty,
224 "CA: CouldAssert%s"
225 "ECA: Evaluate CouldAssert%s"
226 "ATD: AssertTrackingDesired%s"
227 "eATD: Evaluate AssertTrackingDesired%s%s",
228 VTY_NEWLINE, VTY_NEWLINE, VTY_NEWLINE, VTY_NEWLINE, VTY_NEWLINE);
229
230 vty_out(vty,
231 "Interface Address Source Group CA eCA ATD eATD%s",
232 VTY_NEWLINE);
233
234 for (ALL_LIST_ELEMENTS_RO(pim_ifchannel_list, ch_node, ch)) {
235 pim_ifp = ch->interface->info;
236
237 if (!pim_ifp)
238 continue;
239
240 ifaddr = pim_ifp->primary_address;
241
242 char ch_src_str[INET_ADDRSTRLEN];
243 char ch_grp_str[INET_ADDRSTRLEN];
244
245 pim_inet4_dump("<ch_src?>", ch->sg.src,
246 ch_src_str, sizeof(ch_src_str));
247 pim_inet4_dump("<ch_grp?>", ch->sg.grp,
248 ch_grp_str, sizeof(ch_grp_str));
249 vty_out(vty, "%-9s %-15s %-15s %-15s %-3s %-3s %-3s %-4s%s",
250 ch->interface->name,
251 inet_ntoa(ifaddr),
252 ch_src_str,
253 ch_grp_str,
254 PIM_IF_FLAG_TEST_COULD_ASSERT(ch->flags) ? "yes" : "no",
255 pim_macro_ch_could_assert_eval(ch) ? "yes" : "no",
256 PIM_IF_FLAG_TEST_ASSERT_TRACKING_DESIRED(ch->flags) ? "yes" : "no",
257 pim_macro_assert_tracking_desired_eval(ch) ? "yes" : "no",
258 VTY_NEWLINE);
259 } /* scan interface channels */
260 }
261
262 static void pim_show_assert_metric(struct vty *vty)
263 {
264 struct pim_interface *pim_ifp;
265 struct listnode *ch_node;
266 struct pim_ifchannel *ch;
267 struct in_addr ifaddr;
268
269 vty_out(vty,
270 "Interface Address Source Group RPT Pref Metric Address %s",
271 VTY_NEWLINE);
272
273 for (ALL_LIST_ELEMENTS_RO(pim_ifchannel_list, ch_node, ch)) {
274 pim_ifp = ch->interface->info;
275
276 if (!pim_ifp)
277 continue;
278
279 ifaddr = pim_ifp->primary_address;
280
281 char ch_src_str[INET_ADDRSTRLEN];
282 char ch_grp_str[INET_ADDRSTRLEN];
283 char addr_str[INET_ADDRSTRLEN];
284 struct pim_assert_metric am;
285
286 am = pim_macro_spt_assert_metric(&ch->upstream->rpf, pim_ifp->primary_address);
287
288 pim_inet4_dump("<ch_src?>", ch->sg.src,
289 ch_src_str, sizeof(ch_src_str));
290 pim_inet4_dump("<ch_grp?>", ch->sg.grp,
291 ch_grp_str, sizeof(ch_grp_str));
292 pim_inet4_dump("<addr?>", am.ip_address,
293 addr_str, sizeof(addr_str));
294
295 vty_out(vty, "%-9s %-15s %-15s %-15s %-3s %4u %6u %-15s%s",
296 ch->interface->name,
297 inet_ntoa(ifaddr),
298 ch_src_str,
299 ch_grp_str,
300 am.rpt_bit_flag ? "yes" : "no",
301 am.metric_preference,
302 am.route_metric,
303 addr_str,
304 VTY_NEWLINE);
305 } /* scan interface channels */
306 }
307
308 static void pim_show_assert_winner_metric(struct vty *vty)
309 {
310 struct pim_interface *pim_ifp;
311 struct listnode *ch_node;
312 struct pim_ifchannel *ch;
313 struct in_addr ifaddr;
314
315 vty_out(vty,
316 "Interface Address Source Group RPT Pref Metric Address %s",
317 VTY_NEWLINE);
318
319 for (ALL_LIST_ELEMENTS_RO(pim_ifchannel_list, ch_node, ch)) {
320 pim_ifp = ch->interface->info;
321
322 if (!pim_ifp)
323 continue;
324
325 ifaddr = pim_ifp->primary_address;
326
327 char ch_src_str[INET_ADDRSTRLEN];
328 char ch_grp_str[INET_ADDRSTRLEN];
329 char addr_str[INET_ADDRSTRLEN];
330 struct pim_assert_metric *am;
331 char pref_str[5];
332 char metr_str[7];
333
334 am = &ch->ifassert_winner_metric;
335
336 pim_inet4_dump("<ch_src?>", ch->sg.src,
337 ch_src_str, sizeof(ch_src_str));
338 pim_inet4_dump("<ch_grp?>", ch->sg.grp,
339 ch_grp_str, sizeof(ch_grp_str));
340 pim_inet4_dump("<addr?>", am->ip_address,
341 addr_str, sizeof(addr_str));
342
343 if (am->metric_preference == PIM_ASSERT_METRIC_PREFERENCE_MAX)
344 snprintf(pref_str, sizeof(pref_str), "INFI");
345 else
346 snprintf(pref_str, sizeof(pref_str), "%4u", am->metric_preference);
347
348 if (am->route_metric == PIM_ASSERT_ROUTE_METRIC_MAX)
349 snprintf(metr_str, sizeof(metr_str), "INFI");
350 else
351 snprintf(metr_str, sizeof(metr_str), "%6u", am->route_metric);
352
353 vty_out(vty, "%-9s %-15s %-15s %-15s %-3s %-4s %-6s %-15s%s",
354 ch->interface->name,
355 inet_ntoa(ifaddr),
356 ch_src_str,
357 ch_grp_str,
358 am->rpt_bit_flag ? "yes" : "no",
359 pref_str,
360 metr_str,
361 addr_str,
362 VTY_NEWLINE);
363 } /* scan interface channels */
364 }
365
366 static void json_object_pim_ifp_add(struct json_object *json, struct interface *ifp)
367 {
368 struct pim_interface *pim_ifp;
369
370 pim_ifp = ifp->info;
371 json_object_string_add(json, "name", ifp->name);
372 json_object_string_add(json, "state", if_is_up(ifp) ? "up" : "down");
373 json_object_string_add(json, "address", inet_ntoa(pim_ifp->primary_address));
374 json_object_int_add(json, "index", ifp->ifindex);
375
376 if (if_is_multicast(ifp))
377 json_object_boolean_true_add(json, "flagMulticast");
378
379 if (if_is_broadcast(ifp))
380 json_object_boolean_true_add(json, "flagBroadcast");
381
382 if (ifp->flags & IFF_ALLMULTI)
383 json_object_boolean_true_add(json, "flagAllMulticast");
384
385 if (ifp->flags & IFF_PROMISC)
386 json_object_boolean_true_add(json, "flagPromiscuous");
387
388 if (PIM_IF_IS_DELETED(ifp))
389 json_object_boolean_true_add(json, "flagDeleted");
390
391 if (pim_if_lan_delay_enabled(ifp))
392 json_object_boolean_true_add(json, "lanDelayEnabled");
393 }
394
395 static void pim_show_membership(struct vty *vty, u_char uj)
396 {
397 struct pim_interface *pim_ifp;
398 struct listnode *ch_node;
399 struct pim_ifchannel *ch;
400 enum json_type type;
401 json_object *json = NULL;
402 json_object *json_iface = NULL;
403 json_object *json_row = NULL;
404 json_object *json_tmp = NULL;
405
406 json = json_object_new_object();
407
408 for (ALL_LIST_ELEMENTS_RO(pim_ifchannel_list, ch_node, ch)) {
409
410 pim_ifp = ch->interface->info;
411
412 if (!pim_ifp)
413 continue;
414
415 char ch_src_str[INET_ADDRSTRLEN];
416 char ch_grp_str[INET_ADDRSTRLEN];
417
418 pim_inet4_dump("<ch_src?>", ch->sg.src,
419 ch_src_str, sizeof(ch_src_str));
420 pim_inet4_dump("<ch_grp?>", ch->sg.grp,
421 ch_grp_str, sizeof(ch_grp_str));
422
423 json_object_object_get_ex(json, ch->interface->name, &json_iface);
424
425 if (!json_iface) {
426 json_iface = json_object_new_object();
427 json_object_pim_ifp_add(json_iface, ch->interface);
428 json_object_object_add(json, ch->interface->name, json_iface);
429 }
430
431 json_row = json_object_new_object();
432 json_object_string_add(json_row, "source", ch_src_str);
433 json_object_string_add(json_row, "group", ch_grp_str);
434 json_object_string_add(json_row, "localMembership",
435 ch->local_ifmembership == PIM_IFMEMBERSHIP_NOINFO ? "NOINFO" : "INCLUDE");
436 json_object_object_add(json_iface, ch_grp_str, json_row);
437 } /* scan interface channels */
438
439 if (uj) {
440 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
441 } else {
442 vty_out(vty,
443 "Interface Address Source Group Membership%s",
444 VTY_NEWLINE);
445
446 /*
447 * Example of the json data we are traversing
448 *
449 * {
450 * "swp3":{
451 * "name":"swp3",
452 * "state":"up",
453 * "address":"10.1.20.1",
454 * "index":5,
455 * "flagMulticast":true,
456 * "flagBroadcast":true,
457 * "lanDelayEnabled":true,
458 * "226.10.10.10":{
459 * "source":"*",
460 * "group":"226.10.10.10",
461 * "localMembership":"INCLUDE"
462 * }
463 * }
464 * }
465 */
466
467 /* foreach interface */
468 json_object_object_foreach(json, key, val) {
469
470 /* Find all of the keys where the val is an object. In the example
471 * above the only one is 226.10.10.10
472 */
473 json_object_object_foreach(val, if_field_key, if_field_val) {
474 type = json_object_get_type(if_field_val);
475
476 if (type == json_type_object) {
477 vty_out(vty, "%-9s ", key);
478
479 json_object_object_get_ex(val, "address", &json_tmp);
480 vty_out(vty, "%-15s ", json_object_get_string(json_tmp));
481
482 json_object_object_get_ex(if_field_val, "source", &json_tmp);
483 vty_out(vty, "%-15s ", json_object_get_string(json_tmp));
484
485 /* Group */
486 vty_out(vty, "%-15s ", if_field_key);
487
488 json_object_object_get_ex(if_field_val, "localMembership", &json_tmp);
489 vty_out(vty, "%-10s%s", json_object_get_string(json_tmp), VTY_NEWLINE);
490 }
491 }
492 }
493 }
494
495 json_object_free(json);
496 }
497
498 static void pim_print_ifp_flags(struct vty *vty, struct interface *ifp, int mloop)
499 {
500 vty_out(vty, "Flags%s", VTY_NEWLINE);
501 vty_out(vty, "-----%s", VTY_NEWLINE);
502 vty_out(vty, "All Multicast : %s%s", (ifp->flags & IFF_ALLMULTI) ? "yes" : "no", VTY_NEWLINE);
503 vty_out(vty, "Broadcast : %s%s", if_is_broadcast(ifp)? "yes" : "no", VTY_NEWLINE);
504 vty_out(vty, "Deleted : %s%s", PIM_IF_IS_DELETED(ifp) ? "yes" : "no", VTY_NEWLINE);
505 vty_out(vty, "Interface Index : %d%s", ifp->ifindex, VTY_NEWLINE);
506 vty_out(vty, "Multicast : %s%s", if_is_multicast(ifp) ? "yes" : "no", VTY_NEWLINE);
507 vty_out(vty, "Multicast Loop : %d%s", mloop, VTY_NEWLINE);
508 vty_out(vty, "Promiscuous : %s%s", (ifp->flags & IFF_PROMISC) ? "yes" : "no", VTY_NEWLINE);
509 vty_out(vty, "%s", VTY_NEWLINE);
510 vty_out(vty, "%s", VTY_NEWLINE);
511 }
512
513 static void igmp_show_interfaces(struct vty *vty, u_char uj)
514 {
515 struct listnode *node;
516 struct interface *ifp;
517 time_t now;
518 json_object *json = NULL;
519 json_object *json_row = NULL;
520
521 now = pim_time_monotonic_sec();
522
523 if (uj)
524 json = json_object_new_object();
525 else
526 vty_out(vty,
527 "Interface State Address V Querier Query Timer Uptime%s",
528 VTY_NEWLINE);
529
530 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
531 struct pim_interface *pim_ifp;
532 struct listnode *sock_node;
533 struct igmp_sock *igmp;
534
535 pim_ifp = ifp->info;
536
537 if (!pim_ifp)
538 continue;
539
540 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
541 char uptime[10];
542 char query_hhmmss[10];
543
544 pim_time_uptime(uptime, sizeof(uptime), now - igmp->sock_creation);
545 pim_time_timer_to_hhmmss(query_hhmmss, sizeof(query_hhmmss), igmp->t_igmp_query_timer);
546
547 if (uj) {
548 json_row = json_object_new_object();
549 json_object_pim_ifp_add(json_row, ifp);
550 json_object_string_add(json_row, "upTime", uptime);
551 json_object_int_add(json_row, "version", pim_ifp->igmp_version);
552
553 if (igmp->t_igmp_query_timer) {
554 json_object_boolean_true_add(json_row, "querier");
555 json_object_string_add(json_row, "queryTimer", query_hhmmss);
556 }
557
558 json_object_object_add(json, ifp->name, json_row);
559
560 } else {
561 vty_out(vty, "%-9s %5s %15s %d %7s %11s %8s%s",
562 ifp->name,
563 if_is_up(ifp) ? "up" : "down",
564 inet_ntoa(igmp->ifaddr),
565 pim_ifp->igmp_version,
566 igmp->t_igmp_query_timer ? "local" : "other",
567 query_hhmmss,
568 uptime,
569 VTY_NEWLINE);
570 }
571 }
572 }
573
574 if (uj) {
575 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
576 json_object_free(json);
577 }
578 }
579
580 static void igmp_show_interfaces_single(struct vty *vty, const char *ifname, u_char uj)
581 {
582 struct igmp_sock *igmp;
583 struct interface *ifp;
584 struct listnode *node;
585 struct listnode *sock_node;
586 struct pim_interface *pim_ifp;
587 char uptime[10];
588 char query_hhmmss[10];
589 char other_hhmmss[10];
590 int found_ifname = 0;
591 int sqi;
592 int mloop;
593 long gmi_msec; /* Group Membership Interval */
594 long lmqt_msec;
595 long ohpi_msec;
596 long oqpi_msec; /* Other Querier Present Interval */
597 long qri_msec;
598 time_t now;
599
600 json_object *json = NULL;
601 json_object *json_row = NULL;
602
603 if (uj)
604 json = json_object_new_object();
605
606 now = pim_time_monotonic_sec();
607
608 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
609 pim_ifp = ifp->info;
610
611 if (!pim_ifp)
612 continue;
613
614 if (strcmp(ifname, "detail") && strcmp(ifname, ifp->name))
615 continue;
616
617 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
618 found_ifname = 1;
619 pim_time_uptime(uptime, sizeof(uptime), now - igmp->sock_creation);
620 pim_time_timer_to_hhmmss(query_hhmmss, sizeof(query_hhmmss), igmp->t_igmp_query_timer);
621 pim_time_timer_to_hhmmss(other_hhmmss, sizeof(other_hhmmss), igmp->t_other_querier_timer);
622
623 gmi_msec = PIM_IGMP_GMI_MSEC(igmp->querier_robustness_variable,
624 igmp->querier_query_interval,
625 pim_ifp->igmp_query_max_response_time_dsec);
626
627 sqi = PIM_IGMP_SQI(pim_ifp->igmp_default_query_interval);
628
629 oqpi_msec = PIM_IGMP_OQPI_MSEC(igmp->querier_robustness_variable,
630 igmp->querier_query_interval,
631 pim_ifp->igmp_query_max_response_time_dsec);
632
633 lmqt_msec = PIM_IGMP_LMQT_MSEC(pim_ifp->igmp_query_max_response_time_dsec,
634 igmp->querier_robustness_variable);
635
636 ohpi_msec = PIM_IGMP_OHPI_DSEC(igmp->querier_robustness_variable,
637 igmp->querier_query_interval,
638 pim_ifp->igmp_query_max_response_time_dsec) * 100;
639
640 qri_msec = pim_ifp->igmp_query_max_response_time_dsec * 100;
641 mloop = pim_socket_mcastloop_get(pim_ifp->pim_sock_fd);
642
643 if (uj) {
644 json_row = json_object_new_object();
645 json_object_pim_ifp_add(json_row, ifp);
646 json_object_string_add(json_row, "upTime", uptime);
647 json_object_string_add(json_row, "querier", igmp->t_igmp_query_timer ? "local" : "other");
648 json_object_int_add(json_row, "queryStartCount", igmp->startup_query_count);
649 json_object_string_add(json_row, "queryQueryTimer", query_hhmmss);
650 json_object_string_add(json_row, "queryOtherTimer", other_hhmmss);
651 json_object_int_add(json_row, "version", pim_ifp->igmp_version);
652 json_object_int_add(json_row, "timerGroupMembershipIntervalMsec", gmi_msec);
653 json_object_int_add(json_row, "timerLastMemberQueryMsec", lmqt_msec);
654 json_object_int_add(json_row, "timerOlderHostPresentIntervalMsec", ohpi_msec);
655 json_object_int_add(json_row, "timerOtherQuerierPresentIntervalMsec", oqpi_msec);
656 json_object_int_add(json_row, "timerQueryInterval", igmp->querier_query_interval);
657 json_object_int_add(json_row, "timerQueryResponseIntervalMsec", qri_msec);
658 json_object_int_add(json_row, "timerRobustnessVariable", igmp->querier_robustness_variable);
659 json_object_int_add(json_row, "timerStartupQueryInterval", sqi);
660
661 json_object_object_add(json, ifp->name, json_row);
662
663 } else {
664 vty_out(vty, "Interface : %s%s", ifp->name, VTY_NEWLINE);
665 vty_out(vty, "State : %s%s", if_is_up(ifp) ? "up" : "down", VTY_NEWLINE);
666 vty_out(vty, "Address : %s%s", inet_ntoa(pim_ifp->primary_address), VTY_NEWLINE);
667 vty_out(vty, "Uptime : %s%s", uptime, VTY_NEWLINE);
668 vty_out(vty, "Version : %d%s", pim_ifp->igmp_version, VTY_NEWLINE);
669 vty_out(vty, "%s", VTY_NEWLINE);
670 vty_out(vty, "%s", VTY_NEWLINE);
671
672 vty_out(vty, "Querier%s", VTY_NEWLINE);
673 vty_out(vty, "-------%s", VTY_NEWLINE);
674 vty_out(vty, "Querier : %s%s", igmp->t_igmp_query_timer ? "local" : "other", VTY_NEWLINE);
675 vty_out(vty, "Start Count : %d%s", igmp->startup_query_count, VTY_NEWLINE);
676 vty_out(vty, "Query Timer : %s%s", query_hhmmss, VTY_NEWLINE);
677 vty_out(vty, "Other Timer : %s%s", other_hhmmss, VTY_NEWLINE);
678 vty_out(vty, "%s", VTY_NEWLINE);
679 vty_out(vty, "%s", VTY_NEWLINE);
680
681 vty_out(vty, "Timers%s", VTY_NEWLINE);
682 vty_out(vty, "------%s", VTY_NEWLINE);
683 vty_out(vty, "Group Membership Interval : %lis%s", gmi_msec/1000, VTY_NEWLINE);
684 vty_out(vty, "Last Member Query Time : %lis%s", lmqt_msec/1000, VTY_NEWLINE);
685 vty_out(vty, "Older Host Present Interval : %lis%s", ohpi_msec/1000, VTY_NEWLINE);
686 vty_out(vty, "Other Querier Present Interval : %lis%s", oqpi_msec/1000, VTY_NEWLINE);
687 vty_out(vty, "Query Interval : %ds%s", igmp->querier_query_interval, VTY_NEWLINE);
688 vty_out(vty, "Query Response Interval : %lis%s", qri_msec/1000, VTY_NEWLINE);
689 vty_out(vty, "Robustness Variable : %d%s", igmp->querier_robustness_variable, VTY_NEWLINE);
690 vty_out(vty, "Startup Query Interval : %ds%s", sqi, VTY_NEWLINE);
691 vty_out(vty, "%s", VTY_NEWLINE);
692 vty_out(vty, "%s", VTY_NEWLINE);
693
694 pim_print_ifp_flags(vty, ifp, mloop);
695 }
696 }
697 }
698
699 if (uj) {
700 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
701 json_object_free(json);
702 } else {
703 if (!found_ifname)
704 vty_out (vty, "%% No such interface%s", VTY_NEWLINE);
705 }
706 }
707
708 static void igmp_show_interface_join(struct vty *vty)
709 {
710 struct listnode *node;
711 struct interface *ifp;
712 time_t now;
713
714 now = pim_time_monotonic_sec();
715
716 vty_out(vty,
717 "Interface Address Source Group Socket Uptime %s",
718 VTY_NEWLINE);
719
720 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
721 struct pim_interface *pim_ifp;
722 struct listnode *join_node;
723 struct igmp_join *ij;
724 struct in_addr pri_addr;
725 char pri_addr_str[INET_ADDRSTRLEN];
726
727 pim_ifp = ifp->info;
728
729 if (!pim_ifp)
730 continue;
731
732 if (!pim_ifp->igmp_join_list)
733 continue;
734
735 pri_addr = pim_find_primary_addr(ifp);
736 pim_inet4_dump("<pri?>", pri_addr, pri_addr_str, sizeof(pri_addr_str));
737
738 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_join_list, join_node, ij)) {
739 char group_str[INET_ADDRSTRLEN];
740 char source_str[INET_ADDRSTRLEN];
741 char uptime[10];
742
743 pim_time_uptime(uptime, sizeof(uptime), now - ij->sock_creation);
744 pim_inet4_dump("<grp?>", ij->group_addr, group_str, sizeof(group_str));
745 pim_inet4_dump("<src?>", ij->source_addr, source_str, sizeof(source_str));
746
747 vty_out(vty, "%-9s %-15s %-15s %-15s %6d %8s%s",
748 ifp->name,
749 pri_addr_str,
750 source_str,
751 group_str,
752 ij->sock_fd,
753 uptime,
754 VTY_NEWLINE);
755 } /* for (pim_ifp->igmp_join_list) */
756
757 } /* for (iflist) */
758
759 }
760
761 static void pim_show_interfaces_single(struct vty *vty, const char *ifname, u_char uj)
762 {
763 struct in_addr ifaddr;
764 struct interface *ifp;
765 struct listnode *neighnode;
766 struct listnode*node;
767 struct listnode *upnode;
768 struct pim_interface *pim_ifp;
769 struct pim_neighbor *neigh;
770 struct pim_upstream *up;
771 time_t now;
772 char dr_str[INET_ADDRSTRLEN];
773 char dr_uptime[10];
774 char expire[10];
775 char grp_str[INET_ADDRSTRLEN];
776 char hello_period[10];
777 char hello_timer[10];
778 char neigh_src_str[INET_ADDRSTRLEN];
779 char src_str[INET_ADDRSTRLEN];
780 char stat_uptime[10];
781 char uptime[10];
782 int mloop;
783 int found_ifname = 0;
784 int print_header;
785 json_object *json = NULL;
786 json_object *json_row = NULL;
787 json_object *json_pim_neighbor = NULL;
788 json_object *json_pim_neighbors = NULL;
789 json_object *json_group = NULL;
790 json_object *json_group_source = NULL;
791 json_object *json_fhr_sources = NULL;
792 struct pim_secondary_addr *sec_addr;
793 struct listnode *sec_node;
794
795 now = pim_time_monotonic_sec();
796
797 if (uj)
798 json = json_object_new_object();
799
800 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
801 pim_ifp = ifp->info;
802
803 if (!pim_ifp)
804 continue;
805
806 if (pim_ifp->pim_sock_fd < 0)
807 continue;
808
809 if (strcmp(ifname, "detail") && strcmp(ifname, ifp->name))
810 continue;
811
812 found_ifname = 1;
813 ifaddr = pim_ifp->primary_address;
814 pim_inet4_dump("<dr?>", pim_ifp->pim_dr_addr, dr_str, sizeof(dr_str));
815 pim_time_uptime_begin(dr_uptime, sizeof(dr_uptime), now, pim_ifp->pim_dr_election_last);
816 pim_time_timer_to_hhmmss(hello_timer, sizeof(hello_timer), pim_ifp->t_pim_hello_timer);
817 pim_time_mmss(hello_period, sizeof(hello_period), pim_ifp->pim_hello_period);
818 pim_time_uptime(stat_uptime, sizeof(stat_uptime), now - pim_ifp->pim_ifstat_start);
819 mloop = pim_socket_mcastloop_get(pim_ifp->pim_sock_fd);
820
821 if (uj) {
822 char pbuf[PREFIX2STR_BUFFER];
823 json_row = json_object_new_object();
824 json_object_pim_ifp_add(json_row, ifp);
825
826 if (pim_ifp->update_source.s_addr != INADDR_ANY) {
827 json_object_string_add(json_row, "useSource", inet_ntoa(pim_ifp->update_source));
828 }
829 if (pim_ifp->sec_addr_list) {
830 json_object *sec_list = NULL;
831
832 sec_list = json_object_new_array();
833 for (ALL_LIST_ELEMENTS_RO(pim_ifp->sec_addr_list, sec_node, sec_addr)) {
834 json_object_array_add(sec_list,
835 json_object_new_string(prefix2str(&sec_addr->addr,
836 pbuf,
837 sizeof(pbuf))));
838 }
839 json_object_object_add(json_row, "secondaryAddressList", sec_list);
840 }
841
842 // PIM neighbors
843 if (pim_ifp->pim_neighbor_list->count) {
844 json_pim_neighbors = json_object_new_object();
845
846 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list, neighnode, neigh)) {
847 json_pim_neighbor = json_object_new_object();
848 pim_inet4_dump("<src?>", neigh->source_addr, neigh_src_str, sizeof(neigh_src_str));
849 pim_time_uptime(uptime, sizeof(uptime), now - neigh->creation);
850 pim_time_timer_to_hhmmss(expire, sizeof(expire), neigh->t_expire_timer);
851
852 json_object_string_add(json_pim_neighbor, "address", neigh_src_str);
853 json_object_string_add(json_pim_neighbor, "upTime", uptime);
854 json_object_string_add(json_pim_neighbor, "holdtime", expire);
855
856 json_object_object_add(json_pim_neighbors, neigh_src_str, json_pim_neighbor);
857 }
858
859 json_object_object_add(json_row, "neighbors", json_pim_neighbors);
860 }
861
862 json_object_string_add(json_row, "drAddress", dr_str);
863 json_object_int_add(json_row, "drPriority", pim_ifp->pim_dr_priority);
864 json_object_string_add(json_row, "drUptime", dr_uptime);
865 json_object_int_add(json_row, "drElections", pim_ifp->pim_dr_election_count);
866 json_object_int_add(json_row, "drChanges", pim_ifp->pim_dr_election_changes);
867
868 // FHR
869 for (ALL_LIST_ELEMENTS_RO(pim_upstream_list, upnode, up)) {
870 if (ifp == up->rpf.source_nexthop.interface) {
871 if (up->flags & PIM_UPSTREAM_FLAG_MASK_FHR) {
872 if (!json_fhr_sources) {
873 json_fhr_sources = json_object_new_object();
874 }
875
876 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
877 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
878 pim_time_uptime(uptime, sizeof(uptime), now - up->state_transition);
879
880 /* Does this group live in json_fhr_sources? If not create it. */
881 json_object_object_get_ex(json_fhr_sources, grp_str, &json_group);
882
883 if (!json_group) {
884 json_group = json_object_new_object();
885 json_object_object_add(json_fhr_sources, grp_str, json_group);
886 }
887
888 json_group_source = json_object_new_object();
889 json_object_string_add(json_group_source, "source", src_str);
890 json_object_string_add(json_group_source, "group", grp_str);
891 json_object_string_add(json_group_source, "upTime", uptime);
892 json_object_object_add(json_group, src_str, json_group_source);
893 }
894 }
895 }
896
897 if (json_fhr_sources) {
898 json_object_object_add(json_row, "firstHopRouter", json_fhr_sources);
899 }
900
901 json_object_int_add(json_row, "helloPeriod", pim_ifp->pim_hello_period);
902 json_object_string_add(json_row, "helloTimer", hello_timer);
903 json_object_string_add(json_row, "helloStatStart", stat_uptime);
904 json_object_int_add(json_row, "helloReceived", pim_ifp->pim_ifstat_hello_recv);
905 json_object_int_add(json_row, "helloReceivedFailed", pim_ifp->pim_ifstat_hello_recvfail);
906 json_object_int_add(json_row, "helloSend", pim_ifp->pim_ifstat_hello_sent);
907 json_object_int_add(json_row, "hellosendFailed", pim_ifp->pim_ifstat_hello_sendfail);
908 json_object_int_add(json_row, "helloGenerationId", pim_ifp->pim_generation_id);
909 json_object_int_add(json_row, "flagMulticastLoop", mloop);
910
911 json_object_int_add(json_row, "effectivePropagationDelay", pim_if_effective_propagation_delay_msec(ifp));
912 json_object_int_add(json_row, "effectiveOverrideInterval", pim_if_effective_override_interval_msec(ifp));
913 json_object_int_add(json_row, "joinPruneOverrideInterval", pim_if_jp_override_interval_msec(ifp));
914
915 json_object_int_add(json_row, "propagationDelay", pim_ifp->pim_propagation_delay_msec);
916 json_object_int_add(json_row, "propagationDelayHighest", pim_ifp->pim_neighbors_highest_propagation_delay_msec);
917 json_object_int_add(json_row, "overrideInterval", pim_ifp->pim_override_interval_msec);
918 json_object_int_add(json_row, "overrideIntervalHighest", pim_ifp->pim_neighbors_highest_override_interval_msec);
919 json_object_object_add(json, ifp->name, json_row);
920
921 } else {
922 vty_out(vty, "Interface : %s%s", ifp->name, VTY_NEWLINE);
923 vty_out(vty, "State : %s%s", if_is_up(ifp) ? "up" : "down", VTY_NEWLINE);
924 if (pim_ifp->update_source.s_addr != INADDR_ANY) {
925 vty_out(vty, "Use Source : %s%s", inet_ntoa(pim_ifp->update_source), VTY_NEWLINE);
926 }
927 if (pim_ifp->sec_addr_list) {
928 char pbuf[PREFIX2STR_BUFFER];
929 vty_out(vty, "Address : %s (primary)%s",
930 inet_ntoa(ifaddr), VTY_NEWLINE);
931 for (ALL_LIST_ELEMENTS_RO(pim_ifp->sec_addr_list, sec_node, sec_addr)) {
932 vty_out(vty, " %s%s",
933 prefix2str(&sec_addr->addr,
934 pbuf,
935 sizeof(pbuf)), VTY_NEWLINE);
936 }
937 } else {
938 vty_out(vty, "Address : %s%s", inet_ntoa(ifaddr), VTY_NEWLINE);
939 }
940 vty_out(vty, "%s", VTY_NEWLINE);
941
942 // PIM neighbors
943 print_header = 1;
944
945 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list, neighnode, neigh)) {
946
947 if (print_header) {
948 vty_out(vty, "PIM Neighbors%s", VTY_NEWLINE);
949 vty_out(vty, "-------------%s", VTY_NEWLINE);
950 print_header = 0;
951 }
952
953 pim_inet4_dump("<src?>", neigh->source_addr, neigh_src_str, sizeof(neigh_src_str));
954 pim_time_uptime(uptime, sizeof(uptime), now - neigh->creation);
955 pim_time_timer_to_hhmmss(expire, sizeof(expire), neigh->t_expire_timer);
956 vty_out(vty, "%-15s : up for %s, holdtime expires in %s%s", neigh_src_str, uptime, expire, VTY_NEWLINE);
957 }
958
959 if (!print_header) {
960 vty_out(vty, "%s", VTY_NEWLINE);
961 vty_out(vty, "%s", VTY_NEWLINE);
962 }
963
964 vty_out(vty, "Designated Router%s", VTY_NEWLINE);
965 vty_out(vty, "-----------------%s", VTY_NEWLINE);
966 vty_out(vty, "Address : %s%s", dr_str, VTY_NEWLINE);
967 vty_out(vty, "Priority : %d%s", pim_ifp->pim_dr_priority, VTY_NEWLINE);
968 vty_out(vty, "Uptime : %s%s", dr_uptime, VTY_NEWLINE);
969 vty_out(vty, "Elections : %d%s", pim_ifp->pim_dr_election_count, VTY_NEWLINE);
970 vty_out(vty, "Changes : %d%s", pim_ifp->pim_dr_election_changes, VTY_NEWLINE);
971 vty_out(vty, "%s", VTY_NEWLINE);
972 vty_out(vty, "%s", VTY_NEWLINE);
973
974 // FHR
975 print_header = 1;
976 for (ALL_LIST_ELEMENTS_RO(pim_upstream_list, upnode, up)) {
977 if (strcmp(ifp->name, up->rpf.source_nexthop.interface->name) == 0) {
978 if (up->flags & PIM_UPSTREAM_FLAG_MASK_FHR) {
979
980 if (print_header) {
981 vty_out(vty, "FHR - First Hop Router%s", VTY_NEWLINE);
982 vty_out(vty, "----------------------%s", VTY_NEWLINE);
983 print_header = 0;
984 }
985
986 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
987 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
988 pim_time_uptime(uptime, sizeof(uptime), now - up->state_transition);
989 vty_out(vty, "%s : %s is a source, uptime is %s%s", grp_str, src_str, uptime, VTY_NEWLINE);
990 }
991 }
992 }
993
994 if (!print_header) {
995 vty_out(vty, "%s", VTY_NEWLINE);
996 vty_out(vty, "%s", VTY_NEWLINE);
997 }
998
999 vty_out(vty, "Hellos%s", VTY_NEWLINE);
1000 vty_out(vty, "------%s", VTY_NEWLINE);
1001 vty_out(vty, "Period : %d%s", pim_ifp->pim_hello_period, VTY_NEWLINE);
1002 vty_out(vty, "Timer : %s%s", hello_timer, VTY_NEWLINE);
1003 vty_out(vty, "StatStart : %s%s", stat_uptime, VTY_NEWLINE);
1004 vty_out(vty, "Receive : %d%s", pim_ifp->pim_ifstat_hello_recv, VTY_NEWLINE);
1005 vty_out(vty, "Receive Failed : %d%s", pim_ifp->pim_ifstat_hello_recvfail, VTY_NEWLINE);
1006 vty_out(vty, "Send : %d%s", pim_ifp->pim_ifstat_hello_sent, VTY_NEWLINE);
1007 vty_out(vty, "Send Failed : %d%s", pim_ifp->pim_ifstat_hello_sendfail, VTY_NEWLINE);
1008 vty_out(vty, "Generation ID : %08x%s", pim_ifp->pim_generation_id, VTY_NEWLINE);
1009 vty_out(vty, "%s", VTY_NEWLINE);
1010 vty_out(vty, "%s", VTY_NEWLINE);
1011
1012 pim_print_ifp_flags(vty, ifp, mloop);
1013
1014 vty_out(vty, "Join Prune Interval%s", VTY_NEWLINE);
1015 vty_out(vty, "-------------------%s", VTY_NEWLINE);
1016 vty_out(vty, "LAN Delay : %s%s", pim_if_lan_delay_enabled(ifp) ? "yes" : "no", VTY_NEWLINE);
1017 vty_out(vty, "Effective Propagation Delay : %d msec%s", pim_if_effective_propagation_delay_msec(ifp), VTY_NEWLINE);
1018 vty_out(vty, "Effective Override Interval : %d msec%s", pim_if_effective_override_interval_msec(ifp), VTY_NEWLINE);
1019 vty_out(vty, "Join Prune Override Interval : %d msec%s", pim_if_jp_override_interval_msec(ifp), VTY_NEWLINE);
1020 vty_out(vty, "%s", VTY_NEWLINE);
1021 vty_out(vty, "%s", VTY_NEWLINE);
1022
1023 vty_out(vty, "LAN Prune Delay%s", VTY_NEWLINE);
1024 vty_out(vty, "---------------%s", VTY_NEWLINE);
1025 vty_out(vty, "Propagation Delay : %d msec%s", pim_ifp->pim_propagation_delay_msec, VTY_NEWLINE);
1026 vty_out(vty, "Propagation Delay (Highest) : %d msec%s", pim_ifp->pim_neighbors_highest_propagation_delay_msec, VTY_NEWLINE);
1027 vty_out(vty, "Override Interval : %d msec%s", pim_ifp->pim_override_interval_msec, VTY_NEWLINE);
1028 vty_out(vty, "Override Interval (Highest) : %d msec%s", pim_ifp->pim_neighbors_highest_override_interval_msec, VTY_NEWLINE);
1029 vty_out(vty, "%s", VTY_NEWLINE);
1030 vty_out(vty, "%s", VTY_NEWLINE);
1031 }
1032 }
1033
1034 if (uj) {
1035 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1036 json_object_free(json);
1037 } else {
1038 if (!found_ifname)
1039 vty_out (vty, "%% No such interface%s", VTY_NEWLINE);
1040 }
1041 }
1042
1043 static void pim_show_interfaces(struct vty *vty, u_char uj)
1044 {
1045 struct interface *ifp;
1046 struct listnode *node;
1047 struct listnode *upnode;
1048 struct pim_interface *pim_ifp;
1049 struct pim_upstream *up;
1050 int fhr = 0;
1051 int pim_nbrs = 0;
1052 int pim_ifchannels = 0;
1053 json_object *json = NULL;
1054 json_object *json_row = NULL;
1055 json_object *json_tmp;
1056
1057 json = json_object_new_object();
1058
1059 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
1060 pim_ifp = ifp->info;
1061
1062 if (!pim_ifp)
1063 continue;
1064
1065 if (pim_ifp->pim_sock_fd < 0)
1066 continue;
1067
1068 pim_nbrs = pim_ifp->pim_neighbor_list->count;
1069 pim_ifchannels = pim_ifp->pim_ifchannel_list->count;
1070 fhr = 0;
1071
1072 for (ALL_LIST_ELEMENTS_RO(pim_upstream_list, upnode, up))
1073 if (ifp == up->rpf.source_nexthop.interface)
1074 if (up->flags & PIM_UPSTREAM_FLAG_MASK_FHR)
1075 fhr++;
1076
1077 json_row = json_object_new_object();
1078 json_object_pim_ifp_add(json_row, ifp);
1079 json_object_int_add(json_row, "pimNeighbors", pim_nbrs);
1080 json_object_int_add(json_row, "pimIfChannels", pim_ifchannels);
1081 json_object_int_add(json_row, "firstHopRouter", fhr);
1082 json_object_string_add(json_row, "pimDesignatedRouter", inet_ntoa(pim_ifp->pim_dr_addr));
1083
1084 if (pim_ifp->pim_dr_addr.s_addr == pim_ifp->primary_address.s_addr)
1085 json_object_boolean_true_add(json_row, "pimDesignatedRouterLocal");
1086
1087 json_object_object_add(json, ifp->name, json_row);
1088 }
1089
1090 if (uj) {
1091 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1092 } else {
1093 vty_out(vty, "Interface State Address PIM Nbrs PIM DR FHR IfChannels%s", VTY_NEWLINE);
1094
1095 json_object_object_foreach(json, key, val) {
1096 vty_out(vty, "%-9s ", key);
1097
1098 json_object_object_get_ex(val, "state", &json_tmp);
1099 vty_out(vty, "%5s ", json_object_get_string(json_tmp));
1100
1101 json_object_object_get_ex(val, "address", &json_tmp);
1102 vty_out(vty, "%15s ", json_object_get_string(json_tmp));
1103
1104 json_object_object_get_ex(val, "pimNeighbors", &json_tmp);
1105 vty_out(vty, "%8d ", json_object_get_int(json_tmp));
1106
1107 if (json_object_object_get_ex(val, "pimDesignatedRouterLocal", &json_tmp)) {
1108 vty_out(vty, "%15s ", "local");
1109 } else {
1110 json_object_object_get_ex(val, "pimDesignatedRouter", &json_tmp);
1111 vty_out(vty, "%15s ", json_object_get_string(json_tmp));
1112 }
1113
1114 json_object_object_get_ex(val, "firstHopRouter", &json_tmp);
1115 vty_out(vty, "%3d ", json_object_get_int(json_tmp));
1116
1117 json_object_object_get_ex(val, "pimIfChannels", &json_tmp);
1118 vty_out(vty, "%9d%s", json_object_get_int(json_tmp), VTY_NEWLINE);
1119 }
1120 }
1121
1122 json_object_free(json);
1123 }
1124
1125 static void pim_show_interface_traffic (struct vty *vty, u_char uj)
1126 {
1127 struct interface *ifp = NULL;
1128 struct pim_interface *pim_ifp = NULL;
1129 struct listnode *node = NULL;
1130 json_object *json = NULL;
1131 json_object *json_row = NULL;
1132
1133 if (uj)
1134 json = json_object_new_object ();
1135 else
1136 {
1137 vty_out (vty, "%s", VTY_NEWLINE);
1138 vty_out (vty, "%-12s%-17s%-17s%-17s%-17s%-17s%-17s%s", "Interface",
1139 " HELLO", " JOIN", " PRUNE", " REGISTER",
1140 " REGISTER-STOP", " ASSERT", VTY_NEWLINE);
1141 vty_out (vty,
1142 "%-10s%-18s%-17s%-17s%-17s%-17s%-17s%s",
1143 "", " Rx/Tx", " Rx/Tx", " Rx/Tx", " Rx/Tx",
1144 " Rx/Tx", " Rx/Tx", VTY_NEWLINE);
1145 vty_out (vty,
1146 "---------------------------------------------------------------------------------------------------------------%s",
1147 VTY_NEWLINE);
1148 }
1149
1150 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp))
1151 {
1152 pim_ifp = ifp->info;
1153
1154 if (!pim_ifp)
1155 continue;
1156
1157 if (pim_ifp->pim_sock_fd < 0)
1158 continue;
1159 if (uj)
1160 {
1161 json_row = json_object_new_object ();
1162 json_object_pim_ifp_add (json_row, ifp);
1163 json_object_int_add (json_row, "helloRx", pim_ifp->pim_ifstat_hello_recv);
1164 json_object_int_add (json_row, "helloTx", pim_ifp->pim_ifstat_hello_sent);
1165 json_object_int_add (json_row, "joinRx", pim_ifp->pim_ifstat_join_recv);
1166 json_object_int_add (json_row, "joinTx", pim_ifp->pim_ifstat_join_send);
1167 json_object_int_add (json_row, "registerRx", pim_ifp->pim_ifstat_reg_recv);
1168 json_object_int_add (json_row, "registerTx", pim_ifp->pim_ifstat_reg_recv);
1169 json_object_int_add (json_row, "registerStopRx", pim_ifp->pim_ifstat_reg_stop_recv);
1170 json_object_int_add (json_row, "registerStopTx", pim_ifp->pim_ifstat_reg_stop_send);
1171 json_object_int_add (json_row, "assertRx", pim_ifp->pim_ifstat_assert_recv);
1172 json_object_int_add (json_row, "assertTx", pim_ifp->pim_ifstat_assert_send);
1173
1174 json_object_object_add (json, ifp->name, json_row);
1175 }
1176 else
1177 {
1178 vty_out (vty,
1179 "%-10s %8u/%-8u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u %s",
1180 ifp->name, pim_ifp->pim_ifstat_hello_recv,
1181 pim_ifp->pim_ifstat_hello_sent, pim_ifp->pim_ifstat_join_recv,
1182 pim_ifp->pim_ifstat_join_send, pim_ifp->pim_ifstat_prune_recv,
1183 pim_ifp->pim_ifstat_prune_send, pim_ifp->pim_ifstat_reg_recv,
1184 pim_ifp->pim_ifstat_reg_send,
1185 pim_ifp->pim_ifstat_reg_stop_recv,
1186 pim_ifp->pim_ifstat_reg_stop_send,
1187 pim_ifp->pim_ifstat_assert_recv,
1188 pim_ifp->pim_ifstat_assert_send, VTY_NEWLINE);
1189 }
1190 }
1191 if (uj)
1192 {
1193 vty_out (vty, "%s%s", json_object_to_json_string_ext (json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1194 json_object_free (json);
1195 }
1196 }
1197
1198 static void pim_show_interface_traffic_single (struct vty *vty, const char *ifname, u_char uj)
1199 {
1200 struct interface *ifp = NULL;
1201 struct pim_interface *pim_ifp = NULL;
1202 struct listnode *node = NULL;
1203 json_object *json = NULL;
1204 json_object *json_row = NULL;
1205 uint8_t found_ifname = 0;
1206
1207 if (uj)
1208 json = json_object_new_object ();
1209 else
1210 {
1211 vty_out (vty, "%s", VTY_NEWLINE);
1212 vty_out (vty, "%-12s%-17s%-17s%-17s%-17s%-17s%-17s%s", "Interface",
1213 " HELLO", " JOIN", " PRUNE", " REGISTER",
1214 " REGISTER-STOP", " ASSERT", VTY_NEWLINE);
1215 vty_out (vty,
1216 "%-10s%-18s%-17s%-17s%-17s%-17s%-17s%s",
1217 "", " Rx/Tx", " Rx/Tx", " Rx/Tx", " Rx/Tx",
1218 " Rx/Tx", " Rx/Tx", VTY_NEWLINE);
1219 vty_out (vty,
1220 "---------------------------------------------------------------------------------------------------------------%s",
1221 VTY_NEWLINE);
1222 }
1223
1224 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp))
1225 {
1226 if (strcmp (ifname, ifp->name))
1227 continue;
1228
1229 pim_ifp = ifp->info;
1230
1231 if (!pim_ifp)
1232 continue;
1233
1234 if (pim_ifp->pim_sock_fd < 0)
1235 continue;
1236
1237 found_ifname = 1;
1238 if (uj)
1239 {
1240 json_row = json_object_new_object ();
1241 json_object_pim_ifp_add (json_row, ifp);
1242 json_object_int_add (json_row, "helloRx", pim_ifp->pim_ifstat_hello_recv);
1243 json_object_int_add (json_row, "helloTx", pim_ifp->pim_ifstat_hello_sent);
1244 json_object_int_add (json_row, "joinRx", pim_ifp->pim_ifstat_join_recv);
1245 json_object_int_add (json_row, "joinTx", pim_ifp->pim_ifstat_join_send);
1246 json_object_int_add (json_row, "registerRx", pim_ifp->pim_ifstat_reg_recv);
1247 json_object_int_add (json_row, "registerTx", pim_ifp->pim_ifstat_reg_recv);
1248 json_object_int_add (json_row, "registerStopRx", pim_ifp->pim_ifstat_reg_stop_recv);
1249 json_object_int_add (json_row, "registerStopTx", pim_ifp->pim_ifstat_reg_stop_send);
1250 json_object_int_add (json_row, "assertRx", pim_ifp->pim_ifstat_assert_recv);
1251 json_object_int_add (json_row, "assertTx", pim_ifp->pim_ifstat_assert_send);
1252
1253 json_object_object_add (json, ifp->name, json_row);
1254 }
1255 else
1256 {
1257 vty_out (vty,
1258 "%-10s %8u/%-8u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u %7u/%-7u %s",
1259 ifp->name, pim_ifp->pim_ifstat_hello_recv,
1260 pim_ifp->pim_ifstat_hello_sent, pim_ifp->pim_ifstat_join_recv,
1261 pim_ifp->pim_ifstat_join_send, pim_ifp->pim_ifstat_prune_recv,
1262 pim_ifp->pim_ifstat_prune_send, pim_ifp->pim_ifstat_reg_recv,
1263 pim_ifp->pim_ifstat_reg_send,
1264 pim_ifp->pim_ifstat_reg_stop_recv,
1265 pim_ifp->pim_ifstat_reg_stop_send,
1266 pim_ifp->pim_ifstat_assert_recv,
1267 pim_ifp->pim_ifstat_assert_send, VTY_NEWLINE);
1268 }
1269 }
1270 if (uj)
1271 {
1272 vty_out (vty, "%s%s", json_object_to_json_string_ext (json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1273 json_object_free (json);
1274 }
1275 else
1276 {
1277 if (!found_ifname)
1278 vty_out (vty, "%% No such interface%s", VTY_NEWLINE);
1279 }
1280 }
1281
1282 static void pim_show_join(struct vty *vty, u_char uj)
1283 {
1284 struct pim_interface *pim_ifp;
1285 struct in_addr ifaddr;
1286 struct listnode *ch_node;
1287 struct pim_ifchannel *ch;
1288 time_t now;
1289 json_object *json = NULL;
1290 json_object *json_iface = NULL;
1291 json_object *json_row = NULL;
1292 json_object *json_grp = NULL;
1293
1294 now = pim_time_monotonic_sec();
1295
1296 if (uj)
1297 json = json_object_new_object();
1298 else
1299 vty_out(vty,
1300 "Interface Address Source Group State Uptime Expire Prune%s",
1301 VTY_NEWLINE);
1302
1303 for (ALL_LIST_ELEMENTS_RO(pim_ifchannel_list, ch_node, ch)) {
1304
1305 pim_ifp = ch->interface->info;
1306
1307 if (!pim_ifp)
1308 continue;
1309
1310 ifaddr = pim_ifp->primary_address;
1311
1312 char ch_src_str[INET_ADDRSTRLEN];
1313 char ch_grp_str[INET_ADDRSTRLEN];
1314 char uptime[10];
1315 char expire[10];
1316 char prune[10];
1317
1318 pim_inet4_dump("<ch_src?>", ch->sg.src,
1319 ch_src_str, sizeof(ch_src_str));
1320 pim_inet4_dump("<ch_grp?>", ch->sg.grp,
1321 ch_grp_str, sizeof(ch_grp_str));
1322
1323 pim_time_uptime_begin(uptime, sizeof(uptime), now, ch->ifjoin_creation);
1324 pim_time_timer_to_mmss(expire, sizeof(expire),
1325 ch->t_ifjoin_expiry_timer);
1326 pim_time_timer_to_mmss(prune, sizeof(prune),
1327 ch->t_ifjoin_prune_pending_timer);
1328
1329 if (uj) {
1330 json_object_object_get_ex(json, ch->interface->name, &json_iface);
1331
1332 if (!json_iface) {
1333 json_iface = json_object_new_object();
1334 json_object_pim_ifp_add(json_iface, ch->interface);
1335 json_object_object_add(json, ch->interface->name, json_iface);
1336 }
1337
1338 json_row = json_object_new_object();
1339 json_object_string_add(json_row, "source", ch_src_str);
1340 json_object_string_add(json_row, "group", ch_grp_str);
1341 json_object_string_add(json_row, "upTime", uptime);
1342 json_object_string_add(json_row, "expire", expire);
1343 json_object_string_add(json_row, "prune", prune);
1344 json_object_string_add(json_row, "channelJoinName",
1345 pim_ifchannel_ifjoin_name(ch->ifjoin_state, ch->flags));
1346 if (PIM_IF_FLAG_TEST_S_G_RPT(ch->flags))
1347 json_object_int_add(json_row, "SGRpt", 1);
1348
1349 json_object_object_get_ex(json_iface, ch_grp_str, &json_grp);
1350 if (!json_grp)
1351 {
1352 json_grp = json_object_new_object();
1353 json_object_object_add(json_grp, ch_src_str, json_row);
1354 json_object_object_add(json_iface, ch_grp_str, json_grp);
1355 }
1356 else
1357 json_object_object_add(json_grp, ch_src_str, json_row);
1358 } else {
1359 vty_out(vty, "%-9s %-15s %-15s %-15s %-6s %8s %-6s %5s%s",
1360 ch->interface->name,
1361 inet_ntoa(ifaddr),
1362 ch_src_str,
1363 ch_grp_str,
1364 pim_ifchannel_ifjoin_name(ch->ifjoin_state, ch->flags),
1365 uptime,
1366 expire,
1367 prune,
1368 VTY_NEWLINE);
1369 }
1370 } /* scan interface channels */
1371
1372 if (uj) {
1373 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1374 json_object_free(json);
1375 }
1376 }
1377
1378 static void pim_show_neighbors_single(struct vty *vty, const char *neighbor, u_char uj)
1379 {
1380 struct listnode *node;
1381 struct listnode *neighnode;
1382 struct interface *ifp;
1383 struct pim_interface *pim_ifp;
1384 struct pim_neighbor *neigh;
1385 time_t now;
1386 int found_neighbor = 0;
1387 int option_address_list;
1388 int option_dr_priority;
1389 int option_generation_id;
1390 int option_holdtime;
1391 int option_lan_prune_delay;
1392 int option_t_bit;
1393 char uptime[10];
1394 char expire[10];
1395 char neigh_src_str[INET_ADDRSTRLEN];
1396
1397 json_object *json = NULL;
1398 json_object *json_ifp = NULL;
1399 json_object *json_row = NULL;
1400
1401 now = pim_time_monotonic_sec();
1402
1403 if (uj)
1404 json = json_object_new_object();
1405
1406 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
1407 pim_ifp = ifp->info;
1408
1409 if (!pim_ifp)
1410 continue;
1411
1412 if (pim_ifp->pim_sock_fd < 0)
1413 continue;
1414
1415 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list, neighnode, neigh)) {
1416 pim_inet4_dump("<src?>", neigh->source_addr,
1417 neigh_src_str, sizeof(neigh_src_str));
1418
1419 /*
1420 * The user can specify either the interface name or the PIM neighbor IP.
1421 * If this pim_ifp matches neither then skip.
1422 */
1423 if (strcmp(neighbor, "detail") &&
1424 strcmp(neighbor, ifp->name) &&
1425 strcmp(neighbor, neigh_src_str))
1426 continue;
1427
1428 found_neighbor = 1;
1429 pim_time_uptime(uptime, sizeof(uptime), now - neigh->creation);
1430 pim_time_timer_to_hhmmss(expire, sizeof(expire), neigh->t_expire_timer);
1431
1432 option_address_list = 0;
1433 option_dr_priority = 0;
1434 option_generation_id = 0;
1435 option_holdtime = 0;
1436 option_lan_prune_delay = 0;
1437 option_t_bit = 0;
1438
1439 if (PIM_OPTION_IS_SET(neigh->hello_options, PIM_OPTION_MASK_ADDRESS_LIST))
1440 option_address_list = 1;
1441
1442 if (PIM_OPTION_IS_SET(neigh->hello_options, PIM_OPTION_MASK_DR_PRIORITY))
1443 option_dr_priority = 1;
1444
1445 if (PIM_OPTION_IS_SET(neigh->hello_options, PIM_OPTION_MASK_GENERATION_ID))
1446 option_generation_id = 1;
1447
1448 if (PIM_OPTION_IS_SET(neigh->hello_options, PIM_OPTION_MASK_HOLDTIME))
1449 option_holdtime = 1;
1450
1451 if (PIM_OPTION_IS_SET(neigh->hello_options, PIM_OPTION_MASK_LAN_PRUNE_DELAY))
1452 option_lan_prune_delay = 1;
1453
1454 if (PIM_OPTION_IS_SET(neigh->hello_options, PIM_OPTION_MASK_CAN_DISABLE_JOIN_SUPPRESSION))
1455 option_t_bit = 1;
1456
1457 if (uj) {
1458
1459 /* Does this ifp live in json? If not create it. */
1460 json_object_object_get_ex(json, ifp->name, &json_ifp);
1461
1462 if (!json_ifp) {
1463 json_ifp = json_object_new_object();
1464 json_object_pim_ifp_add(json_ifp, ifp);
1465 json_object_object_add(json, ifp->name, json_ifp);
1466 }
1467
1468 json_row = json_object_new_object();
1469 json_object_string_add(json_row, "interface", ifp->name);
1470 json_object_string_add(json_row, "address", neigh_src_str);
1471 json_object_string_add(json_row, "upTime", uptime);
1472 json_object_string_add(json_row, "holdtime", expire);
1473 json_object_int_add(json_row, "drPriority", neigh->dr_priority);
1474 json_object_int_add(json_row, "generationId", neigh->generation_id);
1475
1476 if (option_address_list)
1477 json_object_boolean_true_add(json_row, "helloOptionAddressList");
1478
1479 if (option_dr_priority)
1480 json_object_boolean_true_add(json_row, "helloOptionDrPriority");
1481
1482 if (option_generation_id)
1483 json_object_boolean_true_add(json_row, "helloOptionGenerationId");
1484
1485 if (option_holdtime)
1486 json_object_boolean_true_add(json_row, "helloOptionHoldtime");
1487
1488 if (option_lan_prune_delay)
1489 json_object_boolean_true_add(json_row, "helloOptionLanPruneDelay");
1490
1491 if (option_t_bit)
1492 json_object_boolean_true_add(json_row, "helloOptionTBit");
1493
1494 json_object_object_add(json_ifp, neigh_src_str, json_row);
1495
1496 } else {
1497 vty_out(vty, "Interface : %s%s", ifp->name, VTY_NEWLINE);
1498 vty_out(vty, "Neighbor : %s%s", neigh_src_str, VTY_NEWLINE);
1499 vty_out(vty, " Uptime : %s%s", uptime, VTY_NEWLINE);
1500 vty_out(vty, " Holdtime : %s%s", expire, VTY_NEWLINE);
1501 vty_out(vty, " DR Priority : %d%s", neigh->dr_priority, VTY_NEWLINE);
1502 vty_out(vty, " Generation ID : %08x%s", neigh->generation_id, VTY_NEWLINE);
1503 vty_out(vty, " Override Interval (msec) : %d%s", neigh->override_interval_msec, VTY_NEWLINE);
1504 vty_out(vty, " Propagation Delay (msec) : %d%s", neigh->propagation_delay_msec, VTY_NEWLINE);
1505 vty_out(vty, " Hello Option - Address List : %s%s", option_address_list ? "yes" : "no", VTY_NEWLINE);
1506 vty_out(vty, " Hello Option - DR Priority : %s%s", option_dr_priority ? "yes" : "no", VTY_NEWLINE);
1507 vty_out(vty, " Hello Option - Generation ID : %s%s", option_generation_id? "yes" : "no", VTY_NEWLINE);
1508 vty_out(vty, " Hello Option - Holdtime : %s%s", option_holdtime ? "yes" : "no", VTY_NEWLINE);
1509 vty_out(vty, " Hello Option - LAN Prune Delay : %s%s", option_lan_prune_delay ? "yes" : "no", VTY_NEWLINE);
1510 vty_out(vty, " Hello Option - T-bit : %s%s", option_t_bit ? "yes" : "no", VTY_NEWLINE);
1511 vty_out(vty, "%s", VTY_NEWLINE);
1512 }
1513 }
1514 }
1515
1516 if (uj) {
1517 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1518 json_object_free(json);
1519 } else {
1520 {
1521 if (!found_neighbor)
1522 vty_out (vty, "%% No such interface or neighbor%s", VTY_NEWLINE);
1523 }
1524 }
1525 }
1526
1527 static void
1528 pim_show_state(struct vty *vty, const char *src_or_group, const char *group, u_char uj)
1529 {
1530 struct channel_oil *c_oil;
1531 struct listnode *node;
1532 json_object *json = NULL;
1533 json_object *json_group = NULL;
1534 json_object *json_ifp_in = NULL;
1535 json_object *json_ifp_out = NULL;
1536 json_object *json_source = NULL;
1537 time_t now;
1538 int first_oif;
1539 now = pim_time_monotonic_sec();
1540
1541 if (uj) {
1542 json = json_object_new_object();
1543 } else {
1544 vty_out(vty, "Codes: J -> Pim Join, I -> IGMP Report, S -> Source, * -> Inherited from (*,G)");
1545 vty_out(vty, "%sInstalled Source Group IIF OIL%s", VTY_NEWLINE, VTY_NEWLINE);
1546 }
1547
1548 for (ALL_LIST_ELEMENTS_RO(pim_channel_oil_list, node, c_oil)) {
1549 char grp_str[INET_ADDRSTRLEN];
1550 char src_str[INET_ADDRSTRLEN];
1551 char in_ifname[INTERFACE_NAMSIZ+1];
1552 char out_ifname[INTERFACE_NAMSIZ+1];
1553 int oif_vif_index;
1554 struct interface *ifp_in;
1555 first_oif = 1;
1556
1557 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp, grp_str, sizeof(grp_str));
1558 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin, src_str, sizeof(src_str));
1559 ifp_in = pim_if_find_by_vif_index(c_oil->oil.mfcc_parent);
1560
1561 if (ifp_in)
1562 strcpy(in_ifname, ifp_in->name);
1563 else
1564 strcpy(in_ifname, "<iif?>");
1565
1566 if (src_or_group)
1567 {
1568 if (strcmp(src_or_group, src_str) && strcmp(src_or_group, grp_str))
1569 continue;
1570
1571 if (group && strcmp(group, grp_str))
1572 continue;
1573 }
1574
1575 if (uj) {
1576
1577 /* Find the group, create it if it doesn't exist */
1578 json_object_object_get_ex(json, grp_str, &json_group);
1579
1580 if (!json_group) {
1581 json_group = json_object_new_object();
1582 json_object_object_add(json, grp_str, json_group);
1583 }
1584
1585 /* Find the source nested under the group, create it if it doesn't exist */
1586 json_object_object_get_ex(json_group, src_str, &json_source);
1587
1588 if (!json_source) {
1589 json_source = json_object_new_object();
1590 json_object_object_add(json_group, src_str, json_source);
1591 }
1592
1593 /* Find the inbound interface nested under the source, create it if it doesn't exist */
1594 json_object_object_get_ex(json_source, in_ifname, &json_ifp_in);
1595
1596 if (!json_ifp_in) {
1597 json_ifp_in = json_object_new_object();
1598 json_object_object_add(json_source, in_ifname, json_ifp_in);
1599 json_object_int_add (json_source, "Installed", c_oil->installed);
1600 json_object_int_add (json_source, "RefCount", c_oil->oil_ref_count);
1601 json_object_int_add (json_source, "OilListSize", c_oil->oil_size);
1602 json_object_int_add (json_source, "OilRescan", c_oil->oil_inherited_rescan);
1603 json_object_int_add (json_source, "LastUsed", c_oil->cc.lastused);
1604 json_object_int_add (json_source, "PacketCount", c_oil->cc.pktcnt);
1605 json_object_int_add (json_source, "ByteCount", c_oil->cc.bytecnt);
1606 json_object_int_add (json_source, "WrongInterface", c_oil->cc.wrong_if);
1607 }
1608 } else {
1609 vty_out(vty, "%-9d %-15s %-15s %-7s ",
1610 c_oil->installed,
1611 src_str,
1612 grp_str,
1613 ifp_in->name);
1614 }
1615
1616 for (oif_vif_index = 0; oif_vif_index < MAXVIFS; ++oif_vif_index) {
1617 struct interface *ifp_out;
1618 char oif_uptime[10];
1619 int ttl;
1620
1621 ttl = c_oil->oil.mfcc_ttls[oif_vif_index];
1622 if (ttl < 1)
1623 continue;
1624
1625 ifp_out = pim_if_find_by_vif_index(oif_vif_index);
1626 pim_time_uptime(oif_uptime, sizeof(oif_uptime), now - c_oil->oif_creation[oif_vif_index]);
1627
1628 if (ifp_out)
1629 strcpy(out_ifname, ifp_out->name);
1630 else
1631 strcpy(out_ifname, "<oif?>");
1632
1633 if (uj) {
1634 json_ifp_out = json_object_new_object();
1635 json_object_string_add(json_ifp_out, "source", src_str);
1636 json_object_string_add(json_ifp_out, "group", grp_str);
1637 json_object_string_add(json_ifp_out, "inboundInterface", in_ifname);
1638 json_object_string_add(json_ifp_out, "outboundInterface", out_ifname);
1639 json_object_int_add(json_ifp_out, "installed", c_oil->installed);
1640
1641 json_object_object_add(json_ifp_in, out_ifname, json_ifp_out);
1642 } else {
1643 if (first_oif)
1644 {
1645 first_oif = 0;
1646 vty_out(vty, "%s(%c%c%c%c)", out_ifname,
1647 (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_IGMP) ? 'I' : ' ',
1648 (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_PIM) ? 'J' : ' ',
1649 (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_SOURCE) ? 'S' : ' ',
1650 (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_STAR) ? '*' : ' ');
1651 }
1652 else
1653 vty_out(vty, ", %s(%c%c%c%c)", out_ifname,
1654 (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_IGMP) ? 'I' : ' ',
1655 (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_PIM) ? 'J' : ' ',
1656 (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_SOURCE) ? 'S' : ' ',
1657 (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_STAR) ? '*' : ' ' );
1658 }
1659 }
1660
1661 if (!uj)
1662 vty_out(vty, "%s", VTY_NEWLINE);
1663 }
1664
1665
1666 if (uj) {
1667 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1668 json_object_free(json);
1669 } else {
1670 vty_out(vty, "%s", VTY_NEWLINE);
1671 }
1672 }
1673
1674 static void pim_show_neighbors(struct vty *vty, u_char uj)
1675 {
1676 struct listnode *node;
1677 struct listnode *neighnode;
1678 struct interface *ifp;
1679 struct pim_interface *pim_ifp;
1680 struct pim_neighbor *neigh;
1681 time_t now;
1682 char uptime[10];
1683 char expire[10];
1684 char neigh_src_str[INET_ADDRSTRLEN];
1685 json_object *json = NULL;
1686 json_object *json_ifp_rows = NULL;
1687 json_object *json_row = NULL;
1688
1689 now = pim_time_monotonic_sec();
1690
1691 if (uj) {
1692 json = json_object_new_object();
1693 } else {
1694 vty_out(vty, "Interface Neighbor Uptime Holdtime DR Pri%s", VTY_NEWLINE);
1695 }
1696
1697 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
1698 pim_ifp = ifp->info;
1699
1700 if (!pim_ifp)
1701 continue;
1702
1703 if (pim_ifp->pim_sock_fd < 0)
1704 continue;
1705
1706 if (uj)
1707 json_ifp_rows = json_object_new_object();
1708
1709 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list, neighnode, neigh)) {
1710 pim_inet4_dump("<src?>", neigh->source_addr,
1711 neigh_src_str, sizeof(neigh_src_str));
1712 pim_time_uptime(uptime, sizeof(uptime), now - neigh->creation);
1713 pim_time_timer_to_hhmmss(expire, sizeof(expire), neigh->t_expire_timer);
1714
1715 if (uj) {
1716 json_row = json_object_new_object();
1717 json_object_string_add(json_row, "interface", ifp->name);
1718 json_object_string_add(json_row, "neighbor", neigh_src_str);
1719 json_object_string_add(json_row, "upTime", uptime);
1720 json_object_string_add(json_row, "holdTime", expire);
1721 json_object_int_add(json_row, "holdTimeMax", neigh->holdtime);
1722 json_object_int_add(json_row, "drPriority", neigh->dr_priority);
1723 json_object_object_add(json_ifp_rows, neigh_src_str, json_row);
1724
1725 } else {
1726 vty_out(vty, "%-9s %15s %8s %8s %6d%s",
1727 ifp->name,
1728 neigh_src_str,
1729 uptime,
1730 expire,
1731 neigh->dr_priority,
1732 VTY_NEWLINE);
1733 }
1734 }
1735
1736 if (uj) {
1737 json_object_object_add(json, ifp->name, json_ifp_rows);
1738 json_ifp_rows = NULL;
1739 }
1740 }
1741
1742 if (uj) {
1743 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1744 json_object_free(json);
1745 }
1746 }
1747
1748 static void pim_show_neighbors_secondary(struct vty *vty)
1749 {
1750 struct listnode *node;
1751 struct interface *ifp;
1752
1753 vty_out(vty, "Interface Address Neighbor Secondary %s", VTY_NEWLINE);
1754
1755 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
1756 struct pim_interface *pim_ifp;
1757 struct in_addr ifaddr;
1758 struct listnode *neighnode;
1759 struct pim_neighbor *neigh;
1760
1761 pim_ifp = ifp->info;
1762
1763 if (!pim_ifp)
1764 continue;
1765
1766 if (pim_ifp->pim_sock_fd < 0)
1767 continue;
1768
1769 ifaddr = pim_ifp->primary_address;
1770
1771 for (ALL_LIST_ELEMENTS_RO(pim_ifp->pim_neighbor_list, neighnode, neigh)) {
1772 char neigh_src_str[INET_ADDRSTRLEN];
1773 struct listnode *prefix_node;
1774 struct prefix *p;
1775
1776 if (!neigh->prefix_list)
1777 continue;
1778
1779 pim_inet4_dump("<src?>", neigh->source_addr,
1780 neigh_src_str, sizeof(neigh_src_str));
1781
1782 for (ALL_LIST_ELEMENTS_RO(neigh->prefix_list, prefix_node, p)) {
1783 char neigh_sec_str[PREFIX2STR_BUFFER];
1784
1785 prefix2str(p, neigh_sec_str, sizeof(neigh_sec_str));
1786
1787 vty_out(vty, "%-9s %-15s %-15s %-15s%s",
1788 ifp->name,
1789 inet_ntoa(ifaddr),
1790 neigh_src_str,
1791 neigh_sec_str,
1792 VTY_NEWLINE);
1793 }
1794 }
1795 }
1796 }
1797
1798 static void
1799 json_object_pim_upstream_add (json_object *json, struct pim_upstream *up)
1800 {
1801 if (up->flags & PIM_UPSTREAM_FLAG_MASK_DR_JOIN_DESIRED)
1802 json_object_boolean_true_add(json, "drJoinDesired");
1803
1804 if (up->flags & PIM_UPSTREAM_FLAG_MASK_DR_JOIN_DESIRED_UPDATED)
1805 json_object_boolean_true_add(json, "drJoinDesiredUpdated");
1806
1807 if (up->flags & PIM_UPSTREAM_FLAG_MASK_FHR)
1808 json_object_boolean_true_add(json, "firstHopRouter");
1809
1810 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_IGMP)
1811 json_object_boolean_true_add(json, "sourceIgmp");
1812
1813 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_PIM)
1814 json_object_boolean_true_add(json, "sourcePim");
1815
1816 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_STREAM)
1817 json_object_boolean_true_add(json, "sourceStream");
1818
1819 /* XXX: need to print ths flag in the plain text display as well */
1820 if (up->flags & PIM_UPSTREAM_FLAG_MASK_SRC_MSDP)
1821 json_object_boolean_true_add(json, "sourceMsdp");
1822 }
1823
1824 static const char *
1825 pim_upstream_state2brief_str (enum pim_upstream_state join_state, char *state_str)
1826 {
1827 switch (join_state)
1828 {
1829 case PIM_UPSTREAM_NOTJOINED:
1830 strcpy (state_str, "NotJ");
1831 break;
1832 case PIM_UPSTREAM_JOINED:
1833 strcpy (state_str, "J");
1834 break;
1835 default:
1836 strcpy (state_str, "Unk");
1837 }
1838 return state_str;
1839 }
1840
1841 static const char *
1842 pim_reg_state2brief_str (enum pim_reg_state reg_state, char *state_str)
1843 {
1844 switch (reg_state)
1845 {
1846 case PIM_REG_NOINFO:
1847 strcpy (state_str, "RegNI");
1848 break;
1849 case PIM_REG_JOIN:
1850 strcpy (state_str, "RegJ");
1851 break;
1852 case PIM_REG_JOIN_PENDING:
1853 case PIM_REG_PRUNE:
1854 strcpy (state_str, "RegP");
1855 break;
1856 default:
1857 strcpy (state_str, "Unk");
1858 }
1859 return state_str;
1860 }
1861
1862 static void pim_show_upstream(struct vty *vty, u_char uj)
1863 {
1864 struct listnode *upnode;
1865 struct pim_upstream *up;
1866 time_t now;
1867 json_object *json = NULL;
1868 json_object *json_group = NULL;
1869 json_object *json_row = NULL;
1870
1871 now = pim_time_monotonic_sec();
1872
1873 if (uj)
1874 json = json_object_new_object();
1875 else
1876 vty_out(vty, "Iif Source Group State Uptime JoinTimer RSTimer KATimer RefCnt%s", VTY_NEWLINE);
1877
1878 for (ALL_LIST_ELEMENTS_RO(pim_upstream_list, upnode, up)) {
1879 char src_str[INET_ADDRSTRLEN];
1880 char grp_str[INET_ADDRSTRLEN];
1881 char uptime[10];
1882 char join_timer[10];
1883 char rs_timer[10];
1884 char ka_timer[10];
1885 char msdp_reg_timer[10];
1886 char state_str[PIM_REG_STATE_STR_LEN];
1887
1888 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
1889 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
1890 pim_time_uptime(uptime, sizeof(uptime), now - up->state_transition);
1891 pim_time_timer_to_hhmmss (join_timer, sizeof(join_timer), up->t_join_timer);
1892
1893 /*
1894 * If we have a J/P timer for the neighbor display that
1895 */
1896 if (!up->t_join_timer)
1897 {
1898 struct pim_neighbor *nbr;
1899
1900 nbr = pim_neighbor_find (up->rpf.source_nexthop.interface,
1901 up->rpf.rpf_addr.u.prefix4);
1902 if (nbr)
1903 pim_time_timer_to_hhmmss (join_timer, sizeof(join_timer), nbr->jp_timer);
1904 }
1905
1906 pim_time_timer_to_hhmmss (rs_timer, sizeof (rs_timer), up->t_rs_timer);
1907 pim_time_timer_to_hhmmss (ka_timer, sizeof (ka_timer), up->t_ka_timer);
1908 pim_time_timer_to_hhmmss (msdp_reg_timer, sizeof (msdp_reg_timer), up->t_msdp_reg_timer);
1909
1910 pim_upstream_state2brief_str (up->join_state, state_str);
1911 if (up->reg_state != PIM_REG_NOINFO) {
1912 char tmp_str[PIM_REG_STATE_STR_LEN];
1913
1914 sprintf (state_str + strlen (state_str), ",%s",
1915 pim_reg_state2brief_str (up->reg_state, tmp_str));
1916 }
1917
1918 if (uj) {
1919 json_object_object_get_ex(json, grp_str, &json_group);
1920
1921 if (!json_group) {
1922 json_group = json_object_new_object();
1923 json_object_object_add(json, grp_str, json_group);
1924 }
1925
1926 json_row = json_object_new_object();
1927 json_object_pim_upstream_add(json_row, up);
1928 json_object_string_add(json_row, "inboundInterface", up->rpf.source_nexthop.interface->name);
1929 json_object_string_add(json_row, "source", src_str);
1930 json_object_string_add(json_row, "group", grp_str);
1931 json_object_string_add(json_row, "state", state_str);
1932 json_object_string_add(json_row, "joinState", pim_upstream_state2str (up->join_state));
1933 json_object_string_add(json_row, "regState", pim_reg_state2str (up->reg_state, state_str));
1934 json_object_string_add(json_row, "upTime", uptime);
1935 json_object_string_add(json_row, "joinTimer", join_timer);
1936 json_object_string_add(json_row, "resetTimer", rs_timer);
1937 json_object_string_add(json_row, "keepaliveTimer", ka_timer);
1938 json_object_string_add(json_row, "msdpRegTimer", msdp_reg_timer);
1939 json_object_int_add(json_row, "refCount", up->ref_count);
1940 json_object_int_add(json_row, "sptBit", up->sptbit);
1941 json_object_object_add(json_group, src_str, json_row);
1942 } else {
1943 vty_out(vty, "%-10s%-15s %-15s %-11s %-8s %-9s %-9s %-9s %6d%s",
1944 up->rpf.source_nexthop.interface->name,
1945 src_str,
1946 grp_str,
1947 state_str,
1948 uptime,
1949 join_timer,
1950 rs_timer,
1951 ka_timer,
1952 up->ref_count,
1953 VTY_NEWLINE);
1954 }
1955 }
1956
1957 if (uj) {
1958 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
1959 json_object_free(json);
1960 }
1961 }
1962
1963 static void pim_show_join_desired(struct vty *vty, u_char uj)
1964 {
1965 struct listnode *chnode;
1966 struct pim_interface *pim_ifp;
1967 struct pim_ifchannel *ch;
1968 char src_str[INET_ADDRSTRLEN];
1969 char grp_str[INET_ADDRSTRLEN];
1970 json_object *json = NULL;
1971 json_object *json_group = NULL;
1972 json_object *json_row = NULL;
1973
1974 if (uj)
1975 json = json_object_new_object();
1976 else
1977 vty_out(vty,
1978 "Interface Source Group LostAssert Joins PimInclude JoinDesired EvalJD%s",
1979 VTY_NEWLINE);
1980
1981 /* scan per-interface (S,G) state */
1982 for (ALL_LIST_ELEMENTS_RO(pim_ifchannel_list, chnode, ch)) {
1983 /* scan all interfaces */
1984 pim_ifp = ch->interface->info;
1985 if (!pim_ifp)
1986 continue;
1987
1988 struct pim_upstream *up = ch->upstream;
1989
1990 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
1991 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
1992
1993 if (uj) {
1994 json_object_object_get_ex(json, grp_str, &json_group);
1995
1996 if (!json_group) {
1997 json_group = json_object_new_object();
1998 json_object_object_add(json, grp_str, json_group);
1999 }
2000
2001 json_row = json_object_new_object();
2002 json_object_pim_upstream_add(json_row, up);
2003 json_object_string_add(json_row, "interface", ch->interface->name);
2004 json_object_string_add(json_row, "source", src_str);
2005 json_object_string_add(json_row, "group", grp_str);
2006
2007 if (pim_macro_ch_lost_assert(ch))
2008 json_object_boolean_true_add(json_row, "lostAssert");
2009
2010 if (pim_macro_chisin_joins(ch))
2011 json_object_boolean_true_add(json_row, "joins");
2012
2013 if (pim_macro_chisin_pim_include(ch))
2014 json_object_boolean_true_add(json_row, "pimInclude");
2015
2016 if (pim_upstream_evaluate_join_desired(up))
2017 json_object_boolean_true_add(json_row, "evaluateJoinDesired");
2018
2019 json_object_object_add(json_group, src_str, json_row);
2020
2021 } else {
2022 vty_out(vty, "%-9s %-15s %-15s %-10s %-5s %-10s %-11s %-6s%s",
2023 ch->interface->name,
2024 src_str,
2025 grp_str,
2026 pim_macro_ch_lost_assert(ch) ? "yes" : "no",
2027 pim_macro_chisin_joins(ch) ? "yes" : "no",
2028 pim_macro_chisin_pim_include(ch) ? "yes" : "no",
2029 PIM_UPSTREAM_FLAG_TEST_DR_JOIN_DESIRED(up->flags) ? "yes" : "no",
2030 pim_upstream_evaluate_join_desired(up) ? "yes" : "no",
2031 VTY_NEWLINE);
2032 }
2033 }
2034
2035 if (uj) {
2036 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
2037 json_object_free(json);
2038 }
2039 }
2040
2041 static void pim_show_upstream_rpf(struct vty *vty, u_char uj)
2042 {
2043 struct listnode *upnode;
2044 struct pim_upstream *up;
2045 json_object *json = NULL;
2046 json_object *json_group = NULL;
2047 json_object *json_row = NULL;
2048
2049 if (uj)
2050 json = json_object_new_object();
2051 else
2052 vty_out(vty,
2053 "Source Group RpfIface RibNextHop RpfAddress %s",
2054 VTY_NEWLINE);
2055
2056 for (ALL_LIST_ELEMENTS_RO(pim_upstream_list, upnode, up)) {
2057 char src_str[INET_ADDRSTRLEN];
2058 char grp_str[INET_ADDRSTRLEN];
2059 char rpf_nexthop_str[PREFIX_STRLEN];
2060 char rpf_addr_str[PREFIX_STRLEN];
2061 struct pim_rpf *rpf;
2062 const char *rpf_ifname;
2063
2064 rpf = &up->rpf;
2065
2066 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
2067 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
2068 pim_addr_dump("<nexthop?>", &rpf->source_nexthop.mrib_nexthop_addr, rpf_nexthop_str, sizeof(rpf_nexthop_str));
2069 pim_addr_dump("<rpf?>", &rpf->rpf_addr, rpf_addr_str, sizeof(rpf_addr_str));
2070
2071 rpf_ifname = rpf->source_nexthop.interface ? rpf->source_nexthop.interface->name : "<ifname?>";
2072
2073 if (uj) {
2074 json_object_object_get_ex(json, grp_str, &json_group);
2075
2076 if (!json_group) {
2077 json_group = json_object_new_object();
2078 json_object_object_add(json, grp_str, json_group);
2079 }
2080
2081 json_row = json_object_new_object();
2082 json_object_pim_upstream_add(json_row, up);
2083 json_object_string_add(json_row, "source", src_str);
2084 json_object_string_add(json_row, "group", grp_str);
2085 json_object_string_add(json_row, "rpfInterface", rpf_ifname);
2086 json_object_string_add(json_row, "ribNexthop", rpf_nexthop_str);
2087 json_object_string_add(json_row, "rpfAddress", rpf_addr_str);
2088 json_object_object_add(json_group, src_str, json_row);
2089 } else {
2090 vty_out(vty, "%-15s %-15s %-8s %-15s %-15s%s",
2091 src_str,
2092 grp_str,
2093 rpf_ifname,
2094 rpf_nexthop_str,
2095 rpf_addr_str,
2096 VTY_NEWLINE);
2097 }
2098 }
2099
2100 if (uj) {
2101 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
2102 json_object_free(json);
2103 }
2104 }
2105
2106 static void show_rpf_refresh_stats(struct vty *vty, time_t now, json_object *json)
2107 {
2108 char refresh_uptime[10];
2109
2110 pim_time_uptime_begin(refresh_uptime, sizeof(refresh_uptime), now, qpim_rpf_cache_refresh_last);
2111
2112 if (json) {
2113 json_object_int_add(json, "rpfCacheRefreshDelayMsecs", qpim_rpf_cache_refresh_delay_msec);
2114 json_object_int_add(json, "rpfCacheRefreshTimer", pim_time_timer_remain_msec(qpim_rpf_cache_refresher));
2115 json_object_int_add(json, "rpfCacheRefreshRequests", qpim_rpf_cache_refresh_requests);
2116 json_object_int_add(json, "rpfCacheRefreshEvents", qpim_rpf_cache_refresh_events);
2117 json_object_string_add(json, "rpfCacheRefreshLast", refresh_uptime);
2118 json_object_int_add(json, "nexthopLookups", qpim_nexthop_lookups);
2119 json_object_int_add(json, "nexthopLookupsAvoided", nexthop_lookups_avoided);
2120 } else {
2121 vty_out(vty,
2122 "RPF Cache Refresh Delay: %ld msecs%s"
2123 "RPF Cache Refresh Timer: %ld msecs%s"
2124 "RPF Cache Refresh Requests: %lld%s"
2125 "RPF Cache Refresh Events: %lld%s"
2126 "RPF Cache Refresh Last: %s%s"
2127 "Nexthop Lookups: %lld%s"
2128 "Nexthop Lookups Avoided: %lld%s",
2129 qpim_rpf_cache_refresh_delay_msec, VTY_NEWLINE,
2130 pim_time_timer_remain_msec(qpim_rpf_cache_refresher), VTY_NEWLINE,
2131 (long long)qpim_rpf_cache_refresh_requests, VTY_NEWLINE,
2132 (long long)qpim_rpf_cache_refresh_events, VTY_NEWLINE,
2133 refresh_uptime, VTY_NEWLINE,
2134 (long long) qpim_nexthop_lookups, VTY_NEWLINE,
2135 (long long)nexthop_lookups_avoided, VTY_NEWLINE);
2136 }
2137 }
2138
2139 static void show_scan_oil_stats(struct vty *vty, time_t now)
2140 {
2141 char uptime_scan_oil[10];
2142 char uptime_mroute_add[10];
2143 char uptime_mroute_del[10];
2144
2145 pim_time_uptime_begin(uptime_scan_oil, sizeof(uptime_scan_oil), now, qpim_scan_oil_last);
2146 pim_time_uptime_begin(uptime_mroute_add, sizeof(uptime_mroute_add), now, qpim_mroute_add_last);
2147 pim_time_uptime_begin(uptime_mroute_del, sizeof(uptime_mroute_del), now, qpim_mroute_del_last);
2148
2149 vty_out(vty,
2150 "Scan OIL - Last: %s Events: %lld%s"
2151 "MFC Add - Last: %s Events: %lld%s"
2152 "MFC Del - Last: %s Events: %lld%s",
2153 uptime_scan_oil, (long long) qpim_scan_oil_events, VTY_NEWLINE,
2154 uptime_mroute_add, (long long) qpim_mroute_add_events, VTY_NEWLINE,
2155 uptime_mroute_del, (long long) qpim_mroute_del_events, VTY_NEWLINE);
2156 }
2157
2158 static void pim_show_rpf(struct vty *vty, u_char uj)
2159 {
2160 struct listnode *up_node;
2161 struct pim_upstream *up;
2162 time_t now = pim_time_monotonic_sec();
2163 json_object *json = NULL;
2164 json_object *json_group = NULL;
2165 json_object *json_row = NULL;
2166
2167 if (uj) {
2168 json = json_object_new_object();
2169 show_rpf_refresh_stats(vty, now, json);
2170 } else {
2171 show_rpf_refresh_stats(vty, now, json);
2172 vty_out(vty, "%s", VTY_NEWLINE);
2173 vty_out(vty,
2174 "Source Group RpfIface RpfAddress RibNextHop Metric Pref%s",
2175 VTY_NEWLINE);
2176 }
2177
2178 for (ALL_LIST_ELEMENTS_RO(pim_upstream_list, up_node, up)) {
2179 char src_str[INET_ADDRSTRLEN];
2180 char grp_str[INET_ADDRSTRLEN];
2181 char rpf_addr_str[PREFIX_STRLEN];
2182 char rib_nexthop_str[PREFIX_STRLEN];
2183 const char *rpf_ifname;
2184 struct pim_rpf *rpf = &up->rpf;
2185
2186 pim_inet4_dump("<src?>", up->sg.src, src_str, sizeof(src_str));
2187 pim_inet4_dump("<grp?>", up->sg.grp, grp_str, sizeof(grp_str));
2188 pim_addr_dump("<rpf?>", &rpf->rpf_addr, rpf_addr_str, sizeof(rpf_addr_str));
2189 pim_addr_dump("<nexthop?>", &rpf->source_nexthop.mrib_nexthop_addr, rib_nexthop_str, sizeof(rib_nexthop_str));
2190
2191 rpf_ifname = rpf->source_nexthop.interface ? rpf->source_nexthop.interface->name : "<ifname?>";
2192
2193 if (uj) {
2194 json_object_object_get_ex(json, grp_str, &json_group);
2195
2196 if (!json_group) {
2197 json_group = json_object_new_object();
2198 json_object_object_add(json, grp_str, json_group);
2199 }
2200
2201 json_row = json_object_new_object();
2202 json_object_string_add(json_row, "source", src_str);
2203 json_object_string_add(json_row, "group", grp_str);
2204 json_object_string_add(json_row, "rpfInterface", rpf_ifname);
2205 json_object_string_add(json_row, "rpfAddress", rpf_addr_str);
2206 json_object_string_add(json_row, "ribNexthop", rib_nexthop_str);
2207 json_object_int_add(json_row, "routeMetric", rpf->source_nexthop.mrib_route_metric);
2208 json_object_int_add(json_row, "routePreference", rpf->source_nexthop.mrib_metric_preference);
2209 json_object_object_add(json_group, src_str, json_row);
2210
2211 } else {
2212 vty_out(vty, "%-15s %-15s %-8s %-15s %-15s %6d %4d%s",
2213 src_str,
2214 grp_str,
2215 rpf_ifname,
2216 rpf_addr_str,
2217 rib_nexthop_str,
2218 rpf->source_nexthop.mrib_route_metric,
2219 rpf->source_nexthop.mrib_metric_preference,
2220 VTY_NEWLINE);
2221 }
2222 }
2223
2224 if (uj) {
2225 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
2226 json_object_free(json);
2227 }
2228 }
2229
2230 static int
2231 pim_print_pnc_cache_walkcb (struct hash_backet *backet, void *arg)
2232 {
2233 struct pim_nexthop_cache *pnc = backet->data;
2234 struct vty *vty = arg;
2235 struct nexthop *nh_node = NULL;
2236 ifindex_t first_ifindex;
2237 struct interface *ifp = NULL;
2238
2239 if (!pnc)
2240 return CMD_SUCCESS;
2241
2242 for (nh_node = pnc->nexthop; nh_node; nh_node = nh_node->next)
2243 {
2244 first_ifindex = nh_node->ifindex;
2245 ifp = if_lookup_by_index (first_ifindex, VRF_DEFAULT);
2246
2247 vty_out (vty, "%-15s ", inet_ntoa (pnc->rpf.rpf_addr.u.prefix4));
2248 vty_out (vty, "%-14s ", ifp ? ifp->name : "NULL");
2249 vty_out (vty, "%s ", inet_ntoa (nh_node->gate.ipv4));
2250 vty_out (vty, "%s", VTY_NEWLINE);
2251 }
2252 return CMD_SUCCESS;
2253 }
2254
2255 static void
2256 pim_show_nexthop (struct vty *vty)
2257 {
2258
2259 if (pimg && !pimg->rpf_hash)
2260 {
2261 vty_out (vty, "no nexthop cache %s", VTY_NEWLINE);
2262 return;
2263 }
2264
2265 vty_out (vty, "Number of registered addresses: %lu %s",
2266 pimg->rpf_hash->count, VTY_NEWLINE);
2267 vty_out (vty, "Address Interface Nexthop%s", VTY_NEWLINE);
2268 vty_out (vty, "-------------------------------------------%s", VTY_NEWLINE);
2269
2270 hash_walk (pimg->rpf_hash, pim_print_pnc_cache_walkcb, vty);
2271
2272 }
2273
2274 static void igmp_show_groups(struct vty *vty, u_char uj)
2275 {
2276 struct listnode *ifnode;
2277 struct interface *ifp;
2278 time_t now;
2279 json_object *json = NULL;
2280 json_object *json_iface = NULL;
2281 json_object *json_row = NULL;
2282
2283 now = pim_time_monotonic_sec();
2284
2285 if (uj)
2286 json = json_object_new_object();
2287 else
2288 vty_out(vty, "Interface Address Group Mode Timer Srcs V Uptime %s", VTY_NEWLINE);
2289
2290 /* scan interfaces */
2291 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), ifnode, ifp)) {
2292 struct pim_interface *pim_ifp = ifp->info;
2293 struct listnode *sock_node;
2294 struct igmp_sock *igmp;
2295
2296 if (!pim_ifp)
2297 continue;
2298
2299 /* scan igmp sockets */
2300 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
2301 char ifaddr_str[INET_ADDRSTRLEN];
2302 struct listnode *grpnode;
2303 struct igmp_group *grp;
2304
2305 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str, sizeof(ifaddr_str));
2306
2307 /* scan igmp groups */
2308 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list, grpnode, grp)) {
2309 char group_str[INET_ADDRSTRLEN];
2310 char hhmmss[10];
2311 char uptime[10];
2312
2313 pim_inet4_dump("<group?>", grp->group_addr, group_str, sizeof(group_str));
2314 pim_time_timer_to_hhmmss(hhmmss, sizeof(hhmmss), grp->t_group_timer);
2315 pim_time_uptime(uptime, sizeof(uptime), now - grp->group_creation);
2316
2317 if (uj) {
2318 json_object_object_get_ex(json, ifp->name, &json_iface);
2319
2320 if (!json_iface) {
2321 json_iface = json_object_new_object();
2322 json_object_pim_ifp_add(json_iface, ifp);
2323 json_object_object_add(json, ifp->name, json_iface);
2324 }
2325
2326 json_row = json_object_new_object();
2327 json_object_string_add(json_row, "source", ifaddr_str);
2328 json_object_string_add(json_row, "group", group_str);
2329
2330 if (grp->igmp_version == 3)
2331 json_object_string_add(json_row, "mode", grp->group_filtermode_isexcl ? "EXCLUDE" : "INCLUDE");
2332
2333 json_object_string_add(json_row, "timer", hhmmss);
2334 json_object_int_add(json_row, "sourcesCount", grp->group_source_list ? listcount(grp->group_source_list) : 0);
2335 json_object_int_add(json_row, "version", grp->igmp_version);
2336 json_object_string_add(json_row, "uptime", uptime);
2337 json_object_object_add(json_iface, group_str, json_row);
2338
2339 } else {
2340 vty_out(vty, "%-9s %-15s %-15s %4s %8s %4d %d %8s%s",
2341 ifp->name,
2342 ifaddr_str,
2343 group_str,
2344 grp->igmp_version == 3 ? (grp->group_filtermode_isexcl ? "EXCL" : "INCL") : "----",
2345 hhmmss,
2346 grp->group_source_list ? listcount(grp->group_source_list) : 0,
2347 grp->igmp_version,
2348 uptime,
2349 VTY_NEWLINE);
2350 }
2351 } /* scan igmp groups */
2352 } /* scan igmp sockets */
2353 } /* scan interfaces */
2354
2355 if (uj) {
2356 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
2357 json_object_free(json);
2358 }
2359 }
2360
2361 static void igmp_show_group_retransmission(struct vty *vty)
2362 {
2363 struct listnode *ifnode;
2364 struct interface *ifp;
2365
2366 vty_out(vty, "Interface Address Group RetTimer Counter RetSrcs%s", VTY_NEWLINE);
2367
2368 /* scan interfaces */
2369 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), ifnode, ifp)) {
2370 struct pim_interface *pim_ifp = ifp->info;
2371 struct listnode *sock_node;
2372 struct igmp_sock *igmp;
2373
2374 if (!pim_ifp)
2375 continue;
2376
2377 /* scan igmp sockets */
2378 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
2379 char ifaddr_str[INET_ADDRSTRLEN];
2380 struct listnode *grpnode;
2381 struct igmp_group *grp;
2382
2383 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str, sizeof(ifaddr_str));
2384
2385 /* scan igmp groups */
2386 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list, grpnode, grp)) {
2387 char group_str[INET_ADDRSTRLEN];
2388 char grp_retr_mmss[10];
2389 struct listnode *src_node;
2390 struct igmp_source *src;
2391 int grp_retr_sources = 0;
2392
2393 pim_inet4_dump("<group?>", grp->group_addr, group_str, sizeof(group_str));
2394 pim_time_timer_to_mmss(grp_retr_mmss, sizeof(grp_retr_mmss), grp->t_group_query_retransmit_timer);
2395
2396
2397 /* count group sources with retransmission state */
2398 for (ALL_LIST_ELEMENTS_RO(grp->group_source_list, src_node, src)) {
2399 if (src->source_query_retransmit_count > 0) {
2400 ++grp_retr_sources;
2401 }
2402 }
2403
2404 vty_out(vty, "%-9s %-15s %-15s %-8s %7d %7d%s",
2405 ifp->name,
2406 ifaddr_str,
2407 group_str,
2408 grp_retr_mmss,
2409 grp->group_specific_query_retransmit_count,
2410 grp_retr_sources,
2411 VTY_NEWLINE);
2412
2413 } /* scan igmp groups */
2414 } /* scan igmp sockets */
2415 } /* scan interfaces */
2416 }
2417
2418 static void igmp_show_sources(struct vty *vty)
2419 {
2420 struct listnode *ifnode;
2421 struct interface *ifp;
2422 time_t now;
2423
2424 now = pim_time_monotonic_sec();
2425
2426 vty_out(vty, "Interface Address Group Source Timer Fwd Uptime %s", VTY_NEWLINE);
2427
2428 /* scan interfaces */
2429 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), ifnode, ifp)) {
2430 struct pim_interface *pim_ifp = ifp->info;
2431 struct listnode *sock_node;
2432 struct igmp_sock *igmp;
2433
2434 if (!pim_ifp)
2435 continue;
2436
2437 /* scan igmp sockets */
2438 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
2439 char ifaddr_str[INET_ADDRSTRLEN];
2440 struct listnode *grpnode;
2441 struct igmp_group *grp;
2442
2443 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str, sizeof(ifaddr_str));
2444
2445 /* scan igmp groups */
2446 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list, grpnode, grp)) {
2447 char group_str[INET_ADDRSTRLEN];
2448 struct listnode *srcnode;
2449 struct igmp_source *src;
2450
2451 pim_inet4_dump("<group?>", grp->group_addr, group_str, sizeof(group_str));
2452
2453 /* scan group sources */
2454 for (ALL_LIST_ELEMENTS_RO(grp->group_source_list, srcnode, src)) {
2455 char source_str[INET_ADDRSTRLEN];
2456 char mmss[10];
2457 char uptime[10];
2458
2459 pim_inet4_dump("<source?>", src->source_addr, source_str, sizeof(source_str));
2460
2461 pim_time_timer_to_mmss(mmss, sizeof(mmss), src->t_source_timer);
2462
2463 pim_time_uptime(uptime, sizeof(uptime), now - src->source_creation);
2464
2465 vty_out(vty, "%-9s %-15s %-15s %-15s %5s %3s %8s%s",
2466 ifp->name,
2467 ifaddr_str,
2468 group_str,
2469 source_str,
2470 mmss,
2471 IGMP_SOURCE_TEST_FORWARDING(src->source_flags) ? "Y" : "N",
2472 uptime,
2473 VTY_NEWLINE);
2474
2475 } /* scan group sources */
2476 } /* scan igmp groups */
2477 } /* scan igmp sockets */
2478 } /* scan interfaces */
2479 }
2480
2481 static void igmp_show_source_retransmission(struct vty *vty)
2482 {
2483 struct listnode *ifnode;
2484 struct interface *ifp;
2485
2486 vty_out(vty, "Interface Address Group Source Counter%s", VTY_NEWLINE);
2487
2488 /* scan interfaces */
2489 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), ifnode, ifp)) {
2490 struct pim_interface *pim_ifp = ifp->info;
2491 struct listnode *sock_node;
2492 struct igmp_sock *igmp;
2493
2494 if (!pim_ifp)
2495 continue;
2496
2497 /* scan igmp sockets */
2498 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
2499 char ifaddr_str[INET_ADDRSTRLEN];
2500 struct listnode *grpnode;
2501 struct igmp_group *grp;
2502
2503 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str, sizeof(ifaddr_str));
2504
2505 /* scan igmp groups */
2506 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list, grpnode, grp)) {
2507 char group_str[INET_ADDRSTRLEN];
2508 struct listnode *srcnode;
2509 struct igmp_source *src;
2510
2511 pim_inet4_dump("<group?>", grp->group_addr, group_str, sizeof(group_str));
2512
2513 /* scan group sources */
2514 for (ALL_LIST_ELEMENTS_RO(grp->group_source_list, srcnode, src)) {
2515 char source_str[INET_ADDRSTRLEN];
2516
2517 pim_inet4_dump("<source?>", src->source_addr, source_str, sizeof(source_str));
2518
2519 vty_out(vty, "%-9s %-15s %-15s %-15s %7d%s",
2520 ifp->name,
2521 ifaddr_str,
2522 group_str,
2523 source_str,
2524 src->source_query_retransmit_count,
2525 VTY_NEWLINE);
2526
2527 } /* scan group sources */
2528 } /* scan igmp groups */
2529 } /* scan igmp sockets */
2530 } /* scan interfaces */
2531 }
2532
2533 static void clear_igmp_interfaces()
2534 {
2535 struct listnode *ifnode;
2536 struct listnode *ifnextnode;
2537 struct interface *ifp;
2538
2539 for (ALL_LIST_ELEMENTS (vrf_iflist (VRF_DEFAULT), ifnode, ifnextnode, ifp)) {
2540 pim_if_addr_del_all_igmp(ifp);
2541 }
2542
2543 for (ALL_LIST_ELEMENTS (vrf_iflist (VRF_DEFAULT), ifnode, ifnextnode, ifp)) {
2544 pim_if_addr_add_all(ifp);
2545 }
2546 }
2547
2548 static void clear_pim_interfaces()
2549 {
2550 struct listnode *ifnode;
2551 struct listnode *ifnextnode;
2552 struct interface *ifp;
2553
2554 for (ALL_LIST_ELEMENTS (vrf_iflist (VRF_DEFAULT), ifnode, ifnextnode, ifp)) {
2555 if (ifp->info) {
2556 pim_neighbor_delete_all(ifp, "interface cleared");
2557 }
2558 }
2559 }
2560
2561 static void clear_interfaces()
2562 {
2563 clear_igmp_interfaces();
2564 clear_pim_interfaces();
2565 }
2566
2567 DEFUN (clear_ip_interfaces,
2568 clear_ip_interfaces_cmd,
2569 "clear ip interfaces",
2570 CLEAR_STR
2571 IP_STR
2572 "Reset interfaces\n")
2573 {
2574 clear_interfaces();
2575
2576 return CMD_SUCCESS;
2577 }
2578
2579 DEFUN (clear_ip_igmp_interfaces,
2580 clear_ip_igmp_interfaces_cmd,
2581 "clear ip igmp interfaces",
2582 CLEAR_STR
2583 IP_STR
2584 CLEAR_IP_IGMP_STR
2585 "Reset IGMP interfaces\n")
2586 {
2587 clear_igmp_interfaces();
2588
2589 return CMD_SUCCESS;
2590 }
2591
2592 static void mroute_add_all()
2593 {
2594 struct listnode *node;
2595 struct channel_oil *c_oil;
2596
2597 for (ALL_LIST_ELEMENTS_RO(pim_channel_oil_list, node, c_oil)) {
2598 if (pim_mroute_add(c_oil, __PRETTY_FUNCTION__)) {
2599 /* just log warning */
2600 char source_str[INET_ADDRSTRLEN];
2601 char group_str[INET_ADDRSTRLEN];
2602 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin, source_str, sizeof(source_str));
2603 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp, group_str, sizeof(group_str));
2604 zlog_warn("%s %s: (S,G)=(%s,%s) failure writing MFC",
2605 __FILE__, __PRETTY_FUNCTION__,
2606 source_str, group_str);
2607 }
2608 }
2609 }
2610
2611 static void mroute_del_all()
2612 {
2613 struct listnode *node;
2614 struct channel_oil *c_oil;
2615
2616 for (ALL_LIST_ELEMENTS_RO(pim_channel_oil_list, node, c_oil)) {
2617 if (pim_mroute_del(c_oil, __PRETTY_FUNCTION__)) {
2618 /* just log warning */
2619 char source_str[INET_ADDRSTRLEN];
2620 char group_str[INET_ADDRSTRLEN];
2621 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin, source_str, sizeof(source_str));
2622 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp, group_str, sizeof(group_str));
2623 zlog_warn("%s %s: (S,G)=(%s,%s) failure clearing MFC",
2624 __FILE__, __PRETTY_FUNCTION__,
2625 source_str, group_str);
2626 }
2627 }
2628 }
2629
2630 DEFUN (clear_ip_mroute,
2631 clear_ip_mroute_cmd,
2632 "clear ip mroute",
2633 CLEAR_STR
2634 IP_STR
2635 "Reset multicast routes\n")
2636 {
2637 mroute_del_all();
2638 mroute_add_all();
2639
2640 return CMD_SUCCESS;
2641 }
2642
2643 DEFUN (clear_ip_pim_interfaces,
2644 clear_ip_pim_interfaces_cmd,
2645 "clear ip pim interfaces",
2646 CLEAR_STR
2647 IP_STR
2648 CLEAR_IP_PIM_STR
2649 "Reset PIM interfaces\n")
2650 {
2651 clear_pim_interfaces();
2652
2653 return CMD_SUCCESS;
2654 }
2655
2656 DEFUN (clear_ip_pim_interface_traffic,
2657 clear_ip_pim_interface_traffic_cmd,
2658 "clear ip pim interface traffic",
2659 "Reset functions\n"
2660 "IP information\n"
2661 "PIM clear commands\n"
2662 "Reset PIM interfaces\n"
2663 "Reset Protocol Packet counters\n")
2664 {
2665 struct listnode *ifnode = NULL;
2666 struct listnode *ifnextnode = NULL;
2667 struct interface *ifp = NULL;
2668 struct pim_interface *pim_ifp = NULL;
2669
2670 for (ALL_LIST_ELEMENTS (vrf_iflist (VRF_DEFAULT), ifnode, ifnextnode, ifp))
2671 {
2672 pim_ifp = ifp->info;
2673
2674 if (!pim_ifp)
2675 continue;
2676
2677 pim_ifp->pim_ifstat_hello_recv = 0;
2678 pim_ifp->pim_ifstat_hello_sent = 0;
2679 pim_ifp->pim_ifstat_join_recv = 0;
2680 pim_ifp->pim_ifstat_join_send = 0;
2681 pim_ifp->pim_ifstat_prune_recv = 0;
2682 pim_ifp->pim_ifstat_prune_send = 0;
2683 pim_ifp->pim_ifstat_reg_recv = 0;
2684 pim_ifp->pim_ifstat_reg_send = 0;
2685 pim_ifp->pim_ifstat_reg_stop_recv = 0;
2686 pim_ifp->pim_ifstat_reg_stop_send = 0;
2687 pim_ifp->pim_ifstat_assert_recv = 0;
2688 pim_ifp->pim_ifstat_assert_send = 0;
2689
2690 }
2691
2692 return CMD_SUCCESS;
2693 }
2694
2695 DEFUN (clear_ip_pim_oil,
2696 clear_ip_pim_oil_cmd,
2697 "clear ip pim oil",
2698 CLEAR_STR
2699 IP_STR
2700 CLEAR_IP_PIM_STR
2701 "Rescan PIM OIL (output interface list)\n")
2702 {
2703 pim_scan_oil();
2704
2705 return CMD_SUCCESS;
2706 }
2707
2708 DEFUN (show_ip_igmp_interface,
2709 show_ip_igmp_interface_cmd,
2710 "show ip igmp interface [detail|WORD] [json]",
2711 SHOW_STR
2712 IP_STR
2713 IGMP_STR
2714 "IGMP interface information\n"
2715 "Detailed output\n"
2716 "interface name\n"
2717 "JavaScript Object Notation\n")
2718 {
2719 u_char uj = use_json(argc, argv);
2720 int idx = 0;
2721
2722 if (argv_find(argv, argc, "detail", &idx) ||
2723 argv_find(argv, argc, "WORD", &idx))
2724 igmp_show_interfaces_single(vty, argv[idx]->arg, uj);
2725 else
2726 igmp_show_interfaces(vty, uj);
2727
2728 return CMD_SUCCESS;
2729 }
2730
2731 DEFUN (show_ip_igmp_join,
2732 show_ip_igmp_join_cmd,
2733 "show ip igmp join",
2734 SHOW_STR
2735 IP_STR
2736 IGMP_STR
2737 "IGMP static join information\n")
2738 {
2739 igmp_show_interface_join(vty);
2740
2741 return CMD_SUCCESS;
2742 }
2743
2744 DEFUN (show_ip_igmp_groups,
2745 show_ip_igmp_groups_cmd,
2746 "show ip igmp groups [json]",
2747 SHOW_STR
2748 IP_STR
2749 IGMP_STR
2750 IGMP_GROUP_STR
2751 "JavaScript Object Notation\n")
2752 {
2753 u_char uj = use_json(argc, argv);
2754 igmp_show_groups(vty, uj);
2755
2756 return CMD_SUCCESS;
2757 }
2758
2759 DEFUN (show_ip_igmp_groups_retransmissions,
2760 show_ip_igmp_groups_retransmissions_cmd,
2761 "show ip igmp groups retransmissions",
2762 SHOW_STR
2763 IP_STR
2764 IGMP_STR
2765 IGMP_GROUP_STR
2766 "IGMP group retransmissions\n")
2767 {
2768 igmp_show_group_retransmission(vty);
2769
2770 return CMD_SUCCESS;
2771 }
2772
2773 DEFUN (show_ip_igmp_sources,
2774 show_ip_igmp_sources_cmd,
2775 "show ip igmp sources",
2776 SHOW_STR
2777 IP_STR
2778 IGMP_STR
2779 IGMP_SOURCE_STR)
2780 {
2781 igmp_show_sources(vty);
2782
2783 return CMD_SUCCESS;
2784 }
2785
2786 DEFUN (show_ip_igmp_sources_retransmissions,
2787 show_ip_igmp_sources_retransmissions_cmd,
2788 "show ip igmp sources retransmissions",
2789 SHOW_STR
2790 IP_STR
2791 IGMP_STR
2792 IGMP_SOURCE_STR
2793 "IGMP source retransmissions\n")
2794 {
2795 igmp_show_source_retransmission(vty);
2796
2797 return CMD_SUCCESS;
2798 }
2799
2800 DEFUN (show_ip_pim_assert,
2801 show_ip_pim_assert_cmd,
2802 "show ip pim assert",
2803 SHOW_STR
2804 IP_STR
2805 PIM_STR
2806 "PIM interface assert\n")
2807 {
2808 pim_show_assert(vty);
2809
2810 return CMD_SUCCESS;
2811 }
2812
2813 DEFUN (show_ip_pim_assert_internal,
2814 show_ip_pim_assert_internal_cmd,
2815 "show ip pim assert-internal",
2816 SHOW_STR
2817 IP_STR
2818 PIM_STR
2819 "PIM interface internal assert state\n")
2820 {
2821 pim_show_assert_internal(vty);
2822
2823 return CMD_SUCCESS;
2824 }
2825
2826 DEFUN (show_ip_pim_assert_metric,
2827 show_ip_pim_assert_metric_cmd,
2828 "show ip pim assert-metric",
2829 SHOW_STR
2830 IP_STR
2831 PIM_STR
2832 "PIM interface assert metric\n")
2833 {
2834 pim_show_assert_metric(vty);
2835
2836 return CMD_SUCCESS;
2837 }
2838
2839 DEFUN (show_ip_pim_assert_winner_metric,
2840 show_ip_pim_assert_winner_metric_cmd,
2841 "show ip pim assert-winner-metric",
2842 SHOW_STR
2843 IP_STR
2844 PIM_STR
2845 "PIM interface assert winner metric\n")
2846 {
2847 pim_show_assert_winner_metric(vty);
2848
2849 return CMD_SUCCESS;
2850 }
2851
2852 DEFUN (show_ip_pim_interface,
2853 show_ip_pim_interface_cmd,
2854 "show ip pim interface [detail|WORD] [json]",
2855 SHOW_STR
2856 IP_STR
2857 PIM_STR
2858 "PIM interface information\n"
2859 "Detailed output\n"
2860 "interface name\n"
2861 "JavaScript Object Notation\n")
2862 {
2863 u_char uj = use_json(argc, argv);
2864 int idx = 0;
2865
2866 if (argv_find(argv, argc, "WORD", &idx) ||
2867 argv_find(argv, argc, "detail", &idx))
2868 pim_show_interfaces_single(vty, argv[idx]->arg, uj);
2869
2870 else
2871 pim_show_interfaces(vty, uj);
2872
2873 return CMD_SUCCESS;
2874 }
2875
2876 DEFUN (show_ip_pim_join,
2877 show_ip_pim_join_cmd,
2878 "show ip pim join [json]",
2879 SHOW_STR
2880 IP_STR
2881 PIM_STR
2882 "PIM interface join information\n"
2883 JSON_STR)
2884 {
2885 u_char uj = use_json(argc, argv);
2886 pim_show_join(vty, uj);
2887
2888 return CMD_SUCCESS;
2889 }
2890
2891 DEFUN (show_ip_pim_local_membership,
2892 show_ip_pim_local_membership_cmd,
2893 "show ip pim local-membership [json]",
2894 SHOW_STR
2895 IP_STR
2896 PIM_STR
2897 "PIM interface local-membership\n"
2898 JSON_STR)
2899 {
2900 u_char uj = use_json(argc, argv);
2901 pim_show_membership(vty, uj);
2902
2903 return CMD_SUCCESS;
2904 }
2905
2906 DEFUN (show_ip_pim_neighbor,
2907 show_ip_pim_neighbor_cmd,
2908 "show ip pim neighbor [detail|WORD] [json]",
2909 SHOW_STR
2910 IP_STR
2911 PIM_STR
2912 "PIM neighbor information\n"
2913 "Detailed output\n"
2914 "Name of interface or neighbor\n"
2915 "JavaScript Object Notation\n")
2916 {
2917 u_char uj = use_json(argc, argv);
2918 int idx = 0;
2919
2920 if (argv_find(argv, argc, "detail", &idx) ||
2921 argv_find(argv, argc, "WORD", &idx))
2922 pim_show_neighbors_single(vty, argv[idx]->arg, uj);
2923 else
2924 pim_show_neighbors(vty, uj);
2925
2926 return CMD_SUCCESS;
2927 }
2928
2929 DEFUN (show_ip_pim_secondary,
2930 show_ip_pim_secondary_cmd,
2931 "show ip pim secondary",
2932 SHOW_STR
2933 IP_STR
2934 PIM_STR
2935 "PIM neighbor addresses\n")
2936 {
2937 pim_show_neighbors_secondary(vty);
2938
2939 return CMD_SUCCESS;
2940 }
2941
2942 DEFUN (show_ip_pim_state,
2943 show_ip_pim_state_cmd,
2944 "show ip pim state [A.B.C.D [A.B.C.D]] [json]",
2945 SHOW_STR
2946 IP_STR
2947 PIM_STR
2948 "PIM state information\n"
2949 "Unicast or Multicast address\n"
2950 "Multicast address\n"
2951 "JavaScript Object Notation\n")
2952 {
2953 const char *src_or_group = NULL;
2954 const char *group = NULL;
2955 u_char uj = use_json(argc, argv);
2956 if (uj)
2957 argc--;
2958
2959 if (argc == 6)
2960 {
2961 src_or_group = argv[4]->arg;
2962 group = argv[5]->arg;
2963 }
2964 else if (argc == 5)
2965 src_or_group = argv[4]->arg;
2966
2967 pim_show_state(vty, src_or_group, group, uj);
2968
2969 return CMD_SUCCESS;
2970 }
2971
2972 DEFUN (show_ip_pim_upstream,
2973 show_ip_pim_upstream_cmd,
2974 "show ip pim upstream [json]",
2975 SHOW_STR
2976 IP_STR
2977 PIM_STR
2978 "PIM upstream information\n"
2979 "JavaScript Object Notation\n")
2980 {
2981 u_char uj = use_json(argc, argv);
2982 pim_show_upstream(vty, uj);
2983
2984 return CMD_SUCCESS;
2985 }
2986
2987 DEFUN (show_ip_pim_upstream_join_desired,
2988 show_ip_pim_upstream_join_desired_cmd,
2989 "show ip pim upstream-join-desired [json]",
2990 SHOW_STR
2991 IP_STR
2992 PIM_STR
2993 "PIM upstream join-desired\n"
2994 "JavaScript Object Notation\n")
2995 {
2996 u_char uj = use_json(argc, argv);
2997 pim_show_join_desired(vty, uj);
2998
2999 return CMD_SUCCESS;
3000 }
3001
3002 DEFUN (show_ip_pim_upstream_rpf,
3003 show_ip_pim_upstream_rpf_cmd,
3004 "show ip pim upstream-rpf [json]",
3005 SHOW_STR
3006 IP_STR
3007 PIM_STR
3008 "PIM upstream source rpf\n"
3009 "JavaScript Object Notation\n")
3010 {
3011 u_char uj = use_json(argc, argv);
3012 pim_show_upstream_rpf(vty, uj);
3013
3014 return CMD_SUCCESS;
3015 }
3016
3017 DEFUN (show_ip_pim_rp,
3018 show_ip_pim_rp_cmd,
3019 "show ip pim rp-info [json]",
3020 SHOW_STR
3021 IP_STR
3022 PIM_STR
3023 "PIM RP information\n"
3024 "JavaScript Object Notation\n")
3025 {
3026 u_char uj = use_json(argc, argv);
3027 pim_rp_show_information (vty, uj);
3028
3029 return CMD_SUCCESS;
3030 }
3031
3032 DEFUN (show_ip_pim_rpf,
3033 show_ip_pim_rpf_cmd,
3034 "show ip pim rpf [json]",
3035 SHOW_STR
3036 IP_STR
3037 PIM_STR
3038 "PIM cached source rpf information\n"
3039 "JavaScript Object Notation\n")
3040 {
3041 u_char uj = use_json(argc, argv);
3042 pim_show_rpf(vty, uj);
3043
3044 return CMD_SUCCESS;
3045 }
3046
3047 DEFUN (show_ip_pim_nexthop,
3048 show_ip_pim_nexthop_cmd,
3049 "show ip pim nexthop",
3050 SHOW_STR
3051 IP_STR
3052 PIM_STR
3053 "PIM cached nexthop rpf information\n")
3054 {
3055 pim_show_nexthop (vty);
3056
3057 return CMD_SUCCESS;
3058 }
3059
3060 DEFUN (show_ip_pim_nexthop_lookup,
3061 show_ip_pim_nexthop_lookup_cmd,
3062 "show ip pim nexthop-lookup A.B.C.D A.B.C.D",
3063 SHOW_STR
3064 IP_STR
3065 PIM_STR
3066 "PIM cached nexthop rpf lookup\n"
3067 "Source/RP address\n"
3068 "Multicast Group address\n")
3069 {
3070 struct pim_nexthop_cache pnc;
3071 struct prefix nht_p;
3072 int result = 0;
3073 struct in_addr src_addr, grp_addr;
3074 struct in_addr vif_source;
3075 const char *addr_str, *addr_str1;
3076 struct prefix grp;
3077 struct pim_nexthop nexthop;
3078 char nexthop_addr_str[PREFIX_STRLEN];
3079 char grp_str[PREFIX_STRLEN];
3080
3081 addr_str = argv[4]->arg;
3082 result = inet_pton (AF_INET, addr_str, &src_addr);
3083 if (result <= 0)
3084 {
3085 vty_out (vty, "Bad unicast address %s: errno=%d: %s%s",
3086 addr_str, errno, safe_strerror (errno), VTY_NEWLINE);
3087 return CMD_WARNING;
3088 }
3089
3090 if (pim_is_group_224_4 (src_addr))
3091 {
3092 vty_out (vty, "Invalid argument. Expected Valid Source Address.%s", VTY_NEWLINE);
3093 return CMD_WARNING;
3094 }
3095
3096 addr_str1 = argv[5]->arg;
3097 result = inet_pton (AF_INET, addr_str1, &grp_addr);
3098 if (result <= 0)
3099 {
3100 vty_out (vty, "Bad unicast address %s: errno=%d: %s%s",
3101 addr_str, errno, safe_strerror (errno), VTY_NEWLINE);
3102 return CMD_WARNING;
3103 }
3104
3105 if (!pim_is_group_224_4 (grp_addr))
3106 {
3107 vty_out (vty, "Invalid argument. Expected Valid Multicast Group Address.%s", VTY_NEWLINE);
3108 return CMD_WARNING;
3109 }
3110
3111 if (!pim_rp_set_upstream_addr (&vif_source, src_addr, grp_addr))
3112 return CMD_SUCCESS;
3113
3114 memset (&pnc, 0, sizeof (struct pim_nexthop_cache));
3115 nht_p.family = AF_INET;
3116 nht_p.prefixlen = IPV4_MAX_BITLEN;
3117 nht_p.u.prefix4 = vif_source;
3118 grp.family = AF_INET;
3119 grp.prefixlen = IPV4_MAX_BITLEN;
3120 grp.u.prefix4 = grp_addr;
3121 memset (&nexthop, 0, sizeof (nexthop));
3122
3123 if ((pim_find_or_track_nexthop (&nht_p, NULL, NULL, &pnc)) == 1)
3124 {
3125 //Compute PIM RPF using Cached nexthop
3126 pim_ecmp_nexthop_search (&pnc, &nexthop, &nht_p, &grp, 0);
3127 }
3128 else
3129 pim_ecmp_nexthop_lookup (&nexthop, vif_source, &nht_p, &grp, 0);
3130
3131 pim_addr_dump ("<grp?>", &grp, grp_str, sizeof (grp_str));
3132 pim_addr_dump ("<nexthop?>", &nexthop.mrib_nexthop_addr,
3133 nexthop_addr_str, sizeof (nexthop_addr_str));
3134 vty_out (vty, "Group %s --- Nexthop %s Interface %s %s", grp_str,
3135 nexthop_addr_str, nexthop.interface->name, VTY_NEWLINE);
3136
3137 return CMD_SUCCESS;
3138 }
3139
3140 DEFUN (show_ip_pim_interface_traffic,
3141 show_ip_pim_interface_traffic_cmd,
3142 "show ip pim interface traffic [WORD] [json]",
3143 SHOW_STR
3144 IP_STR
3145 PIM_STR
3146 "PIM interface information\n"
3147 "Protocol Packet counters\n"
3148 "Interface name\n"
3149 "JavaScript Object Notation\n")
3150 {
3151 u_char uj = use_json (argc, argv);
3152 int idx = 0;
3153
3154 if (argv_find(argv, argc, "WORD", &idx))
3155 pim_show_interface_traffic_single (vty, argv[idx]->arg, uj);
3156 else
3157 pim_show_interface_traffic (vty, uj);
3158
3159 return CMD_SUCCESS;
3160 }
3161
3162 static void show_multicast_interfaces(struct vty *vty)
3163 {
3164 struct listnode *node;
3165 struct interface *ifp;
3166
3167 vty_out(vty, "%s", VTY_NEWLINE);
3168
3169 vty_out(vty, "Interface Address ifi Vif PktsIn PktsOut BytesIn BytesOut%s",
3170 VTY_NEWLINE);
3171
3172 for (ALL_LIST_ELEMENTS_RO (vrf_iflist (VRF_DEFAULT), node, ifp)) {
3173 struct pim_interface *pim_ifp;
3174 struct in_addr ifaddr;
3175 struct sioc_vif_req vreq;
3176
3177 pim_ifp = ifp->info;
3178
3179 if (!pim_ifp)
3180 continue;
3181
3182 memset(&vreq, 0, sizeof(vreq));
3183 vreq.vifi = pim_ifp->mroute_vif_index;
3184
3185 if (ioctl(qpim_mroute_socket_fd, SIOCGETVIFCNT, &vreq)) {
3186 zlog_warn("ioctl(SIOCGETVIFCNT=%lu) failure for interface %s vif_index=%d: errno=%d: %s%s",
3187 (unsigned long)SIOCGETVIFCNT,
3188 ifp->name,
3189 pim_ifp->mroute_vif_index,
3190 errno,
3191 safe_strerror(errno),
3192 VTY_NEWLINE);
3193 }
3194
3195 ifaddr = pim_ifp->primary_address;
3196
3197 vty_out(vty, "%-9s %-15s %3d %3d %7lu %7lu %10lu %10lu%s",
3198 ifp->name,
3199 inet_ntoa(ifaddr),
3200 ifp->ifindex,
3201 pim_ifp->mroute_vif_index,
3202 (unsigned long) vreq.icount,
3203 (unsigned long) vreq.ocount,
3204 (unsigned long) vreq.ibytes,
3205 (unsigned long) vreq.obytes,
3206 VTY_NEWLINE);
3207 }
3208 }
3209
3210 DEFUN (show_ip_multicast,
3211 show_ip_multicast_cmd,
3212 "show ip multicast",
3213 SHOW_STR
3214 IP_STR
3215 "Multicast global information\n")
3216 {
3217 time_t now = pim_time_monotonic_sec();
3218
3219 char uptime[10];
3220
3221 vty_out(vty, "Mroute socket descriptor: %d%s",
3222 qpim_mroute_socket_fd,
3223 VTY_NEWLINE);
3224
3225 pim_time_uptime(uptime, sizeof(uptime), now - qpim_mroute_socket_creation);
3226 vty_out(vty, "Mroute socket uptime: %s%s",
3227 uptime,
3228 VTY_NEWLINE);
3229
3230 vty_out(vty, "%s", VTY_NEWLINE);
3231
3232 pim_zebra_zclient_update (vty);
3233 pim_zlookup_show_ip_multicast (vty);
3234
3235 vty_out(vty, "%s", VTY_NEWLINE);
3236 vty_out(vty, "Maximum highest VifIndex: %d%s",
3237 PIM_MAX_USABLE_VIFS,
3238 VTY_NEWLINE);
3239
3240 vty_out (vty, "%s", VTY_NEWLINE);
3241 vty_out (vty, "Upstream Join Timer: %d secs%s",
3242 qpim_t_periodic, VTY_NEWLINE);
3243 vty_out (vty, "Join/Prune Holdtime: %d secs%s",
3244 PIM_JP_HOLDTIME, VTY_NEWLINE);
3245 vty_out (vty, "PIM ECMP: %s%s",
3246 qpim_ecmp_enable ? "Enable" : "Disable", VTY_NEWLINE);
3247 vty_out (vty, "PIM ECMP Rebalance: %s%s",
3248 qpim_ecmp_rebalance_enable ? "Enable" : "Disable", VTY_NEWLINE);
3249
3250 vty_out (vty, "%s", VTY_NEWLINE);
3251
3252 show_rpf_refresh_stats(vty, now, NULL);
3253
3254 vty_out(vty, "%s", VTY_NEWLINE);
3255
3256 show_scan_oil_stats(vty, now);
3257
3258 show_multicast_interfaces(vty);
3259
3260 return CMD_SUCCESS;
3261 }
3262
3263 static void show_mroute(struct vty *vty, u_char uj)
3264 {
3265 struct listnode *node;
3266 struct channel_oil *c_oil;
3267 struct static_route *s_route;
3268 time_t now;
3269 json_object *json = NULL;
3270 json_object *json_group = NULL;
3271 json_object *json_source = NULL;
3272 json_object *json_oil = NULL;
3273 json_object *json_ifp_out = NULL;
3274 int found_oif = 0;
3275 int first = 1;
3276 char grp_str[INET_ADDRSTRLEN];
3277 char src_str[INET_ADDRSTRLEN];
3278 char in_ifname[INTERFACE_NAMSIZ+1];
3279 char out_ifname[INTERFACE_NAMSIZ+1];
3280 int oif_vif_index;
3281 struct interface *ifp_in;
3282 char proto[100];
3283
3284 if (uj) {
3285 json = json_object_new_object();
3286 } else {
3287 vty_out(vty, "Source Group Proto Input Output TTL Uptime%s",
3288 VTY_NEWLINE);
3289 }
3290
3291 now = pim_time_monotonic_sec();
3292
3293 /* print list of PIM and IGMP routes */
3294 for (ALL_LIST_ELEMENTS_RO(pim_channel_oil_list, node, c_oil)) {
3295 found_oif = 0;
3296 first = 1;
3297 if (!c_oil->installed && !uj)
3298 continue;
3299
3300 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp, grp_str, sizeof(grp_str));
3301 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin, src_str, sizeof(src_str));
3302 ifp_in = pim_if_find_by_vif_index(c_oil->oil.mfcc_parent);
3303
3304 if (ifp_in)
3305 strcpy(in_ifname, ifp_in->name);
3306 else
3307 strcpy(in_ifname, "<iif?>");
3308
3309 if (uj) {
3310
3311 /* Find the group, create it if it doesn't exist */
3312 json_object_object_get_ex(json, grp_str, &json_group);
3313
3314 if (!json_group) {
3315 json_group = json_object_new_object();
3316 json_object_object_add(json, grp_str, json_group);
3317 }
3318
3319 /* Find the source nested under the group, create it if it doesn't exist */
3320 json_object_object_get_ex(json_group, src_str, &json_source);
3321
3322 if (!json_source) {
3323 json_source = json_object_new_object();
3324 json_object_object_add(json_group, src_str, json_source);
3325 }
3326
3327 /* Find the inbound interface nested under the source, create it if it doesn't exist */
3328 json_object_int_add(json_source, "installed", c_oil->installed);
3329 json_object_int_add(json_source, "refCount", c_oil->oil_ref_count);
3330 json_object_int_add(json_source, "oilSize", c_oil->oil_size);
3331 json_object_int_add(json_source, "OilInheritedRescan", c_oil->oil_inherited_rescan);
3332 json_object_string_add(json_source, "iif", in_ifname);
3333 json_oil = NULL;
3334 }
3335
3336 for (oif_vif_index = 0; oif_vif_index < MAXVIFS; ++oif_vif_index) {
3337 struct interface *ifp_out;
3338 char oif_uptime[10];
3339 int ttl;
3340
3341 ttl = c_oil->oil.mfcc_ttls[oif_vif_index];
3342 if (ttl < 1)
3343 continue;
3344
3345 ifp_out = pim_if_find_by_vif_index(oif_vif_index);
3346 pim_time_uptime(oif_uptime, sizeof(oif_uptime), now - c_oil->oif_creation[oif_vif_index]);
3347 found_oif = 1;
3348
3349 if (ifp_out)
3350 strcpy(out_ifname, ifp_out->name);
3351 else
3352 strcpy(out_ifname, "<oif?>");
3353
3354 if (uj) {
3355 json_ifp_out = json_object_new_object();
3356 json_object_string_add(json_ifp_out, "source", src_str);
3357 json_object_string_add(json_ifp_out, "group", grp_str);
3358
3359 if (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_PIM)
3360 json_object_boolean_true_add(json_ifp_out, "protocolPim");
3361
3362 if (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_IGMP)
3363 json_object_boolean_true_add(json_ifp_out, "protocolIgmp");
3364
3365 if (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_SOURCE)
3366 json_object_boolean_true_add(json_ifp_out, "protocolSource");
3367
3368 if (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_STAR)
3369 json_object_boolean_true_add(json_ifp_out, "protocolInherited");
3370
3371 json_object_string_add(json_ifp_out, "inboundInterface", in_ifname);
3372 json_object_int_add(json_ifp_out, "iVifI", c_oil->oil.mfcc_parent);
3373 json_object_string_add(json_ifp_out, "outboundInterface", out_ifname);
3374 json_object_int_add(json_ifp_out, "oVifI", oif_vif_index);
3375 json_object_int_add(json_ifp_out, "ttl", ttl);
3376 json_object_string_add(json_ifp_out, "upTime", oif_uptime);
3377 if (!json_oil) {
3378 json_oil = json_object_new_object();
3379 json_object_object_add(json_source, "oil", json_oil);
3380 }
3381 json_object_object_add(json_oil, out_ifname, json_ifp_out);
3382 } else {
3383 if (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_PIM) {
3384 strcpy(proto, "PIM");
3385 }
3386
3387 if (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_IGMP) {
3388 strcpy(proto, "IGMP");
3389 }
3390
3391 if (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_SOURCE) {
3392 strcpy(proto, "SRC");
3393 }
3394
3395 if (c_oil->oif_flags[oif_vif_index] & PIM_OIF_FLAG_PROTO_STAR) {
3396 strcpy(proto, "STAR");
3397 }
3398
3399 vty_out(vty, "%-15s %-15s %-6s %-10s %-10s %-3d %8s%s",
3400 src_str,
3401 grp_str,
3402 proto,
3403 in_ifname,
3404 out_ifname,
3405 ttl,
3406 oif_uptime,
3407 VTY_NEWLINE);
3408
3409 if (first)
3410 {
3411 src_str[0] = '\0';
3412 grp_str[0] = '\0';
3413 in_ifname[0] = '\0';
3414 first = 0;
3415 }
3416 }
3417 }
3418
3419 if (!uj && !found_oif) {
3420 vty_out(vty, "%-15s %-15s %-6s %-10s %-10s %-3d %8s%s",
3421 src_str,
3422 grp_str,
3423 "none",
3424 in_ifname,
3425 "none",
3426 0,
3427 "--:--:--",
3428 VTY_NEWLINE);
3429 }
3430 }
3431
3432 /* Print list of static routes */
3433 for (ALL_LIST_ELEMENTS_RO(qpim_static_route_list, node, s_route)) {
3434 first = 1;
3435
3436 if (!s_route->c_oil.installed)
3437 continue;
3438
3439 pim_inet4_dump("<group?>", s_route->group, grp_str, sizeof(grp_str));
3440 pim_inet4_dump("<source?>", s_route->source, src_str, sizeof(src_str));
3441 ifp_in = pim_if_find_by_vif_index(s_route->iif);
3442 found_oif = 0;
3443
3444 if (ifp_in)
3445 strcpy(in_ifname, ifp_in->name);
3446 else
3447 strcpy(in_ifname, "<iif?>");
3448
3449 if (uj) {
3450
3451 /* Find the group, create it if it doesn't exist */
3452 json_object_object_get_ex(json, grp_str, &json_group);
3453
3454 if (!json_group) {
3455 json_group = json_object_new_object();
3456 json_object_object_add(json, grp_str, json_group);
3457 }
3458
3459 /* Find the source nested under the group, create it if it doesn't exist */
3460 json_object_object_get_ex(json_group, src_str, &json_source);
3461
3462 if (!json_source) {
3463 json_source = json_object_new_object();
3464 json_object_object_add(json_group, src_str, json_source);
3465 }
3466
3467 json_object_string_add(json_source, "iif", in_ifname);
3468 json_oil = NULL;
3469 } else {
3470 strcpy(proto, "STATIC");
3471 }
3472
3473 for (oif_vif_index = 0; oif_vif_index < MAXVIFS; ++oif_vif_index) {
3474 struct interface *ifp_out;
3475 char oif_uptime[10];
3476 int ttl;
3477
3478 ttl = s_route->oif_ttls[oif_vif_index];
3479 if (ttl < 1)
3480 continue;
3481
3482 ifp_out = pim_if_find_by_vif_index(oif_vif_index);
3483 pim_time_uptime(oif_uptime, sizeof(oif_uptime), now - s_route->c_oil.oif_creation[oif_vif_index]);
3484 found_oif = 1;
3485
3486 if (ifp_out)
3487 strcpy(out_ifname, ifp_out->name);
3488 else
3489 strcpy(out_ifname, "<oif?>");
3490
3491 if (uj) {
3492 json_ifp_out = json_object_new_object();
3493 json_object_string_add(json_ifp_out, "source", src_str);
3494 json_object_string_add(json_ifp_out, "group", grp_str);
3495 json_object_boolean_true_add(json_ifp_out, "protocolStatic");
3496 json_object_string_add(json_ifp_out, "inboundInterface", in_ifname);
3497 json_object_int_add(json_ifp_out, "iVifI", c_oil->oil.mfcc_parent);
3498 json_object_string_add(json_ifp_out, "outboundInterface", out_ifname);
3499 json_object_int_add(json_ifp_out, "oVifI", oif_vif_index);
3500 json_object_int_add(json_ifp_out, "ttl", ttl);
3501 json_object_string_add(json_ifp_out, "upTime", oif_uptime);
3502 if (!json_oil) {
3503 json_oil = json_object_new_object();
3504 json_object_object_add(json_source, "oil", json_oil);
3505 }
3506 json_object_object_add(json_oil, out_ifname, json_ifp_out);
3507 } else {
3508 vty_out(vty, "%-15s %-15s %-6s %-10s %-10s %-3d %8s%s",
3509 src_str,
3510 grp_str,
3511 proto,
3512 in_ifname,
3513 out_ifname,
3514 ttl,
3515 oif_uptime,
3516 VTY_NEWLINE);
3517 if (first)
3518 {
3519 src_str[0] = '\0';
3520 grp_str[0] = '\0';
3521 in_ifname[0] = '\0';
3522 first = 0;
3523 }
3524 }
3525 }
3526
3527 if (!uj && !found_oif) {
3528 vty_out(vty, "%-15s %-15s %-6s %-10s %-10s %-3d %8s%s",
3529 src_str,
3530 grp_str,
3531 proto,
3532 in_ifname,
3533 "none",
3534 0,
3535 "--:--:--",
3536 VTY_NEWLINE);
3537 }
3538 }
3539
3540 if (uj) {
3541 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
3542 json_object_free(json);
3543 }
3544 }
3545
3546 DEFUN (show_ip_mroute,
3547 show_ip_mroute_cmd,
3548 "show ip mroute [json]",
3549 SHOW_STR
3550 IP_STR
3551 MROUTE_STR
3552 JSON_STR)
3553 {
3554 u_char uj = use_json(argc, argv);
3555 show_mroute(vty, uj);
3556 return CMD_SUCCESS;
3557 }
3558
3559 static void show_mroute_count(struct vty *vty)
3560 {
3561 struct listnode *node;
3562 struct channel_oil *c_oil;
3563 struct static_route *s_route;
3564
3565 vty_out(vty, "%s", VTY_NEWLINE);
3566
3567 vty_out(vty, "Source Group LastUsed Packets Bytes WrongIf %s",
3568 VTY_NEWLINE);
3569
3570 /* Print PIM and IGMP route counts */
3571 for (ALL_LIST_ELEMENTS_RO(pim_channel_oil_list, node, c_oil)) {
3572 char group_str[INET_ADDRSTRLEN];
3573 char source_str[INET_ADDRSTRLEN];
3574
3575 if (!c_oil->installed)
3576 continue;
3577
3578 pim_mroute_update_counters (c_oil);
3579
3580 pim_inet4_dump("<group?>", c_oil->oil.mfcc_mcastgrp, group_str, sizeof(group_str));
3581 pim_inet4_dump("<source?>", c_oil->oil.mfcc_origin, source_str, sizeof(source_str));
3582
3583 vty_out(vty, "%-15s %-15s %-8llu %-7ld %-10ld %-7ld%s",
3584 source_str,
3585 group_str,
3586 c_oil->cc.lastused/100,
3587 c_oil->cc.pktcnt,
3588 c_oil->cc.bytecnt,
3589 c_oil->cc.wrong_if,
3590 VTY_NEWLINE);
3591 }
3592
3593 /* Print static route counts */
3594 for (ALL_LIST_ELEMENTS_RO(qpim_static_route_list, node, s_route)) {
3595 char group_str[INET_ADDRSTRLEN];
3596 char source_str[INET_ADDRSTRLEN];
3597
3598 if (!s_route->c_oil.installed)
3599 continue;
3600
3601 pim_mroute_update_counters (&s_route->c_oil);
3602
3603 pim_inet4_dump("<group?>", s_route->c_oil.oil.mfcc_mcastgrp, group_str, sizeof(group_str));
3604 pim_inet4_dump("<source?>", s_route->c_oil.oil.mfcc_origin, source_str, sizeof(source_str));
3605
3606 vty_out(vty, "%-15s %-15s %-8llu %-7ld %-10ld %-7ld%s",
3607 source_str,
3608 group_str,
3609 s_route->c_oil.cc.lastused,
3610 s_route->c_oil.cc.pktcnt,
3611 s_route->c_oil.cc.bytecnt,
3612 s_route->c_oil.cc.wrong_if,
3613 VTY_NEWLINE);
3614 }
3615 }
3616
3617 DEFUN (show_ip_mroute_count,
3618 show_ip_mroute_count_cmd,
3619 "show ip mroute count",
3620 SHOW_STR
3621 IP_STR
3622 MROUTE_STR
3623 "Route and packet count data\n")
3624 {
3625 show_mroute_count(vty);
3626 return CMD_SUCCESS;
3627 }
3628
3629 DEFUN (show_ip_rib,
3630 show_ip_rib_cmd,
3631 "show ip rib A.B.C.D",
3632 SHOW_STR
3633 IP_STR
3634 RIB_STR
3635 "Unicast address\n")
3636 {
3637 int idx_ipv4 = 3;
3638 struct in_addr addr;
3639 const char *addr_str;
3640 struct pim_nexthop nexthop;
3641 char nexthop_addr_str[PREFIX_STRLEN];
3642 int result;
3643
3644 memset (&nexthop, 0, sizeof (nexthop));
3645 addr_str = argv[idx_ipv4]->arg;
3646 result = inet_pton(AF_INET, addr_str, &addr);
3647 if (result <= 0) {
3648 vty_out(vty, "Bad unicast address %s: errno=%d: %s%s",
3649 addr_str, errno, safe_strerror(errno), VTY_NEWLINE);
3650 return CMD_WARNING;
3651 }
3652
3653 if (pim_nexthop_lookup(&nexthop, addr, 0)) {
3654 vty_out(vty, "Failure querying RIB nexthop for unicast address %s%s",
3655 addr_str, VTY_NEWLINE);
3656 return CMD_WARNING;
3657 }
3658
3659 vty_out(vty, "Address NextHop Interface Metric Preference%s",
3660 VTY_NEWLINE);
3661
3662 pim_addr_dump("<nexthop?>", &nexthop.mrib_nexthop_addr,
3663 nexthop_addr_str, sizeof(nexthop_addr_str));
3664
3665 vty_out(vty, "%-15s %-15s %-9s %6d %10d%s",
3666 addr_str,
3667 nexthop_addr_str,
3668 nexthop.interface ? nexthop.interface->name : "<ifname?>",
3669 nexthop.mrib_route_metric,
3670 nexthop.mrib_metric_preference,
3671 VTY_NEWLINE);
3672
3673 return CMD_SUCCESS;
3674 }
3675
3676 static void show_ssmpingd(struct vty *vty)
3677 {
3678 struct listnode *node;
3679 struct ssmpingd_sock *ss;
3680 time_t now;
3681
3682 vty_out(vty, "Source Socket Address Port Uptime Requests%s",
3683 VTY_NEWLINE);
3684
3685 if (!qpim_ssmpingd_list)
3686 return;
3687
3688 now = pim_time_monotonic_sec();
3689
3690 for (ALL_LIST_ELEMENTS_RO(qpim_ssmpingd_list, node, ss)) {
3691 char source_str[INET_ADDRSTRLEN];
3692 char ss_uptime[10];
3693 struct sockaddr_in bind_addr;
3694 socklen_t len = sizeof(bind_addr);
3695 char bind_addr_str[INET_ADDRSTRLEN];
3696
3697 pim_inet4_dump("<src?>", ss->source_addr, source_str, sizeof(source_str));
3698
3699 if (pim_socket_getsockname(ss->sock_fd, (struct sockaddr *) &bind_addr, &len)) {
3700 vty_out(vty, "%% Failure reading socket name for ssmpingd source %s on fd=%d%s",
3701 source_str, ss->sock_fd, VTY_NEWLINE);
3702 }
3703
3704 pim_inet4_dump("<addr?>", bind_addr.sin_addr, bind_addr_str, sizeof(bind_addr_str));
3705 pim_time_uptime(ss_uptime, sizeof(ss_uptime), now - ss->creation);
3706
3707 vty_out(vty, "%-15s %6d %-15s %5d %8s %8lld%s",
3708 source_str,
3709 ss->sock_fd,
3710 bind_addr_str,
3711 ntohs(bind_addr.sin_port),
3712 ss_uptime,
3713 (long long)ss->requests,
3714 VTY_NEWLINE);
3715 }
3716 }
3717
3718 DEFUN (show_ip_ssmpingd,
3719 show_ip_ssmpingd_cmd,
3720 "show ip ssmpingd",
3721 SHOW_STR
3722 IP_STR
3723 SHOW_SSMPINGD_STR)
3724 {
3725 show_ssmpingd(vty);
3726 return CMD_SUCCESS;
3727 }
3728
3729 static int
3730 pim_rp_cmd_worker (struct vty *vty, const char *rp, const char *group, const char *plist)
3731 {
3732 int result;
3733
3734 result = pim_rp_new (rp, group, plist);
3735
3736 if (result == PIM_MALLOC_FAIL)
3737 {
3738 vty_out (vty, "%% Out of memory%s", VTY_NEWLINE);
3739 return CMD_WARNING;
3740 }
3741
3742 if (result == PIM_GROUP_BAD_ADDRESS)
3743 {
3744 vty_out (vty, "%% Bad group address specified: %s%s", group, VTY_NEWLINE);
3745 return CMD_WARNING;
3746 }
3747
3748 if (result == PIM_RP_BAD_ADDRESS)
3749 {
3750 vty_out (vty, "%% Bad RP address specified: %s%s", rp, VTY_NEWLINE);
3751 return CMD_WARNING;
3752 }
3753
3754 if (result == PIM_RP_NO_PATH)
3755 {
3756 vty_out (vty, "%% No Path to RP address specified: %s%s", rp, VTY_NEWLINE);
3757 return CMD_WARNING;
3758 }
3759
3760 if (result == PIM_GROUP_OVERLAP)
3761 {
3762 vty_out (vty, "%% Group range specified cannot overlap%s", VTY_NEWLINE);
3763 return CMD_WARNING;
3764 }
3765
3766 if (result == PIM_GROUP_PFXLIST_OVERLAP)
3767 {
3768 vty_out (vty, "%% This group is already covered by a RP prefix-list%s", VTY_NEWLINE);
3769 return CMD_WARNING;
3770 }
3771
3772 if (result == PIM_RP_PFXLIST_IN_USE)
3773 {
3774 vty_out (vty, "%% The same prefix-list cannot be applied to multiple RPs%s", VTY_NEWLINE);
3775 return CMD_WARNING;
3776 }
3777
3778 return CMD_SUCCESS;
3779 }
3780
3781 static int
3782 pim_cmd_spt_switchover (enum pim_spt_switchover spt, const char *plist)
3783 {
3784 pimg->spt.switchover = spt;
3785
3786 switch (pimg->spt.switchover)
3787 {
3788 case PIM_SPT_IMMEDIATE:
3789 if (pimg->spt.plist)
3790 XFREE (MTYPE_PIM_SPT_PLIST_NAME, pimg->spt.plist);
3791
3792 pim_upstream_add_lhr_star_pimreg ();
3793 break;
3794 case PIM_SPT_INFINITY:
3795 pim_upstream_remove_lhr_star_pimreg (plist);
3796
3797 if (pimg->spt.plist)
3798 XFREE (MTYPE_PIM_SPT_PLIST_NAME, pimg->spt.plist);
3799
3800 if (plist)
3801 pimg->spt.plist = XSTRDUP (MTYPE_PIM_SPT_PLIST_NAME, plist);
3802 break;
3803 }
3804
3805 return CMD_SUCCESS;
3806 }
3807
3808 DEFUN (ip_pim_spt_switchover_infinity,
3809 ip_pim_spt_switchover_infinity_cmd,
3810 "ip pim spt-switchover infinity-and-beyond",
3811 IP_STR
3812 PIM_STR
3813 "SPT-Switchover\n"
3814 "Never switch to SPT Tree\n")
3815 {
3816 return pim_cmd_spt_switchover (PIM_SPT_INFINITY, NULL);
3817 }
3818
3819 DEFUN (ip_pim_spt_switchover_infinity_plist,
3820 ip_pim_spt_switchover_infinity_plist_cmd,
3821 "ip pim spt-switchover infinity-and-beyond prefix-list WORD",
3822 IP_STR
3823 PIM_STR
3824 "SPT-Switchover\n"
3825 "Never switch to SPT Tree\n"
3826 "Prefix-List to control which groups to switch\n"
3827 "Prefix-List name\n")
3828 {
3829 return pim_cmd_spt_switchover (PIM_SPT_INFINITY, argv[5]->arg);
3830 }
3831
3832 DEFUN (no_ip_pim_spt_switchover_infinity,
3833 no_ip_pim_spt_switchover_infinity_cmd,
3834 "no ip pim spt-switchover infinity-and-beyond",
3835 NO_STR
3836 IP_STR
3837 PIM_STR
3838 "SPT_Switchover\n"
3839 "Never switch to SPT Tree\n")
3840 {
3841 return pim_cmd_spt_switchover (PIM_SPT_IMMEDIATE, NULL);
3842 }
3843
3844 DEFUN (no_ip_pim_spt_switchover_infinity_plist,
3845 no_ip_pim_spt_switchover_infinity_plist_cmd,
3846 "no ip pim spt-switchover infinity-and-beyond prefix-list WORD",
3847 NO_STR
3848 IP_STR
3849 PIM_STR
3850 "SPT_Switchover\n"
3851 "Never switch to SPT Tree\n"
3852 "Prefix-List to control which groups to switch\n"
3853 "Prefix-List name\n")
3854 {
3855 return pim_cmd_spt_switchover (PIM_SPT_IMMEDIATE, NULL);
3856 }
3857
3858 DEFUN (ip_pim_joinprune_time,
3859 ip_pim_joinprune_time_cmd,
3860 "ip pim join-prune-interval (60-600)",
3861 IP_STR
3862 "pim multicast routing\n"
3863 "Join Prune Send Interval\n"
3864 "Seconds\n")
3865 {
3866 qpim_t_periodic = atoi(argv[3]->arg);
3867 return CMD_SUCCESS;
3868 }
3869
3870 DEFUN (no_ip_pim_joinprune_time,
3871 no_ip_pim_joinprune_time_cmd,
3872 "no ip pim join-prune-interval (60-600)",
3873 NO_STR
3874 IP_STR
3875 "pim multicast routing\n"
3876 "Join Prune Send Interval\n"
3877 "Seconds\n")
3878 {
3879 qpim_t_periodic = PIM_DEFAULT_T_PERIODIC;
3880 return CMD_SUCCESS;
3881 }
3882
3883 DEFUN (ip_pim_register_suppress,
3884 ip_pim_register_suppress_cmd,
3885 "ip pim register-suppress-time (5-60000)",
3886 IP_STR
3887 "pim multicast routing\n"
3888 "Register Suppress Timer\n"
3889 "Seconds\n")
3890 {
3891 qpim_register_suppress_time = atoi (argv[3]->arg);
3892 return CMD_SUCCESS;
3893 }
3894
3895 DEFUN (no_ip_pim_register_suppress,
3896 no_ip_pim_register_suppress_cmd,
3897 "no ip pim register-suppress-time (5-60000)",
3898 NO_STR
3899 IP_STR
3900 "pim multicast routing\n"
3901 "Register Suppress Timer\n"
3902 "Seconds\n")
3903 {
3904 qpim_register_suppress_time = PIM_REGISTER_SUPPRESSION_TIME_DEFAULT;
3905 return CMD_SUCCESS;
3906 }
3907
3908 DEFUN (ip_pim_keep_alive,
3909 ip_pim_keep_alive_cmd,
3910 "ip pim keep-alive-timer (31-60000)",
3911 IP_STR
3912 "pim multicast routing\n"
3913 "Keep alive Timer\n"
3914 "Seconds\n")
3915 {
3916 qpim_keep_alive_time = atoi (argv[3]->arg);
3917 return CMD_SUCCESS;
3918 }
3919
3920 DEFUN (no_ip_pim_keep_alive,
3921 no_ip_pim_keep_alive_cmd,
3922 "no ip pim keep-alive-timer (31-60000)",
3923 NO_STR
3924 IP_STR
3925 "pim multicast routing\n"
3926 "Keep alive Timer\n"
3927 "Seconds\n")
3928 {
3929 qpim_keep_alive_time = PIM_KEEPALIVE_PERIOD;
3930 return CMD_SUCCESS;
3931 }
3932
3933 DEFUN (ip_pim_packets,
3934 ip_pim_packets_cmd,
3935 "ip pim packets (1-100)",
3936 IP_STR
3937 "pim multicast routing\n"
3938 "packets to process at one time per fd\n"
3939 "Number of packets\n")
3940 {
3941 qpim_packet_process = atoi (argv[3]->arg);
3942 return CMD_SUCCESS;
3943 }
3944
3945 DEFUN (no_ip_pim_packets,
3946 no_ip_pim_packets_cmd,
3947 "no ip pim packets (1-100)",
3948 NO_STR
3949 IP_STR
3950 "pim multicast routing\n"
3951 "packets to process at one time per fd\n"
3952 "Number of packets\n")
3953 {
3954 qpim_packet_process = PIM_DEFAULT_PACKET_PROCESS;
3955 return CMD_SUCCESS;
3956 }
3957
3958 DEFUN (ip_pim_v6_secondary,
3959 ip_pim_v6_secondary_cmd,
3960 "ip pim send-v6-secondary",
3961 IP_STR
3962 "pim multicast routing\n"
3963 "Send v6 secondary addresses\n")
3964 {
3965 pimg->send_v6_secondary = 1;
3966
3967 return CMD_SUCCESS;
3968 }
3969
3970 DEFUN (no_ip_pim_v6_secondary,
3971 no_ip_pim_v6_secondary_cmd,
3972 "no ip pim send-v6-secondary",
3973 NO_STR
3974 IP_STR
3975 "pim multicast routing\n"
3976 "Send v6 secondary addresses\n")
3977 {
3978 pimg->send_v6_secondary = 0;
3979
3980 return CMD_SUCCESS;
3981 }
3982
3983 DEFUN (ip_pim_rp,
3984 ip_pim_rp_cmd,
3985 "ip pim rp A.B.C.D [A.B.C.D/M]",
3986 IP_STR
3987 "pim multicast routing\n"
3988 "Rendevous Point\n"
3989 "ip address of RP\n"
3990 "Group Address range to cover\n")
3991 {
3992 int idx_ipv4 = 3;
3993
3994 if (argc == (idx_ipv4 + 1))
3995 return pim_rp_cmd_worker (vty, argv[idx_ipv4]->arg, NULL, NULL);
3996 else
3997 return pim_rp_cmd_worker (vty, argv[idx_ipv4]->arg, argv[idx_ipv4 + 1]->arg, NULL);
3998
3999 }
4000
4001 DEFUN (ip_pim_rp_prefix_list,
4002 ip_pim_rp_prefix_list_cmd,
4003 "ip pim rp A.B.C.D prefix-list WORD",
4004 IP_STR
4005 "pim multicast routing\n"
4006 "Rendevous Point\n"
4007 "ip address of RP\n"
4008 "group prefix-list filter\n"
4009 "Name of a prefix-list\n")
4010 {
4011 return pim_rp_cmd_worker (vty, argv[3]->arg, NULL, argv[5]->arg);
4012 }
4013
4014 static int
4015 pim_no_rp_cmd_worker (struct vty *vty, const char *rp, const char *group,
4016 const char *plist)
4017 {
4018 int result = pim_rp_del (rp, group, plist);
4019
4020 if (result == PIM_GROUP_BAD_ADDRESS)
4021 {
4022 vty_out (vty, "%% Bad group address specified: %s%s", group, VTY_NEWLINE);
4023 return CMD_WARNING;
4024 }
4025
4026 if (result == PIM_RP_BAD_ADDRESS)
4027 {
4028 vty_out (vty, "%% Bad RP address specified: %s%s", rp, VTY_NEWLINE);
4029 return CMD_WARNING;
4030 }
4031
4032 if (result == PIM_RP_NOT_FOUND)
4033 {
4034 vty_out (vty, "%% Unable to find specified RP%s", VTY_NEWLINE);
4035 return CMD_WARNING;
4036 }
4037
4038 return CMD_SUCCESS;
4039 }
4040
4041 DEFUN (no_ip_pim_rp,
4042 no_ip_pim_rp_cmd,
4043 "no ip pim rp A.B.C.D [A.B.C.D/M]",
4044 NO_STR
4045 IP_STR
4046 "pim multicast routing\n"
4047 "Rendevous Point\n"
4048 "ip address of RP\n"
4049 "Group Address range to cover\n")
4050 {
4051 int idx_ipv4 = 4, idx_group = 0;
4052
4053 if (argv_find (argv, argc, "A.B.C.D/M", &idx_group))
4054 return pim_no_rp_cmd_worker (vty, argv[idx_ipv4]->arg, argv[idx_group]->arg, NULL);
4055 else
4056 return pim_no_rp_cmd_worker (vty, argv[idx_ipv4]->arg, NULL, NULL);
4057 }
4058
4059 DEFUN (no_ip_pim_rp_prefix_list,
4060 no_ip_pim_rp_prefix_list_cmd,
4061 "no ip pim rp A.B.C.D prefix-list WORD",
4062 NO_STR
4063 IP_STR
4064 "pim multicast routing\n"
4065 "Rendevous Point\n"
4066 "ip address of RP\n"
4067 "group prefix-list filter\n"
4068 "Name of a prefix-list\n")
4069 {
4070 return pim_no_rp_cmd_worker (vty, argv[4]->arg, NULL, argv[6]->arg);
4071 }
4072
4073 static int
4074 pim_ssm_cmd_worker (struct vty *vty, const char *plist)
4075 {
4076 int result = pim_ssm_range_set (VRF_DEFAULT, plist);
4077
4078 if (result == PIM_SSM_ERR_NONE)
4079 return CMD_SUCCESS;
4080
4081 switch (result)
4082 {
4083 case PIM_SSM_ERR_NO_VRF:
4084 vty_out (vty, "%% VRF doesn't exist%s", VTY_NEWLINE);
4085 break;
4086 case PIM_SSM_ERR_DUP:
4087 vty_out (vty, "%% duplicate config%s", VTY_NEWLINE);
4088 break;
4089 default:
4090 vty_out (vty, "%% ssm range config failed%s", VTY_NEWLINE);
4091 }
4092
4093 return CMD_WARNING;
4094 }
4095
4096 DEFUN (ip_pim_ssm_prefix_list,
4097 ip_pim_ssm_prefix_list_cmd,
4098 "ip pim ssm prefix-list WORD",
4099 IP_STR
4100 "pim multicast routing\n"
4101 "Source Specific Multicast\n"
4102 "group range prefix-list filter\n"
4103 "Name of a prefix-list\n")
4104 {
4105 return pim_ssm_cmd_worker (vty, argv[0]->arg);
4106 }
4107
4108 DEFUN (no_ip_pim_ssm_prefix_list,
4109 no_ip_pim_ssm_prefix_list_cmd,
4110 "no ip pim ssm prefix-list",
4111 NO_STR
4112 IP_STR
4113 "pim multicast routing\n"
4114 "Source Specific Multicast\n"
4115 "group range prefix-list filter\n")
4116 {
4117 return pim_ssm_cmd_worker (vty, NULL);
4118 }
4119
4120 DEFUN (no_ip_pim_ssm_prefix_list_name,
4121 no_ip_pim_ssm_prefix_list_name_cmd,
4122 "no ip pim ssm prefix-list WORD",
4123 NO_STR
4124 IP_STR
4125 "pim multicast routing\n"
4126 "Source Specific Multicast\n"
4127 "group range prefix-list filter\n"
4128 "Name of a prefix-list\n")
4129 {
4130 struct pim_ssm *ssm = pimg->ssm_info;
4131
4132 if (ssm->plist_name && !strcmp(ssm->plist_name, argv[0]->arg))
4133 return pim_ssm_cmd_worker (vty, NULL);
4134
4135 vty_out (vty, "%% pim ssm prefix-list %s doesn't exist%s",
4136 argv[0]->arg, VTY_NEWLINE);
4137
4138 return CMD_WARNING;
4139 }
4140
4141 static void
4142 ip_pim_ssm_show_group_range(struct vty *vty, u_char uj)
4143 {
4144 struct pim_ssm *ssm = pimg->ssm_info;
4145 const char *range_str = ssm->plist_name?ssm->plist_name:PIM_SSM_STANDARD_RANGE;
4146
4147 if (uj)
4148 {
4149 json_object *json;
4150 json = json_object_new_object();
4151 json_object_string_add(json, "ssmGroups", range_str);
4152 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
4153 json_object_free(json);
4154 }
4155 else
4156 vty_out(vty, "SSM group range : %s%s", range_str, VTY_NEWLINE);
4157 }
4158
4159 DEFUN (show_ip_pim_ssm_range,
4160 show_ip_pim_ssm_range_cmd,
4161 "show ip pim group-type [json]",
4162 SHOW_STR
4163 IP_STR
4164 PIM_STR
4165 "PIM group type\n"
4166 "JavaScript Object Notation\n")
4167 {
4168 u_char uj = use_json(argc, argv);
4169 ip_pim_ssm_show_group_range(vty, uj);
4170
4171 return CMD_SUCCESS;
4172 }
4173
4174 static void
4175 ip_pim_ssm_show_group_type(struct vty *vty, u_char uj, const char *group)
4176 {
4177 struct in_addr group_addr;
4178 const char *type_str;
4179 int result;
4180
4181 result = inet_pton(AF_INET, group, &group_addr);
4182 if (result <= 0)
4183 type_str = "invalid";
4184 else
4185 {
4186 if (pim_is_group_224_4 (group_addr))
4187 type_str = pim_is_grp_ssm (group_addr)?"SSM":"ASM";
4188 else
4189 type_str = "not-multicast";
4190 }
4191
4192 if (uj)
4193 {
4194 json_object *json;
4195 json = json_object_new_object();
4196 json_object_string_add(json, "groupType", type_str);
4197 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
4198 json_object_free(json);
4199 }
4200 else
4201 vty_out(vty, "Group type : %s%s", type_str, VTY_NEWLINE);
4202 }
4203
4204 DEFUN (show_ip_pim_group_type,
4205 show_ip_pim_group_type_cmd,
4206 "show ip pim group-type A.B.C.D [json]",
4207 SHOW_STR
4208 IP_STR
4209 PIM_STR
4210 "multicast group type\n"
4211 "group address\n"
4212 "JavaScript Object Notation\n")
4213 {
4214 u_char uj = use_json(argc, argv);
4215 ip_pim_ssm_show_group_type(vty, uj, argv[0]->arg);
4216
4217 return CMD_SUCCESS;
4218 }
4219
4220 DEFUN_HIDDEN (ip_multicast_routing,
4221 ip_multicast_routing_cmd,
4222 "ip multicast-routing",
4223 IP_STR
4224 "Enable IP multicast forwarding\n")
4225 {
4226 return CMD_SUCCESS;
4227 }
4228
4229 DEFUN_HIDDEN (no_ip_multicast_routing,
4230 no_ip_multicast_routing_cmd,
4231 "no ip multicast-routing",
4232 NO_STR
4233 IP_STR
4234 "Global IP configuration subcommands\n"
4235 "Enable IP multicast forwarding\n")
4236 {
4237 vty_out (vty, "Command is Disabled and will be removed in a future version%s", VTY_NEWLINE);
4238 return CMD_SUCCESS;
4239 }
4240
4241 DEFUN (ip_ssmpingd,
4242 ip_ssmpingd_cmd,
4243 "ip ssmpingd [A.B.C.D]",
4244 IP_STR
4245 CONF_SSMPINGD_STR
4246 "Source address\n")
4247 {
4248 int idx_ipv4 = 2;
4249 int result;
4250 struct in_addr source_addr;
4251 const char *source_str = (argc == 3) ? argv[idx_ipv4]->arg : "0.0.0.0";
4252
4253 result = inet_pton(AF_INET, source_str, &source_addr);
4254 if (result <= 0) {
4255 vty_out(vty, "%% Bad source address %s: errno=%d: %s%s",
4256 source_str, errno, safe_strerror(errno), VTY_NEWLINE);
4257 return CMD_WARNING;
4258 }
4259
4260 result = pim_ssmpingd_start(source_addr);
4261 if (result) {
4262 vty_out(vty, "%% Failure starting ssmpingd for source %s: %d%s",
4263 source_str, result, VTY_NEWLINE);
4264 return CMD_WARNING;
4265 }
4266
4267 return CMD_SUCCESS;
4268 }
4269
4270 DEFUN (no_ip_ssmpingd,
4271 no_ip_ssmpingd_cmd,
4272 "no ip ssmpingd [A.B.C.D]",
4273 NO_STR
4274 IP_STR
4275 CONF_SSMPINGD_STR
4276 "Source address\n")
4277 {
4278 int idx_ipv4 = 3;
4279 int result;
4280 struct in_addr source_addr;
4281 const char *source_str = (argc == 4) ? argv[idx_ipv4]->arg : "0.0.0.0";
4282
4283 result = inet_pton(AF_INET, source_str, &source_addr);
4284 if (result <= 0) {
4285 vty_out(vty, "%% Bad source address %s: errno=%d: %s%s",
4286 source_str, errno, safe_strerror(errno), VTY_NEWLINE);
4287 return CMD_WARNING;
4288 }
4289
4290 result = pim_ssmpingd_stop(source_addr);
4291 if (result) {
4292 vty_out(vty, "%% Failure stopping ssmpingd for source %s: %d%s",
4293 source_str, result, VTY_NEWLINE);
4294 return CMD_WARNING;
4295 }
4296
4297 return CMD_SUCCESS;
4298 }
4299
4300 DEFUN (ip_pim_ecmp,
4301 ip_pim_ecmp_cmd,
4302 "ip pim ecmp",
4303 IP_STR
4304 "pim multicast routing\n"
4305 "Enable PIM ECMP \n")
4306 {
4307 qpim_ecmp_enable = 1;
4308
4309 return CMD_SUCCESS;
4310 }
4311
4312 DEFUN (no_ip_pim_ecmp,
4313 no_ip_pim_ecmp_cmd,
4314 "no ip pim ecmp",
4315 NO_STR
4316 IP_STR
4317 "pim multicast routing\n"
4318 "Disable PIM ECMP \n")
4319 {
4320 qpim_ecmp_enable = 0;
4321
4322 return CMD_SUCCESS;
4323 }
4324
4325 DEFUN (ip_pim_ecmp_rebalance,
4326 ip_pim_ecmp_rebalance_cmd,
4327 "ip pim ecmp rebalance",
4328 IP_STR
4329 "pim multicast routing\n"
4330 "Enable PIM ECMP \n"
4331 "Enable PIM ECMP Rebalance\n")
4332 {
4333 qpim_ecmp_enable = 1;
4334 qpim_ecmp_rebalance_enable = 1;
4335
4336 return CMD_SUCCESS;
4337 }
4338
4339 DEFUN (no_ip_pim_ecmp_rebalance,
4340 no_ip_pim_ecmp_rebalance_cmd,
4341 "no ip pim ecmp rebalance",
4342 NO_STR
4343 IP_STR
4344 "pim multicast routing\n"
4345 "Disable PIM ECMP \n"
4346 "Disable PIM ECMP Rebalance\n")
4347 {
4348 qpim_ecmp_rebalance_enable = 0;
4349
4350 return CMD_SUCCESS;
4351 }
4352
4353 static int
4354 pim_cmd_igmp_start (struct vty *vty, struct interface *ifp)
4355 {
4356 struct pim_interface *pim_ifp;
4357 uint8_t need_startup = 0;
4358
4359 pim_ifp = ifp->info;
4360
4361 if (!pim_ifp)
4362 {
4363 pim_ifp = pim_if_new(ifp, 1 /* igmp=true */, 0 /* pim=false */);
4364 if (!pim_ifp)
4365 {
4366 vty_out(vty, "Could not enable IGMP on interface %s%s",
4367 ifp->name, VTY_NEWLINE);
4368 return CMD_WARNING;
4369 }
4370 need_startup = 1;
4371 }
4372 else
4373 {
4374 if (!PIM_IF_TEST_IGMP(pim_ifp->options))
4375 {
4376 PIM_IF_DO_IGMP(pim_ifp->options);
4377 need_startup = 1;
4378 }
4379 }
4380
4381 /* 'ip igmp' executed multiple times, with need_startup
4382 avoid multiple if add all and membership refresh */
4383 if (need_startup)
4384 {
4385 pim_if_addr_add_all(ifp);
4386 pim_if_membership_refresh(ifp);
4387 }
4388
4389 return CMD_SUCCESS;
4390 }
4391
4392 DEFUN (interface_ip_igmp,
4393 interface_ip_igmp_cmd,
4394 "ip igmp",
4395 IP_STR
4396 IFACE_IGMP_STR)
4397 {
4398 VTY_DECLVAR_CONTEXT(interface, ifp);
4399
4400 return pim_cmd_igmp_start(vty, ifp);
4401 }
4402
4403 DEFUN (interface_no_ip_igmp,
4404 interface_no_ip_igmp_cmd,
4405 "no ip igmp",
4406 NO_STR
4407 IP_STR
4408 IFACE_IGMP_STR)
4409 {
4410 VTY_DECLVAR_CONTEXT(interface, ifp);
4411 struct pim_interface *pim_ifp;
4412
4413 pim_ifp = ifp->info;
4414 if (!pim_ifp)
4415 return CMD_SUCCESS;
4416
4417 PIM_IF_DONT_IGMP(pim_ifp->options);
4418
4419 pim_if_membership_clear(ifp);
4420
4421 pim_if_addr_del_all_igmp(ifp);
4422
4423 if (!PIM_IF_TEST_PIM(pim_ifp->options)) {
4424 pim_if_delete(ifp);
4425 }
4426
4427 return CMD_SUCCESS;
4428 }
4429
4430 DEFUN (interface_ip_igmp_join,
4431 interface_ip_igmp_join_cmd,
4432 "ip igmp join A.B.C.D A.B.C.D",
4433 IP_STR
4434 IFACE_IGMP_STR
4435 "IGMP join multicast group\n"
4436 "Multicast group address\n"
4437 "Source address\n")
4438 {
4439 VTY_DECLVAR_CONTEXT(interface, ifp);
4440 int idx_ipv4 = 3;
4441 int idx_ipv4_2 = 4;
4442 const char *group_str;
4443 const char *source_str;
4444 struct in_addr group_addr;
4445 struct in_addr source_addr;
4446 int result;
4447
4448 /* Group address */
4449 group_str = argv[idx_ipv4]->arg;
4450 result = inet_pton(AF_INET, group_str, &group_addr);
4451 if (result <= 0) {
4452 vty_out(vty, "Bad group address %s: errno=%d: %s%s",
4453 group_str, errno, safe_strerror(errno), VTY_NEWLINE);
4454 return CMD_WARNING;
4455 }
4456
4457 /* Source address */
4458 source_str = argv[idx_ipv4_2]->arg;
4459 result = inet_pton(AF_INET, source_str, &source_addr);
4460 if (result <= 0) {
4461 vty_out(vty, "Bad source address %s: errno=%d: %s%s",
4462 source_str, errno, safe_strerror(errno), VTY_NEWLINE);
4463 return CMD_WARNING;
4464 }
4465
4466 result = pim_if_igmp_join_add(ifp, group_addr, source_addr);
4467 if (result) {
4468 vty_out(vty, "%% Failure joining IGMP group %s source %s on interface %s: %d%s",
4469 group_str, source_str, ifp->name, result, VTY_NEWLINE);
4470 return CMD_WARNING;
4471 }
4472
4473 return CMD_SUCCESS;
4474 }
4475
4476 DEFUN (interface_no_ip_igmp_join,
4477 interface_no_ip_igmp_join_cmd,
4478 "no ip igmp join A.B.C.D A.B.C.D",
4479 NO_STR
4480 IP_STR
4481 IFACE_IGMP_STR
4482 "IGMP join multicast group\n"
4483 "Multicast group address\n"
4484 "Source address\n")
4485 {
4486 VTY_DECLVAR_CONTEXT(interface, ifp);
4487 int idx_ipv4 = 4;
4488 int idx_ipv4_2 = 5;
4489 const char *group_str;
4490 const char *source_str;
4491 struct in_addr group_addr;
4492 struct in_addr source_addr;
4493 int result;
4494
4495 /* Group address */
4496 group_str = argv[idx_ipv4]->arg;
4497 result = inet_pton(AF_INET, group_str, &group_addr);
4498 if (result <= 0) {
4499 vty_out(vty, "Bad group address %s: errno=%d: %s%s",
4500 group_str, errno, safe_strerror(errno), VTY_NEWLINE);
4501 return CMD_WARNING;
4502 }
4503
4504 /* Source address */
4505 source_str = argv[idx_ipv4_2]->arg;
4506 result = inet_pton(AF_INET, source_str, &source_addr);
4507 if (result <= 0) {
4508 vty_out(vty, "Bad source address %s: errno=%d: %s%s",
4509 source_str, errno, safe_strerror(errno), VTY_NEWLINE);
4510 return CMD_WARNING;
4511 }
4512
4513 result = pim_if_igmp_join_del(ifp, group_addr, source_addr);
4514 if (result) {
4515 vty_out(vty, "%% Failure leaving IGMP group %s source %s on interface %s: %d%s",
4516 group_str, source_str, ifp->name, result, VTY_NEWLINE);
4517 return CMD_WARNING;
4518 }
4519
4520 return CMD_SUCCESS;
4521 }
4522
4523 /*
4524 CLI reconfiguration affects the interface level (struct pim_interface).
4525 This function propagates the reconfiguration to every active socket
4526 for that interface.
4527 */
4528 static void igmp_sock_query_interval_reconfig(struct igmp_sock *igmp)
4529 {
4530 struct interface *ifp;
4531 struct pim_interface *pim_ifp;
4532
4533 zassert(igmp);
4534
4535 /* other querier present? */
4536
4537 if (igmp->t_other_querier_timer)
4538 return;
4539
4540 /* this is the querier */
4541
4542 zassert(igmp->interface);
4543 zassert(igmp->interface->info);
4544
4545 ifp = igmp->interface;
4546 pim_ifp = ifp->info;
4547
4548 if (PIM_DEBUG_IGMP_TRACE) {
4549 char ifaddr_str[INET_ADDRSTRLEN];
4550 pim_inet4_dump("<ifaddr?>", igmp->ifaddr, ifaddr_str, sizeof(ifaddr_str));
4551 zlog_debug("%s: Querier %s on %s reconfig query_interval=%d",
4552 __PRETTY_FUNCTION__,
4553 ifaddr_str,
4554 ifp->name,
4555 pim_ifp->igmp_default_query_interval);
4556 }
4557
4558 /*
4559 igmp_startup_mode_on() will reset QQI:
4560
4561 igmp->querier_query_interval = pim_ifp->igmp_default_query_interval;
4562 */
4563 igmp_startup_mode_on(igmp);
4564 }
4565
4566 static void igmp_sock_query_reschedule(struct igmp_sock *igmp)
4567 {
4568 if (igmp->t_igmp_query_timer) {
4569 /* other querier present */
4570 zassert(igmp->t_igmp_query_timer);
4571 zassert(!igmp->t_other_querier_timer);
4572
4573 pim_igmp_general_query_off(igmp);
4574 pim_igmp_general_query_on(igmp);
4575
4576 zassert(igmp->t_igmp_query_timer);
4577 zassert(!igmp->t_other_querier_timer);
4578 }
4579 else {
4580 /* this is the querier */
4581
4582 zassert(!igmp->t_igmp_query_timer);
4583 zassert(igmp->t_other_querier_timer);
4584
4585 pim_igmp_other_querier_timer_off(igmp);
4586 pim_igmp_other_querier_timer_on(igmp);
4587
4588 zassert(!igmp->t_igmp_query_timer);
4589 zassert(igmp->t_other_querier_timer);
4590 }
4591 }
4592
4593 static void change_query_interval(struct pim_interface *pim_ifp,
4594 int query_interval)
4595 {
4596 struct listnode *sock_node;
4597 struct igmp_sock *igmp;
4598
4599 pim_ifp->igmp_default_query_interval = query_interval;
4600
4601 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
4602 igmp_sock_query_interval_reconfig(igmp);
4603 igmp_sock_query_reschedule(igmp);
4604 }
4605 }
4606
4607 static void change_query_max_response_time(struct pim_interface *pim_ifp,
4608 int query_max_response_time_dsec)
4609 {
4610 struct listnode *sock_node;
4611 struct igmp_sock *igmp;
4612
4613 pim_ifp->igmp_query_max_response_time_dsec = query_max_response_time_dsec;
4614
4615 /*
4616 Below we modify socket/group/source timers in order to quickly
4617 reflect the change. Otherwise, those timers would eventually catch
4618 up.
4619 */
4620
4621 /* scan all sockets */
4622 for (ALL_LIST_ELEMENTS_RO(pim_ifp->igmp_socket_list, sock_node, igmp)) {
4623 struct listnode *grp_node;
4624 struct igmp_group *grp;
4625
4626 /* reschedule socket general query */
4627 igmp_sock_query_reschedule(igmp);
4628
4629 /* scan socket groups */
4630 for (ALL_LIST_ELEMENTS_RO(igmp->igmp_group_list, grp_node, grp)) {
4631 struct listnode *src_node;
4632 struct igmp_source *src;
4633
4634 /* reset group timers for groups in EXCLUDE mode */
4635 if (grp->group_filtermode_isexcl) {
4636 igmp_group_reset_gmi(grp);
4637 }
4638
4639 /* scan group sources */
4640 for (ALL_LIST_ELEMENTS_RO(grp->group_source_list, src_node, src)) {
4641
4642 /* reset source timers for sources with running timers */
4643 if (src->t_source_timer) {
4644 igmp_source_reset_gmi(igmp, grp, src);
4645 }
4646 }
4647 }
4648 }
4649 }
4650
4651 #define IGMP_QUERY_INTERVAL_MIN (1)
4652 #define IGMP_QUERY_INTERVAL_MAX (1800)
4653
4654 DEFUN (interface_ip_igmp_query_interval,
4655 interface_ip_igmp_query_interval_cmd,
4656 "ip igmp query-interval (1-1800)",
4657 IP_STR
4658 IFACE_IGMP_STR
4659 IFACE_IGMP_QUERY_INTERVAL_STR
4660 "Query interval in seconds\n")
4661 {
4662 VTY_DECLVAR_CONTEXT(interface, ifp);
4663 struct pim_interface *pim_ifp;
4664 int query_interval;
4665 int query_interval_dsec;
4666 int ret;
4667
4668 pim_ifp = ifp->info;
4669
4670 if (!pim_ifp) {
4671 ret = pim_cmd_igmp_start(vty, ifp);
4672 if (ret != CMD_SUCCESS)
4673 return ret;
4674 pim_ifp = ifp->info;
4675 }
4676
4677 query_interval = atoi(argv[3]->arg);
4678 query_interval_dsec = 10 * query_interval;
4679
4680 /*
4681 It seems we don't need to check bounds since command.c does it
4682 already, but we verify them anyway for extra safety.
4683 */
4684 if (query_interval < IGMP_QUERY_INTERVAL_MIN) {
4685 vty_out(vty, "General query interval %d lower than minimum %d%s",
4686 query_interval,
4687 IGMP_QUERY_INTERVAL_MIN,
4688 VTY_NEWLINE);
4689 return CMD_WARNING;
4690 }
4691 if (query_interval > IGMP_QUERY_INTERVAL_MAX) {
4692 vty_out(vty, "General query interval %d higher than maximum %d%s",
4693 query_interval,
4694 IGMP_QUERY_INTERVAL_MAX,
4695 VTY_NEWLINE);
4696 return CMD_WARNING;
4697 }
4698
4699 if (query_interval_dsec <= pim_ifp->igmp_query_max_response_time_dsec) {
4700 vty_out(vty,
4701 "Can't set general query interval %d dsec <= query max response time %d dsec.%s",
4702 query_interval_dsec, pim_ifp->igmp_query_max_response_time_dsec,
4703 VTY_NEWLINE);
4704 return CMD_WARNING;
4705 }
4706
4707 change_query_interval(pim_ifp, query_interval);
4708
4709 return CMD_SUCCESS;
4710 }
4711
4712 DEFUN (interface_no_ip_igmp_query_interval,
4713 interface_no_ip_igmp_query_interval_cmd,
4714 "no ip igmp query-interval",
4715 NO_STR
4716 IP_STR
4717 IFACE_IGMP_STR
4718 IFACE_IGMP_QUERY_INTERVAL_STR)
4719 {
4720 VTY_DECLVAR_CONTEXT(interface, ifp);
4721 struct pim_interface *pim_ifp;
4722 int default_query_interval_dsec;
4723
4724 pim_ifp = ifp->info;
4725
4726 if (!pim_ifp)
4727 return CMD_SUCCESS;
4728
4729 default_query_interval_dsec = IGMP_GENERAL_QUERY_INTERVAL * 10;
4730
4731 if (default_query_interval_dsec <= pim_ifp->igmp_query_max_response_time_dsec) {
4732 vty_out(vty,
4733 "Can't set default general query interval %d dsec <= query max response time %d dsec.%s",
4734 default_query_interval_dsec, pim_ifp->igmp_query_max_response_time_dsec,
4735 VTY_NEWLINE);
4736 return CMD_WARNING;
4737 }
4738
4739 change_query_interval(pim_ifp, IGMP_GENERAL_QUERY_INTERVAL);
4740
4741 return CMD_SUCCESS;
4742 }
4743
4744 DEFUN (interface_ip_igmp_version,
4745 interface_ip_igmp_version_cmd,
4746 "ip igmp version (2-3)",
4747 IP_STR
4748 IFACE_IGMP_STR
4749 "IGMP version\n"
4750 "IGMP version number\n")
4751 {
4752 VTY_DECLVAR_CONTEXT(interface,ifp);
4753 struct pim_interface *pim_ifp = NULL;
4754 int igmp_version, old_version = 0;
4755 int ret;
4756
4757 pim_ifp = ifp->info;
4758
4759 if (!pim_ifp)
4760 {
4761 ret = pim_cmd_igmp_start(vty, ifp);
4762 if (ret != CMD_SUCCESS)
4763 return ret;
4764 pim_ifp = ifp->info;
4765 }
4766
4767 igmp_version = atoi(argv[3]->arg);
4768 old_version = pim_ifp->igmp_version;
4769 pim_ifp->igmp_version = igmp_version;
4770
4771 //Check if IGMP is Enabled otherwise, enable on interface
4772 if (!PIM_IF_TEST_IGMP (pim_ifp->options))
4773 {
4774 PIM_IF_DO_IGMP(pim_ifp->options);
4775 pim_if_addr_add_all(ifp);
4776 pim_if_membership_refresh(ifp);
4777 old_version = igmp_version; //avoid refreshing membership again.
4778 }
4779 /* Current and new version is different refresh existing
4780 membership. Going from 3 -> 2 or 2 -> 3. */
4781 if (old_version != igmp_version)
4782 pim_if_membership_refresh(ifp);
4783
4784 return CMD_SUCCESS;
4785 }
4786
4787 DEFUN (interface_no_ip_igmp_version,
4788 interface_no_ip_igmp_version_cmd,
4789 "no ip igmp version (2-3)",
4790 NO_STR
4791 IP_STR
4792 IFACE_IGMP_STR
4793 "IGMP version\n"
4794 "IGMP version number\n")
4795 {
4796 VTY_DECLVAR_CONTEXT(interface, ifp);
4797 struct pim_interface *pim_ifp;
4798
4799 pim_ifp = ifp->info;
4800
4801 if (!pim_ifp)
4802 return CMD_SUCCESS;
4803
4804 pim_ifp->igmp_version = IGMP_DEFAULT_VERSION;
4805
4806 return CMD_SUCCESS;
4807 }
4808
4809 #define IGMP_QUERY_MAX_RESPONSE_TIME_MIN_DSEC (10)
4810 #define IGMP_QUERY_MAX_RESPONSE_TIME_MAX_DSEC (250)
4811
4812 DEFUN (interface_ip_igmp_query_max_response_time,
4813 interface_ip_igmp_query_max_response_time_cmd,
4814 "ip igmp query-max-response-time (10-250)",
4815 IP_STR
4816 IFACE_IGMP_STR
4817 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_STR
4818 "Query response value in deci-seconds\n")
4819 {
4820 VTY_DECLVAR_CONTEXT(interface, ifp);
4821 struct pim_interface *pim_ifp;
4822 int query_max_response_time;
4823 int ret;
4824
4825 pim_ifp = ifp->info;
4826
4827 if (!pim_ifp) {
4828 ret = pim_cmd_igmp_start(vty, ifp);
4829 if (ret != CMD_SUCCESS)
4830 return ret;
4831 pim_ifp = ifp->info;
4832 }
4833
4834 query_max_response_time = atoi(argv[3]->arg);
4835
4836 if (query_max_response_time >= pim_ifp->igmp_default_query_interval * 10) {
4837 vty_out(vty,
4838 "Can't set query max response time %d sec >= general query interval %d sec%s",
4839 query_max_response_time, pim_ifp->igmp_default_query_interval,
4840 VTY_NEWLINE);
4841 return CMD_WARNING;
4842 }
4843
4844 change_query_max_response_time(pim_ifp, query_max_response_time);
4845
4846 return CMD_SUCCESS;
4847 }
4848
4849 DEFUN (interface_no_ip_igmp_query_max_response_time,
4850 interface_no_ip_igmp_query_max_response_time_cmd,
4851 "no ip igmp query-max-response-time (10-250)",
4852 NO_STR
4853 IP_STR
4854 IFACE_IGMP_STR
4855 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_STR
4856 "Time for response in deci-seconds\n")
4857 {
4858 VTY_DECLVAR_CONTEXT(interface, ifp);
4859 struct pim_interface *pim_ifp;
4860
4861 pim_ifp = ifp->info;
4862
4863 if (!pim_ifp)
4864 return CMD_SUCCESS;
4865
4866 change_query_max_response_time(pim_ifp, IGMP_QUERY_MAX_RESPONSE_TIME_DSEC);
4867
4868 return CMD_SUCCESS;
4869 }
4870
4871 #define IGMP_QUERY_MAX_RESPONSE_TIME_MIN_DSEC (10)
4872 #define IGMP_QUERY_MAX_RESPONSE_TIME_MAX_DSEC (250)
4873
4874 DEFUN_HIDDEN (interface_ip_igmp_query_max_response_time_dsec,
4875 interface_ip_igmp_query_max_response_time_dsec_cmd,
4876 "ip igmp query-max-response-time-dsec (10-250)",
4877 IP_STR
4878 IFACE_IGMP_STR
4879 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_DSEC_STR
4880 "Query response value in deciseconds\n")
4881 {
4882 VTY_DECLVAR_CONTEXT(interface, ifp);
4883 struct pim_interface *pim_ifp;
4884 int query_max_response_time_dsec;
4885 int default_query_interval_dsec;
4886 int ret;
4887
4888 pim_ifp = ifp->info;
4889
4890 if (!pim_ifp) {
4891 ret = pim_cmd_igmp_start(vty, ifp);
4892 if (ret != CMD_SUCCESS)
4893 return ret;
4894 pim_ifp = ifp->info;
4895 }
4896
4897 query_max_response_time_dsec = atoi(argv[4]->arg);
4898
4899 default_query_interval_dsec = 10 * pim_ifp->igmp_default_query_interval;
4900
4901 if (query_max_response_time_dsec >= default_query_interval_dsec) {
4902 vty_out(vty,
4903 "Can't set query max response time %d dsec >= general query interval %d dsec%s",
4904 query_max_response_time_dsec, default_query_interval_dsec,
4905 VTY_NEWLINE);
4906 return CMD_WARNING;
4907 }
4908
4909 change_query_max_response_time(pim_ifp, query_max_response_time_dsec);
4910
4911 return CMD_SUCCESS;
4912 }
4913
4914 DEFUN_HIDDEN (interface_no_ip_igmp_query_max_response_time_dsec,
4915 interface_no_ip_igmp_query_max_response_time_dsec_cmd,
4916 "no ip igmp query-max-response-time-dsec",
4917 NO_STR
4918 IP_STR
4919 IFACE_IGMP_STR
4920 IFACE_IGMP_QUERY_MAX_RESPONSE_TIME_DSEC_STR)
4921 {
4922 VTY_DECLVAR_CONTEXT(interface, ifp);
4923 struct pim_interface *pim_ifp;
4924
4925 pim_ifp = ifp->info;
4926
4927 if (!pim_ifp)
4928 return CMD_SUCCESS;
4929
4930 change_query_max_response_time(pim_ifp, IGMP_QUERY_MAX_RESPONSE_TIME_DSEC);
4931
4932 return CMD_SUCCESS;
4933 }
4934
4935 DEFUN (interface_ip_pim_drprio,
4936 interface_ip_pim_drprio_cmd,
4937 "ip pim drpriority (1-4294967295)",
4938 IP_STR
4939 PIM_STR
4940 "Set the Designated Router Election Priority\n"
4941 "Value of the new DR Priority\n")
4942 {
4943 VTY_DECLVAR_CONTEXT(interface, ifp);
4944 int idx_number = 3;
4945 struct pim_interface *pim_ifp;
4946 uint32_t old_dr_prio;
4947
4948 pim_ifp = ifp->info;
4949
4950 if (!pim_ifp) {
4951 vty_out(vty, "Please enable PIM on interface, first%s", VTY_NEWLINE);
4952 return CMD_WARNING;
4953 }
4954
4955 old_dr_prio = pim_ifp->pim_dr_priority;
4956
4957 pim_ifp->pim_dr_priority = strtol(argv[idx_number]->arg, NULL, 10);
4958
4959 if (old_dr_prio != pim_ifp->pim_dr_priority) {
4960 if (pim_if_dr_election(ifp))
4961 pim_hello_restart_now(ifp);
4962 }
4963
4964 return CMD_SUCCESS;
4965 }
4966
4967 DEFUN (interface_no_ip_pim_drprio,
4968 interface_no_ip_pim_drprio_cmd,
4969 "no ip pim drpriority [(1-4294967295)]",
4970 NO_STR
4971 IP_STR
4972 PIM_STR
4973 "Revert the Designated Router Priority to default\n"
4974 "Old Value of the Priority\n")
4975 {
4976 VTY_DECLVAR_CONTEXT(interface, ifp);
4977 struct pim_interface *pim_ifp;
4978
4979 pim_ifp = ifp->info;
4980
4981 if (!pim_ifp) {
4982 vty_out(vty, "Pim not enabled on this interface%s", VTY_NEWLINE);
4983 return CMD_WARNING;
4984 }
4985
4986 if (pim_ifp->pim_dr_priority != PIM_DEFAULT_DR_PRIORITY) {
4987 pim_ifp->pim_dr_priority = PIM_DEFAULT_DR_PRIORITY;
4988 if (pim_if_dr_election(ifp))
4989 pim_hello_restart_now(ifp);
4990 }
4991
4992 return CMD_SUCCESS;
4993 }
4994
4995 static int
4996 pim_cmd_interface_add (struct interface *ifp)
4997 {
4998 struct pim_interface *pim_ifp = ifp->info;
4999
5000 if (!pim_ifp) {
5001 pim_ifp = pim_if_new(ifp, 0 /* igmp=false */, 1 /* pim=true */);
5002 if (!pim_ifp) {
5003 return 0;
5004 }
5005 }
5006 else {
5007 PIM_IF_DO_PIM(pim_ifp->options);
5008 }
5009
5010 pim_if_addr_add_all(ifp);
5011 pim_if_membership_refresh(ifp);
5012 return 1;
5013 }
5014
5015 DEFUN_HIDDEN (interface_ip_pim_ssm,
5016 interface_ip_pim_ssm_cmd,
5017 "ip pim ssm",
5018 IP_STR
5019 PIM_STR
5020 IFACE_PIM_STR)
5021 {
5022 VTY_DECLVAR_CONTEXT(interface, ifp);
5023
5024 if (!pim_cmd_interface_add(ifp)) {
5025 vty_out(vty, "Could not enable PIM SM on interface%s", VTY_NEWLINE);
5026 return CMD_WARNING;
5027 }
5028
5029 vty_out(vty, "WARN: Enabled PIM SM on interface; configure PIM SSM range if needed%s", VTY_NEWLINE);
5030 return CMD_SUCCESS;
5031 }
5032
5033 DEFUN (interface_ip_pim_sm,
5034 interface_ip_pim_sm_cmd,
5035 "ip pim sm",
5036 IP_STR
5037 PIM_STR
5038 IFACE_PIM_SM_STR)
5039 {
5040 VTY_DECLVAR_CONTEXT(interface, ifp);
5041 if (!pim_cmd_interface_add(ifp)) {
5042 vty_out(vty, "Could not enable PIM SM on interface%s", VTY_NEWLINE);
5043 return CMD_WARNING;
5044 }
5045
5046 pim_if_create_pimreg();
5047
5048 return CMD_SUCCESS;
5049 }
5050
5051 static int
5052 pim_cmd_interface_delete (struct interface *ifp)
5053 {
5054 struct pim_interface *pim_ifp = ifp->info;
5055
5056 if (!pim_ifp)
5057 return 1;
5058
5059 PIM_IF_DONT_PIM(pim_ifp->options);
5060
5061 pim_if_membership_clear(ifp);
5062
5063 /*
5064 pim_sock_delete() removes all neighbors from
5065 pim_ifp->pim_neighbor_list.
5066 */
5067 pim_sock_delete(ifp, "pim unconfigured on interface");
5068
5069 if (!PIM_IF_TEST_IGMP(pim_ifp->options)) {
5070 pim_if_addr_del_all(ifp);
5071 pim_if_delete(ifp);
5072 }
5073
5074 return 1;
5075 }
5076
5077 DEFUN_HIDDEN (interface_no_ip_pim_ssm,
5078 interface_no_ip_pim_ssm_cmd,
5079 "no ip pim ssm",
5080 NO_STR
5081 IP_STR
5082 PIM_STR
5083 IFACE_PIM_STR)
5084 {
5085 VTY_DECLVAR_CONTEXT(interface, ifp);
5086 if (!pim_cmd_interface_delete(ifp)) {
5087 vty_out(vty, "Unable to delete interface information%s", VTY_NEWLINE);
5088 return CMD_WARNING;
5089 }
5090
5091 return CMD_SUCCESS;
5092 }
5093
5094 DEFUN (interface_no_ip_pim_sm,
5095 interface_no_ip_pim_sm_cmd,
5096 "no ip pim sm",
5097 NO_STR
5098 IP_STR
5099 PIM_STR
5100 IFACE_PIM_SM_STR)
5101 {
5102 VTY_DECLVAR_CONTEXT(interface, ifp);
5103 if (!pim_cmd_interface_delete(ifp)) {
5104 vty_out(vty, "Unable to delete interface information%s", VTY_NEWLINE);
5105 return CMD_WARNING;
5106 }
5107
5108 return CMD_SUCCESS;
5109 }
5110
5111 DEFUN (interface_ip_mroute,
5112 interface_ip_mroute_cmd,
5113 "ip mroute INTERFACE A.B.C.D",
5114 IP_STR
5115 "Add multicast route\n"
5116 "Outgoing interface name\n"
5117 "Group address\n")
5118 {
5119 VTY_DECLVAR_CONTEXT(interface, iif);
5120 int idx_interface = 2;
5121 int idx_ipv4 = 3;
5122 struct interface *oif;
5123 const char *oifname;
5124 const char *grp_str;
5125 struct in_addr grp_addr;
5126 struct in_addr src_addr;
5127 int result;
5128
5129 oifname = argv[idx_interface]->arg;
5130 oif = if_lookup_by_name(oifname, VRF_DEFAULT);
5131 if (!oif) {
5132 vty_out(vty, "No such interface name %s%s",
5133 oifname, VTY_NEWLINE);
5134 return CMD_WARNING;
5135 }
5136
5137 grp_str = argv[idx_ipv4]->arg;
5138 result = inet_pton(AF_INET, grp_str, &grp_addr);
5139 if (result <= 0) {
5140 vty_out(vty, "Bad group address %s: errno=%d: %s%s",
5141 grp_str, errno, safe_strerror(errno), VTY_NEWLINE);
5142 return CMD_WARNING;
5143 }
5144
5145 src_addr.s_addr = INADDR_ANY;
5146
5147 if (pim_static_add(iif, oif, grp_addr, src_addr)) {
5148 vty_out(vty, "Failed to add route%s", VTY_NEWLINE);
5149 return CMD_WARNING;
5150 }
5151
5152 return CMD_SUCCESS;
5153 }
5154
5155 DEFUN (interface_ip_mroute_source,
5156 interface_ip_mroute_source_cmd,
5157 "ip mroute INTERFACE A.B.C.D A.B.C.D",
5158 IP_STR
5159 "Add multicast route\n"
5160 "Outgoing interface name\n"
5161 "Group address\n"
5162 "Source address\n")
5163 {
5164 VTY_DECLVAR_CONTEXT(interface, iif);
5165 int idx_interface = 2;
5166 int idx_ipv4 = 3;
5167 int idx_ipv4_2 = 4;
5168 struct interface *oif;
5169 const char *oifname;
5170 const char *grp_str;
5171 struct in_addr grp_addr;
5172 const char *src_str;
5173 struct in_addr src_addr;
5174 int result;
5175
5176 oifname = argv[idx_interface]->arg;
5177 oif = if_lookup_by_name(oifname, VRF_DEFAULT);
5178 if (!oif) {
5179 vty_out(vty, "No such interface name %s%s",
5180 oifname, VTY_NEWLINE);
5181 return CMD_WARNING;
5182 }
5183
5184 grp_str = argv[idx_ipv4]->arg;
5185 result = inet_pton(AF_INET, grp_str, &grp_addr);
5186 if (result <= 0) {
5187 vty_out(vty, "Bad group address %s: errno=%d: %s%s",
5188 grp_str, errno, safe_strerror(errno), VTY_NEWLINE);
5189 return CMD_WARNING;
5190 }
5191
5192 src_str = argv[idx_ipv4_2]->arg;
5193 result = inet_pton(AF_INET, src_str, &src_addr);
5194 if (result <= 0) {
5195 vty_out(vty, "Bad source address %s: errno=%d: %s%s",
5196 src_str, errno, safe_strerror(errno), VTY_NEWLINE);
5197 return CMD_WARNING;
5198 }
5199
5200 if (pim_static_add(iif, oif, grp_addr, src_addr)) {
5201 vty_out(vty, "Failed to add route%s", VTY_NEWLINE);
5202 return CMD_WARNING;
5203 }
5204
5205 return CMD_SUCCESS;
5206 }
5207
5208 DEFUN (interface_no_ip_mroute,
5209 interface_no_ip_mroute_cmd,
5210 "no ip mroute INTERFACE A.B.C.D",
5211 NO_STR
5212 IP_STR
5213 "Add multicast route\n"
5214 "Outgoing interface name\n"
5215 "Group Address\n")
5216 {
5217 VTY_DECLVAR_CONTEXT(interface, iif);
5218 int idx_interface = 3;
5219 int idx_ipv4 = 4;
5220 struct interface *oif;
5221 const char *oifname;
5222 const char *grp_str;
5223 struct in_addr grp_addr;
5224 struct in_addr src_addr;
5225 int result;
5226
5227 oifname = argv[idx_interface]->arg;
5228 oif = if_lookup_by_name(oifname, VRF_DEFAULT);
5229 if (!oif) {
5230 vty_out(vty, "No such interface name %s%s",
5231 oifname, VTY_NEWLINE);
5232 return CMD_WARNING;
5233 }
5234
5235 grp_str = argv[idx_ipv4]->arg;
5236 result = inet_pton(AF_INET, grp_str, &grp_addr);
5237 if (result <= 0) {
5238 vty_out(vty, "Bad group address %s: errno=%d: %s%s",
5239 grp_str, errno, safe_strerror(errno), VTY_NEWLINE);
5240 return CMD_WARNING;
5241 }
5242
5243 src_addr.s_addr = INADDR_ANY;
5244
5245 if (pim_static_del(iif, oif, grp_addr, src_addr)) {
5246 vty_out(vty, "Failed to remove route%s", VTY_NEWLINE);
5247 return CMD_WARNING;
5248 }
5249
5250 return CMD_SUCCESS;
5251 }
5252
5253 DEFUN (interface_no_ip_mroute_source,
5254 interface_no_ip_mroute_source_cmd,
5255 "no ip mroute INTERFACE A.B.C.D A.B.C.D",
5256 NO_STR
5257 IP_STR
5258 "Add multicast route\n"
5259 "Outgoing interface name\n"
5260 "Group Address\n"
5261 "Source Address\n")
5262 {
5263 VTY_DECLVAR_CONTEXT(interface, iif);
5264 int idx_interface = 3;
5265 int idx_ipv4 = 4;
5266 int idx_ipv4_2 = 5;
5267 struct interface *oif;
5268 const char *oifname;
5269 const char *grp_str;
5270 struct in_addr grp_addr;
5271 const char *src_str;
5272 struct in_addr src_addr;
5273 int result;
5274
5275 oifname = argv[idx_interface]->arg;
5276 oif = if_lookup_by_name(oifname, VRF_DEFAULT);
5277 if (!oif) {
5278 vty_out(vty, "No such interface name %s%s",
5279 oifname, VTY_NEWLINE);
5280 return CMD_WARNING;
5281 }
5282
5283 grp_str = argv[idx_ipv4]->arg;
5284 result = inet_pton(AF_INET, grp_str, &grp_addr);
5285 if (result <= 0) {
5286 vty_out(vty, "Bad group address %s: errno=%d: %s%s",
5287 grp_str, errno, safe_strerror(errno), VTY_NEWLINE);
5288 return CMD_WARNING;
5289 }
5290
5291 src_str = argv[idx_ipv4_2]->arg;
5292 result = inet_pton(AF_INET, src_str, &src_addr);
5293 if (result <= 0) {
5294 vty_out(vty, "Bad source address %s: errno=%d: %s%s",
5295 src_str, errno, safe_strerror(errno), VTY_NEWLINE);
5296 return CMD_WARNING;
5297 }
5298
5299 if (pim_static_del(iif, oif, grp_addr, src_addr)) {
5300 vty_out(vty, "Failed to remove route%s", VTY_NEWLINE);
5301 return CMD_WARNING;
5302 }
5303
5304 return CMD_SUCCESS;
5305 }
5306
5307 DEFUN (interface_ip_pim_hello,
5308 interface_ip_pim_hello_cmd,
5309 "ip pim hello (1-180) [(1-180)]",
5310 IP_STR
5311 PIM_STR
5312 IFACE_PIM_HELLO_STR
5313 IFACE_PIM_HELLO_TIME_STR
5314 IFACE_PIM_HELLO_HOLD_STR)
5315 {
5316 VTY_DECLVAR_CONTEXT(interface, ifp);
5317 int idx_time = 3;
5318 int idx_hold = 4;
5319 struct pim_interface *pim_ifp;
5320
5321 pim_ifp = ifp->info;
5322
5323 if (!pim_ifp) {
5324 vty_out(vty, "Pim not enabled on this interface%s", VTY_NEWLINE);
5325 return CMD_WARNING;
5326 }
5327
5328 pim_ifp->pim_hello_period = strtol(argv[idx_time]->arg, NULL, 10);
5329
5330 if (argc == idx_hold + 1)
5331 pim_ifp->pim_default_holdtime = strtol(argv[idx_hold]->arg, NULL, 10);
5332
5333 return CMD_SUCCESS;
5334 }
5335
5336
5337
5338 DEFUN (interface_no_ip_pim_hello,
5339 interface_no_ip_pim_hello_cmd,
5340 "no ip pim hello [(1-180) (1-180)]",
5341 NO_STR
5342 IP_STR
5343 PIM_STR
5344 IFACE_PIM_HELLO_STR
5345 IFACE_PIM_HELLO_TIME_STR
5346 IFACE_PIM_HELLO_HOLD_STR)
5347 {
5348 VTY_DECLVAR_CONTEXT(interface, ifp);
5349 struct pim_interface *pim_ifp;
5350
5351 pim_ifp = ifp->info;
5352
5353 if (!pim_ifp) {
5354 vty_out(vty, "Pim not enabled on this interface%s", VTY_NEWLINE);
5355 return CMD_WARNING;
5356 }
5357
5358 pim_ifp->pim_hello_period = PIM_DEFAULT_HELLO_PERIOD;
5359 pim_ifp->pim_default_holdtime = -1;
5360
5361 return CMD_SUCCESS;
5362 }
5363
5364 DEFUN (debug_igmp,
5365 debug_igmp_cmd,
5366 "debug igmp",
5367 DEBUG_STR
5368 DEBUG_IGMP_STR)
5369 {
5370 PIM_DO_DEBUG_IGMP_EVENTS;
5371 PIM_DO_DEBUG_IGMP_PACKETS;
5372 PIM_DO_DEBUG_IGMP_TRACE;
5373 return CMD_SUCCESS;
5374 }
5375
5376 DEFUN (no_debug_igmp,
5377 no_debug_igmp_cmd,
5378 "no debug igmp",
5379 NO_STR
5380 DEBUG_STR
5381 DEBUG_IGMP_STR)
5382 {
5383 PIM_DONT_DEBUG_IGMP_EVENTS;
5384 PIM_DONT_DEBUG_IGMP_PACKETS;
5385 PIM_DONT_DEBUG_IGMP_TRACE;
5386 return CMD_SUCCESS;
5387 }
5388
5389
5390 DEFUN (debug_igmp_events,
5391 debug_igmp_events_cmd,
5392 "debug igmp events",
5393 DEBUG_STR
5394 DEBUG_IGMP_STR
5395 DEBUG_IGMP_EVENTS_STR)
5396 {
5397 PIM_DO_DEBUG_IGMP_EVENTS;
5398 return CMD_SUCCESS;
5399 }
5400
5401 DEFUN (no_debug_igmp_events,
5402 no_debug_igmp_events_cmd,
5403 "no debug igmp events",
5404 NO_STR
5405 DEBUG_STR
5406 DEBUG_IGMP_STR
5407 DEBUG_IGMP_EVENTS_STR)
5408 {
5409 PIM_DONT_DEBUG_IGMP_EVENTS;
5410 return CMD_SUCCESS;
5411 }
5412
5413
5414 DEFUN (debug_igmp_packets,
5415 debug_igmp_packets_cmd,
5416 "debug igmp packets",
5417 DEBUG_STR
5418 DEBUG_IGMP_STR
5419 DEBUG_IGMP_PACKETS_STR)
5420 {
5421 PIM_DO_DEBUG_IGMP_PACKETS;
5422 return CMD_SUCCESS;
5423 }
5424
5425 DEFUN (no_debug_igmp_packets,
5426 no_debug_igmp_packets_cmd,
5427 "no debug igmp packets",
5428 NO_STR
5429 DEBUG_STR
5430 DEBUG_IGMP_STR
5431 DEBUG_IGMP_PACKETS_STR)
5432 {
5433 PIM_DONT_DEBUG_IGMP_PACKETS;
5434 return CMD_SUCCESS;
5435 }
5436
5437
5438 DEFUN (debug_igmp_trace,
5439 debug_igmp_trace_cmd,
5440 "debug igmp trace",
5441 DEBUG_STR
5442 DEBUG_IGMP_STR
5443 DEBUG_IGMP_TRACE_STR)
5444 {
5445 PIM_DO_DEBUG_IGMP_TRACE;
5446 return CMD_SUCCESS;
5447 }
5448
5449 DEFUN (no_debug_igmp_trace,
5450 no_debug_igmp_trace_cmd,
5451 "no debug igmp trace",
5452 NO_STR
5453 DEBUG_STR
5454 DEBUG_IGMP_STR
5455 DEBUG_IGMP_TRACE_STR)
5456 {
5457 PIM_DONT_DEBUG_IGMP_TRACE;
5458 return CMD_SUCCESS;
5459 }
5460
5461
5462 DEFUN (debug_mroute,
5463 debug_mroute_cmd,
5464 "debug mroute",
5465 DEBUG_STR
5466 DEBUG_MROUTE_STR)
5467 {
5468 PIM_DO_DEBUG_MROUTE;
5469 return CMD_SUCCESS;
5470 }
5471
5472 DEFUN (debug_mroute_detail,
5473 debug_mroute_detail_cmd,
5474 "debug mroute detail",
5475 DEBUG_STR
5476 DEBUG_MROUTE_STR
5477 "detailed\n")
5478 {
5479 PIM_DO_DEBUG_MROUTE_DETAIL;
5480 return CMD_SUCCESS;
5481 }
5482
5483 DEFUN (no_debug_mroute,
5484 no_debug_mroute_cmd,
5485 "no debug mroute",
5486 NO_STR
5487 DEBUG_STR
5488 DEBUG_MROUTE_STR)
5489 {
5490 PIM_DONT_DEBUG_MROUTE;
5491 return CMD_SUCCESS;
5492 }
5493
5494 DEFUN (no_debug_mroute_detail,
5495 no_debug_mroute_detail_cmd,
5496 "no debug mroute detail",
5497 NO_STR
5498 DEBUG_STR
5499 DEBUG_MROUTE_STR
5500 "detailed\n")
5501 {
5502 PIM_DONT_DEBUG_MROUTE_DETAIL;
5503 return CMD_SUCCESS;
5504 }
5505
5506 DEFUN (debug_static,
5507 debug_static_cmd,
5508 "debug static",
5509 DEBUG_STR
5510 DEBUG_STATIC_STR)
5511 {
5512 PIM_DO_DEBUG_STATIC;
5513 return CMD_SUCCESS;
5514 }
5515
5516 DEFUN (no_debug_static,
5517 no_debug_static_cmd,
5518 "no debug static",
5519 NO_STR
5520 DEBUG_STR
5521 DEBUG_STATIC_STR)
5522 {
5523 PIM_DONT_DEBUG_STATIC;
5524 return CMD_SUCCESS;
5525 }
5526
5527
5528 DEFUN (debug_pim,
5529 debug_pim_cmd,
5530 "debug pim",
5531 DEBUG_STR
5532 DEBUG_PIM_STR)
5533 {
5534 PIM_DO_DEBUG_PIM_EVENTS;
5535 PIM_DO_DEBUG_PIM_PACKETS;
5536 PIM_DO_DEBUG_PIM_TRACE;
5537 PIM_DO_DEBUG_MSDP_EVENTS;
5538 PIM_DO_DEBUG_MSDP_PACKETS;
5539 return CMD_SUCCESS;
5540 }
5541
5542 DEFUN (no_debug_pim,
5543 no_debug_pim_cmd,
5544 "no debug pim",
5545 NO_STR
5546 DEBUG_STR
5547 DEBUG_PIM_STR)
5548 {
5549 PIM_DONT_DEBUG_PIM_EVENTS;
5550 PIM_DONT_DEBUG_PIM_PACKETS;
5551 PIM_DONT_DEBUG_PIM_TRACE;
5552 PIM_DONT_DEBUG_MSDP_EVENTS;
5553 PIM_DONT_DEBUG_MSDP_PACKETS;
5554
5555 PIM_DONT_DEBUG_PIM_PACKETDUMP_SEND;
5556 PIM_DONT_DEBUG_PIM_PACKETDUMP_RECV;
5557
5558 return CMD_SUCCESS;
5559 }
5560
5561
5562 DEFUN (debug_pim_events,
5563 debug_pim_events_cmd,
5564 "debug pim events",
5565 DEBUG_STR
5566 DEBUG_PIM_STR
5567 DEBUG_PIM_EVENTS_STR)
5568 {
5569 PIM_DO_DEBUG_PIM_EVENTS;
5570 return CMD_SUCCESS;
5571 }
5572
5573 DEFUN (no_debug_pim_events,
5574 no_debug_pim_events_cmd,
5575 "no debug pim events",
5576 NO_STR
5577 DEBUG_STR
5578 DEBUG_PIM_STR
5579 DEBUG_PIM_EVENTS_STR)
5580 {
5581 PIM_DONT_DEBUG_PIM_EVENTS;
5582 return CMD_SUCCESS;
5583 }
5584
5585 DEFUN (debug_pim_packets,
5586 debug_pim_packets_cmd,
5587 "debug pim packets [<hello|joins|register>]",
5588 DEBUG_STR
5589 DEBUG_PIM_STR
5590 DEBUG_PIM_PACKETS_STR
5591 DEBUG_PIM_HELLO_PACKETS_STR
5592 DEBUG_PIM_J_P_PACKETS_STR
5593 DEBUG_PIM_PIM_REG_PACKETS_STR)
5594 {
5595 int idx = 0;
5596 if (argv_find (argv, argc, "hello", &idx))
5597 {
5598 PIM_DO_DEBUG_PIM_HELLO;
5599 vty_out (vty, "PIM Hello debugging is on%s", VTY_NEWLINE);
5600 }
5601 else if (argv_find (argv, argc ,"joins", &idx))
5602 {
5603 PIM_DO_DEBUG_PIM_J_P;
5604 vty_out (vty, "PIM Join/Prune debugging is on%s", VTY_NEWLINE);
5605 }
5606 else if (argv_find (argv, argc, "register", &idx))
5607 {
5608 PIM_DO_DEBUG_PIM_REG;
5609 vty_out (vty, "PIM Register debugging is on%s", VTY_NEWLINE);
5610 }
5611 else
5612 {
5613 PIM_DO_DEBUG_PIM_PACKETS;
5614 vty_out (vty, "PIM Packet debugging is on %s", VTY_NEWLINE);
5615 }
5616 return CMD_SUCCESS;
5617 }
5618
5619 DEFUN (no_debug_pim_packets,
5620 no_debug_pim_packets_cmd,
5621 "no debug pim packets [<hello|joins|register>]",
5622 NO_STR
5623 DEBUG_STR
5624 DEBUG_PIM_STR
5625 DEBUG_PIM_PACKETS_STR
5626 DEBUG_PIM_HELLO_PACKETS_STR
5627 DEBUG_PIM_J_P_PACKETS_STR
5628 DEBUG_PIM_PIM_REG_PACKETS_STR)
5629 {
5630 int idx = 0;
5631 if (argv_find (argv, argc,"hello",&idx))
5632 {
5633 PIM_DONT_DEBUG_PIM_HELLO;
5634 vty_out (vty, "PIM Hello debugging is off %s", VTY_NEWLINE);
5635 }
5636 else if (argv_find (argv, argc, "joins", &idx))
5637 {
5638 PIM_DONT_DEBUG_PIM_J_P;
5639 vty_out (vty, "PIM Join/Prune debugging is off %s", VTY_NEWLINE);
5640 }
5641 else if (argv_find (argv, argc, "register", &idx))
5642 {
5643 PIM_DONT_DEBUG_PIM_REG;
5644 vty_out (vty, "PIM Register debugging is off%s", VTY_NEWLINE);
5645 }
5646 else
5647 PIM_DONT_DEBUG_PIM_PACKETS;
5648
5649 return CMD_SUCCESS;
5650 }
5651
5652
5653 DEFUN (debug_pim_packetdump_send,
5654 debug_pim_packetdump_send_cmd,
5655 "debug pim packet-dump send",
5656 DEBUG_STR
5657 DEBUG_PIM_STR
5658 DEBUG_PIM_PACKETDUMP_STR
5659 DEBUG_PIM_PACKETDUMP_SEND_STR)
5660 {
5661 PIM_DO_DEBUG_PIM_PACKETDUMP_SEND;
5662 return CMD_SUCCESS;
5663 }
5664
5665 DEFUN (no_debug_pim_packetdump_send,
5666 no_debug_pim_packetdump_send_cmd,
5667 "no debug pim packet-dump send",
5668 NO_STR
5669 DEBUG_STR
5670 DEBUG_PIM_STR
5671 DEBUG_PIM_PACKETDUMP_STR
5672 DEBUG_PIM_PACKETDUMP_SEND_STR)
5673 {
5674 PIM_DONT_DEBUG_PIM_PACKETDUMP_SEND;
5675 return CMD_SUCCESS;
5676 }
5677
5678
5679 DEFUN (debug_pim_packetdump_recv,
5680 debug_pim_packetdump_recv_cmd,
5681 "debug pim packet-dump receive",
5682 DEBUG_STR
5683 DEBUG_PIM_STR
5684 DEBUG_PIM_PACKETDUMP_STR
5685 DEBUG_PIM_PACKETDUMP_RECV_STR)
5686 {
5687 PIM_DO_DEBUG_PIM_PACKETDUMP_RECV;
5688 return CMD_SUCCESS;
5689 }
5690
5691 DEFUN (no_debug_pim_packetdump_recv,
5692 no_debug_pim_packetdump_recv_cmd,
5693 "no debug pim packet-dump receive",
5694 NO_STR
5695 DEBUG_STR
5696 DEBUG_PIM_STR
5697 DEBUG_PIM_PACKETDUMP_STR
5698 DEBUG_PIM_PACKETDUMP_RECV_STR)
5699 {
5700 PIM_DONT_DEBUG_PIM_PACKETDUMP_RECV;
5701 return CMD_SUCCESS;
5702 }
5703
5704
5705 DEFUN (debug_pim_trace,
5706 debug_pim_trace_cmd,
5707 "debug pim trace",
5708 DEBUG_STR
5709 DEBUG_PIM_STR
5710 DEBUG_PIM_TRACE_STR)
5711 {
5712 PIM_DO_DEBUG_PIM_TRACE;
5713 return CMD_SUCCESS;
5714 }
5715
5716 DEFUN (no_debug_pim_trace,
5717 no_debug_pim_trace_cmd,
5718 "no debug pim trace",
5719 NO_STR
5720 DEBUG_STR
5721 DEBUG_PIM_STR
5722 DEBUG_PIM_TRACE_STR)
5723 {
5724 PIM_DONT_DEBUG_PIM_TRACE;
5725 return CMD_SUCCESS;
5726 }
5727
5728
5729 DEFUN (debug_ssmpingd,
5730 debug_ssmpingd_cmd,
5731 "debug ssmpingd",
5732 DEBUG_STR
5733 DEBUG_PIM_STR
5734 DEBUG_SSMPINGD_STR)
5735 {
5736 PIM_DO_DEBUG_SSMPINGD;
5737 return CMD_SUCCESS;
5738 }
5739
5740 DEFUN (no_debug_ssmpingd,
5741 no_debug_ssmpingd_cmd,
5742 "no debug ssmpingd",
5743 NO_STR
5744 DEBUG_STR
5745 DEBUG_PIM_STR
5746 DEBUG_SSMPINGD_STR)
5747 {
5748 PIM_DONT_DEBUG_SSMPINGD;
5749 return CMD_SUCCESS;
5750 }
5751
5752
5753 DEFUN (debug_pim_zebra,
5754 debug_pim_zebra_cmd,
5755 "debug pim zebra",
5756 DEBUG_STR
5757 DEBUG_PIM_STR
5758 DEBUG_PIM_ZEBRA_STR)
5759 {
5760 PIM_DO_DEBUG_ZEBRA;
5761 return CMD_SUCCESS;
5762 }
5763
5764 DEFUN (no_debug_pim_zebra,
5765 no_debug_pim_zebra_cmd,
5766 "no debug pim zebra",
5767 NO_STR
5768 DEBUG_STR
5769 DEBUG_PIM_STR
5770 DEBUG_PIM_ZEBRA_STR)
5771 {
5772 PIM_DONT_DEBUG_ZEBRA;
5773 return CMD_SUCCESS;
5774 }
5775
5776
5777 DEFUN (debug_msdp,
5778 debug_msdp_cmd,
5779 "debug msdp",
5780 DEBUG_STR
5781 DEBUG_MSDP_STR)
5782 {
5783 PIM_DO_DEBUG_MSDP_EVENTS;
5784 PIM_DO_DEBUG_MSDP_PACKETS;
5785 return CMD_SUCCESS;
5786 }
5787
5788 DEFUN (no_debug_msdp,
5789 no_debug_msdp_cmd,
5790 "no debug msdp",
5791 NO_STR
5792 DEBUG_STR
5793 DEBUG_MSDP_STR)
5794 {
5795 PIM_DONT_DEBUG_MSDP_EVENTS;
5796 PIM_DONT_DEBUG_MSDP_PACKETS;
5797 return CMD_SUCCESS;
5798 }
5799
5800 ALIAS (no_debug_msdp,
5801 undebug_msdp_cmd,
5802 "undebug msdp",
5803 UNDEBUG_STR
5804 DEBUG_MSDP_STR)
5805
5806 DEFUN (debug_msdp_events,
5807 debug_msdp_events_cmd,
5808 "debug msdp events",
5809 DEBUG_STR
5810 DEBUG_MSDP_STR
5811 DEBUG_MSDP_EVENTS_STR)
5812 {
5813 PIM_DO_DEBUG_MSDP_EVENTS;
5814 return CMD_SUCCESS;
5815 }
5816
5817 DEFUN (no_debug_msdp_events,
5818 no_debug_msdp_events_cmd,
5819 "no debug msdp events",
5820 NO_STR
5821 DEBUG_STR
5822 DEBUG_MSDP_STR
5823 DEBUG_MSDP_EVENTS_STR)
5824 {
5825 PIM_DONT_DEBUG_MSDP_EVENTS;
5826 return CMD_SUCCESS;
5827 }
5828
5829 ALIAS (no_debug_msdp_events,
5830 undebug_msdp_events_cmd,
5831 "undebug msdp events",
5832 UNDEBUG_STR
5833 DEBUG_MSDP_STR
5834 DEBUG_MSDP_EVENTS_STR)
5835
5836 DEFUN (debug_msdp_packets,
5837 debug_msdp_packets_cmd,
5838 "debug msdp packets",
5839 DEBUG_STR
5840 DEBUG_MSDP_STR
5841 DEBUG_MSDP_PACKETS_STR)
5842 {
5843 PIM_DO_DEBUG_MSDP_PACKETS;
5844 return CMD_SUCCESS;
5845 }
5846
5847 DEFUN (no_debug_msdp_packets,
5848 no_debug_msdp_packets_cmd,
5849 "no debug msdp packets",
5850 NO_STR
5851 DEBUG_STR
5852 DEBUG_MSDP_STR
5853 DEBUG_MSDP_PACKETS_STR)
5854 {
5855 PIM_DONT_DEBUG_MSDP_PACKETS;
5856 return CMD_SUCCESS;
5857 }
5858
5859 ALIAS (no_debug_msdp_packets,
5860 undebug_msdp_packets_cmd,
5861 "undebug msdp packets",
5862 UNDEBUG_STR
5863 DEBUG_MSDP_STR
5864 DEBUG_MSDP_PACKETS_STR)
5865
5866 DEFUN (show_debugging_pim,
5867 show_debugging_pim_cmd,
5868 "show debugging pim",
5869 SHOW_STR
5870 DEBUG_STR
5871 PIM_STR)
5872 {
5873 pim_debug_config_write(vty);
5874 return CMD_SUCCESS;
5875 }
5876
5877 static int
5878 interface_pim_use_src_cmd_worker(struct vty *vty, const char *source)
5879 {
5880 int result;
5881 struct in_addr source_addr;
5882 VTY_DECLVAR_CONTEXT(interface, ifp);
5883
5884 result = inet_pton(AF_INET, source, &source_addr);
5885 if (result <= 0) {
5886 vty_out(vty, "%% Bad source address %s: errno=%d: %s%s",
5887 source, errno, safe_strerror(errno), VTY_NEWLINE);
5888 return CMD_WARNING;
5889 }
5890
5891 result = pim_update_source_set(ifp, source_addr);
5892 switch (result) {
5893 case PIM_SUCCESS:
5894 break;
5895 case PIM_IFACE_NOT_FOUND:
5896 vty_out(vty, "Pim not enabled on this interface%s", VTY_NEWLINE);
5897 break;
5898 case PIM_UPDATE_SOURCE_DUP:
5899 vty_out(vty, "%% Source already set to %s%s", source, VTY_NEWLINE);
5900 break;
5901 default:
5902 vty_out(vty, "%% Source set failed%s", VTY_NEWLINE);
5903 }
5904
5905 return result?CMD_WARNING:CMD_SUCCESS;
5906 }
5907
5908 DEFUN (interface_pim_use_source,
5909 interface_pim_use_source_cmd,
5910 "ip pim use-source A.B.C.D",
5911 IP_STR
5912 "pim multicast routing\n"
5913 "Configure primary IP address\n"
5914 "source ip address\n")
5915 {
5916 return interface_pim_use_src_cmd_worker (vty, argv[3]->arg);
5917 }
5918
5919 DEFUN (interface_no_pim_use_source,
5920 interface_no_pim_use_source_cmd,
5921 "no ip pim use-source",
5922 NO_STR
5923 IP_STR
5924 "pim multicast routing\n"
5925 "Delete source IP address\n")
5926 {
5927 return interface_pim_use_src_cmd_worker (vty, "0.0.0.0");
5928 }
5929
5930 static int
5931 ip_msdp_peer_cmd_worker (struct vty *vty, const char *peer, const char *local)
5932 {
5933 enum pim_msdp_err result;
5934 struct in_addr peer_addr;
5935 struct in_addr local_addr;
5936
5937 result = inet_pton(AF_INET, peer, &peer_addr);
5938 if (result <= 0) {
5939 vty_out(vty, "%% Bad peer address %s: errno=%d: %s%s",
5940 peer, errno, safe_strerror(errno), VTY_NEWLINE);
5941 return CMD_WARNING;
5942 }
5943
5944 result = inet_pton(AF_INET, local, &local_addr);
5945 if (result <= 0) {
5946 vty_out(vty, "%% Bad source address %s: errno=%d: %s%s",
5947 local, errno, safe_strerror(errno), VTY_NEWLINE);
5948 return CMD_WARNING;
5949 }
5950
5951 result = pim_msdp_peer_add(peer_addr, local_addr, "default", NULL/* mp_p */);
5952 switch (result) {
5953 case PIM_MSDP_ERR_NONE:
5954 break;
5955 case PIM_MSDP_ERR_OOM:
5956 vty_out(vty, "%% Out of memory%s", VTY_NEWLINE);
5957 break;
5958 case PIM_MSDP_ERR_PEER_EXISTS:
5959 vty_out(vty, "%% Peer exists%s", VTY_NEWLINE);
5960 break;
5961 case PIM_MSDP_ERR_MAX_MESH_GROUPS:
5962 vty_out(vty, "%% Only one mesh-group allowed currently%s", VTY_NEWLINE);
5963 break;
5964 default:
5965 vty_out(vty, "%% peer add failed%s", VTY_NEWLINE);
5966 }
5967
5968 return result?CMD_WARNING:CMD_SUCCESS;
5969 }
5970
5971 DEFUN_HIDDEN (ip_msdp_peer,
5972 ip_msdp_peer_cmd,
5973 "ip msdp peer A.B.C.D source A.B.C.D",
5974 IP_STR
5975 CFG_MSDP_STR
5976 "Configure MSDP peer\n"
5977 "peer ip address\n"
5978 "Source address for TCP connection\n"
5979 "local ip address\n")
5980 {
5981 return ip_msdp_peer_cmd_worker (vty, argv[3]->arg, argv[5]->arg);
5982 }
5983
5984 static int
5985 ip_no_msdp_peer_cmd_worker (struct vty *vty, const char *peer)
5986 {
5987 enum pim_msdp_err result;
5988 struct in_addr peer_addr;
5989
5990 result = inet_pton(AF_INET, peer, &peer_addr);
5991 if (result <= 0) {
5992 vty_out(vty, "%% Bad peer address %s: errno=%d: %s%s",
5993 peer, errno, safe_strerror(errno), VTY_NEWLINE);
5994 return CMD_WARNING;
5995 }
5996
5997 result = pim_msdp_peer_del(peer_addr);
5998 switch (result) {
5999 case PIM_MSDP_ERR_NONE:
6000 break;
6001 case PIM_MSDP_ERR_NO_PEER:
6002 vty_out(vty, "%% Peer does not exist%s", VTY_NEWLINE);
6003 break;
6004 default:
6005 vty_out(vty, "%% peer del failed%s", VTY_NEWLINE);
6006 }
6007
6008 return result?CMD_WARNING:CMD_SUCCESS;
6009 }
6010
6011 DEFUN_HIDDEN (no_ip_msdp_peer,
6012 no_ip_msdp_peer_cmd,
6013 "no ip msdp peer A.B.C.D",
6014 NO_STR
6015 IP_STR
6016 CFG_MSDP_STR
6017 "Delete MSDP peer\n"
6018 "peer ip address\n")
6019 {
6020 return ip_no_msdp_peer_cmd_worker (vty, argv[4]->arg);
6021 }
6022
6023 static int
6024 ip_msdp_mesh_group_member_cmd_worker(struct vty *vty, const char *mg, const char *mbr)
6025 {
6026 enum pim_msdp_err result;
6027 struct in_addr mbr_ip;
6028
6029 result = inet_pton(AF_INET, mbr, &mbr_ip);
6030 if (result <= 0) {
6031 vty_out(vty, "%% Bad member address %s: errno=%d: %s%s",
6032 mbr, errno, safe_strerror(errno), VTY_NEWLINE);
6033 return CMD_WARNING;
6034 }
6035
6036 result = pim_msdp_mg_mbr_add(mg, mbr_ip);
6037 switch (result) {
6038 case PIM_MSDP_ERR_NONE:
6039 break;
6040 case PIM_MSDP_ERR_OOM:
6041 vty_out(vty, "%% Out of memory%s", VTY_NEWLINE);
6042 break;
6043 case PIM_MSDP_ERR_MG_MBR_EXISTS:
6044 vty_out(vty, "%% mesh-group member exists%s", VTY_NEWLINE);
6045 break;
6046 case PIM_MSDP_ERR_MAX_MESH_GROUPS:
6047 vty_out(vty, "%% Only one mesh-group allowed currently%s", VTY_NEWLINE);
6048 break;
6049 default:
6050 vty_out(vty, "%% member add failed%s", VTY_NEWLINE);
6051 }
6052
6053 return result?CMD_WARNING:CMD_SUCCESS;
6054 }
6055
6056 DEFUN (ip_msdp_mesh_group_member,
6057 ip_msdp_mesh_group_member_cmd,
6058 "ip msdp mesh-group WORD member A.B.C.D",
6059 IP_STR
6060 CFG_MSDP_STR
6061 "Configure MSDP mesh-group\n"
6062 "mesh group name\n"
6063 "mesh group member\n"
6064 "peer ip address\n")
6065 {
6066 return ip_msdp_mesh_group_member_cmd_worker(vty, argv[3]->arg, argv[5]->arg);
6067 }
6068
6069 static int
6070 ip_no_msdp_mesh_group_member_cmd_worker(struct vty *vty, const char *mg, const char *mbr)
6071 {
6072 enum pim_msdp_err result;
6073 struct in_addr mbr_ip;
6074
6075 result = inet_pton(AF_INET, mbr, &mbr_ip);
6076 if (result <= 0) {
6077 vty_out(vty, "%% Bad member address %s: errno=%d: %s%s",
6078 mbr, errno, safe_strerror(errno), VTY_NEWLINE);
6079 return CMD_WARNING;
6080 }
6081
6082 result = pim_msdp_mg_mbr_del(mg, mbr_ip);
6083 switch (result) {
6084 case PIM_MSDP_ERR_NONE:
6085 break;
6086 case PIM_MSDP_ERR_NO_MG:
6087 vty_out(vty, "%% mesh-group does not exist%s", VTY_NEWLINE);
6088 break;
6089 case PIM_MSDP_ERR_NO_MG_MBR:
6090 vty_out(vty, "%% mesh-group member does not exist%s", VTY_NEWLINE);
6091 break;
6092 default:
6093 vty_out(vty, "%% mesh-group member del failed%s", VTY_NEWLINE);
6094 }
6095
6096 return result?CMD_WARNING:CMD_SUCCESS;
6097 }
6098 DEFUN (no_ip_msdp_mesh_group_member,
6099 no_ip_msdp_mesh_group_member_cmd,
6100 "no ip msdp mesh-group WORD member A.B.C.D",
6101 NO_STR
6102 IP_STR
6103 CFG_MSDP_STR
6104 "Delete MSDP mesh-group member\n"
6105 "mesh group name\n"
6106 "mesh group member\n"
6107 "peer ip address\n")
6108 {
6109 return ip_no_msdp_mesh_group_member_cmd_worker(vty, argv[4]->arg, argv[6]->arg);
6110 }
6111
6112 static int
6113 ip_msdp_mesh_group_source_cmd_worker(struct vty *vty, const char *mg, const char *src)
6114 {
6115 enum pim_msdp_err result;
6116 struct in_addr src_ip;
6117
6118 result = inet_pton(AF_INET, src, &src_ip);
6119 if (result <= 0) {
6120 vty_out(vty, "%% Bad source address %s: errno=%d: %s%s",
6121 src, errno, safe_strerror(errno), VTY_NEWLINE);
6122 return CMD_WARNING;
6123 }
6124
6125 result = pim_msdp_mg_src_add(mg, src_ip);
6126 switch (result) {
6127 case PIM_MSDP_ERR_NONE:
6128 break;
6129 case PIM_MSDP_ERR_OOM:
6130 vty_out(vty, "%% Out of memory%s", VTY_NEWLINE);
6131 break;
6132 case PIM_MSDP_ERR_MAX_MESH_GROUPS:
6133 vty_out(vty, "%% Only one mesh-group allowed currently%s", VTY_NEWLINE);
6134 break;
6135 default:
6136 vty_out(vty, "%% source add failed%s", VTY_NEWLINE);
6137 }
6138
6139 return result?CMD_WARNING:CMD_SUCCESS;
6140 }
6141
6142
6143 DEFUN (ip_msdp_mesh_group_source,
6144 ip_msdp_mesh_group_source_cmd,
6145 "ip msdp mesh-group WORD source A.B.C.D",
6146 IP_STR
6147 CFG_MSDP_STR
6148 "Configure MSDP mesh-group\n"
6149 "mesh group name\n"
6150 "mesh group local address\n"
6151 "source ip address for the TCP connection\n")
6152 {
6153 return ip_msdp_mesh_group_source_cmd_worker(vty, argv[3]->arg, argv[5]->arg);
6154 }
6155
6156 static int
6157 ip_no_msdp_mesh_group_source_cmd_worker(struct vty *vty, const char *mg)
6158 {
6159 enum pim_msdp_err result;
6160
6161 result = pim_msdp_mg_src_del(mg);
6162 switch (result) {
6163 case PIM_MSDP_ERR_NONE:
6164 break;
6165 case PIM_MSDP_ERR_NO_MG:
6166 vty_out(vty, "%% mesh-group does not exist%s", VTY_NEWLINE);
6167 break;
6168 default:
6169 vty_out(vty, "%% mesh-group source del failed%s", VTY_NEWLINE);
6170 }
6171
6172 return result?CMD_WARNING:CMD_SUCCESS;
6173 }
6174
6175 static int
6176 ip_no_msdp_mesh_group_cmd_worker(struct vty *vty, const char *mg)
6177 {
6178 enum pim_msdp_err result;
6179
6180 result = pim_msdp_mg_del(mg);
6181 switch (result) {
6182 case PIM_MSDP_ERR_NONE:
6183 break;
6184 case PIM_MSDP_ERR_NO_MG:
6185 vty_out(vty, "%% mesh-group does not exist%s", VTY_NEWLINE);
6186 break;
6187 default:
6188 vty_out(vty, "%% mesh-group source del failed%s", VTY_NEWLINE);
6189 }
6190
6191 return result ? CMD_WARNING : CMD_SUCCESS;
6192 }
6193
6194 DEFUN (no_ip_msdp_mesh_group_source,
6195 no_ip_msdp_mesh_group_source_cmd,
6196 "no ip msdp mesh-group WORD source [A.B.C.D]",
6197 NO_STR
6198 IP_STR
6199 CFG_MSDP_STR
6200 "Delete MSDP mesh-group source\n"
6201 "mesh group name\n"
6202 "mesh group source\n"
6203 "mesh group local address\n")
6204 {
6205 if (argc == 7)
6206 return ip_no_msdp_mesh_group_cmd_worker(vty, argv[6]->arg);
6207 else
6208 return ip_no_msdp_mesh_group_source_cmd_worker(vty, argv[4]->arg);
6209 }
6210
6211 static void
6212 print_empty_json_obj(struct vty *vty)
6213 {
6214 json_object *json;
6215 json = json_object_new_object();
6216 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
6217 json_object_free(json);
6218 }
6219
6220 static void
6221 ip_msdp_show_mesh_group(struct vty *vty, u_char uj)
6222 {
6223 struct listnode *mbrnode;
6224 struct pim_msdp_mg_mbr *mbr;
6225 struct pim_msdp_mg *mg = msdp->mg;
6226 char mbr_str[INET_ADDRSTRLEN];
6227 char src_str[INET_ADDRSTRLEN];
6228 char state_str[PIM_MSDP_STATE_STRLEN];
6229 enum pim_msdp_peer_state state;
6230 json_object *json = NULL;
6231 json_object *json_mg_row = NULL;
6232 json_object *json_members = NULL;
6233 json_object *json_row = NULL;
6234
6235 if (!mg) {
6236 if (uj)
6237 print_empty_json_obj(vty);
6238 return;
6239 }
6240
6241 pim_inet4_dump("<source?>", mg->src_ip, src_str, sizeof(src_str));
6242 if (uj) {
6243 json = json_object_new_object();
6244 /* currently there is only one mesh group but we should still make
6245 * it a dict with mg-name as key */
6246 json_mg_row = json_object_new_object();
6247 json_object_string_add(json_mg_row, "name", mg->mesh_group_name);
6248 json_object_string_add(json_mg_row, "source", src_str);
6249 } else {
6250 vty_out(vty, "Mesh group : %s%s", mg->mesh_group_name, VTY_NEWLINE);
6251 vty_out(vty, " Source : %s%s", src_str, VTY_NEWLINE);
6252 vty_out(vty, " Member State%s", VTY_NEWLINE);
6253 }
6254
6255 for (ALL_LIST_ELEMENTS_RO(mg->mbr_list, mbrnode, mbr)) {
6256 pim_inet4_dump("<mbr?>", mbr->mbr_ip, mbr_str, sizeof(mbr_str));
6257 if (mbr->mp) {
6258 state = mbr->mp->state;
6259 } else {
6260 state = PIM_MSDP_DISABLED;
6261 }
6262 pim_msdp_state_dump(state, state_str, sizeof(state_str));
6263 if (uj) {
6264 json_row = json_object_new_object();
6265 json_object_string_add(json_row, "member", mbr_str);
6266 json_object_string_add(json_row, "state", state_str);
6267 if (!json_members) {
6268 json_members = json_object_new_object();
6269 json_object_object_add(json_mg_row, "members", json_members);
6270 }
6271 json_object_object_add(json_members, mbr_str, json_row);
6272 } else {
6273 vty_out(vty, " %-15s %11s%s",
6274 mbr_str, state_str, VTY_NEWLINE);
6275 }
6276 }
6277
6278 if (uj) {
6279 json_object_object_add(json, mg->mesh_group_name, json_mg_row);
6280 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
6281 json_object_free(json);
6282 }
6283 }
6284
6285 DEFUN (show_ip_msdp_mesh_group,
6286 show_ip_msdp_mesh_group_cmd,
6287 "show ip msdp mesh-group [json]",
6288 SHOW_STR
6289 IP_STR
6290 MSDP_STR
6291 "MSDP mesh-group information\n"
6292 "JavaScript Object Notation\n")
6293 {
6294 u_char uj = use_json(argc, argv);
6295 ip_msdp_show_mesh_group(vty, uj);
6296
6297 return CMD_SUCCESS;
6298 }
6299
6300 static void
6301 ip_msdp_show_peers(struct vty *vty, u_char uj)
6302 {
6303 struct listnode *mpnode;
6304 struct pim_msdp_peer *mp;
6305 char peer_str[INET_ADDRSTRLEN];
6306 char local_str[INET_ADDRSTRLEN];
6307 char state_str[PIM_MSDP_STATE_STRLEN];
6308 char timebuf[PIM_MSDP_UPTIME_STRLEN];
6309 int64_t now;
6310 json_object *json = NULL;
6311 json_object *json_row = NULL;
6312
6313
6314 if (uj) {
6315 json = json_object_new_object();
6316 } else {
6317 vty_out(vty, "Peer Local State Uptime SaCnt%s", VTY_NEWLINE);
6318 }
6319
6320 for (ALL_LIST_ELEMENTS_RO(msdp->peer_list, mpnode, mp)) {
6321 if (mp->state == PIM_MSDP_ESTABLISHED) {
6322 now = pim_time_monotonic_sec();
6323 pim_time_uptime(timebuf, sizeof(timebuf), now - mp->uptime);
6324 } else {
6325 strcpy(timebuf, "-");
6326 }
6327 pim_inet4_dump("<peer?>", mp->peer, peer_str, sizeof(peer_str));
6328 pim_inet4_dump("<local?>", mp->local, local_str, sizeof(local_str));
6329 pim_msdp_state_dump(mp->state, state_str, sizeof(state_str));
6330 if (uj) {
6331 json_row = json_object_new_object();
6332 json_object_string_add(json_row, "peer", peer_str);
6333 json_object_string_add(json_row, "local", local_str);
6334 json_object_string_add(json_row, "state", state_str);
6335 json_object_string_add(json_row, "upTime", timebuf);
6336 json_object_int_add(json_row, "saCount", mp->sa_cnt);
6337 json_object_object_add(json, peer_str, json_row);
6338 } else {
6339 vty_out(vty, "%-15s %15s %11s %8s %6d%s",
6340 peer_str, local_str, state_str,
6341 timebuf, mp->sa_cnt, VTY_NEWLINE);
6342 }
6343 }
6344
6345 if (uj) {
6346 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
6347 json_object_free(json);
6348 }
6349 }
6350
6351 static void
6352 ip_msdp_show_peers_detail(struct vty *vty, const char *peer, u_char uj)
6353 {
6354 struct listnode *mpnode;
6355 struct pim_msdp_peer *mp;
6356 char peer_str[INET_ADDRSTRLEN];
6357 char local_str[INET_ADDRSTRLEN];
6358 char state_str[PIM_MSDP_STATE_STRLEN];
6359 char timebuf[PIM_MSDP_UPTIME_STRLEN];
6360 char katimer[PIM_MSDP_TIMER_STRLEN];
6361 char crtimer[PIM_MSDP_TIMER_STRLEN];
6362 char holdtimer[PIM_MSDP_TIMER_STRLEN];
6363 int64_t now;
6364 json_object *json = NULL;
6365 json_object *json_row = NULL;
6366
6367 if (uj) {
6368 json = json_object_new_object();
6369 }
6370
6371 for (ALL_LIST_ELEMENTS_RO(msdp->peer_list, mpnode, mp)) {
6372 pim_inet4_dump("<peer?>", mp->peer, peer_str, sizeof(peer_str));
6373 if (strcmp(peer, "detail") &&
6374 strcmp(peer, peer_str))
6375 continue;
6376
6377 if (mp->state == PIM_MSDP_ESTABLISHED) {
6378 now = pim_time_monotonic_sec();
6379 pim_time_uptime(timebuf, sizeof(timebuf), now - mp->uptime);
6380 } else {
6381 strcpy(timebuf, "-");
6382 }
6383 pim_inet4_dump("<local?>", mp->local, local_str, sizeof(local_str));
6384 pim_msdp_state_dump(mp->state, state_str, sizeof(state_str));
6385 pim_time_timer_to_hhmmss(katimer, sizeof(katimer), mp->ka_timer);
6386 pim_time_timer_to_hhmmss(crtimer, sizeof(crtimer), mp->cr_timer);
6387 pim_time_timer_to_hhmmss(holdtimer, sizeof(holdtimer), mp->hold_timer);
6388
6389 if (uj) {
6390 json_row = json_object_new_object();
6391 json_object_string_add(json_row, "peer", peer_str);
6392 json_object_string_add(json_row, "local", local_str);
6393 json_object_string_add(json_row, "meshGroupName", mp->mesh_group_name);
6394 json_object_string_add(json_row, "state", state_str);
6395 json_object_string_add(json_row, "upTime", timebuf);
6396 json_object_string_add(json_row, "keepAliveTimer", katimer);
6397 json_object_string_add(json_row, "connRetryTimer", crtimer);
6398 json_object_string_add(json_row, "holdTimer", holdtimer);
6399 json_object_string_add(json_row, "lastReset", mp->last_reset);
6400 json_object_int_add(json_row, "connAttempts", mp->conn_attempts);
6401 json_object_int_add(json_row, "establishedChanges", mp->est_flaps);
6402 json_object_int_add(json_row, "saCount", mp->sa_cnt);
6403 json_object_int_add(json_row, "kaSent", mp->ka_tx_cnt);
6404 json_object_int_add(json_row, "kaRcvd", mp->ka_rx_cnt);
6405 json_object_int_add(json_row, "saSent", mp->sa_tx_cnt);
6406 json_object_int_add(json_row, "saRcvd", mp->sa_rx_cnt);
6407 json_object_object_add(json, peer_str, json_row);
6408 } else {
6409 vty_out(vty, "Peer : %s%s", peer_str, VTY_NEWLINE);
6410 vty_out(vty, " Local : %s%s", local_str, VTY_NEWLINE);
6411 vty_out(vty, " Mesh Group : %s%s", mp->mesh_group_name, VTY_NEWLINE);
6412 vty_out(vty, " State : %s%s", state_str, VTY_NEWLINE);
6413 vty_out(vty, " Uptime : %s%s", timebuf, VTY_NEWLINE);
6414
6415 vty_out(vty, " Keepalive Timer : %s%s", katimer, VTY_NEWLINE);
6416 vty_out(vty, " Conn Retry Timer : %s%s", crtimer, VTY_NEWLINE);
6417 vty_out(vty, " Hold Timer : %s%s", holdtimer, VTY_NEWLINE);
6418 vty_out(vty, " Last Reset : %s%s", mp->last_reset, VTY_NEWLINE);
6419 vty_out(vty, " Conn Attempts : %d%s", mp->conn_attempts, VTY_NEWLINE);
6420 vty_out(vty, " Established Changes : %d%s", mp->est_flaps, VTY_NEWLINE);
6421 vty_out(vty, " SA Count : %d%s", mp->sa_cnt, VTY_NEWLINE);
6422 vty_out(vty, " Statistics :%s", VTY_NEWLINE);
6423 vty_out(vty, " Sent Rcvd%s", VTY_NEWLINE);
6424 vty_out(vty, " Keepalives : %10d %10d%s",
6425 mp->ka_tx_cnt, mp->ka_rx_cnt, VTY_NEWLINE);
6426 vty_out(vty, " SAs : %10d %10d%s",
6427 mp->sa_tx_cnt, mp->sa_rx_cnt, VTY_NEWLINE);
6428 vty_out(vty, "%s", VTY_NEWLINE);
6429 }
6430 }
6431
6432 if (uj) {
6433 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
6434 json_object_free(json);
6435 }
6436 }
6437
6438 DEFUN (show_ip_msdp_peer_detail,
6439 show_ip_msdp_peer_detail_cmd,
6440 "show ip msdp peer [detail|A.B.C.D] [json]",
6441 SHOW_STR
6442 IP_STR
6443 MSDP_STR
6444 "MSDP peer information\n"
6445 "Detailed output\n"
6446 "peer ip address\n"
6447 "JavaScript Object Notation\n")
6448 {
6449 u_char uj = use_json(argc, argv);
6450 if (uj)
6451 argc--;
6452
6453 if (argc == 4)
6454 ip_msdp_show_peers_detail(vty, argv[4]->arg, uj);
6455 else
6456 ip_msdp_show_peers(vty, uj);
6457
6458 return CMD_SUCCESS;
6459 }
6460
6461 static void
6462 ip_msdp_show_sa(struct vty *vty, u_char uj)
6463 {
6464 struct listnode *sanode;
6465 struct pim_msdp_sa *sa;
6466 char src_str[INET_ADDRSTRLEN];
6467 char grp_str[INET_ADDRSTRLEN];
6468 char rp_str[INET_ADDRSTRLEN];
6469 char timebuf[PIM_MSDP_UPTIME_STRLEN];
6470 char spt_str[8];
6471 char local_str[8];
6472 int64_t now;
6473 json_object *json = NULL;
6474 json_object *json_group = NULL;
6475 json_object *json_row = NULL;
6476
6477 if (uj) {
6478 json = json_object_new_object();
6479 } else {
6480 vty_out(vty, "Source Group RP Local SPT Uptime%s", VTY_NEWLINE);
6481 }
6482
6483 for (ALL_LIST_ELEMENTS_RO(msdp->sa_list, sanode, sa)) {
6484 now = pim_time_monotonic_sec();
6485 pim_time_uptime(timebuf, sizeof(timebuf), now - sa->uptime);
6486 pim_inet4_dump("<src?>", sa->sg.src, src_str, sizeof(src_str));
6487 pim_inet4_dump("<grp?>", sa->sg.grp, grp_str, sizeof(grp_str));
6488 if (sa->flags & PIM_MSDP_SAF_PEER) {
6489 pim_inet4_dump("<rp?>", sa->rp, rp_str, sizeof(rp_str));
6490 if (sa->up) {
6491 strcpy(spt_str, "yes");
6492 } else {
6493 strcpy(spt_str, "no");
6494 }
6495 } else {
6496 strcpy(rp_str, "-");
6497 strcpy(spt_str, "-");
6498 }
6499 if (sa->flags & PIM_MSDP_SAF_LOCAL) {
6500 strcpy(local_str, "yes");
6501 } else {
6502 strcpy(local_str, "no");
6503 }
6504 if (uj) {
6505 json_object_object_get_ex(json, grp_str, &json_group);
6506
6507 if (!json_group) {
6508 json_group = json_object_new_object();
6509 json_object_object_add(json, grp_str, json_group);
6510 }
6511
6512 json_row = json_object_new_object();
6513 json_object_string_add(json_row, "source", src_str);
6514 json_object_string_add(json_row, "group", grp_str);
6515 json_object_string_add(json_row, "rp", rp_str);
6516 json_object_string_add(json_row, "local", local_str);
6517 json_object_string_add(json_row, "sptSetup", spt_str);
6518 json_object_string_add(json_row, "upTime", timebuf);
6519 json_object_object_add(json_group, src_str, json_row);
6520 } else {
6521 vty_out(vty, "%-15s %15s %15s %5c %3c %8s%s",
6522 src_str, grp_str, rp_str, local_str[0], spt_str[0], timebuf, VTY_NEWLINE);
6523 }
6524 }
6525
6526
6527 if (uj) {
6528 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
6529 json_object_free(json);
6530 }
6531 }
6532
6533 static void
6534 ip_msdp_show_sa_entry_detail(struct pim_msdp_sa *sa, const char *src_str,
6535 const char *grp_str, struct vty *vty,
6536 u_char uj, json_object *json)
6537 {
6538 char rp_str[INET_ADDRSTRLEN];
6539 char peer_str[INET_ADDRSTRLEN];
6540 char timebuf[PIM_MSDP_UPTIME_STRLEN];
6541 char spt_str[8];
6542 char local_str[8];
6543 char statetimer[PIM_MSDP_TIMER_STRLEN];
6544 int64_t now;
6545 json_object *json_group = NULL;
6546 json_object *json_row = NULL;
6547
6548 now = pim_time_monotonic_sec();
6549 pim_time_uptime(timebuf, sizeof(timebuf), now - sa->uptime);
6550 if (sa->flags & PIM_MSDP_SAF_PEER) {
6551 pim_inet4_dump("<rp?>", sa->rp, rp_str, sizeof(rp_str));
6552 pim_inet4_dump("<peer?>", sa->peer, peer_str, sizeof(peer_str));
6553 if (sa->up) {
6554 strcpy(spt_str, "yes");
6555 } else {
6556 strcpy(spt_str, "no");
6557 }
6558 } else {
6559 strcpy(rp_str, "-");
6560 strcpy(peer_str, "-");
6561 strcpy(spt_str, "-");
6562 }
6563 if (sa->flags & PIM_MSDP_SAF_LOCAL) {
6564 strcpy(local_str, "yes");
6565 } else {
6566 strcpy(local_str, "no");
6567 }
6568 pim_time_timer_to_hhmmss(statetimer, sizeof(statetimer), sa->sa_state_timer);
6569 if (uj) {
6570 json_object_object_get_ex(json, grp_str, &json_group);
6571
6572 if (!json_group) {
6573 json_group = json_object_new_object();
6574 json_object_object_add(json, grp_str, json_group);
6575 }
6576
6577 json_row = json_object_new_object();
6578 json_object_string_add(json_row, "source", src_str);
6579 json_object_string_add(json_row, "group", grp_str);
6580 json_object_string_add(json_row, "rp", rp_str);
6581 json_object_string_add(json_row, "local", local_str);
6582 json_object_string_add(json_row, "sptSetup", spt_str);
6583 json_object_string_add(json_row, "upTime", timebuf);
6584 json_object_string_add(json_row, "stateTimer", statetimer);
6585 json_object_object_add(json_group, src_str, json_row);
6586 } else {
6587 vty_out(vty, "SA : %s%s", sa->sg_str, VTY_NEWLINE);
6588 vty_out(vty, " RP : %s%s", rp_str, VTY_NEWLINE);
6589 vty_out(vty, " Peer : %s%s", peer_str, VTY_NEWLINE);
6590 vty_out(vty, " Local : %s%s", local_str, VTY_NEWLINE);
6591 vty_out(vty, " SPT Setup : %s%s", spt_str, VTY_NEWLINE);
6592 vty_out(vty, " Uptime : %s%s", timebuf, VTY_NEWLINE);
6593 vty_out(vty, " State Timer : %s%s", statetimer, VTY_NEWLINE);
6594 vty_out(vty, "%s", VTY_NEWLINE);
6595 }
6596 }
6597
6598 static void
6599 ip_msdp_show_sa_detail(struct vty *vty, u_char uj)
6600 {
6601 struct listnode *sanode;
6602 struct pim_msdp_sa *sa;
6603 char src_str[INET_ADDRSTRLEN];
6604 char grp_str[INET_ADDRSTRLEN];
6605 json_object *json = NULL;
6606
6607 if (uj) {
6608 json = json_object_new_object();
6609 }
6610
6611 for (ALL_LIST_ELEMENTS_RO(msdp->sa_list, sanode, sa)) {
6612 pim_inet4_dump("<src?>", sa->sg.src, src_str, sizeof(src_str));
6613 pim_inet4_dump("<grp?>", sa->sg.grp, grp_str, sizeof(grp_str));
6614 ip_msdp_show_sa_entry_detail(sa, src_str, grp_str, vty, uj, json);
6615 }
6616
6617 if (uj) {
6618 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
6619 json_object_free(json);
6620 }
6621 }
6622
6623 DEFUN (show_ip_msdp_sa_detail,
6624 show_ip_msdp_sa_detail_cmd,
6625 "show ip msdp sa detail [json]",
6626 SHOW_STR
6627 IP_STR
6628 MSDP_STR
6629 "MSDP active-source information\n"
6630 "Detailed output\n"
6631 "JavaScript Object Notation\n")
6632 {
6633 u_char uj = use_json(argc, argv);
6634 ip_msdp_show_sa_detail(vty, uj);
6635
6636 return CMD_SUCCESS;
6637 }
6638
6639 static void
6640 ip_msdp_show_sa_addr(struct vty *vty, const char *addr, u_char uj)
6641 {
6642 struct listnode *sanode;
6643 struct pim_msdp_sa *sa;
6644 char src_str[INET_ADDRSTRLEN];
6645 char grp_str[INET_ADDRSTRLEN];
6646 json_object *json = NULL;
6647
6648 if (uj) {
6649 json = json_object_new_object();
6650 }
6651
6652 for (ALL_LIST_ELEMENTS_RO(msdp->sa_list, sanode, sa)) {
6653 pim_inet4_dump("<src?>", sa->sg.src, src_str, sizeof(src_str));
6654 pim_inet4_dump("<grp?>", sa->sg.grp, grp_str, sizeof(grp_str));
6655 if (!strcmp(addr, src_str) || !strcmp(addr, grp_str)) {
6656 ip_msdp_show_sa_entry_detail(sa, src_str, grp_str, vty, uj, json);
6657 }
6658 }
6659
6660 if (uj) {
6661 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
6662 json_object_free(json);
6663 }
6664 }
6665
6666 static void
6667 ip_msdp_show_sa_sg(struct vty *vty, const char *src, const char *grp, u_char uj)
6668 {
6669 struct listnode *sanode;
6670 struct pim_msdp_sa *sa;
6671 char src_str[INET_ADDRSTRLEN];
6672 char grp_str[INET_ADDRSTRLEN];
6673 json_object *json = NULL;
6674
6675 if (uj) {
6676 json = json_object_new_object();
6677 }
6678
6679 for (ALL_LIST_ELEMENTS_RO(msdp->sa_list, sanode, sa)) {
6680 pim_inet4_dump("<src?>", sa->sg.src, src_str, sizeof(src_str));
6681 pim_inet4_dump("<grp?>", sa->sg.grp, grp_str, sizeof(grp_str));
6682 if (!strcmp(src, src_str) && !strcmp(grp, grp_str)) {
6683 ip_msdp_show_sa_entry_detail(sa, src_str, grp_str, vty, uj, json);
6684 }
6685 }
6686
6687 if (uj) {
6688 vty_out (vty, "%s%s", json_object_to_json_string_ext(json, JSON_C_TO_STRING_PRETTY), VTY_NEWLINE);
6689 json_object_free(json);
6690 }
6691 }
6692
6693 DEFUN (show_ip_msdp_sa_sg,
6694 show_ip_msdp_sa_sg_cmd,
6695 "show ip msdp sa [A.B.C.D [A.B.C.D]] [json]",
6696 SHOW_STR
6697 IP_STR
6698 MSDP_STR
6699 "MSDP active-source information\n"
6700 "source or group ip\n"
6701 "group ip\n"
6702 "JavaScript Object Notation\n")
6703 {
6704 u_char uj = use_json(argc, argv);
6705
6706 int idx = 0;
6707 char *src_ip = argv_find (argv, argc, "A.B.C.D", &idx) ? argv[idx++]->arg : NULL;
6708 char *grp_ip = idx < argc && argv_find (argv, argc, "A.B.C.D", &idx) ?
6709 argv[idx]->arg : NULL;
6710
6711 if (src_ip && grp_ip)
6712 ip_msdp_show_sa_sg(vty, src_ip, grp_ip, uj);
6713 else if (src_ip)
6714 ip_msdp_show_sa_addr(vty, src_ip, uj);
6715 else
6716 ip_msdp_show_sa(vty, uj);
6717
6718 return CMD_SUCCESS;
6719 }
6720
6721 void pim_cmd_init()
6722 {
6723 install_node (&pim_global_node, pim_global_config_write); /* PIM_NODE */
6724 install_node (&interface_node, pim_interface_config_write); /* INTERFACE_NODE */
6725 if_cmd_init ();
6726
6727 install_node (&debug_node, pim_debug_config_write);
6728
6729 install_element (CONFIG_NODE, &ip_multicast_routing_cmd);
6730 install_element (CONFIG_NODE, &no_ip_multicast_routing_cmd);
6731 install_element (CONFIG_NODE, &ip_pim_rp_cmd);
6732 install_element (CONFIG_NODE, &no_ip_pim_rp_cmd);
6733 install_element (CONFIG_NODE, &ip_pim_rp_prefix_list_cmd);
6734 install_element (CONFIG_NODE, &no_ip_pim_rp_prefix_list_cmd);
6735 install_element (CONFIG_NODE, &no_ip_pim_ssm_prefix_list_cmd);
6736 install_element (CONFIG_NODE, &no_ip_pim_ssm_prefix_list_name_cmd);
6737 install_element (CONFIG_NODE, &ip_pim_ssm_prefix_list_cmd);
6738 install_element (CONFIG_NODE, &ip_pim_register_suppress_cmd);
6739 install_element (CONFIG_NODE, &no_ip_pim_register_suppress_cmd);
6740 install_element (CONFIG_NODE, &ip_pim_spt_switchover_infinity_cmd);
6741 install_element (CONFIG_NODE, &ip_pim_spt_switchover_infinity_plist_cmd);
6742 install_element (CONFIG_NODE, &no_ip_pim_spt_switchover_infinity_cmd);
6743 install_element (CONFIG_NODE, &no_ip_pim_spt_switchover_infinity_plist_cmd);
6744 install_element (CONFIG_NODE, &ip_pim_joinprune_time_cmd);
6745 install_element (CONFIG_NODE, &no_ip_pim_joinprune_time_cmd);
6746 install_element (CONFIG_NODE, &ip_pim_keep_alive_cmd);
6747 install_element (CONFIG_NODE, &no_ip_pim_keep_alive_cmd);
6748 install_element (CONFIG_NODE, &ip_pim_packets_cmd);
6749 install_element (CONFIG_NODE, &no_ip_pim_packets_cmd);
6750 install_element (CONFIG_NODE, &ip_pim_v6_secondary_cmd);
6751 install_element (CONFIG_NODE, &no_ip_pim_v6_secondary_cmd);
6752 install_element (CONFIG_NODE, &ip_ssmpingd_cmd);
6753 install_element (CONFIG_NODE, &no_ip_ssmpingd_cmd);
6754 install_element (CONFIG_NODE, &ip_msdp_peer_cmd);
6755 install_element (CONFIG_NODE, &no_ip_msdp_peer_cmd);
6756 install_element (CONFIG_NODE, &ip_pim_ecmp_cmd);
6757 install_element (CONFIG_NODE, &no_ip_pim_ecmp_cmd);
6758 install_element (CONFIG_NODE, &ip_pim_ecmp_rebalance_cmd);
6759 install_element (CONFIG_NODE, &no_ip_pim_ecmp_rebalance_cmd);
6760
6761 install_element (INTERFACE_NODE, &interface_ip_igmp_cmd);
6762 install_element (INTERFACE_NODE, &interface_no_ip_igmp_cmd);
6763 install_element (INTERFACE_NODE, &interface_ip_igmp_join_cmd);
6764 install_element (INTERFACE_NODE, &interface_no_ip_igmp_join_cmd);
6765 install_element (INTERFACE_NODE, &interface_ip_igmp_version_cmd);
6766 install_element (INTERFACE_NODE, &interface_no_ip_igmp_version_cmd);
6767 install_element (INTERFACE_NODE, &interface_ip_igmp_query_interval_cmd);
6768 install_element (INTERFACE_NODE, &interface_no_ip_igmp_query_interval_cmd);
6769 install_element (INTERFACE_NODE, &interface_ip_igmp_query_max_response_time_cmd);
6770 install_element (INTERFACE_NODE, &interface_no_ip_igmp_query_max_response_time_cmd);
6771 install_element (INTERFACE_NODE, &interface_ip_igmp_query_max_response_time_dsec_cmd);
6772 install_element (INTERFACE_NODE, &interface_no_ip_igmp_query_max_response_time_dsec_cmd);
6773 install_element (INTERFACE_NODE, &interface_ip_pim_ssm_cmd);
6774 install_element (INTERFACE_NODE, &interface_no_ip_pim_ssm_cmd);
6775 install_element (INTERFACE_NODE, &interface_ip_pim_sm_cmd);
6776 install_element (INTERFACE_NODE, &interface_no_ip_pim_sm_cmd);
6777 install_element (INTERFACE_NODE, &interface_ip_pim_drprio_cmd);
6778 install_element (INTERFACE_NODE, &interface_no_ip_pim_drprio_cmd);
6779 install_element (INTERFACE_NODE, &interface_ip_pim_hello_cmd);
6780 install_element (INTERFACE_NODE, &interface_no_ip_pim_hello_cmd);
6781
6782 // Static mroutes NEB
6783 install_element (INTERFACE_NODE, &interface_ip_mroute_cmd);
6784 install_element (INTERFACE_NODE, &interface_ip_mroute_source_cmd);
6785 install_element (INTERFACE_NODE, &interface_no_ip_mroute_cmd);
6786 install_element (INTERFACE_NODE, &interface_no_ip_mroute_source_cmd);
6787
6788 install_element (VIEW_NODE, &show_ip_igmp_interface_cmd);
6789 install_element (VIEW_NODE, &show_ip_igmp_join_cmd);
6790 install_element (VIEW_NODE, &show_ip_igmp_groups_cmd);
6791 install_element (VIEW_NODE, &show_ip_igmp_groups_retransmissions_cmd);
6792 install_element (VIEW_NODE, &show_ip_igmp_sources_cmd);
6793 install_element (VIEW_NODE, &show_ip_igmp_sources_retransmissions_cmd);
6794 install_element (VIEW_NODE, &show_ip_pim_assert_cmd);
6795 install_element (VIEW_NODE, &show_ip_pim_assert_internal_cmd);
6796 install_element (VIEW_NODE, &show_ip_pim_assert_metric_cmd);
6797 install_element (VIEW_NODE, &show_ip_pim_assert_winner_metric_cmd);
6798 install_element (VIEW_NODE, &show_ip_pim_interface_traffic_cmd);
6799 install_element (VIEW_NODE, &show_ip_pim_interface_cmd);
6800 install_element (VIEW_NODE, &show_ip_pim_join_cmd);
6801 install_element (VIEW_NODE, &show_ip_pim_local_membership_cmd);
6802 install_element (VIEW_NODE, &show_ip_pim_neighbor_cmd);
6803 install_element (VIEW_NODE, &show_ip_pim_rpf_cmd);
6804 install_element (VIEW_NODE, &show_ip_pim_secondary_cmd);
6805 install_element (VIEW_NODE, &show_ip_pim_state_cmd);
6806 install_element (VIEW_NODE, &show_ip_pim_upstream_cmd);
6807 install_element (VIEW_NODE, &show_ip_pim_upstream_join_desired_cmd);
6808 install_element (VIEW_NODE, &show_ip_pim_upstream_rpf_cmd);
6809 install_element (VIEW_NODE, &show_ip_pim_rp_cmd);
6810 install_element (VIEW_NODE, &show_ip_multicast_cmd);
6811 install_element (VIEW_NODE, &show_ip_mroute_cmd);
6812 install_element (VIEW_NODE, &show_ip_mroute_count_cmd);
6813 install_element (VIEW_NODE, &show_ip_rib_cmd);
6814 install_element (VIEW_NODE, &show_ip_ssmpingd_cmd);
6815 install_element (VIEW_NODE, &show_debugging_pim_cmd);
6816 install_element (VIEW_NODE, &show_ip_pim_nexthop_cmd);
6817 install_element (VIEW_NODE, &show_ip_pim_nexthop_lookup_cmd);
6818
6819 install_element (ENABLE_NODE, &clear_ip_interfaces_cmd);
6820 install_element (ENABLE_NODE, &clear_ip_igmp_interfaces_cmd);
6821 install_element (ENABLE_NODE, &clear_ip_mroute_cmd);
6822 install_element (ENABLE_NODE, &clear_ip_pim_interfaces_cmd);
6823 install_element (ENABLE_NODE, &clear_ip_pim_interface_traffic_cmd);
6824 install_element (ENABLE_NODE, &clear_ip_pim_oil_cmd);
6825
6826 install_element (ENABLE_NODE, &debug_igmp_cmd);
6827 install_element (ENABLE_NODE, &no_debug_igmp_cmd);
6828 install_element (ENABLE_NODE, &debug_igmp_events_cmd);
6829 install_element (ENABLE_NODE, &no_debug_igmp_events_cmd);
6830 install_element (ENABLE_NODE, &debug_igmp_packets_cmd);
6831 install_element (ENABLE_NODE, &no_debug_igmp_packets_cmd);
6832 install_element (ENABLE_NODE, &debug_igmp_trace_cmd);
6833 install_element (ENABLE_NODE, &no_debug_igmp_trace_cmd);
6834 install_element (ENABLE_NODE, &debug_mroute_cmd);
6835 install_element (ENABLE_NODE, &debug_mroute_detail_cmd);
6836 install_element (ENABLE_NODE, &no_debug_mroute_cmd);
6837 install_element (ENABLE_NODE, &no_debug_mroute_detail_cmd);
6838 install_element (ENABLE_NODE, &debug_static_cmd);
6839 install_element (ENABLE_NODE, &no_debug_static_cmd);
6840 install_element (ENABLE_NODE, &debug_pim_cmd);
6841 install_element (ENABLE_NODE, &no_debug_pim_cmd);
6842 install_element (ENABLE_NODE, &debug_pim_events_cmd);
6843 install_element (ENABLE_NODE, &no_debug_pim_events_cmd);
6844 install_element (ENABLE_NODE, &debug_pim_packets_cmd);
6845 install_element (ENABLE_NODE, &no_debug_pim_packets_cmd);
6846 install_element (ENABLE_NODE, &debug_pim_packetdump_send_cmd);
6847 install_element (ENABLE_NODE, &no_debug_pim_packetdump_send_cmd);
6848 install_element (ENABLE_NODE, &debug_pim_packetdump_recv_cmd);
6849 install_element (ENABLE_NODE, &no_debug_pim_packetdump_recv_cmd);
6850 install_element (ENABLE_NODE, &debug_pim_trace_cmd);
6851 install_element (ENABLE_NODE, &no_debug_pim_trace_cmd);
6852 install_element (ENABLE_NODE, &debug_ssmpingd_cmd);
6853 install_element (ENABLE_NODE, &no_debug_ssmpingd_cmd);
6854 install_element (ENABLE_NODE, &debug_pim_zebra_cmd);
6855 install_element (ENABLE_NODE, &no_debug_pim_zebra_cmd);
6856 install_element (ENABLE_NODE, &debug_msdp_cmd);
6857 install_element (ENABLE_NODE, &no_debug_msdp_cmd);
6858 install_element (ENABLE_NODE, &undebug_msdp_cmd);
6859 install_element (ENABLE_NODE, &debug_msdp_events_cmd);
6860 install_element (ENABLE_NODE, &no_debug_msdp_events_cmd);
6861 install_element (ENABLE_NODE, &undebug_msdp_events_cmd);
6862 install_element (ENABLE_NODE, &debug_msdp_packets_cmd);
6863 install_element (ENABLE_NODE, &no_debug_msdp_packets_cmd);
6864 install_element (ENABLE_NODE, &undebug_msdp_packets_cmd);
6865
6866 install_element (CONFIG_NODE, &debug_igmp_cmd);
6867 install_element (CONFIG_NODE, &no_debug_igmp_cmd);
6868 install_element (CONFIG_NODE, &debug_igmp_events_cmd);
6869 install_element (CONFIG_NODE, &no_debug_igmp_events_cmd);
6870 install_element (CONFIG_NODE, &debug_igmp_packets_cmd);
6871 install_element (CONFIG_NODE, &no_debug_igmp_packets_cmd);
6872 install_element (CONFIG_NODE, &debug_igmp_trace_cmd);
6873 install_element (CONFIG_NODE, &no_debug_igmp_trace_cmd);
6874 install_element (CONFIG_NODE, &debug_mroute_cmd);
6875 install_element (CONFIG_NODE, &debug_mroute_detail_cmd);
6876 install_element (CONFIG_NODE, &no_debug_mroute_cmd);
6877 install_element (CONFIG_NODE, &no_debug_mroute_detail_cmd);
6878 install_element (CONFIG_NODE, &debug_static_cmd);
6879 install_element (CONFIG_NODE, &no_debug_static_cmd);
6880 install_element (CONFIG_NODE, &debug_pim_cmd);
6881 install_element (CONFIG_NODE, &no_debug_pim_cmd);
6882 install_element (CONFIG_NODE, &debug_pim_events_cmd);
6883 install_element (CONFIG_NODE, &no_debug_pim_events_cmd);
6884 install_element (CONFIG_NODE, &debug_pim_packets_cmd);
6885 install_element (CONFIG_NODE, &no_debug_pim_packets_cmd);
6886 install_element (CONFIG_NODE, &debug_pim_trace_cmd);
6887 install_element (CONFIG_NODE, &no_debug_pim_trace_cmd);
6888 install_element (CONFIG_NODE, &debug_ssmpingd_cmd);
6889 install_element (CONFIG_NODE, &no_debug_ssmpingd_cmd);
6890 install_element (CONFIG_NODE, &debug_pim_zebra_cmd);
6891 install_element (CONFIG_NODE, &no_debug_pim_zebra_cmd);
6892 install_element (CONFIG_NODE, &debug_msdp_cmd);
6893 install_element (CONFIG_NODE, &no_debug_msdp_cmd);
6894 install_element (CONFIG_NODE, &undebug_msdp_cmd);
6895 install_element (CONFIG_NODE, &debug_msdp_events_cmd);
6896 install_element (CONFIG_NODE, &no_debug_msdp_events_cmd);
6897 install_element (CONFIG_NODE, &undebug_msdp_events_cmd);
6898 install_element (CONFIG_NODE, &debug_msdp_packets_cmd);
6899 install_element (CONFIG_NODE, &no_debug_msdp_packets_cmd);
6900 install_element (CONFIG_NODE, &undebug_msdp_packets_cmd);
6901 install_element (CONFIG_NODE, &ip_msdp_mesh_group_member_cmd);
6902 install_element (CONFIG_NODE, &no_ip_msdp_mesh_group_member_cmd);
6903 install_element (CONFIG_NODE, &ip_msdp_mesh_group_source_cmd);
6904 install_element (CONFIG_NODE, &no_ip_msdp_mesh_group_source_cmd);
6905 install_element (VIEW_NODE, &show_ip_msdp_peer_detail_cmd);
6906 install_element (VIEW_NODE, &show_ip_msdp_sa_detail_cmd);
6907 install_element (VIEW_NODE, &show_ip_msdp_sa_sg_cmd);
6908 install_element (VIEW_NODE, &show_ip_msdp_mesh_group_cmd);
6909 install_element (VIEW_NODE, &show_ip_pim_ssm_range_cmd);
6910 install_element (VIEW_NODE, &show_ip_pim_group_type_cmd);
6911 install_element (INTERFACE_NODE, &interface_pim_use_source_cmd);
6912 install_element (INTERFACE_NODE, &interface_no_pim_use_source_cmd);
6913 }