summaryrefslogtreecommitdiffstats
path: root/sys/kern/uipc_sockbuf.c
diff options
context:
space:
mode:
authortanimura <tanimura@FreeBSD.org>2002-04-27 08:24:29 +0000
committertanimura <tanimura@FreeBSD.org>2002-04-27 08:24:29 +0000
commitdbb4756491715a06ce4578841f6eba43fc62fa70 (patch)
tree982f8e96f9de9e219deaa4ea138a8334dc086c3c /sys/kern/uipc_sockbuf.c
parentbcaaa89ad095d7a175927dd7e531fa6f278ed18e (diff)
downloadFreeBSD-src-dbb4756491715a06ce4578841f6eba43fc62fa70.zip
FreeBSD-src-dbb4756491715a06ce4578841f6eba43fc62fa70.tar.gz
Add a global sx sigio_lock to protect the pointer to the sigio object
of a socket. This avoids lock order reversal caused by locking a process in pgsigio(). sowakeup() and the callers of it (sowwakeup, soisconnected, etc.) now require sigio_lock to be locked. Provide sowwakeup_locked(), soisconnected_locked(), and so on in case where we have to modify a socket and wake up a process atomically.
Diffstat (limited to 'sys/kern/uipc_sockbuf.c')
-rw-r--r--sys/kern/uipc_sockbuf.c82
1 files changed, 69 insertions, 13 deletions
diff --git a/sys/kern/uipc_sockbuf.c b/sys/kern/uipc_sockbuf.c
index 1a8afd3..56fb012 100644
--- a/sys/kern/uipc_sockbuf.c
+++ b/sys/kern/uipc_sockbuf.c
@@ -107,11 +107,46 @@ soisconnecting(so)
}
void
+soisconnected_locked(so)
+ struct socket *so;
+{
+ struct socket *head = so->so_head;
+
+ SIGIO_ASSERT(SX_SLOCKED); /* XXX */
+ so->so_state &= ~(SS_ISCONNECTING|SS_ISDISCONNECTING|SS_ISCONFIRMING);
+ so->so_state |= SS_ISCONNECTED;
+ if (head && (so->so_state & SS_INCOMP)) {
+ if ((so->so_options & SO_ACCEPTFILTER) != 0) {
+ so->so_upcall = head->so_accf->so_accept_filter->accf_callback;
+ so->so_upcallarg = head->so_accf->so_accept_filter_arg;
+ so->so_rcv.sb_flags |= SB_UPCALL;
+ so->so_options &= ~SO_ACCEPTFILTER;
+ SIGIO_SUNLOCK(); /* XXX */
+ so->so_upcall(so, so->so_upcallarg, 0);
+ SIGIO_SLOCK();
+ return;
+ }
+ TAILQ_REMOVE(&head->so_incomp, so, so_list);
+ head->so_incqlen--;
+ so->so_state &= ~SS_INCOMP;
+ TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
+ so->so_state |= SS_COMP;
+ sorwakeup_locked(head);
+ wakeup_one(&head->so_timeo);
+ } else {
+ wakeup(&so->so_timeo);
+ sorwakeup_locked(so);
+ sowwakeup_locked(so);
+ }
+}
+
+void
soisconnected(so)
struct socket *so;
{
struct socket *head = so->so_head;
+ SIGIO_SLOCK();
so->so_state &= ~(SS_ISCONNECTING|SS_ISDISCONNECTING|SS_ISCONFIRMING);
so->so_state |= SS_ISCONNECTED;
if (head && (so->so_state & SS_INCOMP)) {
@@ -120,6 +155,7 @@ soisconnected(so)
so->so_upcallarg = head->so_accf->so_accept_filter_arg;
so->so_rcv.sb_flags |= SB_UPCALL;
so->so_options &= ~SO_ACCEPTFILTER;
+ SIGIO_SUNLOCK();
so->so_upcall(so, so->so_upcallarg, 0);
return;
}
@@ -129,13 +165,14 @@ soisconnected(so)
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
head->so_qlen++;
so->so_state |= SS_COMP;
- sorwakeup(head);
+ sorwakeup_locked(head);
wakeup_one(&head->so_timeo);
} else {
wakeup(&so->so_timeo);
- sorwakeup(so);
- sowwakeup(so);
+ sorwakeup_locked(so);
+ sowwakeup_locked(so);
}
+ SIGIO_SUNLOCK();
}
void
@@ -143,23 +180,36 @@ soisdisconnecting(so)
register struct socket *so;
{
+ SIGIO_SLOCK();
so->so_state &= ~SS_ISCONNECTING;
so->so_state |= (SS_ISDISCONNECTING|SS_CANTRCVMORE|SS_CANTSENDMORE);
wakeup((caddr_t)&so->so_timeo);
- sowwakeup(so);
- sorwakeup(so);
+ sowwakeup_locked(so);
+ sorwakeup_locked(so);
+ SIGIO_SUNLOCK();
}
void
-soisdisconnected(so)
+soisdisconnected_locked(so)
register struct socket *so;
{
+ SIGIO_ASSERT(SX_LOCKED);
so->so_state &= ~(SS_ISCONNECTING|SS_ISCONNECTED|SS_ISDISCONNECTING);
so->so_state |= (SS_CANTRCVMORE|SS_CANTSENDMORE|SS_ISDISCONNECTED);
wakeup((caddr_t)&so->so_timeo);
- sowwakeup(so);
- sorwakeup(so);
+ sowwakeup_locked(so);
+ sorwakeup_locked(so);
+}
+
+void
+soisdisconnected(so)
+ register struct socket *so;
+{
+
+ SIGIO_SLOCK();
+ soisdisconnected_locked(so);
+ SIGIO_SUNLOCK();
}
/*
@@ -215,9 +265,9 @@ sonewconn(head, connstatus)
head->so_incqlen++;
}
if (connstatus) {
- sorwakeup(head);
- wakeup((caddr_t)&head->so_timeo);
- so->so_state |= connstatus;
+ SIGIO_SLOCK();
+ sorwakeup_locked(head);
+ SIGIO_SUNLOCK();
}
return (so);
}
@@ -237,8 +287,10 @@ socantsendmore(so)
struct socket *so;
{
+ SIGIO_SLOCK();
so->so_state |= SS_CANTSENDMORE;
- sowwakeup(so);
+ sowwakeup_locked(so);
+ SIGIO_SUNLOCK();
}
void
@@ -246,8 +298,10 @@ socantrcvmore(so)
struct socket *so;
{
+ SIGIO_SLOCK();
so->so_state |= SS_CANTRCVMORE;
- sorwakeup(so);
+ sorwakeup_locked(so);
+ SIGIO_SUNLOCK();
}
/*
@@ -296,6 +350,8 @@ sowakeup(so, sb)
register struct socket *so;
register struct sockbuf *sb;
{
+ SIGIO_ASSERT(SX_LOCKED);
+
selwakeup(&sb->sb_sel);
sb->sb_flags &= ~SB_SEL;
if (sb->sb_flags & SB_WAIT) {
OpenPOWER on IntegriCloud