pimd: Create special pimreg interface

The linux kernel wants a pimreg vif device.  The pimd
code wants a 'struct interface *' for anything it works
with.  Since the pimreg vif device is not a real linux
device that zebra knows about.  Cheat by creating
a pimreg interface pointer and setup the code to
properly be able to handle the registration of the vif
device.

Signed-off-by: Donald Sharp <sharpd@cumulusnetworks.com>
This commit is contained in:
Donald Sharp 2015-10-29 06:35:12 -07:00 committed by Donald Sharp
parent 37653d4f68
commit c992c9a0c0
4 changed files with 27 additions and 8 deletions

View file

@ -3283,6 +3283,8 @@ DEFUN (interface_ip_pim_sm,
return CMD_WARNING;
}
pim_if_create_pimreg();
return CMD_SUCCESS;
}

View file

@ -41,6 +41,8 @@
#include "pim_time.h"
#include "pim_ssmpingd.h"
struct interface *pim_regiface = NULL;
static void pim_if_igmp_join_del_all(struct interface *ifp);
void pim_if_init()
@ -145,8 +147,6 @@ struct pim_interface *pim_if_new(struct interface *ifp, int igmp, int pim)
pim_sock_reset(ifp);
zassert(PIM_IF_TEST_PIM(pim_ifp->options) || PIM_IF_TEST_IGMP(pim_ifp->options));
if (PIM_MROUTE_IS_ENABLED) {
pim_if_add_vif(ifp);
}
@ -615,6 +615,7 @@ int pim_if_add_vif(struct interface *ifp)
{
struct pim_interface *pim_ifp = ifp->info;
struct in_addr ifaddr;
unsigned char flags;
zassert(pim_ifp);
@ -640,14 +641,15 @@ int pim_if_add_vif(struct interface *ifp)
}
ifaddr = pim_ifp->primary_address;
if (PIM_INADDR_IS_ANY(ifaddr)) {
if (ifp->ifindex != PIM_OIF_PIM_REGISTER_VIF && PIM_INADDR_IS_ANY(ifaddr)) {
zlog_warn("%s: could not get address for interface %s ifindex=%d",
__PRETTY_FUNCTION__,
ifp->name, ifp->ifindex);
return -4;
}
if (pim_mroute_add_vif(ifp->ifindex, ifaddr, 0)) {
flags = (ifp->ifindex == PIM_OIF_PIM_REGISTER_VIF) ? VIFF_REGISTER : 0;
if (pim_mroute_add_vif(ifp->ifindex, ifaddr, flags)) {
/* pim_mroute_add_vif reported error */
return -5;
}
@ -657,7 +659,8 @@ int pim_if_add_vif(struct interface *ifp)
/*
Update highest vif_index
*/
if (pim_ifp->mroute_vif_index > qpim_mroute_oif_highest_vif_index) {
if (pim_ifp->mroute_vif_index != PIM_OIF_PIM_REGISTER_VIF &&
pim_ifp->mroute_vif_index > qpim_mroute_oif_highest_vif_index) {
qpim_mroute_oif_highest_vif_index = pim_ifp->mroute_vif_index;
}
@ -1197,3 +1200,18 @@ void pim_if_update_assert_tracking_desired(struct interface *ifp)
pim_ifchannel_update_assert_tracking_desired(ch);
}
}
/*
* PIM wants to have an interface pointer for everything it does.
* The pimreg is a special interface that we have that is not
* quite an inteface but a VIF is created for it.
*/
void pim_if_create_pimreg (void)
{
if (!pim_regiface) {
pim_regiface = if_create("pimreg", strlen("pimreg"));
pim_regiface->ifindex = PIM_OIF_PIM_REGISTER_VIF;
pim_if_new(pim_regiface, 0, 0);
}
}

View file

@ -107,6 +107,7 @@ struct pim_interface {
uint32_t pim_ifstat_hello_recvfail;
};
extern struct interface *pim_regiface;
/*
if default_holdtime is set (>= 0), use it;
otherwise default_holdtime is 3.5 * hello_period
@ -166,4 +167,5 @@ void pim_if_update_join_desired(struct pim_interface *pim_ifp);
void pim_if_update_assert_tracking_desired(struct interface *ifp);
void pim_if_create_pimreg(void);
#endif /* PIM_IFACE_H */

View file

@ -194,9 +194,6 @@ int pim_channel_add_oif(struct channel_oil *channel_oil,
}
#endif
zassert(qpim_mroute_oif_highest_vif_index < MAXVIFS);
zassert(pim_ifp->mroute_vif_index <= qpim_mroute_oif_highest_vif_index);
/* Prevent single protocol from subscribing same interface to
channel (S,G) multiple times */
if (channel_oil->oif_flags[pim_ifp->mroute_vif_index] & proto_mask) {