diff options
author | mvs <mvs@cvs.openbsd.org> | 2021-05-01 16:13:14 +0000 |
---|---|---|
committer | mvs <mvs@cvs.openbsd.org> | 2021-05-01 16:13:14 +0000 |
commit | 93bb6fd7698cf751377d0213f6abf5cef1277386 (patch) | |
tree | efc9f1a02d3a275e5c204b6f83f978cfb9e9ae66 /sys/net/rtsock.c | |
parent | 58b3d2ba77817cf3ac54cf921bf87b95844589c4 (diff) |
Implement per-socket `so_lock' rwlock(9) and use it to protect routing
(PF_ROUTE) sockets. This can be done because we have no cases where one
thread should lock two sockets simultaneously.
Against the previous version rtm_senddesync_timer() execution was moved
to process context.
Also this time `so_lock' used for routing sockets only but in the future
it will be used to other socket types too.
tested by claudio@
ok claudio@ bluhm@
Diffstat (limited to 'sys/net/rtsock.c')
-rw-r--r-- | sys/net/rtsock.c | 51 |
1 files changed, 35 insertions, 16 deletions
diff --git a/sys/net/rtsock.c b/sys/net/rtsock.c index ac7c163128e..3f8985d6d27 100644 --- a/sys/net/rtsock.c +++ b/sys/net/rtsock.c @@ -1,4 +1,4 @@ -/* $OpenBSD: rtsock.c,v 1.311 2021/04/26 08:21:36 claudio Exp $ */ +/* $OpenBSD: rtsock.c,v 1.312 2021/05/01 16:13:13 mvs Exp $ */ /* $NetBSD: rtsock.c,v 1.18 1996/03/29 00:32:10 cgd Exp $ */ /* @@ -143,7 +143,7 @@ int rt_setsource(unsigned int, struct sockaddr *); /* * Locks used to protect struct members * I immutable after creation - * sK solock (kernel lock) + * s solock */ struct rtpcb { struct socket *rop_socket; /* [I] */ @@ -151,12 +151,12 @@ struct rtpcb { SRPL_ENTRY(rtpcb) rop_list; struct refcnt rop_refcnt; struct timeout rop_timeout; - unsigned int rop_msgfilter; /* [sK] */ - unsigned int rop_flagfilter; /* [sK] */ - unsigned int rop_flags; /* [sK] */ - u_int rop_rtableid; /* [sK] */ + unsigned int rop_msgfilter; /* [s] */ + unsigned int rop_flagfilter; /* [s] */ + unsigned int rop_flags; /* [s] */ + u_int rop_rtableid; /* [s] */ unsigned short rop_proto; /* [I] */ - u_char rop_priority; /* [sK] */ + u_char rop_priority; /* [s] */ }; #define sotortpcb(so) ((struct rtpcb *)(so)->so_pcb) @@ -188,7 +188,7 @@ route_prinit(void) rw_init(&rtptable.rtp_lk, "rtsock"); SRPL_INIT(&rtptable.rtp_list); pool_init(&rtpcb_pool, sizeof(struct rtpcb), 0, - IPL_NONE, PR_WAITOK, "rtpcb", NULL); + IPL_SOFTNET, PR_WAITOK, "rtpcb", NULL); } void @@ -312,7 +312,8 @@ route_attach(struct socket *so, int proto) rop = pool_get(&rtpcb_pool, PR_WAITOK|PR_ZERO); so->so_pcb = rop; /* Init the timeout structure */ - timeout_set(&rop->rop_timeout, rtm_senddesync_timer, so); + timeout_set_flags(&rop->rop_timeout, rtm_senddesync_timer, so, + TIMEOUT_PROC); refcnt_init(&rop->rop_refcnt); rop->rop_socket = so; @@ -352,9 +353,13 @@ route_detach(struct socket *so) rop_list); rw_exit(&rtptable.rtp_lk); + sounlock(so, SL_LOCKED); + /* wait for all references to drop */ refcnt_finalize(&rop->rop_refcnt, "rtsockrefs"); + solock(so); + so->so_pcb = NULL; KASSERT((so->so_state & SS_NOFDREF) == 0); pool_put(&rtpcb_pool, rop); @@ -676,7 +681,7 @@ route_output(struct mbuf *m, struct socket *so, struct sockaddr *dstaddr, struct rtentry *rt = NULL; struct rt_addrinfo info; struct ifnet *ifp; - int len, seq, error = 0; + int len, seq, useloopback, error = 0; u_int tableid; u_int8_t prio; u_char vers, type; @@ -686,6 +691,16 @@ route_output(struct mbuf *m, struct socket *so, struct sockaddr *dstaddr, return (ENOBUFS); if ((m->m_flags & M_PKTHDR) == 0) panic("route_output"); + + useloopback = so->so_options & SO_USELOOPBACK; + + /* + * The socket can't be closed concurrently because the file + * descriptor reference is still held. + */ + + sounlock(so, SL_LOCKED); + len = m->m_pkthdr.len; if (len < offsetof(struct rt_msghdr, rtm_hdrlen) + 1 || len != mtod(m, struct rt_msghdr *)->rtm_msglen) { @@ -859,13 +874,10 @@ route_output(struct mbuf *m, struct socket *so, struct sockaddr *dstaddr, /* * Check to see if we don't want our own messages. */ - if (!(so->so_options & SO_USELOOPBACK)) { - if (rtptable.rtp_count <= 1) { + if (!useloopback) { + if (rtptable.rtp_count == 0) { /* no other listener and no loopback of messages */ -fail: - free(rtm, M_RTABLE, len); - m_freem(m); - return (error); + goto fail; } } if (m_copyback(m, 0, len, rtm, M_NOWAIT)) { @@ -877,6 +889,13 @@ fail: if (m) route_input(m, so, info.rti_info[RTAX_DST] ? info.rti_info[RTAX_DST]->sa_family : AF_UNSPEC); + solock(so); + + return (error); +fail: + free(rtm, M_RTABLE, len); + m_freem(m); + solock(so); return (error); } |