zebra: mrib: static route support
With the MRIB being independent from the Unicast RIB, there's currently
now way to add static routes to the MRIB. Address that by adding a
separate set of commands for MRIB static routes.
Combines these original patches:
- zebra: mrib: ip mroute command to add unicast route to MRIB for multicast RPF.
- zebra: mrib: no ip mroute: Fix removal of static multicast RPF route.
- zebra: mrib: remove unused static_add/delete_ipv4
- zebra: Cleanups to zebra_rib.
- pimd: Merge pim-only branch.
Cc: Everton Marques <everton.marques@gmail.com>
Cc: Balaji G <balajig81@gmail.com>
Signed-off-by: David Lamparter <equinox@opensourcerouting.org>
diff --git a/zebra/rib.h b/zebra/rib.h
index d40b17e..5eedfde 100644
--- a/zebra/rib.h
+++ b/zebra/rib.h
@@ -430,12 +430,12 @@
extern unsigned long rib_score_proto (u_char proto);
extern int
-static_add_ipv4 (struct prefix *p, struct in_addr *gate, const char *ifname,
- u_char flags, u_char distance, u_int32_t vrf_id);
-
+static_add_ipv4_safi (safi_t safi, struct prefix *p, struct in_addr *gate,
+ const char *ifname, u_char flags, u_char distance,
+ u_int32_t vrf_id);
extern int
-static_delete_ipv4 (struct prefix *p, struct in_addr *gate, const char *ifname,
- u_char distance, u_int32_t vrf_id);
+static_delete_ipv4_safi (safi_t safi, struct prefix *p, struct in_addr *gate,
+ const char *ifname, u_char distance, u_int32_t vrf_id);
#ifdef HAVE_IPV6
extern int
diff --git a/zebra/zebra_rib.c b/zebra/zebra_rib.c
index 08ce964..a07036e 100644
--- a/zebra/zebra_rib.c
+++ b/zebra/zebra_rib.c
@@ -2276,14 +2276,14 @@
/* Install static route into rib. */
static void
-static_install_ipv4 (struct prefix *p, struct static_ipv4 *si)
+static_install_ipv4 (safi_t safi, struct prefix *p, struct static_ipv4 *si)
{
struct rib *rib;
struct route_node *rn;
struct route_table *table;
/* Lookup table. */
- table = vrf_table (AFI_IP, SAFI_UNICAST, 0);
+ table = vrf_table (AFI_IP, safi, 0);
if (! table)
return;
@@ -2368,7 +2368,7 @@
/* Uninstall static route from RIB. */
static void
-static_uninstall_ipv4 (struct prefix *p, struct static_ipv4 *si)
+static_uninstall_ipv4 (safi_t safi, struct prefix *p, struct static_ipv4 *si)
{
struct route_node *rn;
struct rib *rib;
@@ -2376,7 +2376,7 @@
struct route_table *table;
/* Lookup table. */
- table = vrf_table (AFI_IP, SAFI_UNICAST, 0);
+ table = vrf_table (AFI_IP, safi, 0);
if (! table)
return;
@@ -2427,10 +2427,10 @@
route_unlock_node (rn);
}
-/* Add static route into static route configuration. */
int
-static_add_ipv4 (struct prefix *p, struct in_addr *gate, const char *ifname,
- u_char flags, u_char distance, u_int32_t vrf_id)
+static_add_ipv4_safi (safi_t safi, struct prefix *p, struct in_addr *gate,
+ const char *ifname, u_char flags, u_char distance,
+ u_int32_t vrf_id)
{
u_char type = 0;
struct route_node *rn;
@@ -2441,7 +2441,7 @@
struct route_table *stable;
/* Lookup table. */
- stable = vrf_static_table (AFI_IP, SAFI_UNICAST, vrf_id);
+ stable = vrf_static_table (AFI_IP, safi, vrf_id);
if (! stable)
return -1;
@@ -2475,7 +2475,7 @@
/* Distance changed. */
if (update)
- static_delete_ipv4 (p, gate, ifname, update->distance, vrf_id);
+ static_delete_ipv4_safi (safi, p, gate, ifname, update->distance, vrf_id);
/* Make new static route structure. */
si = XCALLOC (MTYPE_STATIC_IPV4, sizeof (struct static_ipv4));
@@ -2517,15 +2517,14 @@
si->next = cp;
/* Install into rib. */
- static_install_ipv4 (p, si);
+ static_install_ipv4 (safi, p, si);
return 1;
}
-/* Delete static route from static route configuration. */
int
-static_delete_ipv4 (struct prefix *p, struct in_addr *gate, const char *ifname,
- u_char distance, u_int32_t vrf_id)
+static_delete_ipv4_safi (safi_t safi, struct prefix *p, struct in_addr *gate,
+ const char *ifname, u_char distance, u_int32_t vrf_id)
{
u_char type = 0;
struct route_node *rn;
@@ -2533,7 +2532,7 @@
struct route_table *stable;
/* Lookup table. */
- stable = vrf_static_table (AFI_IP, SAFI_UNICAST, vrf_id);
+ stable = vrf_static_table (AFI_IP, safi, vrf_id);
if (! stable)
return -1;
@@ -2565,7 +2564,7 @@
}
/* Install into rib. */
- static_uninstall_ipv4 (p, si);
+ static_uninstall_ipv4 (safi, p, si);
/* Unlink static route from linked list. */
if (si->prev)
@@ -2586,7 +2585,6 @@
return 1;
}
-
#ifdef HAVE_IPV6
static int
rib_bogus_ipv6 (int type, struct prefix_ipv6 *p,
diff --git a/zebra/zebra_vty.c b/zebra/zebra_vty.c
index 9d6c1dd..6802ece 100644
--- a/zebra/zebra_vty.c
+++ b/zebra/zebra_vty.c
@@ -30,11 +30,14 @@
#include "zebra/zserv.h"
-/* General fucntion for static route. */
+static int do_show_ip_route(struct vty *vty, safi_t safi);
+
+/* General function for static route. */
static int
-zebra_static_ipv4 (struct vty *vty, int add_cmd, const char *dest_str,
- const char *mask_str, const char *gate_str,
- const char *flag_str, const char *distance_str)
+zebra_static_ipv4_safi (struct vty *vty, safi_t safi, int add_cmd,
+ const char *dest_str, const char *mask_str,
+ const char *gate_str, const char *flag_str,
+ const char *distance_str)
{
int ret;
u_char distance;
@@ -81,9 +84,9 @@
return CMD_WARNING;
}
if (add_cmd)
- static_add_ipv4 (&p, NULL, NULL, ZEBRA_FLAG_BLACKHOLE, distance, 0);
+ static_add_ipv4_safi (safi, &p, NULL, NULL, ZEBRA_FLAG_BLACKHOLE, distance, 0);
else
- static_delete_ipv4 (&p, NULL, NULL, distance, 0);
+ static_delete_ipv4_safi (safi, &p, NULL, NULL, distance, 0);
return CMD_SUCCESS;
}
@@ -107,9 +110,9 @@
if (gate_str == NULL)
{
if (add_cmd)
- static_add_ipv4 (&p, NULL, NULL, flag, distance, 0);
+ static_add_ipv4_safi (safi, &p, NULL, NULL, flag, distance, 0);
else
- static_delete_ipv4 (&p, NULL, NULL, distance, 0);
+ static_delete_ipv4_safi (safi, &p, NULL, NULL, distance, 0);
return CMD_SUCCESS;
}
@@ -123,13 +126,58 @@
ifname = gate_str;
if (add_cmd)
- static_add_ipv4 (&p, ifname ? NULL : &gate, ifname, flag, distance, 0);
+ static_add_ipv4_safi (safi, &p, ifname ? NULL : &gate, ifname, flag, distance, 0);
else
- static_delete_ipv4 (&p, ifname ? NULL : &gate, ifname, distance, 0);
+ static_delete_ipv4_safi (safi, &p, ifname ? NULL : &gate, ifname, distance, 0);
return CMD_SUCCESS;
}
+static int
+zebra_static_ipv4 (struct vty *vty, int add_cmd, const char *dest_str,
+ const char *mask_str, const char *gate_str,
+ const char *flag_str, const char *distance_str)
+{
+ return zebra_static_ipv4_safi(vty, SAFI_UNICAST, add_cmd, dest_str, mask_str, gate_str, flag_str, distance_str);
+}
+
+/* Static unicast routes for multicast RPF lookup. */
+DEFUN (ip_mroute,
+ ip_mroute_cmd,
+ "ip mroute A.B.C.D/M (A.B.C.D|INTERFACE) [<1-255>]",
+ IP_STR
+ "Configure static unicast route into MRIB for multicast RPF lookup\n"
+ "IP destination prefix (e.g. 10.0.0.0/8)\n"
+ "Nexthop address\n"
+ "Nexthop interface name\n"
+ "Distance\n")
+{
+ return zebra_static_ipv4_safi(vty, SAFI_MULTICAST, 1, argv[0], NULL, argv[1], NULL, argv[2]);
+}
+
+DEFUN (no_ip_mroute,
+ no_ip_mroute_cmd,
+ "no ip mroute A.B.C.D/M (A.B.C.D|INTERFACE) [<1-255>]",
+ IP_STR
+ "Configure static unicast route into MRIB for multicast RPF lookup\n"
+ "IP destination prefix (e.g. 10.0.0.0/8)\n"
+ "Nexthop address\n"
+ "Nexthop interface name\n"
+ "Distance\n")
+{
+ return zebra_static_ipv4_safi(vty, SAFI_MULTICAST, 0, argv[0], NULL, argv[1], NULL, argv[2]);
+}
+
+DEFUN (show_ip_rpf,
+ show_ip_rpf_cmd,
+ "show ip rpf",
+ SHOW_STR
+ IP_STR
+ "Display RPF information for multicast source\n")
+{
+ return do_show_ip_route(vty, SAFI_MULTICAST);
+}
+
/* Static route configuration. */
DEFUN (ip_route,
ip_route_cmd,
@@ -788,12 +836,16 @@
IP_STR
"IP routing table\n")
{
+ return do_show_ip_route(vty, SAFI_UNICAST);
+}
+
+static int do_show_ip_route(struct vty *vty, safi_t safi) {
struct route_table *table;
struct route_node *rn;
struct rib *rib;
int first = 1;
- table = vrf_table (AFI_IP, SAFI_UNICAST, 0);
+ table = vrf_table (AFI_IP, safi, 0);
if (! table)
return CMD_SUCCESS;
@@ -1195,7 +1247,7 @@
/* Write IPv4 static route configuration. */
static int
-static_config_ipv4 (struct vty *vty)
+static_config_ipv4 (struct vty *vty, safi_t safi, const char *cmd)
{
struct route_node *rn;
struct static_ipv4 *si;
@@ -1205,14 +1257,14 @@
write = 0;
/* Lookup table. */
- stable = vrf_static_table (AFI_IP, SAFI_UNICAST, 0);
+ stable = vrf_static_table (AFI_IP, safi, 0);
if (! stable)
return -1;
for (rn = route_top (stable); rn; rn = route_next (rn))
for (si = rn->info; si; si = si->next)
{
- vty_out (vty, "ip route %s/%d", inet_ntoa (rn->p.u.prefix4),
+ vty_out (vty, "%s %s/%d", cmd, inet_ntoa (rn->p.u.prefix4),
rn->p.prefixlen);
switch (si->type)
@@ -2146,7 +2198,8 @@
{
int write = 0;
- write += static_config_ipv4 (vty);
+ write += static_config_ipv4 (vty, SAFI_UNICAST, "ip route");
+ write += static_config_ipv4 (vty, SAFI_MULTICAST, "ip mroute");
#ifdef HAVE_IPV6
write += static_config_ipv6 (vty);
#endif /* HAVE_IPV6 */
@@ -2185,6 +2238,8 @@
install_node (&ip_node, zebra_ip_config);
install_node (&protocol_node, config_write_protocol);
+ install_element (CONFIG_NODE, &ip_mroute_cmd);
+ install_element (CONFIG_NODE, &no_ip_mroute_cmd);
install_element (CONFIG_NODE, &ip_protocol_cmd);
install_element (CONFIG_NODE, &no_ip_protocol_cmd);
install_element (VIEW_NODE, &show_ip_protocol_cmd);
@@ -2233,6 +2288,8 @@
install_element (VIEW_NODE, &show_ip_mroute_cmd);
install_element (ENABLE_NODE, &show_ip_mroute_cmd);
+ install_element (VIEW_NODE, &show_ip_rpf_cmd);
+ install_element (ENABLE_NODE, &show_ip_rpf_cmd);
#ifdef HAVE_IPV6
install_element (CONFIG_NODE, &ipv6_route_cmd);