Date: Tue, 16 Dec 2008 23:06:36 +0000 (UTC) From: Kip Macy <kmacy@FreeBSD.org> To: src-committers@freebsd.org, svn-src-all@freebsd.org, svn-src-head@freebsd.org Subject: svn commit: r186196 - head/sys/netinet6 Message-ID: <200812162306.mBGN6agp049992@svn.freebsd.org>
next in thread | raw e-mail | index | archive | help
Author: kmacy Date: Tue Dec 16 23:06:36 2008 New Revision: 186196 URL: http://svn.freebsd.org/changeset/base/186196 Log: - Simplify handling of the deferring of mbuf transmit until after lle lock drop - add a couple of comments to clarify intent Modified: head/sys/netinet6/nd6.c Modified: head/sys/netinet6/nd6.c ============================================================================== --- head/sys/netinet6/nd6.c Tue Dec 16 22:16:34 2008 (r186195) +++ head/sys/netinet6/nd6.c Tue Dec 16 23:06:36 2008 (r186196) @@ -1407,7 +1407,7 @@ nd6_cache_lladdr(struct ifnet *ifp, stru int newstate = 0; uint16_t router = 0; struct sockaddr_in6 sin6; - struct mbuf *tail = NULL, *chain = NULL; + struct mbuf *chain = NULL; int static_route = 0; IF_AFDATA_UNLOCK_ASSERT(ifp); @@ -1526,11 +1526,14 @@ nd6_cache_lladdr(struct ifnet *ifp, stru * just set the 2nd argument as the * 1st one. */ - nd6_output_lle(ifp, ifp, m_hold, L3_ADDR_SIN6(ln), NULL, ln, &tail); - if ((tail != NULL) && chain == (NULL)) - chain = tail; + nd6_output_lle(ifp, ifp, m_hold, L3_ADDR_SIN6(ln), NULL, ln, &chain); } - if (chain) + /* + * If we have mbufs in the chain we need to do + * deferred transmit. Copy the address from the + * llentry before dropping the lock down below. + */ + if (chain != NULL) memcpy(&sin6, L3_ADDR_SIN6(ln), sizeof(sin6)); } } else if (ln->ln_state == ND6_LLINFO_INCOMPLETE) { @@ -1706,7 +1709,7 @@ nd6_output(struct ifnet *ifp, struct ifn int nd6_output_lle(struct ifnet *ifp, struct ifnet *origifp, struct mbuf *m0, struct sockaddr_in6 *dst, struct rtentry *rt0, struct llentry *lle, - struct mbuf **tail) + struct mbuf **chain) { INIT_VNET_INET6(curvnet); struct mbuf *m = m0; @@ -1720,7 +1723,7 @@ nd6_output_lle(struct ifnet *ifp, struct LLE_WLOCK_ASSERT(lle); - KASSERT(tail != NULL, (" lle locked but no tail pointer passed")); + KASSERT(chain != NULL, (" lle locked but no mbuf chain pointer passed")); } #endif if (IN6_IS_ADDR_MULTICAST(&dst->sin6_addr)) @@ -1888,12 +1891,25 @@ nd6_output_lle(struct ifnet *ifp, struct #ifdef MAC mac_netinet6_nd6_send(ifp, m); #endif + /* + * We were passed in a pointer to an lle with the lock held + * this means that we can't call if_output as we will + * recurse on the lle lock - so what we do is we create + * a list of mbufs to send and transmit them in the caller + * after the lock is dropped + */ if (lle != NULL) { - if (*tail == NULL) - *tail = m; - else { - (*tail)->m_nextpkt = m; - *tail = m; + if (*chain == NULL) + *chain = m; + else { + struct mbuf *m = *chain; + + /* + * append mbuf to end of deferred chain + */ + while (m->m_nextpkt != NULL) + m = m->m_nextpkt; + m->m_nextpkt = m; } return (error); }
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200812162306.mBGN6agp049992>