summaryrefslogtreecommitdiff
path: root/sys/net
diff options
context:
space:
mode:
authormvs <mvs@cvs.openbsd.org>2021-02-27 11:44:49 +0000
committermvs <mvs@cvs.openbsd.org>2021-02-27 11:44:49 +0000
commit169c7c6cb320625de800acad4c319082e1ad369c (patch)
tree10cad931d8baa79c3f0bef97ea626f05297c0302 /sys/net
parent23b4db207bdec76016ab5fae5b3b6e009045b6ad (diff)
Rework route_input() and rtm_sendup(). While we perform foreach loop
in route_input() we drop solock() after we checked socket state. We pass mbuf(9) to this socket at next loops, while it referenced as `last'. Socket's state could be changed by concurrent thread while it's not locked. Since we perform socket's checks and output in same iteration, the logic which prevents mbuf(9) chain copy for the last socket in list was removed. ok bluhm@ claudio@
Diffstat (limited to 'sys/net')
-rw-r--r--sys/net/rtsock.c44
1 files changed, 12 insertions, 32 deletions
diff --git a/sys/net/rtsock.c b/sys/net/rtsock.c
index f7bb66b6c97..6fb80d9e4da 100644
--- a/sys/net/rtsock.c
+++ b/sys/net/rtsock.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: rtsock.c,v 1.306 2021/02/25 02:48:21 dlg Exp $ */
+/* $OpenBSD: rtsock.c,v 1.307 2021/02/27 11:44:48 mvs Exp $ */
/* $NetBSD: rtsock.c,v 1.18 1996/03/29 00:32:10 cgd Exp $ */
/*
@@ -118,7 +118,7 @@ int route_arp_conflict(struct rtentry *, struct rt_addrinfo *);
int route_cleargateway(struct rtentry *, void *, unsigned int);
void rtm_senddesync_timer(void *);
void rtm_senddesync(struct socket *);
-int rtm_sendup(struct socket *, struct mbuf *, int);
+int rtm_sendup(struct socket *, struct mbuf *);
int rtm_getifa(struct rt_addrinfo *, unsigned int);
int rtm_output(struct rt_msghdr *, struct rtentry **, struct rt_addrinfo *,
@@ -489,7 +489,6 @@ route_input(struct mbuf *m0, struct socket *so0, sa_family_t sa_family)
struct rtpcb *rop;
struct rt_msghdr *rtm;
struct mbuf *m = m0;
- struct socket *last = NULL;
struct srp_ref sr;
int s;
@@ -519,11 +518,8 @@ route_input(struct mbuf *m0, struct socket *so0, sa_family_t sa_family)
*/
if ((so0 == so && !(so0->so_options & SO_USELOOPBACK)) ||
!(so->so_state & SS_ISCONNECTED) ||
- (so->so_state & SS_CANTRCVMORE)) {
-next:
- sounlock(so, s);
- continue;
- }
+ (so->so_state & SS_CANTRCVMORE))
+ goto next;
/* filter messages that the process does not want */
rtm = mtod(m, struct rt_msghdr *);
@@ -568,43 +564,27 @@ next:
*/
if ((rop->rop_flags & ROUTECB_FLAG_FLUSH) != 0)
goto next;
- sounlock(so, s);
- if (last) {
- s = solock(last);
- rtm_sendup(last, m, 1);
- sounlock(last, s);
- refcnt_rele_wake(&sotortpcb(last)->rop_refcnt);
- }
- /* keep a reference for last */
- refcnt_take(&rop->rop_refcnt);
- last = rop->rop_socket;
+ rtm_sendup(so, m);
+next:
+ sounlock(so, s);
}
SRPL_LEAVE(&sr);
- if (last) {
- s = solock(last);
- rtm_sendup(last, m, 0);
- sounlock(last, s);
- refcnt_rele_wake(&sotortpcb(last)->rop_refcnt);
- } else
- m_freem(m);
+ m_freem(m);
}
int
-rtm_sendup(struct socket *so, struct mbuf *m0, int more)
+rtm_sendup(struct socket *so, struct mbuf *m0)
{
struct rtpcb *rop = sotortpcb(so);
struct mbuf *m;
soassertlocked(so);
- if (more) {
- m = m_copym(m0, 0, M_COPYALL, M_NOWAIT);
- if (m == NULL)
- return (ENOMEM);
- } else
- m = m0;
+ m = m_copym(m0, 0, M_COPYALL, M_NOWAIT);
+ if (m == NULL)
+ return (ENOMEM);
if (sbspace(so, &so->so_rcv) < (2 * MSIZE) ||
sbappendaddr(so, &so->so_rcv, &route_src, m, NULL) == 0) {