Change from using if_delmulti() to if_delmulti_ifma() as it simplifies the code
and is safe to use if the ifp has disappeared. Suggested by: bms
This commit is contained in:
parent
c8285e3723
commit
d74fd34568
@ -453,6 +453,7 @@ lacp_port_create(struct lagg_port *lgp)
|
|||||||
lp->lp_ifp = ifp;
|
lp->lp_ifp = ifp;
|
||||||
lp->lp_lagg = lgp;
|
lp->lp_lagg = lgp;
|
||||||
lp->lp_lsc = lsc;
|
lp->lp_lsc = lsc;
|
||||||
|
lp->lp_ifma = rifma;
|
||||||
|
|
||||||
LIST_INSERT_HEAD(&lsc->lsc_ports, lp, lp_next);
|
LIST_INSERT_HEAD(&lsc->lsc_ports, lp, lp_next);
|
||||||
|
|
||||||
@ -471,9 +472,7 @@ void
|
|||||||
lacp_port_destroy(struct lagg_port *lgp)
|
lacp_port_destroy(struct lagg_port *lgp)
|
||||||
{
|
{
|
||||||
struct lacp_port *lp = LACP_PORT(lgp);
|
struct lacp_port *lp = LACP_PORT(lgp);
|
||||||
struct ifnet *ifp = lgp->lp_ifp;
|
int i;
|
||||||
struct sockaddr_dl sdl;
|
|
||||||
int i, error;
|
|
||||||
|
|
||||||
LAGG_LOCK_ASSERT(lgp->lp_lagg);
|
LAGG_LOCK_ASSERT(lgp->lp_lagg);
|
||||||
|
|
||||||
@ -486,18 +485,7 @@ lacp_port_destroy(struct lagg_port *lgp)
|
|||||||
lacp_unselect(lp);
|
lacp_unselect(lp);
|
||||||
lgp->lp_flags &= ~LAGG_PORT_DISABLED;
|
lgp->lp_flags &= ~LAGG_PORT_DISABLED;
|
||||||
|
|
||||||
bzero((char *)&sdl, sizeof(sdl));
|
if_delmulti_ifma(lp->lp_ifma);
|
||||||
sdl.sdl_len = sizeof(sdl);
|
|
||||||
sdl.sdl_family = AF_LINK;
|
|
||||||
sdl.sdl_index = ifp->if_index;
|
|
||||||
sdl.sdl_type = IFT_ETHER;
|
|
||||||
sdl.sdl_alen = ETHER_ADDR_LEN;
|
|
||||||
|
|
||||||
bcopy(ðermulticastaddr_slowprotocols,
|
|
||||||
LLADDR(&sdl), ETHER_ADDR_LEN);
|
|
||||||
error = if_delmulti(ifp, (struct sockaddr *)&sdl);
|
|
||||||
if (error)
|
|
||||||
printf("%s: DELMULTI failed on %s\n", __func__, lgp->lp_ifname);
|
|
||||||
|
|
||||||
LIST_REMOVE(lp, lp_next);
|
LIST_REMOVE(lp, lp_next);
|
||||||
free(lp, M_DEVBUF);
|
free(lp, M_DEVBUF);
|
||||||
|
@ -190,6 +190,7 @@ struct lacp_port {
|
|||||||
int lp_flags;
|
int lp_flags;
|
||||||
u_int lp_media; /* XXX redundant */
|
u_int lp_media; /* XXX redundant */
|
||||||
int lp_timer[LACP_NTIMER];
|
int lp_timer[LACP_NTIMER];
|
||||||
|
struct ifmultiaddr *lp_ifma;
|
||||||
|
|
||||||
struct lacp_aggregator *lp_aggregator;
|
struct lacp_aggregator *lp_aggregator;
|
||||||
};
|
};
|
||||||
|
@ -838,48 +838,16 @@ out:
|
|||||||
static int
|
static int
|
||||||
lagg_ether_setmulti(struct lagg_softc *sc)
|
lagg_ether_setmulti(struct lagg_softc *sc)
|
||||||
{
|
{
|
||||||
struct ifnet *trifp = sc->sc_ifp;
|
|
||||||
struct ifnet *ifp;
|
|
||||||
struct ifmultiaddr *ifma, *rifma = NULL;
|
|
||||||
struct lagg_port *lp;
|
struct lagg_port *lp;
|
||||||
struct lagg_mc *mc;
|
|
||||||
struct sockaddr_dl sdl;
|
|
||||||
int error;
|
|
||||||
|
|
||||||
LAGG_LOCK_ASSERT(sc);
|
LAGG_LOCK_ASSERT(sc);
|
||||||
|
|
||||||
bzero((char *)&sdl, sizeof(sdl));
|
|
||||||
sdl.sdl_len = sizeof(sdl);
|
|
||||||
sdl.sdl_family = AF_LINK;
|
|
||||||
sdl.sdl_type = IFT_ETHER;
|
|
||||||
sdl.sdl_alen = ETHER_ADDR_LEN;
|
|
||||||
|
|
||||||
/* First, remove any existing filter entries. */
|
/* First, remove any existing filter entries. */
|
||||||
lagg_ether_purgemulti(sc);
|
lagg_ether_purgemulti(sc);
|
||||||
|
|
||||||
/* Now program new ones. */
|
|
||||||
TAILQ_FOREACH(ifma, &trifp->if_multiaddrs, ifma_link) {
|
|
||||||
if (ifma->ifma_addr->sa_family != AF_LINK)
|
|
||||||
continue;
|
|
||||||
mc = malloc(sizeof(struct lagg_mc), M_DEVBUF, M_NOWAIT);
|
|
||||||
if (mc == NULL)
|
|
||||||
return (ENOMEM);
|
|
||||||
bcopy(LLADDR((struct sockaddr_dl *)ifma->ifma_addr),
|
|
||||||
(char *)&mc->mc_addr, ETHER_ADDR_LEN);
|
|
||||||
SLIST_INSERT_HEAD(&sc->sc_mc_head, mc, mc_entries);
|
|
||||||
bcopy(LLADDR((struct sockaddr_dl *)ifma->ifma_addr),
|
|
||||||
LLADDR(&sdl), ETHER_ADDR_LEN);
|
|
||||||
|
|
||||||
/* do all the ports */
|
|
||||||
SLIST_FOREACH(lp, &sc->sc_ports, lp_entries) {
|
SLIST_FOREACH(lp, &sc->sc_ports, lp_entries) {
|
||||||
ifp = lp->lp_ifp;
|
lagg_ether_cmdmulti(lp, 1);
|
||||||
sdl.sdl_index = ifp->if_index;
|
|
||||||
error = if_addmulti(ifp, (struct sockaddr *)&sdl, &rifma);
|
|
||||||
if (error)
|
|
||||||
return (error);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -887,9 +855,10 @@ static int
|
|||||||
lagg_ether_cmdmulti(struct lagg_port *lp, int set)
|
lagg_ether_cmdmulti(struct lagg_port *lp, int set)
|
||||||
{
|
{
|
||||||
struct lagg_softc *sc = lp->lp_lagg;
|
struct lagg_softc *sc = lp->lp_lagg;
|
||||||
struct ifnet *ifp = lp->lp_ifp;;
|
struct ifnet *ifp = lp->lp_ifp;
|
||||||
|
struct ifnet *trifp = sc->sc_ifp;
|
||||||
struct lagg_mc *mc;
|
struct lagg_mc *mc;
|
||||||
struct ifmultiaddr *rifma = NULL;
|
struct ifmultiaddr *ifma, *rifma = NULL;
|
||||||
struct sockaddr_dl sdl;
|
struct sockaddr_dl sdl;
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
@ -902,18 +871,27 @@ lagg_ether_cmdmulti(struct lagg_port *lp, int set)
|
|||||||
sdl.sdl_alen = ETHER_ADDR_LEN;
|
sdl.sdl_alen = ETHER_ADDR_LEN;
|
||||||
sdl.sdl_index = ifp->if_index;
|
sdl.sdl_index = ifp->if_index;
|
||||||
|
|
||||||
SLIST_FOREACH(mc, &sc->sc_mc_head, mc_entries) {
|
if (set) {
|
||||||
bcopy((char *)&mc->mc_addr, LLADDR(&sdl), ETHER_ADDR_LEN);
|
TAILQ_FOREACH(ifma, &trifp->if_multiaddrs, ifma_link) {
|
||||||
|
if (ifma->ifma_addr->sa_family != AF_LINK)
|
||||||
|
continue;
|
||||||
|
bcopy(LLADDR((struct sockaddr_dl *)ifma->ifma_addr),
|
||||||
|
LLADDR(&sdl), ETHER_ADDR_LEN);
|
||||||
|
|
||||||
if (set)
|
|
||||||
error = if_addmulti(ifp, (struct sockaddr *)&sdl, &rifma);
|
error = if_addmulti(ifp, (struct sockaddr *)&sdl, &rifma);
|
||||||
else
|
if (error)
|
||||||
error = if_delmulti(ifp, (struct sockaddr *)&sdl);
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
printf("cmdmulti error on %s, set = %d\n",
|
|
||||||
ifp->if_xname, set);
|
|
||||||
return (error);
|
return (error);
|
||||||
|
mc = malloc(sizeof(struct lagg_mc), M_DEVBUF, M_NOWAIT);
|
||||||
|
if (mc == NULL)
|
||||||
|
return (ENOMEM);
|
||||||
|
mc->mc_ifma = rifma;
|
||||||
|
SLIST_INSERT_HEAD(&lp->lp_mc_head, mc, mc_entries);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
while ((mc = SLIST_FIRST(&lp->lp_mc_head)) != NULL) {
|
||||||
|
SLIST_REMOVE(&lp->lp_mc_head, mc, lagg_mc, mc_entries);
|
||||||
|
if_delmulti_ifma(mc->mc_ifma);
|
||||||
|
free(mc, M_DEVBUF);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (0);
|
return (0);
|
||||||
@ -923,18 +901,11 @@ static void
|
|||||||
lagg_ether_purgemulti(struct lagg_softc *sc)
|
lagg_ether_purgemulti(struct lagg_softc *sc)
|
||||||
{
|
{
|
||||||
struct lagg_port *lp;
|
struct lagg_port *lp;
|
||||||
struct lagg_mc *mc;
|
|
||||||
|
|
||||||
LAGG_LOCK_ASSERT(sc);
|
LAGG_LOCK_ASSERT(sc);
|
||||||
|
|
||||||
/* remove from ports */
|
|
||||||
SLIST_FOREACH(lp, &sc->sc_ports, lp_entries)
|
SLIST_FOREACH(lp, &sc->sc_ports, lp_entries)
|
||||||
lagg_ether_cmdmulti(lp, 0);
|
lagg_ether_cmdmulti(lp, 0);
|
||||||
|
|
||||||
while ((mc = SLIST_FIRST(&sc->sc_mc_head)) != NULL) {
|
|
||||||
SLIST_REMOVE(&sc->sc_mc_head, mc, lagg_mc, mc_entries);
|
|
||||||
free(mc, M_DEVBUF);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Handle a ref counted flag that should be set on the lagg port as well */
|
/* Handle a ref counted flag that should be set on the lagg port as well */
|
||||||
|
@ -136,11 +136,7 @@ struct lagg_lb {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct lagg_mc {
|
struct lagg_mc {
|
||||||
union {
|
struct ifmultiaddr *mc_ifma;
|
||||||
struct ether_multi *mcu_enm;
|
|
||||||
} mc_u;
|
|
||||||
struct sockaddr_storage mc_addr;
|
|
||||||
|
|
||||||
SLIST_ENTRY(lagg_mc) mc_entries;
|
SLIST_ENTRY(lagg_mc) mc_entries;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -156,8 +152,6 @@ struct lagg_softc {
|
|||||||
SLIST_HEAD(__tplhd, lagg_port) sc_ports; /* list of interfaces */
|
SLIST_HEAD(__tplhd, lagg_port) sc_ports; /* list of interfaces */
|
||||||
SLIST_ENTRY(lagg_softc) sc_entries;
|
SLIST_ENTRY(lagg_softc) sc_entries;
|
||||||
|
|
||||||
SLIST_HEAD(__mclhd, lagg_mc) sc_mc_head; /* multicast addresses */
|
|
||||||
|
|
||||||
/* lagg protocol callbacks */
|
/* lagg protocol callbacks */
|
||||||
int (*sc_detach)(struct lagg_softc *);
|
int (*sc_detach)(struct lagg_softc *);
|
||||||
int (*sc_start)(struct lagg_softc *, struct mbuf *);
|
int (*sc_start)(struct lagg_softc *, struct mbuf *);
|
||||||
@ -183,6 +177,8 @@ struct lagg_port {
|
|||||||
void *lh_cookie; /* if state hook */
|
void *lh_cookie; /* if state hook */
|
||||||
caddr_t lp_psc; /* protocol data */
|
caddr_t lp_psc; /* protocol data */
|
||||||
|
|
||||||
|
SLIST_HEAD(__mclhd, lagg_mc) lp_mc_head; /* multicast addresses */
|
||||||
|
|
||||||
/* Redirected callbacks */
|
/* Redirected callbacks */
|
||||||
int (*lp_ioctl)(struct ifnet *, u_long, caddr_t);
|
int (*lp_ioctl)(struct ifnet *, u_long, caddr_t);
|
||||||
int (*lp_output)(struct ifnet *, struct mbuf *, struct sockaddr *,
|
int (*lp_output)(struct ifnet *, struct mbuf *, struct sockaddr *,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user