summaryrefslogtreecommitdiffstats
path: root/freebsd/sys/kern/uipc_sockbuf.c
diff options
context:
space:
mode:
authorSebastian Huber <sebastian.huber@embedded-brains.de>2018-08-07 14:56:50 +0200
committerSebastian Huber <sebastian.huber@embedded-brains.de>2018-09-21 10:29:37 +0200
commitc37f9fba70085fedc8eede7559489d2321393005 (patch)
tree042455ebf1fa89a277a825f72e1ed805d0b4d296 /freebsd/sys/kern/uipc_sockbuf.c
parentUpdate to FreeBSD head 2017-06-01 (diff)
downloadrtems-libbsd-c37f9fba70085fedc8eede7559489d2321393005.tar.bz2
Update to FreeBSD head 2017-08-01
Git mirror commit f5002f5e5f78cae9f0269d812dc0aedb0339312c. Update #3472.
Diffstat (limited to 'freebsd/sys/kern/uipc_sockbuf.c')
-rw-r--r--freebsd/sys/kern/uipc_sockbuf.c82
1 files changed, 73 insertions, 9 deletions
diff --git a/freebsd/sys/kern/uipc_sockbuf.c b/freebsd/sys/kern/uipc_sockbuf.c
index 04193c29..4b710a2c 100644
--- a/freebsd/sys/kern/uipc_sockbuf.c
+++ b/freebsd/sys/kern/uipc_sockbuf.c
@@ -316,14 +316,14 @@ sowakeup(struct socket *so, struct sockbuf *sb)
SOCKBUF_LOCK_ASSERT(sb);
- selwakeuppri(&sb->sb_sel, PSOCK);
- if (!SEL_WAITING(&sb->sb_sel))
+ selwakeuppri(sb->sb_sel, PSOCK);
+ if (!SEL_WAITING(sb->sb_sel))
sb->sb_flags &= ~SB_SEL;
if (sb->sb_flags & SB_WAIT) {
sb->sb_flags &= ~SB_WAIT;
wakeup(&sb->sb_acc);
}
- KNOTE_LOCKED(&sb->sb_sel.si_note, 0);
+ KNOTE_LOCKED(&sb->sb_sel->si_note, 0);
if (sb->sb_upcall != NULL) {
ret = sb->sb_upcall(so, sb->sb_upcallarg, M_NOWAIT);
if (ret == SU_ISCONNECTED) {
@@ -336,7 +336,7 @@ sowakeup(struct socket *so, struct sockbuf *sb)
if (sb->sb_flags & SB_AIO)
sowakeup_aio(so, sb);
SOCKBUF_UNLOCK(sb);
- if (ret == SU_ISCONNECTED)
+ if (ret == SU_ISCONNECTED && !(so->so_state & SS_ISDISCONNECTED))
soisconnected(so);
if ((so->so_state & SS_ASYNC) && so->so_sigio != NULL)
pgsigio(&so->so_sigio, SIGIO, 0);
@@ -457,14 +457,78 @@ sbreserve_locked(struct sockbuf *sb, u_long cc, struct socket *so,
}
int
-sbreserve(struct sockbuf *sb, u_long cc, struct socket *so,
- struct thread *td)
+sbsetopt(struct socket *so, int cmd, u_long cc)
{
+ struct sockbuf *sb;
+ short *flags;
+ u_int *hiwat, *lowat;
int error;
- SOCKBUF_LOCK(sb);
- error = sbreserve_locked(sb, cc, so, td);
- SOCKBUF_UNLOCK(sb);
+ SOCK_LOCK(so);
+ if (SOLISTENING(so)) {
+ switch (cmd) {
+ case SO_SNDLOWAT:
+ case SO_SNDBUF:
+ lowat = &so->sol_sbsnd_lowat;
+ hiwat = &so->sol_sbsnd_hiwat;
+ flags = &so->sol_sbsnd_flags;
+ break;
+ case SO_RCVLOWAT:
+ case SO_RCVBUF:
+ lowat = &so->sol_sbrcv_lowat;
+ hiwat = &so->sol_sbrcv_hiwat;
+ flags = &so->sol_sbrcv_flags;
+ break;
+ }
+ } else {
+ switch (cmd) {
+ case SO_SNDLOWAT:
+ case SO_SNDBUF:
+ sb = &so->so_snd;
+ break;
+ case SO_RCVLOWAT:
+ case SO_RCVBUF:
+ sb = &so->so_rcv;
+ break;
+ }
+ flags = &sb->sb_flags;
+ hiwat = &sb->sb_hiwat;
+ lowat = &sb->sb_lowat;
+ SOCKBUF_LOCK(sb);
+ }
+
+ error = 0;
+ switch (cmd) {
+ case SO_SNDBUF:
+ case SO_RCVBUF:
+ if (SOLISTENING(so)) {
+ if (cc > sb_max_adj) {
+ error = ENOBUFS;
+ break;
+ }
+ *hiwat = cc;
+ if (*lowat > *hiwat)
+ *lowat = *hiwat;
+ } else {
+ if (!sbreserve_locked(sb, cc, so, curthread))
+ error = ENOBUFS;
+ }
+ if (error == 0)
+ *flags &= ~SB_AUTOSIZE;
+ break;
+ case SO_SNDLOWAT:
+ case SO_RCVLOWAT:
+ /*
+ * Make sure the low-water is never greater than the
+ * high-water.
+ */
+ *lowat = (cc > *hiwat) ? *hiwat : cc;
+ break;
+ }
+
+ if (!SOLISTENING(so))
+ SOCKBUF_UNLOCK(sb);
+ SOCK_UNLOCK(so);
return (error);
}