Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/netinet6 Fix racy in6m_sol
details: https://anonhg.NetBSD.org/src/rev/b29358adacc0
branches: trunk
changeset: 351871:b29358adacc0
user: ozaki-r <ozaki-r%NetBSD.org@localhost>
date: Thu Mar 02 09:16:46 2017 +0000
description:
Fix racy in6m_sol
Relook up the entry instead of reusing it, which makes locking simple.
diffstat:
sys/netinet6/in6.c | 54 +++++++++++++++++++++++++++++++++----------------
sys/netinet6/in6_var.h | 4 ++-
sys/netinet6/mld6.c | 18 ++++++++++++++-
3 files changed, 55 insertions(+), 21 deletions(-)
diffs (181 lines):
diff -r 739f257da870 -r b29358adacc0 sys/netinet6/in6.c
--- a/sys/netinet6/in6.c Thu Mar 02 05:35:01 2017 +0000
+++ b/sys/netinet6/in6.c Thu Mar 02 09:16:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: in6.c,v 1.242 2017/03/02 05:27:39 ozaki-r Exp $ */
+/* $NetBSD: in6.c,v 1.243 2017/03/02 09:16:46 ozaki-r Exp $ */
/* $KAME: in6.c,v 1.198 2001/07/18 09:12:38 itojun Exp $ */
/*
@@ -62,7 +62,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: in6.c,v 1.242 2017/03/02 05:27:39 ozaki-r Exp $");
+__KERNEL_RCSID(0, "$NetBSD: in6.c,v 1.243 2017/03/02 09:16:46 ozaki-r Exp $");
#ifdef _KERNEL_OPT
#include "opt_inet.h"
@@ -779,8 +779,30 @@
}
static int
+in6_get_llsol_addr(struct in6_addr *llsol, struct ifnet *ifp,
+ struct in6_addr *ip6)
+{
+ int error;
+
+ memset(llsol, 0, sizeof(struct in6_addr));
+ llsol->s6_addr16[0] = htons(0xff02);
+ llsol->s6_addr32[1] = 0;
+ llsol->s6_addr32[2] = htonl(1);
+ llsol->s6_addr32[3] = ip6->s6_addr32[3];
+ llsol->s6_addr8[12] = 0xff;
+
+ error = in6_setscope(llsol, ifp, NULL);
+ if (error != 0) {
+ /* XXX: should not happen */
+ log(LOG_ERR, "%s: in6_setscope failed\n", __func__);
+ }
+
+ return error;
+}
+
+static int
in6_join_mcastgroups(struct in6_aliasreq *ifra, struct in6_ifaddr *ia,
- struct ifnet *ifp, struct in6_multi **in6m_sol, int flags)
+ struct ifnet *ifp, int flags)
{
int error;
struct sockaddr_in6 mltaddr, mltmask;
@@ -790,20 +812,10 @@
int dad_delay;
char ip6buf[INET6_ADDRSTRLEN];
- KASSERT(in6m_sol != NULL);
-
/* join solicited multicast addr for new host id */
- memset(&llsol, 0, sizeof(struct in6_addr));
- llsol.s6_addr16[0] = htons(0xff02);
- llsol.s6_addr32[1] = 0;
- llsol.s6_addr32[2] = htonl(1);
- llsol.s6_addr32[3] = ifra->ifra_addr.sin6_addr.s6_addr32[3];
- llsol.s6_addr8[12] = 0xff;
- if ((error = in6_setscope(&llsol, ifp, NULL)) != 0) {
- /* XXX: should not happen */
- log(LOG_ERR, "%s: in6_setscope failed\n", __func__);
+ error = in6_get_llsol_addr(&llsol, ifp, &ifra->ifra_addr.sin6_addr);
+ if (error != 0)
goto out;
- }
dad_delay = 0;
if ((flags & IN6_IFAUPDATE_DADDELAY)) {
/*
@@ -828,7 +840,6 @@
mutex_enter(&in6_ifaddr_lock);
LIST_INSERT_HEAD(&ia->ia6_memberships, imm, i6mm_chain);
mutex_exit(&in6_ifaddr_lock);
- *in6m_sol = imm->i6mm_maddr;
sockaddr_in6_init(&mltmask, &in6mask32, 0, 0, 0);
@@ -997,7 +1008,6 @@
int error = 0, hostIsNew = 0, plen = -1;
struct sockaddr_in6 dst6;
struct in6_addrlifetime *lt;
- struct in6_multi *in6m_sol = NULL;
int dad_delay, was_tentative;
struct in6_ifaddr *ia = iap ? *iap : NULL;
char ip6buf[INET6_ADDRSTRLEN];
@@ -1283,7 +1293,7 @@
/* join necessary multicast groups */
if ((ifp->if_flags & IFF_MULTICAST) != 0) {
- error = in6_join_mcastgroups(ifra, ia, ifp, &in6m_sol, flags);
+ error = in6_join_mcastgroups(ifra, ia, ifp, flags);
if (error != 0)
goto cleanup;
}
@@ -1306,6 +1316,8 @@
dad_delay = 0;
if ((flags & IN6_IFAUPDATE_DADDELAY)) {
+ struct in6_addr llsol;
+ struct in6_multi *in6m_sol = NULL;
/*
* We need to impose a delay before sending an NS
* for DAD. Check if we also needed a delay for the
@@ -1315,10 +1327,16 @@
* safe).
*/
mindelay = 0;
+ error = in6_get_llsol_addr(&llsol, ifp,
+ &ifra->ifra_addr.sin6_addr);
+ in6_multi_lock(RW_READER);
+ if (error == 0)
+ in6m_sol = in6_lookup_multi(&llsol, ifp);
if (in6m_sol != NULL &&
in6m_sol->in6m_state == MLD_REPORTPENDING) {
mindelay = in6m_sol->in6m_timer;
}
+ in6_multi_unlock();
maxdelay = MAX_RTR_SOLICITATION_DELAY * hz;
if (maxdelay - mindelay == 0)
dad_delay = 0;
diff -r 739f257da870 -r b29358adacc0 sys/netinet6/in6_var.h
--- a/sys/netinet6/in6_var.h Thu Mar 02 05:35:01 2017 +0000
+++ b/sys/netinet6/in6_var.h Thu Mar 02 09:16:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: in6_var.h,v 1.95 2017/03/02 05:27:39 ozaki-r Exp $ */
+/* $NetBSD: in6_var.h,v 1.96 2017/03/02 09:16:46 ozaki-r Exp $ */
/* $KAME: in6_var.h,v 1.81 2002/06/08 11:16:51 itojun Exp $ */
/*
@@ -681,6 +681,8 @@
void in6_init(void);
+void in6_multi_lock(int);
+void in6_multi_unlock(void);
struct in6_multi *
in6_lookup_multi(const struct in6_addr *, const struct ifnet *);
bool in6_multi_group(const struct in6_addr *, const struct ifnet *);
diff -r 739f257da870 -r b29358adacc0 sys/netinet6/mld6.c
--- a/sys/netinet6/mld6.c Thu Mar 02 05:35:01 2017 +0000
+++ b/sys/netinet6/mld6.c Thu Mar 02 09:16:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: mld6.c,v 1.86 2017/03/02 05:27:39 ozaki-r Exp $ */
+/* $NetBSD: mld6.c,v 1.87 2017/03/02 09:16:46 ozaki-r Exp $ */
/* $KAME: mld6.c,v 1.25 2001/01/16 14:14:18 itojun Exp $ */
/*
@@ -102,7 +102,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: mld6.c,v 1.86 2017/03/02 05:27:39 ozaki-r Exp $");
+__KERNEL_RCSID(0, "$NetBSD: mld6.c,v 1.87 2017/03/02 09:16:46 ozaki-r Exp $");
#ifdef _KERNEL_OPT
#include "opt_inet.h"
@@ -901,6 +901,20 @@
rw_exit(&in6_multilock);
}
+void
+in6_multi_lock(int op)
+{
+
+ rw_enter(&in6_multilock, op);
+}
+
+void
+in6_multi_unlock(void)
+{
+
+ rw_exit(&in6_multilock);
+}
+
struct in6_multi_mship *
in6_joingroup(struct ifnet *ifp, struct in6_addr *addr,
int *errorp, int timer)
Home |
Main Index |
Thread Index |
Old Index