From 94a27aa4e3b5a378752e334c879f8e3d7680f1c9 Mon Sep 17 00:00:00 2001 From: martin Date: Mon, 23 Feb 2015 19:15:59 +0000 Subject: [PATCH] Rearange interface detachement slightly: before we free the INET6 specific per-interface data, make sure to call nd6_purge() with it to remove routing entries pointing to the going interface. When we should happen to call this function again later, with the data already gone, just return. Fixes PR kern/49682, ok: christos. --- sys/netinet6/in6.c | 6 +++--- sys/netinet6/in6_ifattach.c | 8 ++++---- sys/netinet6/nd6.c | 37 ++++++++++++++++++++++++++----------- sys/netinet6/nd6.h | 8 ++++---- sys/netinet6/nd6_nbr.c | 6 +++--- sys/netinet6/nd6_rtr.c | 20 ++++++++++++++------ 6 files changed, 54 insertions(+), 31 deletions(-) diff --git a/sys/netinet6/in6.c b/sys/netinet6/in6.c index bfa92bcf59ff..4297ac061c56 100644 --- a/sys/netinet6/in6.c +++ b/sys/netinet6/in6.c @@ -1,4 +1,4 @@ -/* $NetBSD: in6.c,v 1.181 2015/02/20 22:13:48 rjs Exp $ */ +/* $NetBSD: in6.c,v 1.182 2015/02/23 19:15:59 martin Exp $ */ /* $KAME: in6.c,v 1.198 2001/07/18 09:12:38 itojun Exp $ */ /* @@ -62,7 +62,7 @@ */ #include -__KERNEL_RCSID(0, "$NetBSD: in6.c,v 1.181 2015/02/20 22:13:48 rjs Exp $"); +__KERNEL_RCSID(0, "$NetBSD: in6.c,v 1.182 2015/02/23 19:15:59 martin Exp $"); #include "opt_inet.h" #include "opt_compat_netbsd.h" @@ -2318,7 +2318,7 @@ in6_domifdetach(struct ifnet *ifp, void *aux) { struct in6_ifextra *ext = (struct in6_ifextra *)aux; - nd6_ifdetach(ext->nd_ifinfo); + nd6_ifdetach(ifp, ext); free(ext->in6_ifstat, M_IFADDR); free(ext->icmp6_ifstat, M_IFADDR); scope6_ifdetach(ext->scope6_id); diff --git a/sys/netinet6/in6_ifattach.c b/sys/netinet6/in6_ifattach.c index 14e22a139502..cca24817ecd0 100644 --- a/sys/netinet6/in6_ifattach.c +++ b/sys/netinet6/in6_ifattach.c @@ -1,4 +1,4 @@ -/* $NetBSD: in6_ifattach.c,v 1.94 2014/11/14 17:34:23 maxv Exp $ */ +/* $NetBSD: in6_ifattach.c,v 1.95 2015/02/23 19:15:59 martin Exp $ */ /* $KAME: in6_ifattach.c,v 1.124 2001/07/18 08:32:51 jinmei Exp $ */ /* @@ -31,7 +31,7 @@ */ #include -__KERNEL_RCSID(0, "$NetBSD: in6_ifattach.c,v 1.94 2014/11/14 17:34:23 maxv Exp $"); +__KERNEL_RCSID(0, "$NetBSD: in6_ifattach.c,v 1.95 2015/02/23 19:15:59 martin Exp $"); #include #include @@ -848,7 +848,7 @@ in6_ifdetach(struct ifnet *ifp) ip6_mrouter_detach(ifp); /* remove neighbor management table */ - nd6_purge(ifp); + nd6_purge(ifp, NULL); /* XXX this code is duplicated in in6_purgeif() --dyoung */ /* nuke any of IPv6 addresses we have */ @@ -919,7 +919,7 @@ in6_ifdetach(struct ifnet *ifp) * prefixes after removing all addresses above. * (Or can we just delay calling nd6_purge until at this point?) */ - nd6_purge(ifp); + nd6_purge(ifp, NULL); } int diff --git a/sys/netinet6/nd6.c b/sys/netinet6/nd6.c index bc808d6b7214..fff21bfe33ed 100644 --- a/sys/netinet6/nd6.c +++ b/sys/netinet6/nd6.c @@ -1,4 +1,4 @@ -/* $NetBSD: nd6.c,v 1.157 2015/02/17 15:14:28 christos Exp $ */ +/* $NetBSD: nd6.c,v 1.158 2015/02/23 19:15:59 martin Exp $ */ /* $KAME: nd6.c,v 1.279 2002/06/08 11:16:51 itojun Exp $ */ /* @@ -31,7 +31,7 @@ */ #include -__KERNEL_RCSID(0, "$NetBSD: nd6.c,v 1.157 2015/02/17 15:14:28 christos Exp $"); +__KERNEL_RCSID(0, "$NetBSD: nd6.c,v 1.158 2015/02/23 19:15:59 martin Exp $"); #include "bridge.h" #include "carp.h" @@ -205,10 +205,11 @@ nd6_ifattach(struct ifnet *ifp) } void -nd6_ifdetach(struct nd_ifinfo *nd) +nd6_ifdetach(struct ifnet *ifp, struct in6_ifextra *ext) { - free(nd, M_IP6NDP); + nd6_purge(ifp, ext); + free(ext->nd_ifinfo, M_IP6NDP); } void @@ -556,7 +557,7 @@ nd6_timer(void *ignored_arg) TAILQ_FOREACH_SAFE(dr, &nd_defrouter, dr_entry, next_dr) { if (dr->expire && dr->expire < time_second) { - defrtrlist_del(dr); + defrtrlist_del(dr, NULL); } } @@ -746,12 +747,22 @@ nd6_accepts_rtadv(const struct nd_ifinfo *ndi) * ifp goes away. */ void -nd6_purge(struct ifnet *ifp) +nd6_purge(struct ifnet *ifp, struct in6_ifextra *ext) { struct llinfo_nd6 *ln, *nln; struct nd_defrouter *dr, *ndr; struct nd_prefix *pr, *npr; + /* + * During detach, the ND info might be already removed, but + * then is explitly passed as argument. + * Otherwise get it from ifp->if_afdata. + */ + if (ext == NULL) + ext = ifp->if_afdata[AF_INET6]; + if (ext == NULL) + return; + /* * Nuke default router list entries toward ifp. * We defer removal of default router list entries that is installed @@ -762,16 +773,20 @@ nd6_purge(struct ifnet *ifp) if (dr->installed) continue; - if (dr->ifp == ifp) - defrtrlist_del(dr); + if (dr->ifp == ifp) { + KASSERT(ext != NULL); + defrtrlist_del(dr, ext); + } } TAILQ_FOREACH_SAFE(dr, &nd_defrouter, dr_entry, ndr) { if (!dr->installed) continue; - if (dr->ifp == ifp) - defrtrlist_del(dr); + if (dr->ifp == ifp) { + KASSERT(ext != NULL); + defrtrlist_del(dr, ext); + } } /* Nuke prefix list entries toward ifp */ @@ -1797,7 +1812,7 @@ nd6_ioctl(u_long cmd, void *data, struct ifnet *ifp) s = splsoftnet(); defrouter_reset(); TAILQ_FOREACH_SAFE(drtr, &nd_defrouter, dr_entry, next) { - defrtrlist_del(drtr); + defrtrlist_del(drtr, NULL); } defrouter_select(); splx(s); diff --git a/sys/netinet6/nd6.h b/sys/netinet6/nd6.h index 875275ac2dd9..739a8feee787 100644 --- a/sys/netinet6/nd6.h +++ b/sys/netinet6/nd6.h @@ -1,4 +1,4 @@ -/* $NetBSD: nd6.h,v 1.61 2014/12/16 11:42:27 roy Exp $ */ +/* $NetBSD: nd6.h,v 1.62 2015/02/23 19:15:59 martin Exp $ */ /* $KAME: nd6.h,v 1.95 2002/06/08 11:31:06 itojun Exp $ */ /* @@ -407,7 +407,7 @@ union nd_opts { /* nd6.c */ void nd6_init(void); struct nd_ifinfo *nd6_ifattach(struct ifnet *); -void nd6_ifdetach(struct nd_ifinfo *); +void nd6_ifdetach(struct ifnet *, struct in6_ifextra *); int nd6_is_addr_neighbor(const struct sockaddr_in6 *, struct ifnet *); void nd6_option_init(void *, int, union nd_opts *); struct nd_opt_hdr *nd6_option(union nd_opts *); @@ -417,7 +417,7 @@ void nd6_rtmsg(int, struct rtentry *); void nd6_setmtu(struct ifnet *); void nd6_llinfo_settimer(struct llinfo_nd6 *, long); void nd6_timer(void *); -void nd6_purge(struct ifnet *); +void nd6_purge(struct ifnet *, struct in6_ifextra *); void nd6_nud_hint(struct rtentry *, struct in6_addr *, int); int nd6_resolve(struct ifnet *, struct rtentry *, struct mbuf *, struct sockaddr *, u_char *); @@ -454,7 +454,7 @@ void prelist_del(struct nd_prefix *); void defrouter_addreq(struct nd_defrouter *); void defrouter_reset(void); void defrouter_select(void); -void defrtrlist_del(struct nd_defrouter *); +void defrtrlist_del(struct nd_defrouter *, struct in6_ifextra *); void prelist_remove(struct nd_prefix *); int nd6_prelist_add(struct nd_prefixctl *, struct nd_defrouter *, struct nd_prefix **); diff --git a/sys/netinet6/nd6_nbr.c b/sys/netinet6/nd6_nbr.c index ea3d462750bc..dfc9a4bcb328 100644 --- a/sys/netinet6/nd6_nbr.c +++ b/sys/netinet6/nd6_nbr.c @@ -1,4 +1,4 @@ -/* $NetBSD: nd6_nbr.c,v 1.103 2014/12/16 11:42:27 roy Exp $ */ +/* $NetBSD: nd6_nbr.c,v 1.104 2015/02/23 19:15:59 martin Exp $ */ /* $KAME: nd6_nbr.c,v 1.61 2001/02/10 16:06:14 jinmei Exp $ */ /* @@ -31,7 +31,7 @@ */ #include -__KERNEL_RCSID(0, "$NetBSD: nd6_nbr.c,v 1.103 2014/12/16 11:42:27 roy Exp $"); +__KERNEL_RCSID(0, "$NetBSD: nd6_nbr.c,v 1.104 2015/02/23 19:15:59 martin Exp $"); #include "opt_inet.h" #include "opt_ipsec.h" @@ -804,7 +804,7 @@ nd6_na_input(struct mbuf *m, int off, int icmp6len) s = splsoftnet(); dr = defrouter_lookup(in6, rt->rt_ifp); if (dr) - defrtrlist_del(dr); + defrtrlist_del(dr, NULL); else if (!ip6_forwarding) { /* * Even if the neighbor is not in the default diff --git a/sys/netinet6/nd6_rtr.c b/sys/netinet6/nd6_rtr.c index ce6ac1154959..4c6f171ea84c 100644 --- a/sys/netinet6/nd6_rtr.c +++ b/sys/netinet6/nd6_rtr.c @@ -1,4 +1,4 @@ -/* $NetBSD: nd6_rtr.c,v 1.95 2014/12/16 11:42:27 roy Exp $ */ +/* $NetBSD: nd6_rtr.c,v 1.96 2015/02/23 19:15:59 martin Exp $ */ /* $KAME: nd6_rtr.c,v 1.95 2001/02/07 08:09:47 itojun Exp $ */ /* @@ -31,7 +31,7 @@ */ #include -__KERNEL_RCSID(0, "$NetBSD: nd6_rtr.c,v 1.95 2014/12/16 11:42:27 roy Exp $"); +__KERNEL_RCSID(0, "$NetBSD: nd6_rtr.c,v 1.96 2015/02/23 19:15:59 martin Exp $"); #include #include @@ -483,12 +483,20 @@ defrouter_lookup(const struct in6_addr *addr, struct ifnet *ifp) } void -defrtrlist_del(struct nd_defrouter *dr) +defrtrlist_del(struct nd_defrouter *dr, struct in6_ifextra *ext) { - struct nd_ifinfo *ndi = ND_IFINFO(dr->ifp); struct nd_defrouter *deldr = NULL; struct nd_prefix *pr; - struct in6_ifextra *ext = dr->ifp->if_afdata[AF_INET6]; + struct nd_ifinfo *ndi; + + if (ext == NULL) + ext = dr->ifp->if_afdata[AF_INET6]; + + /* detach already in progress, can not do anything */ + if (ext == NULL) + return; + + ndi = ext->nd_ifinfo; /* * Flush all the routing table entries that use the router @@ -749,7 +757,7 @@ defrtrlist_update(struct nd_defrouter *newdr) if ((dr = defrouter_lookup(&newdr->rtaddr, newdr->ifp)) != NULL) { /* entry exists */ if (newdr->rtlifetime == 0) { - defrtrlist_del(dr); + defrtrlist_del(dr, ext); dr = NULL; } else { int oldpref = rtpref(dr);