Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys Ensure to call if_mcast_op with holding IFNET_LOCK
details: https://anonhg.NetBSD.org/src/rev/01a0e659183c
branches: trunk
changeset: 358159:01a0e659183c
user: ozaki-r <ozaki-r%NetBSD.org@localhost>
date: Fri Dec 15 04:03:46 2017 +0000
description:
Ensure to call if_mcast_op with holding IFNET_LOCK
Note that CARP doesn't deal with IFNET_LOCK yet.
diffstat:
sys/net/if.c | 25 ++++++++++++++--
sys/net/if.h | 3 +-
sys/net/if_vlan.c | 10 +++++-
sys/netinet/in.c | 7 +++-
sys/netinet/in_pcb.c | 7 +++-
sys/netinet/ip_output.c | 22 +++++++++++---
sys/netinet6/in6.c | 11 +++++--
sys/netinet6/in6_pcb.c | 11 ++++++-
sys/netinet6/ip6_output.c | 21 ++++++++++++--
sys/netinet6/nd6.c | 69 ++++++++++++++++++++++++++++++++++++++++------
sys/netinet6/nd6_rtr.c | 6 ++-
11 files changed, 158 insertions(+), 34 deletions(-)
diffs (truncated from 553 to 300 lines):
diff -r 0503d7089196 -r 01a0e659183c sys/net/if.c
--- a/sys/net/if.c Fri Dec 15 02:24:22 2017 +0000
+++ b/sys/net/if.c Fri Dec 15 04:03:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if.c,v 1.414 2017/12/14 05:46:54 ozaki-r Exp $ */
+/* $NetBSD: if.c,v 1.415 2017/12/15 04:03:46 ozaki-r Exp $ */
/*-
* Copyright (c) 1999, 2000, 2001, 2008 The NetBSD Foundation, Inc.
@@ -90,7 +90,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if.c,v 1.414 2017/12/14 05:46:54 ozaki-r Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if.c,v 1.415 2017/12/15 04:03:46 ozaki-r Exp $");
#if defined(_KERNEL_OPT)
#include "opt_inet.h"
@@ -2721,12 +2721,27 @@
psref_release(psref, &ifp->if_psref, ifnet_psref_class);
}
+/*
+ * Return ifp having idx. Return NULL if not found. Normally if_byindex
+ * should be used.
+ */
+ifnet_t *
+_if_byindex(u_int idx)
+{
+
+ return (__predict_true(idx < if_indexlim)) ? ifindex2ifnet[idx] : NULL;
+}
+
+/*
+ * Return ifp having idx. Return NULL if not found or the found ifp is
+ * already deactivated.
+ */
ifnet_t *
if_byindex(u_int idx)
{
ifnet_t *ifp;
- ifp = (__predict_true(idx < if_indexlim)) ? ifindex2ifnet[idx] : NULL;
+ ifp = _if_byindex(idx);
if (ifp != NULL && if_is_deactivated(ifp))
ifp = NULL;
return ifp;
@@ -3570,6 +3585,10 @@
int rc;
struct ifreq ifr;
+ /* CARP still doesn't deal with the lock yet */
+#if !defined(NCARP) || (NCARP == 0)
+ KASSERT(IFNET_LOCKED(ifp));
+#endif
if (ifp->if_mcastop != NULL)
rc = (*ifp->if_mcastop)(ifp, cmd, sa);
else {
diff -r 0503d7089196 -r 01a0e659183c sys/net/if.h
--- a/sys/net/if.h Fri Dec 15 02:24:22 2017 +0000
+++ b/sys/net/if.h Fri Dec 15 04:03:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if.h,v 1.253 2017/12/11 03:29:20 ozaki-r Exp $ */
+/* $NetBSD: if.h,v 1.254 2017/12/15 04:03:46 ozaki-r Exp $ */
/*-
* Copyright (c) 1999, 2000, 2001 The NetBSD Foundation, Inc.
@@ -1026,6 +1026,7 @@
struct ifnet *ifunit(const char *);
struct ifnet *if_get(const char *, struct psref *);
ifnet_t *if_byindex(u_int);
+ifnet_t *_if_byindex(u_int);
ifnet_t *if_get_byindex(u_int, struct psref *);
ifnet_t *if_get_bylla(const void *, unsigned char, struct psref *);
void if_put(const struct ifnet *, struct psref *);
diff -r 0503d7089196 -r 01a0e659183c sys/net/if_vlan.c
--- a/sys/net/if_vlan.c Fri Dec 15 02:24:22 2017 +0000
+++ b/sys/net/if_vlan.c Fri Dec 15 04:03:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if_vlan.c,v 1.119 2017/12/11 03:29:20 ozaki-r Exp $ */
+/* $NetBSD: if_vlan.c,v 1.120 2017/12/15 04:03:46 ozaki-r Exp $ */
/*-
* Copyright (c) 2000, 2001 The NetBSD Foundation, Inc.
@@ -78,7 +78,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_vlan.c,v 1.119 2017/12/11 03:29:20 ozaki-r Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_vlan.c,v 1.120 2017/12/15 04:03:46 ozaki-r Exp $");
#ifdef _KERNEL_OPT
#include "opt_inet.h"
@@ -1160,7 +1160,9 @@
mib = ifv->ifv_mib;
KERNEL_LOCK_UNLESS_IFP_MPSAFE(mib->ifvm_p);
+ IFNET_LOCK(mib->ifvm_p);
error = if_mcast_op(mib->ifvm_p, SIOCADDMULTI, sa);
+ IFNET_UNLOCK(mib->ifvm_p);
KERNEL_UNLOCK_UNLESS_IFP_MPSAFE(mib->ifvm_p);
if (error != 0)
@@ -1201,7 +1203,9 @@
/* We no longer use this multicast address. Tell parent so. */
mib = ifv->ifv_mib;
+ IFNET_LOCK(mib->ifvm_p);
error = if_mcast_op(mib->ifvm_p, SIOCDELMULTI, sa);
+ IFNET_UNLOCK(mib->ifvm_p);
if (error == 0) {
/* And forget about this address. */
@@ -1236,8 +1240,10 @@
}
while ((mc = LIST_FIRST(&ifv->ifv_mc_listhead)) != NULL) {
+ IFNET_LOCK(mib->ifvm_p);
(void)if_mcast_op(mib->ifvm_p, SIOCDELMULTI,
(const struct sockaddr *)&mc->mc_addr);
+ IFNET_UNLOCK(mib->ifvm_p);
LIST_REMOVE(mc, mc_entries);
free(mc, M_DEVBUF);
}
diff -r 0503d7089196 -r 01a0e659183c sys/netinet/in.c
--- a/sys/netinet/in.c Fri Dec 15 02:24:22 2017 +0000
+++ b/sys/netinet/in.c Fri Dec 15 04:03:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: in.c,v 1.210 2017/11/17 07:37:12 ozaki-r Exp $ */
+/* $NetBSD: in.c,v 1.211 2017/12/15 04:03:46 ozaki-r Exp $ */
/*
* Copyright (C) 1995, 1996, 1997, and 1998 WIDE Project.
@@ -91,7 +91,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: in.c,v 1.210 2017/11/17 07:37:12 ozaki-r Exp $");
+__KERNEL_RCSID(0, "$NetBSD: in.c,v 1.211 2017/12/15 04:03:46 ozaki-r Exp $");
#include "arp.h"
@@ -912,11 +912,14 @@
void
in_purgeif(struct ifnet *ifp) /* MUST be called at splsoftnet() */
{
+
+ IFNET_LOCK(ifp);
if_purgeaddrs(ifp, AF_INET, in_purgeaddr);
igmp_purgeif(ifp); /* manipulates pools */
#ifdef MROUTING
ip_mrouter_detach(ifp);
#endif
+ IFNET_UNLOCK(ifp);
}
/*
diff -r 0503d7089196 -r 01a0e659183c sys/netinet/in_pcb.c
--- a/sys/netinet/in_pcb.c Fri Dec 15 02:24:22 2017 +0000
+++ b/sys/netinet/in_pcb.c Fri Dec 15 04:03:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: in_pcb.c,v 1.179 2017/08/10 04:31:58 ryo Exp $ */
+/* $NetBSD: in_pcb.c,v 1.180 2017/12/15 04:03:46 ozaki-r Exp $ */
/*
* Copyright (C) 1995, 1996, 1997, and 1998 WIDE Project.
@@ -93,7 +93,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: in_pcb.c,v 1.179 2017/08/10 04:31:58 ryo Exp $");
+__KERNEL_RCSID(0, "$NetBSD: in_pcb.c,v 1.180 2017/12/15 04:03:46 ozaki-r Exp $");
#ifdef _KERNEL_OPT
#include "opt_inet.h"
@@ -777,7 +777,10 @@
need_unlock = true;
}
+ /* IFNET_LOCK must be taken after solock */
+ IFNET_LOCK(ifp);
in_purgeifmcast(inp->inp_moptions, ifp);
+ IFNET_UNLOCK(ifp);
if (need_unlock)
inp_unlock(inp);
diff -r 0503d7089196 -r 01a0e659183c sys/netinet/ip_output.c
--- a/sys/netinet/ip_output.c Fri Dec 15 02:24:22 2017 +0000
+++ b/sys/netinet/ip_output.c Fri Dec 15 04:03:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: ip_output.c,v 1.286 2017/12/11 05:47:18 ryo Exp $ */
+/* $NetBSD: ip_output.c,v 1.287 2017/12/15 04:03:46 ozaki-r Exp $ */
/*
* Copyright (C) 1995, 1996, 1997, and 1998 WIDE Project.
@@ -91,7 +91,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: ip_output.c,v 1.286 2017/12/11 05:47:18 ryo Exp $");
+__KERNEL_RCSID(0, "$NetBSD: ip_output.c,v 1.287 2017/12/15 04:03:46 ozaki-r Exp $");
#ifdef _KERNEL_OPT
#include "opt_inet.h"
@@ -1771,7 +1771,10 @@
* Everything looks good; add a new record to the multicast
* address list for the given interface.
*/
- if ((imo->imo_membership[i] = in_addmulti(&ia, ifp)) == NULL) {
+ IFNET_LOCK(ifp);
+ imo->imo_membership[i] = in_addmulti(&ia, ifp);
+ IFNET_UNLOCK(ifp);
+ if (imo->imo_membership[i] == NULL) {
error = ENOBUFS;
goto out;
}
@@ -1830,7 +1833,9 @@
* Give up the multicast address record to which the
* membership points.
*/
+ IFNET_LOCK(ifp);
in_delmulti(imo->imo_membership[i]);
+ IFNET_UNLOCK(ifp);
/*
* Remove the gap in the membership array.
@@ -2023,8 +2028,15 @@
/* The owner of imo (inp) should be protected by solock */
if (imo != NULL) {
- for (i = 0; i < imo->imo_num_memberships; ++i)
- in_delmulti(imo->imo_membership[i]);
+ for (i = 0; i < imo->imo_num_memberships; ++i) {
+ struct in_multi *inm = imo->imo_membership[i];
+ struct ifnet *ifp = inm->inm_ifp;
+ IFNET_LOCK(ifp);
+ in_delmulti(inm);
+ /* ifp should not leave thanks to solock */
+ IFNET_UNLOCK(ifp);
+ }
+
kmem_intr_free(imo, sizeof(*imo));
}
}
diff -r 0503d7089196 -r 01a0e659183c sys/netinet6/in6.c
--- a/sys/netinet6/in6.c Fri Dec 15 02:24:22 2017 +0000
+++ b/sys/netinet6/in6.c Fri Dec 15 04:03:46 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: in6.c,v 1.254 2017/11/23 07:09:20 ozaki-r Exp $ */
+/* $NetBSD: in6.c,v 1.255 2017/12/15 04:03: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.254 2017/11/23 07:09:20 ozaki-r Exp $");
+__KERNEL_RCSID(0, "$NetBSD: in6.c,v 1.255 2017/12/15 04:03:46 ozaki-r Exp $");
#ifdef _KERNEL_OPT
#include "opt_inet.h"
@@ -1385,6 +1385,7 @@
struct in6_multi_mship *imm;
KASSERT(!ifa_held(ifa));
+ KASSERT(IFNET_LOCKED(ifp));
ifa->ifa_flags |= IFA_DESTROYING;
@@ -1400,12 +1401,14 @@
/*
* leave from multicast groups we have joined for the interface
*/
+ again:
mutex_enter(&in6_ifaddr_lock);
while ((imm = LIST_FIRST(&ia->ia6_memberships)) != NULL) {
LIST_REMOVE(imm, i6mm_chain);
mutex_exit(&in6_ifaddr_lock);
+ KASSERT(imm->i6mm_maddr->in6m_ifp == ifp);
in6_leavegroup(imm);
- mutex_enter(&in6_ifaddr_lock);
+ goto again;
}
mutex_exit(&in6_ifaddr_lock);
@@ -1456,7 +1459,9 @@
in6_purgeif(struct ifnet *ifp)
{
+ IFNET_LOCK(ifp);
in6_ifdetach(ifp);
+ IFNET_UNLOCK(ifp);
}
void
diff -r 0503d7089196 -r 01a0e659183c sys/netinet6/in6_pcb.c
--- a/sys/netinet6/in6_pcb.c Fri Dec 15 02:24:22 2017 +0000
+++ b/sys/netinet6/in6_pcb.c Fri Dec 15 04:03:46 2017 +0000
@@ -1,4 +1,4 @@
Home |
Main Index |
Thread Index |
Old Index