Back out my lats commit of locking down a socket, it conflicts with hsu's work.
Requested by: hsu
This commit is contained in:
parent
c82593cf2c
commit
4cc20ab1f0
@ -116,17 +116,15 @@ svr4_sys_read(td, uap)
|
|||||||
|
|
||||||
if (fp->f_type == DTYPE_SOCKET) {
|
if (fp->f_type == DTYPE_SOCKET) {
|
||||||
so = (struct socket *)fp->f_data;
|
so = (struct socket *)fp->f_data;
|
||||||
SOCK_LOCK(so);
|
|
||||||
so_state = so->so_state;
|
|
||||||
#if defined(GROTTY_READ_HACK)
|
|
||||||
so->so_state &= ~SS_NBIO;
|
|
||||||
#endif
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
DPRINTF(("fd %d is a socket\n", SCARG(uap, fd)));
|
DPRINTF(("fd %d is a socket\n", SCARG(uap, fd)));
|
||||||
if (so_state & SS_ASYNC) {
|
if (so->so_state & SS_ASYNC) {
|
||||||
DPRINTF(("fd %d is an ASYNC socket!\n", SCARG(uap, fd)));
|
DPRINTF(("fd %d is an ASYNC socket!\n", SCARG(uap, fd)));
|
||||||
}
|
}
|
||||||
DPRINTF(("Here are its flags: 0x%x\n", so_state));
|
DPRINTF(("Here are its flags: 0x%x\n", so->so_state));
|
||||||
|
#if defined(GROTTY_READ_HACK)
|
||||||
|
so_state = so->so_state;
|
||||||
|
so->so_state &= ~SS_NBIO;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
rv = read(td, &ra);
|
rv = read(td, &ra);
|
||||||
@ -142,9 +140,7 @@ svr4_sys_read(td, uap)
|
|||||||
|
|
||||||
#if defined(GROTTY_READ_HACK)
|
#if defined(GROTTY_READ_HACK)
|
||||||
if (so) { /* We've already checked to see if this is a socket */
|
if (so) { /* We've already checked to see if this is a socket */
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state = so_state;
|
so->so_state = so_state;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
fdrop(fp, td);
|
fdrop(fp, td);
|
||||||
|
@ -29,12 +29,10 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/fcntl.h>
|
#include <sys/proc.h>
|
||||||
#include <sys/file.h>
|
#include <sys/file.h>
|
||||||
#include <sys/filedesc.h>
|
#include <sys/filedesc.h>
|
||||||
#include <sys/lock.h>
|
#include <sys/fcntl.h>
|
||||||
#include <sys/proc.h>
|
|
||||||
#include <sys/mutex.h>
|
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#include <sys/socketvar.h>
|
#include <sys/socketvar.h>
|
||||||
#include <sys/systm.h>
|
#include <sys/systm.h>
|
||||||
@ -95,7 +93,6 @@ svr4_sys_ioctl(td, uap)
|
|||||||
char c;
|
char c;
|
||||||
int num;
|
int num;
|
||||||
int argsiz;
|
int argsiz;
|
||||||
int sostate;
|
|
||||||
|
|
||||||
svr4_decode_cmd(SCARG(uap, com), dir, &c, &num, &argsiz);
|
svr4_decode_cmd(SCARG(uap, com), dir, &c, &num, &argsiz);
|
||||||
|
|
||||||
@ -116,10 +113,7 @@ svr4_sys_ioctl(td, uap)
|
|||||||
#if defined(DEBUG_SVR4)
|
#if defined(DEBUG_SVR4)
|
||||||
if (fp->f_type == DTYPE_SOCKET) {
|
if (fp->f_type == DTYPE_SOCKET) {
|
||||||
struct socket *so = (struct socket *)fp->f_data;
|
struct socket *so = (struct socket *)fp->f_data;
|
||||||
SOCK_LOCK(so);
|
DPRINTF(("<<< IN: so_state = 0x%x\n", so->so_state));
|
||||||
sostate = so->so_state;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
DPRINTF(("<<< IN: so_state = 0x%x\n", sostate));
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -164,10 +158,7 @@ svr4_sys_ioctl(td, uap)
|
|||||||
struct socket *so;
|
struct socket *so;
|
||||||
|
|
||||||
so = (struct socket *)fp->f_data;
|
so = (struct socket *)fp->f_data;
|
||||||
SOCK_LOCK(so);
|
DPRINTF((">>> OUT: so_state = 0x%x\n", so->so_state));
|
||||||
sostate = so->so_state;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
DPRINTF((">>> OUT: so_state = 0x%x\n", sostate));
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
error = (*fun)(fp, td, retval, SCARG(uap, fd), cmd, SCARG(uap, data));
|
error = (*fun)(fp, td, retval, SCARG(uap, fd), cmd, SCARG(uap, data));
|
||||||
|
@ -203,33 +203,25 @@ fifo_open(ap)
|
|||||||
}
|
}
|
||||||
fip->fi_readers = fip->fi_writers = 0;
|
fip->fi_readers = fip->fi_writers = 0;
|
||||||
wso->so_snd.sb_lowat = PIPE_BUF;
|
wso->so_snd.sb_lowat = PIPE_BUF;
|
||||||
SOCK_LOCK(rso);
|
|
||||||
rso->so_state |= SS_CANTRCVMORE;
|
rso->so_state |= SS_CANTRCVMORE;
|
||||||
SOCK_UNLOCK(rso);
|
|
||||||
}
|
}
|
||||||
if (ap->a_mode & FREAD) {
|
if (ap->a_mode & FREAD) {
|
||||||
fip->fi_readers++;
|
fip->fi_readers++;
|
||||||
if (fip->fi_readers == 1) {
|
if (fip->fi_readers == 1) {
|
||||||
SOCK_LOCK(fip->fi_writesock);
|
|
||||||
fip->fi_writesock->so_state &= ~SS_CANTSENDMORE;
|
fip->fi_writesock->so_state &= ~SS_CANTSENDMORE;
|
||||||
if (fip->fi_writers > 0) {
|
if (fip->fi_writers > 0) {
|
||||||
wakeup((caddr_t)&fip->fi_writers);
|
wakeup((caddr_t)&fip->fi_writers);
|
||||||
sowwakeup(fip->fi_writesock);
|
sowwakeup(fip->fi_writesock);
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(fip->fi_writesock);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (ap->a_mode & FWRITE) {
|
if (ap->a_mode & FWRITE) {
|
||||||
fip->fi_writers++;
|
fip->fi_writers++;
|
||||||
if (fip->fi_writers == 1) {
|
if (fip->fi_writers == 1) {
|
||||||
SOCK_LOCK(fip->fi_readsock);
|
|
||||||
fip->fi_readsock->so_state &= ~SS_CANTRCVMORE;
|
fip->fi_readsock->so_state &= ~SS_CANTRCVMORE;
|
||||||
SOCK_UNLOCK(fip->fi_readsock);
|
|
||||||
if (fip->fi_readers > 0) {
|
if (fip->fi_readers > 0) {
|
||||||
wakeup((caddr_t)&fip->fi_readers);
|
wakeup((caddr_t)&fip->fi_readers);
|
||||||
SOCK_LOCK(fip->fi_writesock);
|
|
||||||
sorwakeup(fip->fi_writesock);
|
sorwakeup(fip->fi_writesock);
|
||||||
SOCK_UNLOCK(fip->fi_writesock);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -290,21 +282,15 @@ fifo_read(ap)
|
|||||||
#endif
|
#endif
|
||||||
if (uio->uio_resid == 0)
|
if (uio->uio_resid == 0)
|
||||||
return (0);
|
return (0);
|
||||||
if (ap->a_ioflag & IO_NDELAY) {
|
if (ap->a_ioflag & IO_NDELAY)
|
||||||
SOCK_LOCK(rso);
|
|
||||||
rso->so_state |= SS_NBIO;
|
rso->so_state |= SS_NBIO;
|
||||||
SOCK_UNLOCK(rso);
|
|
||||||
}
|
|
||||||
startresid = uio->uio_resid;
|
startresid = uio->uio_resid;
|
||||||
VOP_UNLOCK(ap->a_vp, 0, td);
|
VOP_UNLOCK(ap->a_vp, 0, td);
|
||||||
error = soreceive(rso, (struct sockaddr **)0, uio, (struct mbuf **)0,
|
error = soreceive(rso, (struct sockaddr **)0, uio, (struct mbuf **)0,
|
||||||
(struct mbuf **)0, (int *)0);
|
(struct mbuf **)0, (int *)0);
|
||||||
vn_lock(ap->a_vp, LK_EXCLUSIVE | LK_RETRY, td);
|
vn_lock(ap->a_vp, LK_EXCLUSIVE | LK_RETRY, td);
|
||||||
if (ap->a_ioflag & IO_NDELAY) {
|
if (ap->a_ioflag & IO_NDELAY)
|
||||||
SOCK_LOCK(rso);
|
|
||||||
rso->so_state &= ~SS_NBIO;
|
rso->so_state &= ~SS_NBIO;
|
||||||
SOCK_UNLOCK(rso);
|
|
||||||
}
|
|
||||||
return (error);
|
return (error);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -329,20 +315,14 @@ fifo_write(ap)
|
|||||||
if (ap->a_uio->uio_rw != UIO_WRITE)
|
if (ap->a_uio->uio_rw != UIO_WRITE)
|
||||||
panic("fifo_write mode");
|
panic("fifo_write mode");
|
||||||
#endif
|
#endif
|
||||||
if (ap->a_ioflag & IO_NDELAY) {
|
if (ap->a_ioflag & IO_NDELAY)
|
||||||
SOCK_LOCK(wso);
|
|
||||||
wso->so_state |= SS_NBIO;
|
wso->so_state |= SS_NBIO;
|
||||||
SOCK_UNLOCK(wso);
|
|
||||||
}
|
|
||||||
VOP_UNLOCK(ap->a_vp, 0, td);
|
VOP_UNLOCK(ap->a_vp, 0, td);
|
||||||
error = sosend(wso, (struct sockaddr *)0, ap->a_uio, 0,
|
error = sosend(wso, (struct sockaddr *)0, ap->a_uio, 0,
|
||||||
(struct mbuf *)0, 0, td);
|
(struct mbuf *)0, 0, td);
|
||||||
vn_lock(ap->a_vp, LK_EXCLUSIVE | LK_RETRY, td);
|
vn_lock(ap->a_vp, LK_EXCLUSIVE | LK_RETRY, td);
|
||||||
if (ap->a_ioflag & IO_NDELAY) {
|
if (ap->a_ioflag & IO_NDELAY)
|
||||||
SOCK_LOCK(wso);
|
|
||||||
wso->so_state &= ~SS_NBIO;
|
wso->so_state &= ~SS_NBIO;
|
||||||
SOCK_UNLOCK(wso);
|
|
||||||
}
|
|
||||||
return (error);
|
return (error);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -432,13 +412,10 @@ filt_fiforead(struct knote *kn, long hint)
|
|||||||
struct socket *so = (struct socket *)kn->kn_hook;
|
struct socket *so = (struct socket *)kn->kn_hook;
|
||||||
|
|
||||||
kn->kn_data = so->so_rcv.sb_cc;
|
kn->kn_data = so->so_rcv.sb_cc;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTRCVMORE) {
|
if (so->so_state & SS_CANTRCVMORE) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
kn->kn_flags |= EV_EOF;
|
kn->kn_flags |= EV_EOF;
|
||||||
return (1);
|
return (1);
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
kn->kn_flags &= ~EV_EOF;
|
kn->kn_flags &= ~EV_EOF;
|
||||||
return (kn->kn_data > 0);
|
return (kn->kn_data > 0);
|
||||||
}
|
}
|
||||||
@ -459,13 +436,10 @@ filt_fifowrite(struct knote *kn, long hint)
|
|||||||
struct socket *so = (struct socket *)kn->kn_hook;
|
struct socket *so = (struct socket *)kn->kn_hook;
|
||||||
|
|
||||||
kn->kn_data = sbspace(&so->so_snd);
|
kn->kn_data = sbspace(&so->so_snd);
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTSENDMORE) {
|
if (so->so_state & SS_CANTSENDMORE) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
kn->kn_flags |= EV_EOF;
|
kn->kn_flags |= EV_EOF;
|
||||||
return (1);
|
return (1);
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
kn->kn_flags &= ~EV_EOF;
|
kn->kn_flags &= ~EV_EOF;
|
||||||
return (kn->kn_data >= so->so_snd.sb_lowat);
|
return (kn->kn_data >= so->so_snd.sb_lowat);
|
||||||
}
|
}
|
||||||
|
@ -43,26 +43,24 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/fcntl.h>
|
|
||||||
#include <sys/file.h>
|
|
||||||
#include <sys/filedesc.h>
|
|
||||||
#include <sys/kernel.h>
|
|
||||||
#include <sys/lock.h>
|
|
||||||
#include <sys/malloc.h>
|
|
||||||
#include <sys/mbuf.h>
|
|
||||||
#include <sys/mount.h>
|
|
||||||
#include <sys/mutex.h>
|
|
||||||
#include <sys/namei.h>
|
|
||||||
#include <sys/proc.h>
|
|
||||||
#include <sys/socket.h>
|
|
||||||
#include <sys/socketvar.h>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <sys/systm.h>
|
#include <sys/systm.h>
|
||||||
#include <sys/sysproto.h>
|
#include <sys/sysproto.h>
|
||||||
|
#include <sys/kernel.h>
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
|
#include <sys/proc.h>
|
||||||
|
#include <sys/filedesc.h>
|
||||||
|
#include <sys/vnode.h>
|
||||||
|
#include <sys/fcntl.h>
|
||||||
|
#include <sys/file.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/mount.h>
|
||||||
|
#include <sys/malloc.h>
|
||||||
|
#include <sys/namei.h>
|
||||||
|
#include <sys/mbuf.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <sys/socketvar.h>
|
||||||
#include <sys/un.h>
|
#include <sys/un.h>
|
||||||
#include <sys/unpcb.h>
|
#include <sys/unpcb.h>
|
||||||
#include <sys/vnode.h>
|
|
||||||
#include <fs/portalfs/portal.h>
|
#include <fs/portalfs/portal.h>
|
||||||
|
|
||||||
static int portal_fileid = PORTAL_ROOTFILEID+1;
|
static int portal_fileid = PORTAL_ROOTFILEID+1;
|
||||||
@ -184,12 +182,8 @@ portal_connect(so, so2)
|
|||||||
if (so->so_type != so2->so_type)
|
if (so->so_type != so2->so_type)
|
||||||
return (EPROTOTYPE);
|
return (EPROTOTYPE);
|
||||||
|
|
||||||
SOCK_LOCK(so2);
|
if ((so2->so_options & SO_ACCEPTCONN) == 0)
|
||||||
if ((so2->so_options & SO_ACCEPTCONN) == 0) {
|
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
return (ECONNREFUSED);
|
return (ECONNREFUSED);
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
|
|
||||||
if ((so3 = sonewconn(so2, 0)) == 0)
|
if ((so3 = sonewconn(so2, 0)) == 0)
|
||||||
return (ECONNREFUSED);
|
return (ECONNREFUSED);
|
||||||
@ -286,17 +280,14 @@ portal_open(ap)
|
|||||||
* and keep polling the reference count. XXX.
|
* and keep polling the reference count. XXX.
|
||||||
*/
|
*/
|
||||||
s = splnet();
|
s = splnet();
|
||||||
SOCK_LOCK(so);
|
|
||||||
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
||||||
if (fmp->pm_server->f_count == 1) {
|
if (fmp->pm_server->f_count == 1) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ECONNREFUSED;
|
error = ECONNREFUSED;
|
||||||
splx(s);
|
splx(s);
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
(void) msleep((caddr_t) &so->so_timeo, SOCK_MTX(so), PSOCK, "portalcon", 5 * hz);
|
(void) tsleep((caddr_t) &so->so_timeo, PSOCK, "portalcon", 5 * hz);
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
|
|
||||||
if (so->so_error) {
|
if (so->so_error) {
|
||||||
|
@ -1777,22 +1777,19 @@ fgetsock(struct thread *td, int fd, struct socket **spp, u_int *fflagp)
|
|||||||
*spp = (struct socket *)fp->f_data;
|
*spp = (struct socket *)fp->f_data;
|
||||||
if (fflagp)
|
if (fflagp)
|
||||||
*fflagp = fp->f_flag;
|
*fflagp = fp->f_flag;
|
||||||
SOCK_LOCK(*spp);
|
|
||||||
soref(*spp);
|
soref(*spp);
|
||||||
SOCK_UNLOCK(*spp);
|
|
||||||
}
|
}
|
||||||
FILEDESC_UNLOCK(td->td_proc->p_fd);
|
FILEDESC_UNLOCK(td->td_proc->p_fd);
|
||||||
return(error);
|
return(error);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Drop the reference count on the the socket and release the lock.
|
* Drop the reference count on the the socket and XXX release the SX lock in
|
||||||
* The last reference closes the socket. The socket must be unlocked.
|
* the future. The last reference closes the socket.
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
fputsock(struct socket *so)
|
fputsock(struct socket *so)
|
||||||
{
|
{
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorele(so);
|
sorele(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -104,16 +104,13 @@ soo_ioctl(fp, cmd, data, td)
|
|||||||
switch (cmd) {
|
switch (cmd) {
|
||||||
|
|
||||||
case FIONBIO:
|
case FIONBIO:
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (*(int *)data)
|
if (*(int *)data)
|
||||||
so->so_state |= SS_NBIO;
|
so->so_state |= SS_NBIO;
|
||||||
else
|
else
|
||||||
so->so_state &= ~SS_NBIO;
|
so->so_state &= ~SS_NBIO;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
|
|
||||||
case FIOASYNC:
|
case FIOASYNC:
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (*(int *)data) {
|
if (*(int *)data) {
|
||||||
so->so_state |= SS_ASYNC;
|
so->so_state |= SS_ASYNC;
|
||||||
so->so_rcv.sb_flags |= SB_ASYNC;
|
so->so_rcv.sb_flags |= SB_ASYNC;
|
||||||
@ -123,7 +120,6 @@ soo_ioctl(fp, cmd, data, td)
|
|||||||
so->so_rcv.sb_flags &= ~SB_ASYNC;
|
so->so_rcv.sb_flags &= ~SB_ASYNC;
|
||||||
so->so_snd.sb_flags &= ~SB_ASYNC;
|
so->so_snd.sb_flags &= ~SB_ASYNC;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
|
|
||||||
case FIONREAD:
|
case FIONREAD:
|
||||||
@ -145,9 +141,7 @@ soo_ioctl(fp, cmd, data, td)
|
|||||||
return (0);
|
return (0);
|
||||||
|
|
||||||
case SIOCATMARK:
|
case SIOCATMARK:
|
||||||
SOCK_LOCK(so);
|
|
||||||
*(int *)data = (so->so_state&SS_RCVATMARK) != 0;
|
*(int *)data = (so->so_state&SS_RCVATMARK) != 0;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
@ -187,13 +181,11 @@ soo_stat(fp, ub, td)
|
|||||||
* If SS_CANTRCVMORE is set, but there's still data left in the
|
* If SS_CANTRCVMORE is set, but there's still data left in the
|
||||||
* receive buffer, the socket is still readable.
|
* receive buffer, the socket is still readable.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_CANTRCVMORE) == 0 ||
|
if ((so->so_state & SS_CANTRCVMORE) == 0 ||
|
||||||
so->so_rcv.sb_cc != 0)
|
so->so_rcv.sb_cc != 0)
|
||||||
ub->st_mode |= S_IRUSR | S_IRGRP | S_IROTH;
|
ub->st_mode |= S_IRUSR | S_IRGRP | S_IROTH;
|
||||||
if ((so->so_state & SS_CANTSENDMORE) == 0)
|
if ((so->so_state & SS_CANTSENDMORE) == 0)
|
||||||
ub->st_mode |= S_IWUSR | S_IWGRP | S_IWOTH;
|
ub->st_mode |= S_IWUSR | S_IWGRP | S_IWOTH;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
ub->st_size = so->so_rcv.sb_cc;
|
ub->st_size = so->so_rcv.sb_cc;
|
||||||
ub->st_uid = so->so_cred->cr_uid;
|
ub->st_uid = so->so_cred->cr_uid;
|
||||||
ub->st_gid = so->so_cred->cr_gid;
|
ub->st_gid = so->so_cred->cr_gid;
|
||||||
|
@ -35,13 +35,11 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/domain.h>
|
|
||||||
#include <sys/kernel.h>
|
|
||||||
#include <sys/lock.h>
|
|
||||||
#include <sys/mbuf.h>
|
|
||||||
#include <sys/mutex.h>
|
|
||||||
#include <sys/protosw.h>
|
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
|
#include <sys/protosw.h>
|
||||||
|
#include <sys/domain.h>
|
||||||
|
#include <sys/mbuf.h>
|
||||||
|
#include <sys/kernel.h>
|
||||||
#include <sys/socketvar.h>
|
#include <sys/socketvar.h>
|
||||||
#include <sys/systm.h>
|
#include <sys/systm.h>
|
||||||
#include <vm/uma.h>
|
#include <vm/uma.h>
|
||||||
|
@ -102,56 +102,71 @@ soisconnecting(so)
|
|||||||
register struct socket *so;
|
register struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
so->so_state &= ~(SS_ISCONNECTED|SS_ISDISCONNECTING);
|
so->so_state &= ~(SS_ISCONNECTED|SS_ISDISCONNECTING);
|
||||||
so->so_state |= SS_ISCONNECTING;
|
so->so_state |= SS_ISCONNECTING;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
soisconnected_locked(so)
|
||||||
|
struct socket *so;
|
||||||
|
{
|
||||||
|
struct socket *head = so->so_head;
|
||||||
|
|
||||||
|
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;
|
||||||
|
so->so_upcall(so, so->so_upcallarg, 0);
|
||||||
|
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);
|
||||||
|
head->so_qlen++;
|
||||||
|
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
|
void
|
||||||
soisconnected(so)
|
soisconnected(so)
|
||||||
struct socket *so;
|
struct socket *so;
|
||||||
{
|
{
|
||||||
struct socket *head;
|
struct socket *head = so->so_head;
|
||||||
so_upcall_t *upcp;
|
|
||||||
void *upcarg;
|
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
head = so->so_head;
|
|
||||||
so->so_state &= ~(SS_ISCONNECTING|SS_ISDISCONNECTING|SS_ISCONFIRMING);
|
so->so_state &= ~(SS_ISCONNECTING|SS_ISDISCONNECTING|SS_ISCONFIRMING);
|
||||||
so->so_state |= SS_ISCONNECTED;
|
so->so_state |= SS_ISCONNECTED;
|
||||||
if (head && (so->so_state & SS_INCOMP)) {
|
if (head && (so->so_state & SS_INCOMP)) {
|
||||||
if ((so->so_options & SO_ACCEPTFILTER) != 0) {
|
if ((so->so_options & SO_ACCEPTFILTER) != 0) {
|
||||||
SOCK_UNLOCK(so);
|
so->so_upcall = head->so_accf->so_accept_filter->accf_callback;
|
||||||
SOCK_LOCK(head);
|
so->so_upcallarg = head->so_accf->so_accept_filter_arg;
|
||||||
upcp = head->so_accf->so_accept_filter->accf_callback;
|
|
||||||
upcarg = head->so_accf->so_accept_filter_arg;
|
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_upcall = upcp;
|
|
||||||
so->so_upcallarg = upcarg;
|
|
||||||
so->so_rcv.sb_flags |= SB_UPCALL;
|
so->so_rcv.sb_flags |= SB_UPCALL;
|
||||||
so->so_options &= ~SO_ACCEPTFILTER;
|
so->so_options &= ~SO_ACCEPTFILTER;
|
||||||
SOCK_UNLOCK(so);
|
so->so_upcall(so, so->so_upcallarg, 0);
|
||||||
so->so_upcall(so, upcarg, 0);
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
so->so_state &= ~SS_INCOMP;
|
|
||||||
so->so_state |= SS_COMP;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
SOCK_LOCK(head);
|
|
||||||
TAILQ_REMOVE(&head->so_incomp, so, so_list);
|
TAILQ_REMOVE(&head->so_incomp, so, so_list);
|
||||||
head->so_incqlen--;
|
head->so_incqlen--;
|
||||||
|
so->so_state &= ~SS_INCOMP;
|
||||||
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
|
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
|
||||||
head->so_qlen++;
|
head->so_qlen++;
|
||||||
sorwakeup(head);
|
so->so_state |= SS_COMP;
|
||||||
|
sorwakeup_locked(head);
|
||||||
wakeup_one(&head->so_timeo);
|
wakeup_one(&head->so_timeo);
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
} else {
|
} else {
|
||||||
wakeup(&so->so_timeo);
|
wakeup(&so->so_timeo);
|
||||||
sorwakeup(so);
|
sorwakeup_locked(so);
|
||||||
sowwakeup(so);
|
sowwakeup_locked(so);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -160,12 +175,23 @@ soisdisconnecting(so)
|
|||||||
register struct socket *so;
|
register struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
so->so_state &= ~SS_ISCONNECTING;
|
so->so_state &= ~SS_ISCONNECTING;
|
||||||
so->so_state |= (SS_ISDISCONNECTING|SS_CANTRCVMORE|SS_CANTSENDMORE);
|
so->so_state |= (SS_ISDISCONNECTING|SS_CANTRCVMORE|SS_CANTSENDMORE);
|
||||||
wakeup((caddr_t)&so->so_timeo);
|
wakeup((caddr_t)&so->so_timeo);
|
||||||
sowwakeup(so);
|
sowwakeup_locked(so);
|
||||||
sorwakeup(so);
|
sorwakeup_locked(so);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
soisdisconnected_locked(so)
|
||||||
|
register struct socket *so;
|
||||||
|
{
|
||||||
|
|
||||||
|
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_locked(so);
|
||||||
|
sorwakeup_locked(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -173,12 +199,7 @@ soisdisconnected(so)
|
|||||||
register struct socket *so;
|
register struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
soisdisconnected_locked(so);
|
||||||
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);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -203,32 +224,25 @@ sonewconn(head, connstatus)
|
|||||||
so = soalloc(0);
|
so = soalloc(0);
|
||||||
if (so == NULL)
|
if (so == NULL)
|
||||||
return ((struct socket *)0);
|
return ((struct socket *)0);
|
||||||
SOCK_LOCK(head);
|
|
||||||
if ((head->so_options & SO_ACCEPTFILTER) != 0)
|
if ((head->so_options & SO_ACCEPTFILTER) != 0)
|
||||||
connstatus = 0;
|
connstatus = 0;
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
so->so_head = head;
|
so->so_head = head;
|
||||||
so->so_type = head->so_type;
|
so->so_type = head->so_type;
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_options = head->so_options &~ SO_ACCEPTCONN;
|
so->so_options = head->so_options &~ SO_ACCEPTCONN;
|
||||||
so->so_linger = head->so_linger;
|
so->so_linger = head->so_linger;
|
||||||
so->so_state = head->so_state | SS_NOFDREF;
|
so->so_state = head->so_state | SS_NOFDREF;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
so->so_proto = head->so_proto;
|
so->so_proto = head->so_proto;
|
||||||
so->so_timeo = head->so_timeo;
|
so->so_timeo = head->so_timeo;
|
||||||
so->so_cred = crhold(head->so_cred);
|
so->so_cred = crhold(head->so_cred);
|
||||||
if (soreserve(so, head->so_snd.sb_hiwat, head->so_rcv.sb_hiwat) ||
|
if (soreserve(so, head->so_snd.sb_hiwat, head->so_rcv.sb_hiwat) ||
|
||||||
(*so->so_proto->pr_usrreqs->pru_attach)(so, 0, NULL)) {
|
(*so->so_proto->pr_usrreqs->pru_attach)(so, 0, NULL)) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
return ((struct socket *)0);
|
return ((struct socket *)0);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (connstatus) {
|
if (connstatus) {
|
||||||
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
|
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_COMP;
|
so->so_state |= SS_COMP;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
head->so_qlen++;
|
head->so_qlen++;
|
||||||
} else {
|
} else {
|
||||||
if (head->so_incqlen > head->so_qlimit) {
|
if (head->so_incqlen > head->so_qlimit) {
|
||||||
@ -237,19 +251,13 @@ sonewconn(head, connstatus)
|
|||||||
(void) soabort(sp);
|
(void) soabort(sp);
|
||||||
}
|
}
|
||||||
TAILQ_INSERT_TAIL(&head->so_incomp, so, so_list);
|
TAILQ_INSERT_TAIL(&head->so_incomp, so, so_list);
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_INCOMP;
|
so->so_state |= SS_INCOMP;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
head->so_incqlen++;
|
head->so_incqlen++;
|
||||||
}
|
}
|
||||||
if (connstatus) {
|
if (connstatus) {
|
||||||
SOCK_LOCK(head);
|
sorwakeup_locked(head);
|
||||||
sorwakeup(head);
|
|
||||||
wakeup((caddr_t)&head->so_timeo);
|
wakeup((caddr_t)&head->so_timeo);
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= connstatus;
|
so->so_state |= connstatus;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
return (so);
|
return (so);
|
||||||
}
|
}
|
||||||
@ -269,10 +277,8 @@ socantsendmore(so)
|
|||||||
struct socket *so;
|
struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_CANTSENDMORE;
|
so->so_state |= SS_CANTSENDMORE;
|
||||||
sowwakeup(so);
|
sowwakeup_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -280,10 +286,8 @@ socantrcvmore(so)
|
|||||||
struct socket *so;
|
struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_CANTRCVMORE;
|
so->so_state |= SS_CANTRCVMORE;
|
||||||
sorwakeup(so);
|
sorwakeup_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -332,7 +336,6 @@ sowakeup(so, sb)
|
|||||||
register struct socket *so;
|
register struct socket *so;
|
||||||
register struct sockbuf *sb;
|
register struct sockbuf *sb;
|
||||||
{
|
{
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
|
|
||||||
selwakeup(&sb->sb_sel);
|
selwakeup(&sb->sb_sel);
|
||||||
sb->sb_flags &= ~SB_SEL;
|
sb->sb_flags &= ~SB_SEL;
|
||||||
@ -340,23 +343,13 @@ sowakeup(so, sb)
|
|||||||
sb->sb_flags &= ~SB_WAIT;
|
sb->sb_flags &= ~SB_WAIT;
|
||||||
wakeup((caddr_t)&sb->sb_cc);
|
wakeup((caddr_t)&sb->sb_cc);
|
||||||
}
|
}
|
||||||
if ((so->so_state & SS_ASYNC) && so->so_sigio != NULL) {
|
if ((so->so_state & SS_ASYNC) && so->so_sigio != NULL)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
pgsigio(&so->so_sigio, SIGIO, 0);
|
pgsigio(&so->so_sigio, SIGIO, 0);
|
||||||
SOCK_LOCK(so);
|
if (sb->sb_flags & SB_UPCALL)
|
||||||
}
|
|
||||||
if (sb->sb_flags & SB_UPCALL) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
(*so->so_upcall)(so, so->so_upcallarg, M_DONTWAIT);
|
(*so->so_upcall)(so, so->so_upcallarg, M_DONTWAIT);
|
||||||
SOCK_LOCK(so);
|
if (sb->sb_flags & SB_AIO)
|
||||||
}
|
|
||||||
if (sb->sb_flags & SB_AIO) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
aio_swake(so, sb);
|
aio_swake(so, sb);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
KNOTE(&sb->sb_sel.si_note, 0);
|
KNOTE(&sb->sb_sel.si_note, 0);
|
||||||
SOCK_LOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -966,11 +959,9 @@ sotoxsocket(struct socket *so, struct xsocket *xso)
|
|||||||
xso->xso_len = sizeof *xso;
|
xso->xso_len = sizeof *xso;
|
||||||
xso->xso_so = so;
|
xso->xso_so = so;
|
||||||
xso->so_type = so->so_type;
|
xso->so_type = so->so_type;
|
||||||
SOCK_LOCK(so);
|
|
||||||
xso->so_options = so->so_options;
|
xso->so_options = so->so_options;
|
||||||
xso->so_linger = so->so_linger;
|
xso->so_linger = so->so_linger;
|
||||||
xso->so_state = so->so_state;
|
xso->so_state = so->so_state;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
xso->so_pcb = so->so_pcb;
|
xso->so_pcb = so->so_pcb;
|
||||||
xso->xso_protocol = so->so_proto->pr_protocol;
|
xso->xso_protocol = so->so_proto->pr_protocol;
|
||||||
xso->xso_family = so->so_proto->pr_domain->dom_family;
|
xso->xso_family = so->so_proto->pr_domain->dom_family;
|
||||||
|
@ -82,7 +82,6 @@ static struct filterops sowrite_filtops =
|
|||||||
|
|
||||||
uma_zone_t socket_zone;
|
uma_zone_t socket_zone;
|
||||||
so_gen_t so_gencnt; /* generation count for sockets */
|
so_gen_t so_gencnt; /* generation count for sockets */
|
||||||
struct mtx socq_lock;
|
|
||||||
|
|
||||||
MALLOC_DEFINE(M_SONAME, "soname", "socket name");
|
MALLOC_DEFINE(M_SONAME, "soname", "socket name");
|
||||||
MALLOC_DEFINE(M_PCB, "pcb", "protocol control block");
|
MALLOC_DEFINE(M_PCB, "pcb", "protocol control block");
|
||||||
@ -129,8 +128,7 @@ soalloc(waitok)
|
|||||||
if (so) {
|
if (so) {
|
||||||
/* XXX race condition for reentrant kernel */
|
/* XXX race condition for reentrant kernel */
|
||||||
so->so_gencnt = ++so_gencnt;
|
so->so_gencnt = ++so_gencnt;
|
||||||
mtx_init(&so->so_rcv.sb_mtx, "sockbuf rcv", NULL, MTX_DEF);
|
/* sx_init(&so->so_sxlock, "socket sxlock"); */
|
||||||
mtx_init(&so->so_snd.sb_mtx, "sockbuf snd", NULL, MTX_DEF);
|
|
||||||
TAILQ_INIT(&so->so_aiojobq);
|
TAILQ_INIT(&so->so_aiojobq);
|
||||||
++numopensockets;
|
++numopensockets;
|
||||||
}
|
}
|
||||||
@ -175,17 +173,14 @@ socreate(dom, aso, type, proto, cred, td)
|
|||||||
if (so == NULL)
|
if (so == NULL)
|
||||||
return (ENOBUFS);
|
return (ENOBUFS);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
TAILQ_INIT(&so->so_incomp);
|
TAILQ_INIT(&so->so_incomp);
|
||||||
TAILQ_INIT(&so->so_comp);
|
TAILQ_INIT(&so->so_comp);
|
||||||
so->so_type = type;
|
so->so_type = type;
|
||||||
so->so_cred = crhold(cred);
|
so->so_cred = crhold(cred);
|
||||||
so->so_proto = prp;
|
so->so_proto = prp;
|
||||||
soref(so);
|
soref(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = (*prp->pr_usrreqs->pru_attach)(so, proto, td);
|
error = (*prp->pr_usrreqs->pru_attach)(so, proto, td);
|
||||||
if (error) {
|
if (error) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_NOFDREF;
|
so->so_state |= SS_NOFDREF;
|
||||||
sorele(so);
|
sorele(so);
|
||||||
return (error);
|
return (error);
|
||||||
@ -212,9 +207,7 @@ static void
|
|||||||
sodealloc(struct socket *so)
|
sodealloc(struct socket *so)
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
KASSERT(so->so_count == 0, ("sodealloc(): so_count %d", so->so_count));
|
KASSERT(so->so_count == 0, ("sodealloc(): so_count %d", so->so_count));
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
so->so_gencnt = ++so_gencnt;
|
so->so_gencnt = ++so_gencnt;
|
||||||
if (so->so_rcv.sb_hiwat)
|
if (so->so_rcv.sb_hiwat)
|
||||||
(void)chgsbsize(so->so_cred->cr_uidinfo,
|
(void)chgsbsize(so->so_cred->cr_uidinfo,
|
||||||
@ -234,8 +227,7 @@ sodealloc(struct socket *so)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
crfree(so->so_cred);
|
crfree(so->so_cred);
|
||||||
mtx_destroy(&so->so_rcv.sb_mtx);
|
/* sx_destroy(&so->so_sxlock); */
|
||||||
mtx_destroy(&so->so_snd.sb_mtx);
|
|
||||||
uma_zfree(socket_zone, so);
|
uma_zfree(socket_zone, so);
|
||||||
--numopensockets;
|
--numopensockets;
|
||||||
}
|
}
|
||||||
@ -254,11 +246,8 @@ solisten(so, backlog, td)
|
|||||||
splx(s);
|
splx(s);
|
||||||
return (error);
|
return (error);
|
||||||
}
|
}
|
||||||
if (TAILQ_EMPTY(&so->so_comp)) {
|
if (TAILQ_EMPTY(&so->so_comp))
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_options |= SO_ACCEPTCONN;
|
so->so_options |= SO_ACCEPTCONN;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
if (backlog < 0 || backlog > somaxconn)
|
if (backlog < 0 || backlog > somaxconn)
|
||||||
backlog = somaxconn;
|
backlog = somaxconn;
|
||||||
so->so_qlimit = backlog;
|
so->so_qlimit = backlog;
|
||||||
@ -272,21 +261,15 @@ sofree(so)
|
|||||||
{
|
{
|
||||||
struct socket *head = so->so_head;
|
struct socket *head = so->so_head;
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
|
|
||||||
KASSERT(so->so_count == 0, ("socket %p so_count not 0", so));
|
KASSERT(so->so_count == 0, ("socket %p so_count not 0", so));
|
||||||
|
|
||||||
if (so->so_pcb || (so->so_state & SS_NOFDREF) == 0) {
|
if (so->so_pcb || (so->so_state & SS_NOFDREF) == 0)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
if (head != NULL) {
|
if (head != NULL) {
|
||||||
if (so->so_state & SS_INCOMP) {
|
if (so->so_state & SS_INCOMP) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
TAILQ_REMOVE(&head->so_incomp, so, so_list);
|
TAILQ_REMOVE(&head->so_incomp, so, so_list);
|
||||||
head->so_incqlen--;
|
head->so_incqlen--;
|
||||||
} else if (so->so_state & SS_COMP) {
|
} else if (so->so_state & SS_COMP) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/*
|
/*
|
||||||
* We must not decommission a socket that's
|
* We must not decommission a socket that's
|
||||||
* on the accept(2) queue. If we do, then
|
* on the accept(2) queue. If we do, then
|
||||||
@ -295,15 +278,11 @@ sofree(so)
|
|||||||
*/
|
*/
|
||||||
return;
|
return;
|
||||||
} else {
|
} else {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
panic("sofree: not queued");
|
panic("sofree: not queued");
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_INCOMP;
|
so->so_state &= ~SS_INCOMP;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
so->so_head = NULL;
|
so->so_head = NULL;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbrelease(&so->so_snd, so);
|
sbrelease(&so->so_snd, so);
|
||||||
sorflush(so);
|
sorflush(so);
|
||||||
sodealloc(so);
|
sodealloc(so);
|
||||||
@ -326,11 +305,9 @@ soclose(so)
|
|||||||
int error = 0;
|
int error = 0;
|
||||||
|
|
||||||
funsetown(&so->so_sigio);
|
funsetown(&so->so_sigio);
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_options & SO_ACCEPTCONN) {
|
if (so->so_options & SO_ACCEPTCONN) {
|
||||||
struct socket *sp, *sonext;
|
struct socket *sp, *sonext;
|
||||||
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sp = TAILQ_FIRST(&so->so_incomp);
|
sp = TAILQ_FIRST(&so->so_incomp);
|
||||||
for (; sp != NULL; sp = sonext) {
|
for (; sp != NULL; sp = sonext) {
|
||||||
sonext = TAILQ_NEXT(sp, so_list);
|
sonext = TAILQ_NEXT(sp, so_list);
|
||||||
@ -341,49 +318,38 @@ soclose(so)
|
|||||||
/* Dequeue from so_comp since sofree() won't do it */
|
/* Dequeue from so_comp since sofree() won't do it */
|
||||||
TAILQ_REMOVE(&so->so_comp, sp, so_list);
|
TAILQ_REMOVE(&so->so_comp, sp, so_list);
|
||||||
so->so_qlen--;
|
so->so_qlen--;
|
||||||
SOCK_LOCK(sp);
|
|
||||||
sp->so_state &= ~SS_COMP;
|
sp->so_state &= ~SS_COMP;
|
||||||
SOCK_UNLOCK(sp);
|
|
||||||
sp->so_head = NULL;
|
sp->so_head = NULL;
|
||||||
(void) soabort(sp);
|
(void) soabort(sp);
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
}
|
}
|
||||||
if (so->so_pcb == 0)
|
if (so->so_pcb == 0)
|
||||||
goto discard;
|
goto discard;
|
||||||
if (so->so_state & SS_ISCONNECTED) {
|
if (so->so_state & SS_ISCONNECTED) {
|
||||||
if ((so->so_state & SS_ISDISCONNECTING) == 0) {
|
if ((so->so_state & SS_ISDISCONNECTING) == 0) {
|
||||||
error = sodisconnect(so);
|
error = sodisconnect(so);
|
||||||
if (error) {
|
if (error)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto drop;
|
goto drop;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (so->so_options & SO_LINGER) {
|
if (so->so_options & SO_LINGER) {
|
||||||
if ((so->so_state & SS_ISDISCONNECTING) &&
|
if ((so->so_state & SS_ISDISCONNECTING) &&
|
||||||
(so->so_state & SS_NBIO)) {
|
(so->so_state & SS_NBIO))
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto drop;
|
goto drop;
|
||||||
}
|
|
||||||
while (so->so_state & SS_ISCONNECTED) {
|
while (so->so_state & SS_ISCONNECTED) {
|
||||||
error = msleep((caddr_t)&so->so_timeo, SOCK_MTX(so),
|
error = tsleep((caddr_t)&so->so_timeo,
|
||||||
PSOCK | PCATCH, "soclos", so->so_linger * hz);
|
PSOCK | PCATCH, "soclos", so->so_linger * hz);
|
||||||
if (error)
|
if (error)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
drop:
|
drop:
|
||||||
SOCK_ASSERT(so, MA_NOTOWNED);
|
|
||||||
if (so->so_pcb) {
|
if (so->so_pcb) {
|
||||||
int error2 = (*so->so_proto->pr_usrreqs->pru_detach)(so);
|
int error2 = (*so->so_proto->pr_usrreqs->pru_detach)(so);
|
||||||
if (error == 0)
|
if (error == 0)
|
||||||
error = error2;
|
error = error2;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
discard:
|
discard:
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
if (so->so_state & SS_NOFDREF)
|
if (so->so_state & SS_NOFDREF)
|
||||||
panic("soclose: NOFDREF");
|
panic("soclose: NOFDREF");
|
||||||
so->so_state |= SS_NOFDREF;
|
so->so_state |= SS_NOFDREF;
|
||||||
@ -403,7 +369,6 @@ soabort(so)
|
|||||||
|
|
||||||
error = (*so->so_proto->pr_usrreqs->pru_abort)(so);
|
error = (*so->so_proto->pr_usrreqs->pru_abort)(so);
|
||||||
if (error) {
|
if (error) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so); /* note: does not decrement the ref count */
|
sotryfree(so); /* note: does not decrement the ref count */
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
@ -418,11 +383,9 @@ soaccept(so, nam)
|
|||||||
int s = splnet();
|
int s = splnet();
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_NOFDREF) == 0)
|
if ((so->so_state & SS_NOFDREF) == 0)
|
||||||
panic("soaccept: !NOFDREF");
|
panic("soaccept: !NOFDREF");
|
||||||
so->so_state &= ~SS_NOFDREF;
|
so->so_state &= ~SS_NOFDREF;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = (*so->so_proto->pr_usrreqs->pru_accept)(so, nam);
|
error = (*so->so_proto->pr_usrreqs->pru_accept)(so, nam);
|
||||||
splx(s);
|
splx(s);
|
||||||
return (error);
|
return (error);
|
||||||
@ -437,11 +400,8 @@ soconnect(so, nam, td)
|
|||||||
int s;
|
int s;
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_ACCEPTCONN)
|
||||||
if (so->so_options & SO_ACCEPTCONN) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (EOPNOTSUPP);
|
return (EOPNOTSUPP);
|
||||||
}
|
|
||||||
s = splnet();
|
s = splnet();
|
||||||
/*
|
/*
|
||||||
* If protocol is connection-based, can only connect once.
|
* If protocol is connection-based, can only connect once.
|
||||||
@ -449,23 +409,12 @@ soconnect(so, nam, td)
|
|||||||
* This allows user to disconnect by connecting to, e.g.,
|
* This allows user to disconnect by connecting to, e.g.,
|
||||||
* a null address.
|
* a null address.
|
||||||
*/
|
*/
|
||||||
if (so->so_state & (SS_ISCONNECTED|SS_ISCONNECTING)) {
|
if (so->so_state & (SS_ISCONNECTED|SS_ISCONNECTING) &&
|
||||||
if (so->so_proto->pr_flags & PR_CONNREQUIRED) {
|
((so->so_proto->pr_flags & PR_CONNREQUIRED) ||
|
||||||
SOCK_UNLOCK(so);
|
(error = sodisconnect(so))))
|
||||||
error = EISCONN;
|
error = EISCONN;
|
||||||
goto done;
|
else
|
||||||
} else {
|
error = (*so->so_proto->pr_usrreqs->pru_connect)(so, nam, td);
|
||||||
error = sodisconnect(so);
|
|
||||||
if (error) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = EISCONN;
|
|
||||||
goto done;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = (*so->so_proto->pr_usrreqs->pru_connect)(so, nam, td);
|
|
||||||
done:
|
|
||||||
splx(s);
|
splx(s);
|
||||||
return (error);
|
return (error);
|
||||||
}
|
}
|
||||||
@ -490,7 +439,6 @@ sodisconnect(so)
|
|||||||
int s = splnet();
|
int s = splnet();
|
||||||
int error;
|
int error;
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
||||||
error = ENOTCONN;
|
error = ENOTCONN;
|
||||||
goto bad;
|
goto bad;
|
||||||
@ -499,9 +447,7 @@ sodisconnect(so)
|
|||||||
error = EALREADY;
|
error = EALREADY;
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = (*so->so_proto->pr_usrreqs->pru_disconnect)(so);
|
error = (*so->so_proto->pr_usrreqs->pru_disconnect)(so);
|
||||||
SOCK_LOCK(so);
|
|
||||||
bad:
|
bad:
|
||||||
splx(s);
|
splx(s);
|
||||||
return (error);
|
return (error);
|
||||||
@ -560,22 +506,14 @@ sosend(so, addr, uio, top, control, flags, td)
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
dontroute =
|
dontroute =
|
||||||
(flags & MSG_DONTROUTE) && (so->so_options & SO_DONTROUTE) == 0 &&
|
(flags & MSG_DONTROUTE) && (so->so_options & SO_DONTROUTE) == 0 &&
|
||||||
(so->so_proto->pr_flags & PR_ATOMIC);
|
(so->so_proto->pr_flags & PR_ATOMIC);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (td)
|
if (td)
|
||||||
td->td_proc->p_stats->p_ru.ru_msgsnd++;
|
td->td_proc->p_stats->p_ru.ru_msgsnd++;
|
||||||
if (control)
|
if (control)
|
||||||
clen = control->m_len;
|
clen = control->m_len;
|
||||||
#define snderr(errno) \
|
#define snderr(errno) { error = errno; splx(s); goto release; }
|
||||||
do { \
|
|
||||||
error = errno; \
|
|
||||||
SOCK_UNLOCK(so); \
|
|
||||||
splx(s); \
|
|
||||||
goto release; \
|
|
||||||
} while(0);
|
|
||||||
|
|
||||||
restart:
|
restart:
|
||||||
error = sblock(&so->so_snd, SBLOCKWAIT(flags));
|
error = sblock(&so->so_snd, SBLOCKWAIT(flags));
|
||||||
@ -583,13 +521,11 @@ restart:
|
|||||||
goto out;
|
goto out;
|
||||||
do {
|
do {
|
||||||
s = splnet();
|
s = splnet();
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTSENDMORE)
|
if (so->so_state & SS_CANTSENDMORE)
|
||||||
snderr(EPIPE);
|
snderr(EPIPE);
|
||||||
if (so->so_error) {
|
if (so->so_error) {
|
||||||
error = so->so_error;
|
error = so->so_error;
|
||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
@ -609,21 +545,16 @@ restart:
|
|||||||
snderr(so->so_proto->pr_flags & PR_CONNREQUIRED ?
|
snderr(so->so_proto->pr_flags & PR_CONNREQUIRED ?
|
||||||
ENOTCONN : EDESTADDRREQ);
|
ENOTCONN : EDESTADDRREQ);
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
space = sbspace(&so->so_snd);
|
space = sbspace(&so->so_snd);
|
||||||
if (flags & MSG_OOB)
|
if (flags & MSG_OOB)
|
||||||
space += 1024;
|
space += 1024;
|
||||||
if ((atomic && resid > so->so_snd.sb_hiwat) ||
|
if ((atomic && resid > so->so_snd.sb_hiwat) ||
|
||||||
clen > so->so_snd.sb_hiwat) {
|
clen > so->so_snd.sb_hiwat)
|
||||||
SOCK_LOCK(so);
|
|
||||||
snderr(EMSGSIZE);
|
snderr(EMSGSIZE);
|
||||||
}
|
|
||||||
if (space < resid + clen &&
|
if (space < resid + clen &&
|
||||||
(atomic || space < so->so_snd.sb_lowat || space < clen)) {
|
(atomic || space < so->so_snd.sb_lowat || space < clen)) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_NBIO)
|
if (so->so_state & SS_NBIO)
|
||||||
snderr(EWOULDBLOCK);
|
snderr(EWOULDBLOCK);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbunlock(&so->so_snd);
|
sbunlock(&so->so_snd);
|
||||||
error = sbwait(&so->so_snd);
|
error = sbwait(&so->so_snd);
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -691,11 +622,8 @@ nopages:
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} while (space > 0 && atomic);
|
} while (space > 0 && atomic);
|
||||||
if (dontroute) {
|
if (dontroute)
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_options |= SO_DONTROUTE;
|
so->so_options |= SO_DONTROUTE;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
s = splnet(); /* XXX */
|
s = splnet(); /* XXX */
|
||||||
/*
|
/*
|
||||||
* XXX all the SS_CANTSENDMORE checks previously
|
* XXX all the SS_CANTSENDMORE checks previously
|
||||||
@ -721,11 +649,8 @@ nopages:
|
|||||||
(resid > 0 && space > 0) ? PRUS_MORETOCOME : 0,
|
(resid > 0 && space > 0) ? PRUS_MORETOCOME : 0,
|
||||||
top, addr, control, td);
|
top, addr, control, td);
|
||||||
splx(s);
|
splx(s);
|
||||||
if (dontroute) {
|
if (dontroute)
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_options &= ~SO_DONTROUTE;
|
so->so_options &= ~SO_DONTROUTE;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
clen = 0;
|
clen = 0;
|
||||||
control = 0;
|
control = 0;
|
||||||
top = 0;
|
top = 0;
|
||||||
@ -805,15 +730,10 @@ bad:
|
|||||||
}
|
}
|
||||||
if (mp)
|
if (mp)
|
||||||
*mp = (struct mbuf *)0;
|
*mp = (struct mbuf *)0;
|
||||||
SOCK_LOCK(so);
|
if (so->so_state & SS_ISCONFIRMING && uio->uio_resid)
|
||||||
if (so->so_state & SS_ISCONFIRMING && uio->uio_resid) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
(*pr->pr_usrreqs->pru_rcvd)(so, 0);
|
(*pr->pr_usrreqs->pru_rcvd)(so, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
restart:
|
restart:
|
||||||
SOCK_ASSERT(so, MA_NOTOWNED);
|
|
||||||
error = sblock(&so->so_rcv, SBLOCKWAIT(flags));
|
error = sblock(&so->so_rcv, SBLOCKWAIT(flags));
|
||||||
if (error)
|
if (error)
|
||||||
return (error);
|
return (error);
|
||||||
@ -847,37 +767,28 @@ restart:
|
|||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTRCVMORE) {
|
if (so->so_state & SS_CANTRCVMORE) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (m)
|
if (m)
|
||||||
goto dontblock;
|
goto dontblock;
|
||||||
else
|
else
|
||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
for (; m; m = m->m_next)
|
for (; m; m = m->m_next)
|
||||||
if (m->m_type == MT_OOBDATA || (m->m_flags & M_EOR)) {
|
if (m->m_type == MT_OOBDATA || (m->m_flags & M_EOR)) {
|
||||||
m = so->so_rcv.sb_mb;
|
m = so->so_rcv.sb_mb;
|
||||||
goto dontblock;
|
goto dontblock;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & (SS_ISCONNECTED|SS_ISCONNECTING)) == 0 &&
|
if ((so->so_state & (SS_ISCONNECTED|SS_ISCONNECTING)) == 0 &&
|
||||||
(so->so_proto->pr_flags & PR_CONNREQUIRED)) {
|
(so->so_proto->pr_flags & PR_CONNREQUIRED)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ENOTCONN;
|
error = ENOTCONN;
|
||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
if (uio->uio_resid == 0) {
|
if (uio->uio_resid == 0)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto release;
|
goto release;
|
||||||
}
|
|
||||||
if ((so->so_state & SS_NBIO) || (flags & MSG_DONTWAIT)) {
|
if ((so->so_state & SS_NBIO) || (flags & MSG_DONTWAIT)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = EWOULDBLOCK;
|
error = EWOULDBLOCK;
|
||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbunlock(&so->so_rcv);
|
sbunlock(&so->so_rcv);
|
||||||
error = sbwait(&so->so_rcv);
|
error = sbwait(&so->so_rcv);
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -947,9 +858,7 @@ dontblock:
|
|||||||
else
|
else
|
||||||
KASSERT(m->m_type == MT_DATA || m->m_type == MT_HEADER,
|
KASSERT(m->m_type == MT_DATA || m->m_type == MT_HEADER,
|
||||||
("m->m_type == %d", m->m_type));
|
("m->m_type == %d", m->m_type));
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_RCVATMARK;
|
so->so_state &= ~SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
len = uio->uio_resid;
|
len = uio->uio_resid;
|
||||||
if (so->so_oobmark && len > so->so_oobmark - offset)
|
if (so->so_oobmark && len > so->so_oobmark - offset)
|
||||||
len = so->so_oobmark - offset;
|
len = so->so_oobmark - offset;
|
||||||
@ -1007,9 +916,7 @@ dontblock:
|
|||||||
if ((flags & MSG_PEEK) == 0) {
|
if ((flags & MSG_PEEK) == 0) {
|
||||||
so->so_oobmark -= len;
|
so->so_oobmark -= len;
|
||||||
if (so->so_oobmark == 0) {
|
if (so->so_oobmark == 0) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_RCVATMARK;
|
so->so_state |= SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -1029,12 +936,8 @@ dontblock:
|
|||||||
*/
|
*/
|
||||||
while (flags & MSG_WAITALL && m == 0 && uio->uio_resid > 0 &&
|
while (flags & MSG_WAITALL && m == 0 && uio->uio_resid > 0 &&
|
||||||
!sosendallatonce(so) && !nextrecord) {
|
!sosendallatonce(so) && !nextrecord) {
|
||||||
SOCK_LOCK(so);
|
if (so->so_error || so->so_state & SS_CANTRCVMORE)
|
||||||
if (so->so_error || so->so_state & SS_CANTRCVMORE) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/*
|
/*
|
||||||
* Notify the protocol that some data has been
|
* Notify the protocol that some data has been
|
||||||
* drained before blocking.
|
* drained before blocking.
|
||||||
@ -1064,15 +967,12 @@ dontblock:
|
|||||||
if (pr->pr_flags & PR_WANTRCVD && so->so_pcb)
|
if (pr->pr_flags & PR_WANTRCVD && so->so_pcb)
|
||||||
(*pr->pr_usrreqs->pru_rcvd)(so, flags);
|
(*pr->pr_usrreqs->pru_rcvd)(so, flags);
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (orig_resid == uio->uio_resid && orig_resid &&
|
if (orig_resid == uio->uio_resid && orig_resid &&
|
||||||
(flags & MSG_EOR) == 0 && (so->so_state & SS_CANTRCVMORE) == 0) {
|
(flags & MSG_EOR) == 0 && (so->so_state & SS_CANTRCVMORE) == 0) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbunlock(&so->so_rcv);
|
sbunlock(&so->so_rcv);
|
||||||
splx(s);
|
splx(s);
|
||||||
goto restart;
|
goto restart;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
if (flagsp)
|
if (flagsp)
|
||||||
*flagsp |= flags;
|
*flagsp |= flags;
|
||||||
@ -1114,10 +1014,7 @@ sorflush(so)
|
|||||||
socantrcvmore(so);
|
socantrcvmore(so);
|
||||||
sbunlock(sb);
|
sbunlock(sb);
|
||||||
asb = *sb;
|
asb = *sb;
|
||||||
#define RANGEOF(type, start, end) (offsetof(type, end) - offsetof(type, start))
|
bzero((caddr_t)sb, sizeof (*sb));
|
||||||
bzero((caddr_t)&sb->sb_startzero,
|
|
||||||
(unsigned) RANGEOF(struct sockbuf, sb_startzero, sb_endzero));
|
|
||||||
#undef RANGEOF
|
|
||||||
splx(s);
|
splx(s);
|
||||||
if (pr->pr_flags & PR_RIGHTS && pr->pr_domain->dom_dispose)
|
if (pr->pr_flags & PR_RIGHTS && pr->pr_domain->dom_dispose)
|
||||||
(*pr->pr_domain->dom_dispose)(asb.sb_mb);
|
(*pr->pr_domain->dom_dispose)(asb.sb_mb);
|
||||||
@ -1136,13 +1033,10 @@ do_setopt_accept_filter(so, sopt)
|
|||||||
int error = 0;
|
int error = 0;
|
||||||
|
|
||||||
/* do not set/remove accept filters on non listen sockets */
|
/* do not set/remove accept filters on non listen sockets */
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_options & SO_ACCEPTCONN) == 0) {
|
if ((so->so_options & SO_ACCEPTCONN) == 0) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = EINVAL;
|
error = EINVAL;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
/* removing the filter */
|
/* removing the filter */
|
||||||
if (sopt == NULL) {
|
if (sopt == NULL) {
|
||||||
@ -1157,9 +1051,7 @@ do_setopt_accept_filter(so, sopt)
|
|||||||
FREE(af, M_ACCF);
|
FREE(af, M_ACCF);
|
||||||
so->so_accf = NULL;
|
so->so_accf = NULL;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_options &= ~SO_ACCEPTFILTER;
|
so->so_options &= ~SO_ACCEPTFILTER;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
/* adding a filter */
|
/* adding a filter */
|
||||||
@ -1199,9 +1091,7 @@ do_setopt_accept_filter(so, sopt)
|
|||||||
}
|
}
|
||||||
af->so_accept_filter = afp;
|
af->so_accept_filter = afp;
|
||||||
so->so_accf = af;
|
so->so_accf = af;
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_options |= SO_ACCEPTFILTER;
|
so->so_options |= SO_ACCEPTFILTER;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
out:
|
out:
|
||||||
if (afap != NULL)
|
if (afap != NULL)
|
||||||
FREE(afap, M_TEMP);
|
FREE(afap, M_TEMP);
|
||||||
@ -1273,13 +1163,11 @@ sosetopt(so, sopt)
|
|||||||
if (error)
|
if (error)
|
||||||
goto bad;
|
goto bad;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_linger = l.l_linger;
|
so->so_linger = l.l_linger;
|
||||||
if (l.l_onoff)
|
if (l.l_onoff)
|
||||||
so->so_options |= SO_LINGER;
|
so->so_options |= SO_LINGER;
|
||||||
else
|
else
|
||||||
so->so_options &= ~SO_LINGER;
|
so->so_options &= ~SO_LINGER;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SO_DEBUG:
|
case SO_DEBUG:
|
||||||
@ -1295,12 +1183,10 @@ sosetopt(so, sopt)
|
|||||||
sizeof optval);
|
sizeof optval);
|
||||||
if (error)
|
if (error)
|
||||||
goto bad;
|
goto bad;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (optval)
|
if (optval)
|
||||||
so->so_options |= sopt->sopt_name;
|
so->so_options |= sopt->sopt_name;
|
||||||
else
|
else
|
||||||
so->so_options &= ~sopt->sopt_name;
|
so->so_options &= ~sopt->sopt_name;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SO_SNDBUF:
|
case SO_SNDBUF:
|
||||||
@ -1447,30 +1333,23 @@ sogetopt(so, sopt)
|
|||||||
switch (sopt->sopt_name) {
|
switch (sopt->sopt_name) {
|
||||||
#ifdef INET
|
#ifdef INET
|
||||||
case SO_ACCEPTFILTER:
|
case SO_ACCEPTFILTER:
|
||||||
|
if ((so->so_options & SO_ACCEPTCONN) == 0)
|
||||||
|
return (EINVAL);
|
||||||
MALLOC(afap, struct accept_filter_arg *, sizeof(*afap),
|
MALLOC(afap, struct accept_filter_arg *, sizeof(*afap),
|
||||||
M_TEMP, M_WAITOK | M_ZERO);
|
M_TEMP, M_WAITOK | M_ZERO);
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_options & SO_ACCEPTCONN) == 0) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
FREE(afap, M_TEMP);
|
|
||||||
return (EINVAL);
|
|
||||||
}
|
|
||||||
if ((so->so_options & SO_ACCEPTFILTER) != 0) {
|
if ((so->so_options & SO_ACCEPTFILTER) != 0) {
|
||||||
strcpy(afap->af_name, so->so_accf->so_accept_filter->accf_name);
|
strcpy(afap->af_name, so->so_accf->so_accept_filter->accf_name);
|
||||||
if (so->so_accf->so_accept_filter_str != NULL)
|
if (so->so_accf->so_accept_filter_str != NULL)
|
||||||
strcpy(afap->af_arg, so->so_accf->so_accept_filter_str);
|
strcpy(afap->af_arg, so->so_accf->so_accept_filter_str);
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = sooptcopyout(sopt, afap, sizeof(*afap));
|
error = sooptcopyout(sopt, afap, sizeof(*afap));
|
||||||
FREE(afap, M_TEMP);
|
FREE(afap, M_TEMP);
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
case SO_LINGER:
|
case SO_LINGER:
|
||||||
SOCK_LOCK(so);
|
|
||||||
l.l_onoff = so->so_options & SO_LINGER;
|
l.l_onoff = so->so_options & SO_LINGER;
|
||||||
l.l_linger = so->so_linger;
|
l.l_linger = so->so_linger;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = sooptcopyout(sopt, &l, sizeof l);
|
error = sooptcopyout(sopt, &l, sizeof l);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1483,9 +1362,7 @@ sogetopt(so, sopt)
|
|||||||
case SO_BROADCAST:
|
case SO_BROADCAST:
|
||||||
case SO_OOBINLINE:
|
case SO_OOBINLINE:
|
||||||
case SO_TIMESTAMP:
|
case SO_TIMESTAMP:
|
||||||
SOCK_LOCK(so);
|
|
||||||
optval = so->so_options & sopt->sopt_name;
|
optval = so->so_options & sopt->sopt_name;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
integer:
|
integer:
|
||||||
error = sooptcopyout(sopt, &optval, sizeof optval);
|
error = sooptcopyout(sopt, &optval, sizeof optval);
|
||||||
break;
|
break;
|
||||||
@ -1659,31 +1536,22 @@ sopoll(struct socket *so, int events, struct ucred *cred, struct thread *td)
|
|||||||
int revents = 0;
|
int revents = 0;
|
||||||
int s = splnet();
|
int s = splnet();
|
||||||
|
|
||||||
if (events & (POLLIN | POLLRDNORM)) {
|
if (events & (POLLIN | POLLRDNORM))
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (soreadable(so))
|
if (soreadable(so))
|
||||||
revents |= events & (POLLIN | POLLRDNORM);
|
revents |= events & (POLLIN | POLLRDNORM);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (events & POLLINIGNEOF)
|
if (events & POLLINIGNEOF)
|
||||||
if (so->so_rcv.sb_cc >= so->so_rcv.sb_lowat ||
|
if (so->so_rcv.sb_cc >= so->so_rcv.sb_lowat ||
|
||||||
!TAILQ_EMPTY(&so->so_comp) || so->so_error)
|
!TAILQ_EMPTY(&so->so_comp) || so->so_error)
|
||||||
revents |= POLLINIGNEOF;
|
revents |= POLLINIGNEOF;
|
||||||
|
|
||||||
if (events & (POLLOUT | POLLWRNORM)) {
|
if (events & (POLLOUT | POLLWRNORM))
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (sowriteable(so))
|
if (sowriteable(so))
|
||||||
revents |= events & (POLLOUT | POLLWRNORM);
|
revents |= events & (POLLOUT | POLLWRNORM);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (events & (POLLPRI | POLLRDBAND)) {
|
if (events & (POLLPRI | POLLRDBAND))
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_oobmark || (so->so_state & SS_RCVATMARK))
|
if (so->so_oobmark || (so->so_state & SS_RCVATMARK))
|
||||||
revents |= events & (POLLPRI | POLLRDBAND);
|
revents |= events & (POLLPRI | POLLRDBAND);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (revents == 0) {
|
if (revents == 0) {
|
||||||
if (events &
|
if (events &
|
||||||
@ -1712,12 +1580,10 @@ sokqfilter(struct file *fp, struct knote *kn)
|
|||||||
|
|
||||||
switch (kn->kn_filter) {
|
switch (kn->kn_filter) {
|
||||||
case EVFILT_READ:
|
case EVFILT_READ:
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_options & SO_ACCEPTCONN)
|
if (so->so_options & SO_ACCEPTCONN)
|
||||||
kn->kn_fop = &solisten_filtops;
|
kn->kn_fop = &solisten_filtops;
|
||||||
else
|
else
|
||||||
kn->kn_fop = &soread_filtops;
|
kn->kn_fop = &soread_filtops;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sb = &so->so_rcv;
|
sb = &so->so_rcv;
|
||||||
break;
|
break;
|
||||||
case EVFILT_WRITE:
|
case EVFILT_WRITE:
|
||||||
@ -1754,18 +1620,13 @@ filt_soread(struct knote *kn, long hint)
|
|||||||
struct socket *so = (struct socket *)kn->kn_fp->f_data;
|
struct socket *so = (struct socket *)kn->kn_fp->f_data;
|
||||||
|
|
||||||
kn->kn_data = so->so_rcv.sb_cc;
|
kn->kn_data = so->so_rcv.sb_cc;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTRCVMORE) {
|
if (so->so_state & SS_CANTRCVMORE) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
kn->kn_flags |= EV_EOF;
|
kn->kn_flags |= EV_EOF;
|
||||||
kn->kn_fflags = so->so_error;
|
kn->kn_fflags = so->so_error;
|
||||||
return (1);
|
return (1);
|
||||||
}
|
}
|
||||||
if (so->so_error) { /* temporary udp error */
|
if (so->so_error) /* temporary udp error */
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (1);
|
return (1);
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (kn->kn_sfflags & NOTE_LOWAT)
|
if (kn->kn_sfflags & NOTE_LOWAT)
|
||||||
return (kn->kn_data >= kn->kn_sdata);
|
return (kn->kn_data >= kn->kn_sdata);
|
||||||
return (kn->kn_data >= so->so_rcv.sb_lowat);
|
return (kn->kn_data >= so->so_rcv.sb_lowat);
|
||||||
@ -1790,23 +1651,16 @@ filt_sowrite(struct knote *kn, long hint)
|
|||||||
struct socket *so = (struct socket *)kn->kn_fp->f_data;
|
struct socket *so = (struct socket *)kn->kn_fp->f_data;
|
||||||
|
|
||||||
kn->kn_data = sbspace(&so->so_snd);
|
kn->kn_data = sbspace(&so->so_snd);
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTSENDMORE) {
|
if (so->so_state & SS_CANTSENDMORE) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
kn->kn_flags |= EV_EOF;
|
kn->kn_flags |= EV_EOF;
|
||||||
kn->kn_fflags = so->so_error;
|
kn->kn_fflags = so->so_error;
|
||||||
return (1);
|
return (1);
|
||||||
}
|
}
|
||||||
if (so->so_error) { /* temporary udp error */
|
if (so->so_error) /* temporary udp error */
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (1);
|
return (1);
|
||||||
}
|
|
||||||
if (((so->so_state & SS_ISCONNECTED) == 0) &&
|
if (((so->so_state & SS_ISCONNECTED) == 0) &&
|
||||||
(so->so_proto->pr_flags & PR_CONNREQUIRED)) {
|
(so->so_proto->pr_flags & PR_CONNREQUIRED))
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (kn->kn_sfflags & NOTE_LOWAT)
|
if (kn->kn_sfflags & NOTE_LOWAT)
|
||||||
return (kn->kn_data >= kn->kn_sdata);
|
return (kn->kn_data >= kn->kn_sdata);
|
||||||
return (kn->kn_data >= so->so_snd.sb_lowat);
|
return (kn->kn_data >= so->so_snd.sb_lowat);
|
||||||
|
@ -102,56 +102,71 @@ soisconnecting(so)
|
|||||||
register struct socket *so;
|
register struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
so->so_state &= ~(SS_ISCONNECTED|SS_ISDISCONNECTING);
|
so->so_state &= ~(SS_ISCONNECTED|SS_ISDISCONNECTING);
|
||||||
so->so_state |= SS_ISCONNECTING;
|
so->so_state |= SS_ISCONNECTING;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
soisconnected_locked(so)
|
||||||
|
struct socket *so;
|
||||||
|
{
|
||||||
|
struct socket *head = so->so_head;
|
||||||
|
|
||||||
|
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;
|
||||||
|
so->so_upcall(so, so->so_upcallarg, 0);
|
||||||
|
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);
|
||||||
|
head->so_qlen++;
|
||||||
|
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
|
void
|
||||||
soisconnected(so)
|
soisconnected(so)
|
||||||
struct socket *so;
|
struct socket *so;
|
||||||
{
|
{
|
||||||
struct socket *head;
|
struct socket *head = so->so_head;
|
||||||
so_upcall_t *upcp;
|
|
||||||
void *upcarg;
|
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
head = so->so_head;
|
|
||||||
so->so_state &= ~(SS_ISCONNECTING|SS_ISDISCONNECTING|SS_ISCONFIRMING);
|
so->so_state &= ~(SS_ISCONNECTING|SS_ISDISCONNECTING|SS_ISCONFIRMING);
|
||||||
so->so_state |= SS_ISCONNECTED;
|
so->so_state |= SS_ISCONNECTED;
|
||||||
if (head && (so->so_state & SS_INCOMP)) {
|
if (head && (so->so_state & SS_INCOMP)) {
|
||||||
if ((so->so_options & SO_ACCEPTFILTER) != 0) {
|
if ((so->so_options & SO_ACCEPTFILTER) != 0) {
|
||||||
SOCK_UNLOCK(so);
|
so->so_upcall = head->so_accf->so_accept_filter->accf_callback;
|
||||||
SOCK_LOCK(head);
|
so->so_upcallarg = head->so_accf->so_accept_filter_arg;
|
||||||
upcp = head->so_accf->so_accept_filter->accf_callback;
|
|
||||||
upcarg = head->so_accf->so_accept_filter_arg;
|
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_upcall = upcp;
|
|
||||||
so->so_upcallarg = upcarg;
|
|
||||||
so->so_rcv.sb_flags |= SB_UPCALL;
|
so->so_rcv.sb_flags |= SB_UPCALL;
|
||||||
so->so_options &= ~SO_ACCEPTFILTER;
|
so->so_options &= ~SO_ACCEPTFILTER;
|
||||||
SOCK_UNLOCK(so);
|
so->so_upcall(so, so->so_upcallarg, 0);
|
||||||
so->so_upcall(so, upcarg, 0);
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
so->so_state &= ~SS_INCOMP;
|
|
||||||
so->so_state |= SS_COMP;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
SOCK_LOCK(head);
|
|
||||||
TAILQ_REMOVE(&head->so_incomp, so, so_list);
|
TAILQ_REMOVE(&head->so_incomp, so, so_list);
|
||||||
head->so_incqlen--;
|
head->so_incqlen--;
|
||||||
|
so->so_state &= ~SS_INCOMP;
|
||||||
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
|
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
|
||||||
head->so_qlen++;
|
head->so_qlen++;
|
||||||
sorwakeup(head);
|
so->so_state |= SS_COMP;
|
||||||
|
sorwakeup_locked(head);
|
||||||
wakeup_one(&head->so_timeo);
|
wakeup_one(&head->so_timeo);
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
} else {
|
} else {
|
||||||
wakeup(&so->so_timeo);
|
wakeup(&so->so_timeo);
|
||||||
sorwakeup(so);
|
sorwakeup_locked(so);
|
||||||
sowwakeup(so);
|
sowwakeup_locked(so);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -160,12 +175,23 @@ soisdisconnecting(so)
|
|||||||
register struct socket *so;
|
register struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
so->so_state &= ~SS_ISCONNECTING;
|
so->so_state &= ~SS_ISCONNECTING;
|
||||||
so->so_state |= (SS_ISDISCONNECTING|SS_CANTRCVMORE|SS_CANTSENDMORE);
|
so->so_state |= (SS_ISDISCONNECTING|SS_CANTRCVMORE|SS_CANTSENDMORE);
|
||||||
wakeup((caddr_t)&so->so_timeo);
|
wakeup((caddr_t)&so->so_timeo);
|
||||||
sowwakeup(so);
|
sowwakeup_locked(so);
|
||||||
sorwakeup(so);
|
sorwakeup_locked(so);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
soisdisconnected_locked(so)
|
||||||
|
register struct socket *so;
|
||||||
|
{
|
||||||
|
|
||||||
|
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_locked(so);
|
||||||
|
sorwakeup_locked(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -173,12 +199,7 @@ soisdisconnected(so)
|
|||||||
register struct socket *so;
|
register struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
soisdisconnected_locked(so);
|
||||||
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);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -203,32 +224,25 @@ sonewconn(head, connstatus)
|
|||||||
so = soalloc(0);
|
so = soalloc(0);
|
||||||
if (so == NULL)
|
if (so == NULL)
|
||||||
return ((struct socket *)0);
|
return ((struct socket *)0);
|
||||||
SOCK_LOCK(head);
|
|
||||||
if ((head->so_options & SO_ACCEPTFILTER) != 0)
|
if ((head->so_options & SO_ACCEPTFILTER) != 0)
|
||||||
connstatus = 0;
|
connstatus = 0;
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
so->so_head = head;
|
so->so_head = head;
|
||||||
so->so_type = head->so_type;
|
so->so_type = head->so_type;
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_options = head->so_options &~ SO_ACCEPTCONN;
|
so->so_options = head->so_options &~ SO_ACCEPTCONN;
|
||||||
so->so_linger = head->so_linger;
|
so->so_linger = head->so_linger;
|
||||||
so->so_state = head->so_state | SS_NOFDREF;
|
so->so_state = head->so_state | SS_NOFDREF;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
so->so_proto = head->so_proto;
|
so->so_proto = head->so_proto;
|
||||||
so->so_timeo = head->so_timeo;
|
so->so_timeo = head->so_timeo;
|
||||||
so->so_cred = crhold(head->so_cred);
|
so->so_cred = crhold(head->so_cred);
|
||||||
if (soreserve(so, head->so_snd.sb_hiwat, head->so_rcv.sb_hiwat) ||
|
if (soreserve(so, head->so_snd.sb_hiwat, head->so_rcv.sb_hiwat) ||
|
||||||
(*so->so_proto->pr_usrreqs->pru_attach)(so, 0, NULL)) {
|
(*so->so_proto->pr_usrreqs->pru_attach)(so, 0, NULL)) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
return ((struct socket *)0);
|
return ((struct socket *)0);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (connstatus) {
|
if (connstatus) {
|
||||||
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
|
TAILQ_INSERT_TAIL(&head->so_comp, so, so_list);
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_COMP;
|
so->so_state |= SS_COMP;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
head->so_qlen++;
|
head->so_qlen++;
|
||||||
} else {
|
} else {
|
||||||
if (head->so_incqlen > head->so_qlimit) {
|
if (head->so_incqlen > head->so_qlimit) {
|
||||||
@ -237,19 +251,13 @@ sonewconn(head, connstatus)
|
|||||||
(void) soabort(sp);
|
(void) soabort(sp);
|
||||||
}
|
}
|
||||||
TAILQ_INSERT_TAIL(&head->so_incomp, so, so_list);
|
TAILQ_INSERT_TAIL(&head->so_incomp, so, so_list);
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_INCOMP;
|
so->so_state |= SS_INCOMP;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
head->so_incqlen++;
|
head->so_incqlen++;
|
||||||
}
|
}
|
||||||
if (connstatus) {
|
if (connstatus) {
|
||||||
SOCK_LOCK(head);
|
sorwakeup_locked(head);
|
||||||
sorwakeup(head);
|
|
||||||
wakeup((caddr_t)&head->so_timeo);
|
wakeup((caddr_t)&head->so_timeo);
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= connstatus;
|
so->so_state |= connstatus;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
return (so);
|
return (so);
|
||||||
}
|
}
|
||||||
@ -269,10 +277,8 @@ socantsendmore(so)
|
|||||||
struct socket *so;
|
struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_CANTSENDMORE;
|
so->so_state |= SS_CANTSENDMORE;
|
||||||
sowwakeup(so);
|
sowwakeup_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -280,10 +286,8 @@ socantrcvmore(so)
|
|||||||
struct socket *so;
|
struct socket *so;
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_CANTRCVMORE;
|
so->so_state |= SS_CANTRCVMORE;
|
||||||
sorwakeup(so);
|
sorwakeup_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -332,7 +336,6 @@ sowakeup(so, sb)
|
|||||||
register struct socket *so;
|
register struct socket *so;
|
||||||
register struct sockbuf *sb;
|
register struct sockbuf *sb;
|
||||||
{
|
{
|
||||||
SOCK_ASSERT(so, MA_OWNED);
|
|
||||||
|
|
||||||
selwakeup(&sb->sb_sel);
|
selwakeup(&sb->sb_sel);
|
||||||
sb->sb_flags &= ~SB_SEL;
|
sb->sb_flags &= ~SB_SEL;
|
||||||
@ -340,23 +343,13 @@ sowakeup(so, sb)
|
|||||||
sb->sb_flags &= ~SB_WAIT;
|
sb->sb_flags &= ~SB_WAIT;
|
||||||
wakeup((caddr_t)&sb->sb_cc);
|
wakeup((caddr_t)&sb->sb_cc);
|
||||||
}
|
}
|
||||||
if ((so->so_state & SS_ASYNC) && so->so_sigio != NULL) {
|
if ((so->so_state & SS_ASYNC) && so->so_sigio != NULL)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
pgsigio(&so->so_sigio, SIGIO, 0);
|
pgsigio(&so->so_sigio, SIGIO, 0);
|
||||||
SOCK_LOCK(so);
|
if (sb->sb_flags & SB_UPCALL)
|
||||||
}
|
|
||||||
if (sb->sb_flags & SB_UPCALL) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
(*so->so_upcall)(so, so->so_upcallarg, M_DONTWAIT);
|
(*so->so_upcall)(so, so->so_upcallarg, M_DONTWAIT);
|
||||||
SOCK_LOCK(so);
|
if (sb->sb_flags & SB_AIO)
|
||||||
}
|
|
||||||
if (sb->sb_flags & SB_AIO) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
aio_swake(so, sb);
|
aio_swake(so, sb);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
KNOTE(&sb->sb_sel.si_note, 0);
|
KNOTE(&sb->sb_sel.si_note, 0);
|
||||||
SOCK_LOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -966,11 +959,9 @@ sotoxsocket(struct socket *so, struct xsocket *xso)
|
|||||||
xso->xso_len = sizeof *xso;
|
xso->xso_len = sizeof *xso;
|
||||||
xso->xso_so = so;
|
xso->xso_so = so;
|
||||||
xso->so_type = so->so_type;
|
xso->so_type = so->so_type;
|
||||||
SOCK_LOCK(so);
|
|
||||||
xso->so_options = so->so_options;
|
xso->so_options = so->so_options;
|
||||||
xso->so_linger = so->so_linger;
|
xso->so_linger = so->so_linger;
|
||||||
xso->so_state = so->so_state;
|
xso->so_state = so->so_state;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
xso->so_pcb = so->so_pcb;
|
xso->so_pcb = so->so_pcb;
|
||||||
xso->xso_protocol = so->so_proto->pr_protocol;
|
xso->xso_protocol = so->so_proto->pr_protocol;
|
||||||
xso->xso_family = so->so_proto->pr_domain->dom_family;
|
xso->xso_family = so->so_proto->pr_domain->dom_family;
|
||||||
|
@ -245,15 +245,12 @@ accept1(td, uap, compat)
|
|||||||
if (error)
|
if (error)
|
||||||
goto done2;
|
goto done2;
|
||||||
s = splnet();
|
s = splnet();
|
||||||
SOCK_LOCK(head);
|
|
||||||
if ((head->so_options & SO_ACCEPTCONN) == 0) {
|
if ((head->so_options & SO_ACCEPTCONN) == 0) {
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
error = EINVAL;
|
error = EINVAL;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
if ((head->so_state & SS_NBIO) && TAILQ_EMPTY(&head->so_comp)) {
|
if ((head->so_state & SS_NBIO) && TAILQ_EMPTY(&head->so_comp)) {
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
error = EWOULDBLOCK;
|
error = EWOULDBLOCK;
|
||||||
goto done;
|
goto done;
|
||||||
@ -263,16 +260,13 @@ accept1(td, uap, compat)
|
|||||||
head->so_error = ECONNABORTED;
|
head->so_error = ECONNABORTED;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
error = msleep((caddr_t)&head->so_timeo,
|
error = tsleep((caddr_t)&head->so_timeo, PSOCK | PCATCH,
|
||||||
SOCK_MTX(head), PSOCK | PCATCH,
|
|
||||||
"accept", 0);
|
"accept", 0);
|
||||||
if (error) {
|
if (error) {
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
if (head->so_error) {
|
if (head->so_error) {
|
||||||
error = head->so_error;
|
error = head->so_error;
|
||||||
head->so_error = 0;
|
head->so_error = 0;
|
||||||
@ -311,17 +305,13 @@ accept1(td, uap, compat)
|
|||||||
/* connection has been removed from the listen queue */
|
/* connection has been removed from the listen queue */
|
||||||
KNOTE(&head->so_rcv.sb_sel.si_note, 0);
|
KNOTE(&head->so_rcv.sb_sel.si_note, 0);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_COMP;
|
so->so_state &= ~SS_COMP;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
so->so_head = NULL;
|
so->so_head = NULL;
|
||||||
if (head->so_sigio != NULL)
|
if (head->so_sigio != NULL)
|
||||||
fsetown(fgetown(head->so_sigio), &so->so_sigio);
|
fsetown(fgetown(head->so_sigio), &so->so_sigio);
|
||||||
|
|
||||||
FILE_LOCK(nfp);
|
FILE_LOCK(nfp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soref(so); /* file descriptor reference */
|
soref(so); /* file descriptor reference */
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
nfp->f_data = (caddr_t)so; /* nfp has ref count from falloc */
|
nfp->f_data = (caddr_t)so; /* nfp has ref count from falloc */
|
||||||
nfp->f_flag = fflag;
|
nfp->f_flag = fflag;
|
||||||
nfp->f_ops = &socketops;
|
nfp->f_ops = &socketops;
|
||||||
@ -442,30 +432,24 @@ connect(td, uap)
|
|||||||
mtx_lock(&Giant);
|
mtx_lock(&Giant);
|
||||||
if ((error = fgetsock(td, uap->s, &so, NULL)) != 0)
|
if ((error = fgetsock(td, uap->s, &so, NULL)) != 0)
|
||||||
goto done2;
|
goto done2;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_NBIO) && (so->so_state & SS_ISCONNECTING)) {
|
if ((so->so_state & SS_NBIO) && (so->so_state & SS_ISCONNECTING)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = EALREADY;
|
error = EALREADY;
|
||||||
goto done1;
|
goto done1;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = getsockaddr(&sa, uap->name, uap->namelen);
|
error = getsockaddr(&sa, uap->name, uap->namelen);
|
||||||
if (error)
|
if (error)
|
||||||
goto done1;
|
goto done1;
|
||||||
error = soconnect(so, sa, td);
|
error = soconnect(so, sa, td);
|
||||||
if (error)
|
if (error)
|
||||||
goto bad;
|
goto bad;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_NBIO) && (so->so_state & SS_ISCONNECTING)) {
|
if ((so->so_state & SS_NBIO) && (so->so_state & SS_ISCONNECTING)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
FREE(sa, M_SONAME);
|
FREE(sa, M_SONAME);
|
||||||
error = EINPROGRESS;
|
error = EINPROGRESS;
|
||||||
goto done1;
|
goto done1;
|
||||||
}
|
}
|
||||||
s = splnet();
|
s = splnet();
|
||||||
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
||||||
error = msleep((caddr_t)&so->so_timeo,
|
error = tsleep((caddr_t)&so->so_timeo, PSOCK | PCATCH, "connec", 0);
|
||||||
SOCK_MTX(so), PSOCK | PCATCH, "connec", 0);
|
|
||||||
if (error)
|
if (error)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -473,12 +457,9 @@ connect(td, uap)
|
|||||||
error = so->so_error;
|
error = so->so_error;
|
||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
bad:
|
bad:
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_ISCONNECTING;
|
so->so_state &= ~SS_ISCONNECTING;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
FREE(sa, M_SONAME);
|
FREE(sa, M_SONAME);
|
||||||
if (error == ERESTART)
|
if (error == ERESTART)
|
||||||
error = EINTR;
|
error = EINTR;
|
||||||
@ -1431,13 +1412,10 @@ getpeername1(td, uap, compat)
|
|||||||
mtx_lock(&Giant);
|
mtx_lock(&Giant);
|
||||||
if ((error = fgetsock(td, uap->fdes, &so, NULL)) != 0)
|
if ((error = fgetsock(td, uap->fdes, &so, NULL)) != 0)
|
||||||
goto done2;
|
goto done2;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & (SS_ISCONNECTED|SS_ISCONFIRMING)) == 0) {
|
if ((so->so_state & (SS_ISCONNECTED|SS_ISCONFIRMING)) == 0) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ENOTCONN;
|
error = ENOTCONN;
|
||||||
goto done1;
|
goto done1;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = copyin((caddr_t)uap->alen, (caddr_t)&len, sizeof (len));
|
error = copyin((caddr_t)uap->alen, (caddr_t)&len, sizeof (len));
|
||||||
if (error)
|
if (error)
|
||||||
goto done1;
|
goto done1;
|
||||||
@ -1695,13 +1673,10 @@ sendfile(struct thread *td, struct sendfile_args *uap)
|
|||||||
error = EINVAL;
|
error = EINVAL;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ENOTCONN;
|
error = ENOTCONN;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (uap->offset < 0) {
|
if (uap->offset < 0) {
|
||||||
error = EINVAL;
|
error = EINVAL;
|
||||||
goto done;
|
goto done;
|
||||||
@ -1764,17 +1739,14 @@ retry_lookup:
|
|||||||
* Optimize the non-blocking case by looking at the socket space
|
* Optimize the non-blocking case by looking at the socket space
|
||||||
* before going to the extra work of constituting the sf_buf.
|
* before going to the extra work of constituting the sf_buf.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_NBIO) && sbspace(&so->so_snd) <= 0) {
|
if ((so->so_state & SS_NBIO) && sbspace(&so->so_snd) <= 0) {
|
||||||
if (so->so_state & SS_CANTSENDMORE)
|
if (so->so_state & SS_CANTSENDMORE)
|
||||||
error = EPIPE;
|
error = EPIPE;
|
||||||
else
|
else
|
||||||
error = EAGAIN;
|
error = EAGAIN;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbunlock(&so->so_snd);
|
sbunlock(&so->so_snd);
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/*
|
/*
|
||||||
* Attempt to look up the page.
|
* Attempt to look up the page.
|
||||||
*
|
*
|
||||||
@ -1897,7 +1869,6 @@ retry_space:
|
|||||||
* blocks before the pru_send (or more accurately, any blocking
|
* blocks before the pru_send (or more accurately, any blocking
|
||||||
* results in a loop back to here to re-check).
|
* results in a loop back to here to re-check).
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_CANTSENDMORE) || so->so_error) {
|
if ((so->so_state & SS_CANTSENDMORE) || so->so_error) {
|
||||||
if (so->so_state & SS_CANTSENDMORE) {
|
if (so->so_state & SS_CANTSENDMORE) {
|
||||||
error = EPIPE;
|
error = EPIPE;
|
||||||
@ -1905,7 +1876,6 @@ retry_space:
|
|||||||
error = so->so_error;
|
error = so->so_error;
|
||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
sbunlock(&so->so_snd);
|
sbunlock(&so->so_snd);
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -1918,14 +1888,12 @@ retry_space:
|
|||||||
*/
|
*/
|
||||||
if (sbspace(&so->so_snd) < so->so_snd.sb_lowat) {
|
if (sbspace(&so->so_snd) < so->so_snd.sb_lowat) {
|
||||||
if (so->so_state & SS_NBIO) {
|
if (so->so_state & SS_NBIO) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
sbunlock(&so->so_snd);
|
sbunlock(&so->so_snd);
|
||||||
splx(s);
|
splx(s);
|
||||||
error = EAGAIN;
|
error = EAGAIN;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = sbwait(&so->so_snd);
|
error = sbwait(&so->so_snd);
|
||||||
/*
|
/*
|
||||||
* An error from sbwait usually indicates that we've
|
* An error from sbwait usually indicates that we've
|
||||||
@ -1940,7 +1908,6 @@ retry_space:
|
|||||||
}
|
}
|
||||||
goto retry_space;
|
goto retry_space;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = (*so->so_proto->pr_usrreqs->pru_send)(so, 0, m, 0, 0, td);
|
error = (*so->so_proto->pr_usrreqs->pru_send)(so, 0, m, 0, 0, td);
|
||||||
splx(s);
|
splx(s);
|
||||||
if (error) {
|
if (error) {
|
||||||
|
@ -104,7 +104,6 @@ uipc_abort(struct socket *so)
|
|||||||
return EINVAL;
|
return EINVAL;
|
||||||
unp_drop(unp, ECONNABORTED);
|
unp_drop(unp, ECONNABORTED);
|
||||||
unp_detach(unp);
|
unp_detach(unp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -250,9 +249,7 @@ uipc_rcvd(struct socket *so, int flags)
|
|||||||
(void)chgsbsize(so2->so_cred->cr_uidinfo, &so2->so_snd.sb_hiwat,
|
(void)chgsbsize(so2->so_cred->cr_uidinfo, &so2->so_snd.sb_hiwat,
|
||||||
newhiwat, RLIM_INFINITY);
|
newhiwat, RLIM_INFINITY);
|
||||||
unp->unp_cc = so->so_rcv.sb_cc;
|
unp->unp_cc = so->so_rcv.sb_cc;
|
||||||
SOCK_LOCK(so2);
|
|
||||||
sowwakeup(so2);
|
sowwakeup(so2);
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -309,9 +306,7 @@ uipc_send(struct socket *so, int flags, struct mbuf *m, struct sockaddr *nam,
|
|||||||
else
|
else
|
||||||
from = &sun_noname;
|
from = &sun_noname;
|
||||||
if (sbappendaddr(&so2->so_rcv, from, m, control)) {
|
if (sbappendaddr(&so2->so_rcv, from, m, control)) {
|
||||||
SOCK_LOCK(so2);
|
|
||||||
sorwakeup(so2);
|
sorwakeup(so2);
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
m = 0;
|
m = 0;
|
||||||
control = 0;
|
control = 0;
|
||||||
} else
|
} else
|
||||||
@ -327,9 +322,7 @@ uipc_send(struct socket *so, int flags, struct mbuf *m, struct sockaddr *nam,
|
|||||||
* Note: A better implementation would complain
|
* Note: A better implementation would complain
|
||||||
* if not equal to the peer's address.
|
* if not equal to the peer's address.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (nam) {
|
if (nam) {
|
||||||
error = unp_connect(so, nam, td);
|
error = unp_connect(so, nam, td);
|
||||||
if (error)
|
if (error)
|
||||||
@ -341,11 +334,9 @@ uipc_send(struct socket *so, int flags, struct mbuf *m, struct sockaddr *nam,
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (so->so_state & SS_CANTSENDMORE) {
|
if (so->so_state & SS_CANTSENDMORE) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = EPIPE;
|
error = EPIPE;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (unp->unp_conn == 0)
|
if (unp->unp_conn == 0)
|
||||||
panic("uipc_send connected but no connection?");
|
panic("uipc_send connected but no connection?");
|
||||||
so2 = unp->unp_conn->unp_socket;
|
so2 = unp->unp_conn->unp_socket;
|
||||||
@ -367,9 +358,7 @@ uipc_send(struct socket *so, int flags, struct mbuf *m, struct sockaddr *nam,
|
|||||||
(void)chgsbsize(so->so_cred->cr_uidinfo, &so->so_snd.sb_hiwat,
|
(void)chgsbsize(so->so_cred->cr_uidinfo, &so->so_snd.sb_hiwat,
|
||||||
newhiwat, RLIM_INFINITY);
|
newhiwat, RLIM_INFINITY);
|
||||||
unp->unp_conn->unp_cc = so2->so_rcv.sb_cc;
|
unp->unp_conn->unp_cc = so2->so_rcv.sb_cc;
|
||||||
SOCK_LOCK(so2);
|
|
||||||
sorwakeup(so2);
|
sorwakeup(so2);
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
m = 0;
|
m = 0;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -574,9 +563,7 @@ unp_detach(unp)
|
|||||||
unp_disconnect(unp);
|
unp_disconnect(unp);
|
||||||
while (!LIST_EMPTY(&unp->unp_refs))
|
while (!LIST_EMPTY(&unp->unp_refs))
|
||||||
unp_drop(LIST_FIRST(&unp->unp_refs), ECONNRESET);
|
unp_drop(LIST_FIRST(&unp->unp_refs), ECONNRESET);
|
||||||
SOCK_LOCK(unp->unp_socket);
|
|
||||||
soisdisconnected(unp->unp_socket);
|
soisdisconnected(unp->unp_socket);
|
||||||
SOCK_UNLOCK(unp->unp_socket);
|
|
||||||
unp->unp_socket->so_pcb = 0;
|
unp->unp_socket->so_pcb = 0;
|
||||||
if (unp_rights) {
|
if (unp_rights) {
|
||||||
/*
|
/*
|
||||||
@ -710,14 +697,8 @@ unp_connect(so, nam, td)
|
|||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
if (so->so_proto->pr_flags & PR_CONNREQUIRED) {
|
if (so->so_proto->pr_flags & PR_CONNREQUIRED) {
|
||||||
SOCK_LOCK(so2);
|
if ((so2->so_options & SO_ACCEPTCONN) == 0 ||
|
||||||
if ((so2->so_options & SO_ACCEPTCONN) == 0) {
|
(so3 = sonewconn(so2, 0)) == 0) {
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
error = ECONNREFUSED;
|
|
||||||
goto bad;
|
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
if ((so3 = sonewconn(so2, 0)) == 0) {
|
|
||||||
error = ECONNREFUSED;
|
error = ECONNREFUSED;
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
@ -775,19 +756,13 @@ unp_connect2(so, so2)
|
|||||||
|
|
||||||
case SOCK_DGRAM:
|
case SOCK_DGRAM:
|
||||||
LIST_INSERT_HEAD(&unp2->unp_refs, unp, unp_reflink);
|
LIST_INSERT_HEAD(&unp2->unp_refs, unp, unp_reflink);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SOCK_STREAM:
|
case SOCK_STREAM:
|
||||||
unp2->unp_conn = unp;
|
unp2->unp_conn = unp;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
SOCK_LOCK(so2);
|
|
||||||
soisconnected(so2);
|
soisconnected(so2);
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -809,19 +784,13 @@ unp_disconnect(unp)
|
|||||||
|
|
||||||
case SOCK_DGRAM:
|
case SOCK_DGRAM:
|
||||||
LIST_REMOVE(unp, unp_reflink);
|
LIST_REMOVE(unp, unp_reflink);
|
||||||
SOCK_LOCK(unp->unp_socket);
|
|
||||||
unp->unp_socket->so_state &= ~SS_ISCONNECTED;
|
unp->unp_socket->so_state &= ~SS_ISCONNECTED;
|
||||||
SOCK_UNLOCK(unp->unp_socket);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SOCK_STREAM:
|
case SOCK_STREAM:
|
||||||
SOCK_LOCK(unp->unp_socket);
|
|
||||||
soisdisconnected(unp->unp_socket);
|
soisdisconnected(unp->unp_socket);
|
||||||
SOCK_UNLOCK(unp->unp_socket);
|
|
||||||
SOCK_LOCK(unp2->unp_socket);
|
|
||||||
unp2->unp_conn = 0;
|
unp2->unp_conn = 0;
|
||||||
soisdisconnected(unp2->unp_socket);
|
soisdisconnected(unp2->unp_socket);
|
||||||
SOCK_UNLOCK(unp2->unp_socket);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1440,10 +1440,8 @@ no_kqueue:
|
|||||||
*/
|
*/
|
||||||
so = (struct socket *)fp->f_data;
|
so = (struct socket *)fp->f_data;
|
||||||
s = splnet();
|
s = splnet();
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (((opcode == LIO_READ) && (!soreadable(so))) || ((opcode ==
|
if (((opcode == LIO_READ) && (!soreadable(so))) || ((opcode ==
|
||||||
LIO_WRITE) && (!sowriteable(so)))) {
|
LIO_WRITE) && (!sowriteable(so)))) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
TAILQ_INSERT_TAIL(&so->so_aiojobq, aiocbe, list);
|
TAILQ_INSERT_TAIL(&so->so_aiojobq, aiocbe, list);
|
||||||
TAILQ_INSERT_TAIL(&ki->kaio_sockqueue, aiocbe, plist);
|
TAILQ_INSERT_TAIL(&ki->kaio_sockqueue, aiocbe, plist);
|
||||||
if (opcode == LIO_READ)
|
if (opcode == LIO_READ)
|
||||||
@ -1457,7 +1455,6 @@ no_kqueue:
|
|||||||
error = 0;
|
error = 0;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,9 +36,7 @@
|
|||||||
|
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/domain.h>
|
#include <sys/domain.h>
|
||||||
#include <sys/lock.h>
|
|
||||||
#include <sys/malloc.h>
|
#include <sys/malloc.h>
|
||||||
#include <sys/mutex.h>
|
|
||||||
#include <sys/protosw.h>
|
#include <sys/protosw.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#include <sys/socketvar.h>
|
#include <sys/socketvar.h>
|
||||||
@ -100,7 +98,6 @@ raw_detach(rp)
|
|||||||
struct socket *so = rp->rcb_socket;
|
struct socket *so = rp->rcb_socket;
|
||||||
|
|
||||||
so->so_pcb = 0;
|
so->so_pcb = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
LIST_REMOVE(rp, list);
|
LIST_REMOVE(rp, list);
|
||||||
#ifdef notdef
|
#ifdef notdef
|
||||||
@ -124,12 +121,8 @@ raw_disconnect(rp)
|
|||||||
m_freem(dtom(rp->rcb_faddr));
|
m_freem(dtom(rp->rcb_faddr));
|
||||||
rp->rcb_faddr = 0;
|
rp->rcb_faddr = 0;
|
||||||
#endif
|
#endif
|
||||||
SOCK_LOCK(rp->rcb_socket);
|
if (rp->rcb_socket->so_state & SS_NOFDREF)
|
||||||
if (rp->rcb_socket->so_state & SS_NOFDREF) {
|
|
||||||
SOCK_UNLOCK(rp->rcb_socket);
|
|
||||||
raw_detach(rp);
|
raw_detach(rp);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(rp->rcb_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef notdef
|
#ifdef notdef
|
||||||
|
@ -37,7 +37,6 @@
|
|||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/lock.h>
|
#include <sys/lock.h>
|
||||||
#include <sys/mbuf.h>
|
#include <sys/mbuf.h>
|
||||||
#include <sys/mutex.h>
|
|
||||||
#include <sys/protosw.h>
|
#include <sys/protosw.h>
|
||||||
#include <sys/signalvar.h>
|
#include <sys/signalvar.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
@ -106,9 +105,7 @@ raw_input(m0, proto, src, dst)
|
|||||||
/* should notify about lost packet */
|
/* should notify about lost packet */
|
||||||
m_freem(n);
|
m_freem(n);
|
||||||
else {
|
else {
|
||||||
SOCK_LOCK(last);
|
|
||||||
sorwakeup(last);
|
sorwakeup(last);
|
||||||
SOCK_UNLOCK(last);
|
|
||||||
sockets++;
|
sockets++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -120,9 +117,7 @@ raw_input(m0, proto, src, dst)
|
|||||||
m, (struct mbuf *)0) == 0)
|
m, (struct mbuf *)0) == 0)
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
else {
|
else {
|
||||||
SOCK_LOCK(last);
|
|
||||||
sorwakeup(last);
|
sorwakeup(last);
|
||||||
SOCK_UNLOCK(last);
|
|
||||||
sockets++;
|
sockets++;
|
||||||
}
|
}
|
||||||
} else
|
} else
|
||||||
@ -150,11 +145,8 @@ raw_uabort(struct socket *so)
|
|||||||
if (rp == 0)
|
if (rp == 0)
|
||||||
return EINVAL;
|
return EINVAL;
|
||||||
raw_disconnect(rp);
|
raw_disconnect(rp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so); /* XXX huh? called after the sofree()? */
|
soisdisconnected(so); /* XXX huh? called after the sofree()? */
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -211,9 +203,7 @@ raw_udisconnect(struct socket *so)
|
|||||||
return ENOTCONN;
|
return ENOTCONN;
|
||||||
}
|
}
|
||||||
raw_disconnect(rp);
|
raw_disconnect(rp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -147,10 +147,8 @@ rts_attach(struct socket *so, int proto, struct thread *td)
|
|||||||
}
|
}
|
||||||
rp->rcb_faddr = &route_src;
|
rp->rcb_faddr = &route_src;
|
||||||
route_cb.any_count++;
|
route_cb.any_count++;
|
||||||
SOCK_LOCK(so);
|
soisconnected_locked(so);
|
||||||
soisconnected(so);
|
|
||||||
so->so_options |= SO_USELOOPBACK;
|
so->so_options |= SO_USELOOPBACK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -474,9 +472,7 @@ flush:
|
|||||||
/*
|
/*
|
||||||
* Check to see if we don't want our own messages.
|
* Check to see if we don't want our own messages.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_options & SO_USELOOPBACK) == 0) {
|
if ((so->so_options & SO_USELOOPBACK) == 0) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (route_cb.any_count <= 1) {
|
if (route_cb.any_count <= 1) {
|
||||||
if (rtm)
|
if (rtm)
|
||||||
Free(rtm);
|
Free(rtm);
|
||||||
@ -485,8 +481,7 @@ flush:
|
|||||||
}
|
}
|
||||||
/* There is another listener, so construct message */
|
/* There is another listener, so construct message */
|
||||||
rp = sotorawcb(so);
|
rp = sotorawcb(so);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (rtm) {
|
if (rtm) {
|
||||||
m_copyback(m, 0, rtm->rtm_msglen, (caddr_t)rtm);
|
m_copyback(m, 0, rtm->rtm_msglen, (caddr_t)rtm);
|
||||||
if (m->m_pkthdr.len < rtm->rtm_msglen) {
|
if (m->m_pkthdr.len < rtm->rtm_msglen) {
|
||||||
|
@ -410,9 +410,7 @@ ddp_input( m, ifp, elh, phase )
|
|||||||
/*
|
/*
|
||||||
* And wake up whatever might be waiting for it
|
* And wake up whatever might be waiting for it
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(ddp->ddp_socket);
|
|
||||||
sorwakeup( ddp->ddp_socket );
|
sorwakeup( ddp->ddp_socket );
|
||||||
SOCK_UNLOCK(ddp->ddp_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
|
@ -110,11 +110,8 @@ ddp_connect(struct socket *so, struct sockaddr *nam, struct thread *td)
|
|||||||
s = splnet();
|
s = splnet();
|
||||||
error = at_pcbconnect( ddp, nam, td );
|
error = at_pcbconnect( ddp, nam, td );
|
||||||
splx(s);
|
splx(s);
|
||||||
if ( error == 0 ) {
|
if ( error == 0 )
|
||||||
SOCK_LOCK( so );
|
|
||||||
soisconnected( so );
|
soisconnected( so );
|
||||||
SOCK_UNLOCK( so );
|
|
||||||
}
|
|
||||||
return(error);
|
return(error);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,9 +134,7 @@ ddp_disconnect(struct socket *so)
|
|||||||
at_pcbdisconnect( ddp );
|
at_pcbdisconnect( ddp );
|
||||||
ddp->ddp_fsat.sat_addr.s_node = ATADDR_ANYNODE;
|
ddp->ddp_fsat.sat_addr.s_node = ATADDR_ANYNODE;
|
||||||
splx(s);
|
splx(s);
|
||||||
SOCK_LOCK( so );
|
|
||||||
soisdisconnected( so );
|
soisdisconnected( so );
|
||||||
SOCK_UNLOCK( so );
|
|
||||||
return(0);
|
return(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -209,9 +204,7 @@ ddp_abort(struct socket *so)
|
|||||||
if ( ddp == NULL ) {
|
if ( ddp == NULL ) {
|
||||||
return(EINVAL);
|
return(EINVAL);
|
||||||
}
|
}
|
||||||
SOCK_LOCK( so );
|
|
||||||
soisdisconnected( so );
|
soisdisconnected( so );
|
||||||
SOCK_UNLOCK( so );
|
|
||||||
s = splnet();
|
s = splnet();
|
||||||
at_pcbdetach( so, ddp );
|
at_pcbdetach( so, ddp );
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -447,7 +440,6 @@ at_pcballoc( struct socket *so )
|
|||||||
static void
|
static void
|
||||||
at_pcbdetach( struct socket *so, struct ddpcb *ddp)
|
at_pcbdetach( struct socket *so, struct ddpcb *ddp)
|
||||||
{
|
{
|
||||||
SOCK_LOCK( so );
|
|
||||||
soisdisconnected( so );
|
soisdisconnected( so );
|
||||||
so->so_pcb = 0;
|
so->so_pcb = 0;
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
|
@ -110,11 +110,8 @@ ddp_connect(struct socket *so, struct sockaddr *nam, struct thread *td)
|
|||||||
s = splnet();
|
s = splnet();
|
||||||
error = at_pcbconnect( ddp, nam, td );
|
error = at_pcbconnect( ddp, nam, td );
|
||||||
splx(s);
|
splx(s);
|
||||||
if ( error == 0 ) {
|
if ( error == 0 )
|
||||||
SOCK_LOCK( so );
|
|
||||||
soisconnected( so );
|
soisconnected( so );
|
||||||
SOCK_UNLOCK( so );
|
|
||||||
}
|
|
||||||
return(error);
|
return(error);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,9 +134,7 @@ ddp_disconnect(struct socket *so)
|
|||||||
at_pcbdisconnect( ddp );
|
at_pcbdisconnect( ddp );
|
||||||
ddp->ddp_fsat.sat_addr.s_node = ATADDR_ANYNODE;
|
ddp->ddp_fsat.sat_addr.s_node = ATADDR_ANYNODE;
|
||||||
splx(s);
|
splx(s);
|
||||||
SOCK_LOCK( so );
|
|
||||||
soisdisconnected( so );
|
soisdisconnected( so );
|
||||||
SOCK_UNLOCK( so );
|
|
||||||
return(0);
|
return(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -209,9 +204,7 @@ ddp_abort(struct socket *so)
|
|||||||
if ( ddp == NULL ) {
|
if ( ddp == NULL ) {
|
||||||
return(EINVAL);
|
return(EINVAL);
|
||||||
}
|
}
|
||||||
SOCK_LOCK( so );
|
|
||||||
soisdisconnected( so );
|
soisdisconnected( so );
|
||||||
SOCK_UNLOCK( so );
|
|
||||||
s = splnet();
|
s = splnet();
|
||||||
at_pcbdetach( so, ddp );
|
at_pcbdetach( so, ddp );
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -447,7 +440,6 @@ at_pcballoc( struct socket *so )
|
|||||||
static void
|
static void
|
||||||
at_pcbdetach( struct socket *so, struct ddpcb *ddp)
|
at_pcbdetach( struct socket *so, struct ddpcb *ddp)
|
||||||
{
|
{
|
||||||
SOCK_LOCK( so );
|
|
||||||
soisdisconnected( so );
|
soisdisconnected( so );
|
||||||
so->so_pcb = 0;
|
so->so_pcb = 0;
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
|
@ -771,24 +771,19 @@ atm_aal5_cpcs_data(tok, m)
|
|||||||
* Ensure that the socket is able to receive data and
|
* Ensure that the socket is able to receive data and
|
||||||
* that there's room in the socket buffer
|
* that there's room in the socket buffer
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (((so->so_state & SS_ISCONNECTED) == 0) ||
|
if (((so->so_state & SS_ISCONNECTED) == 0) ||
|
||||||
(so->so_state & SS_CANTRCVMORE) ||
|
(so->so_state & SS_CANTRCVMORE) ||
|
||||||
(len > sbspace(&so->so_rcv))) {
|
(len > sbspace(&so->so_rcv))) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
atm_sock_stat.as_indrop[atp->atp_type]++;
|
atm_sock_stat.as_indrop[atp->atp_type]++;
|
||||||
KB_FREEALL(m);
|
KB_FREEALL(m);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Queue the data and notify the user
|
* Queue the data and notify the user
|
||||||
*/
|
*/
|
||||||
sbappendrecord(&so->so_rcv, m);
|
sbappendrecord(&so->so_rcv, m);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -840,12 +835,9 @@ atm_aal5_ctloutput(so, sopt)
|
|||||||
|
|
||||||
case T_ATM_ADD_LEAF:
|
case T_ATM_ADD_LEAF:
|
||||||
case T_ATM_DROP_LEAF:
|
case T_ATM_DROP_LEAF:
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
ATM_RETERR(ENOTCONN);
|
ATM_RETERR(ENOTCONN);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case T_ATM_CAUSE:
|
case T_ATM_CAUSE:
|
||||||
@ -853,12 +845,9 @@ atm_aal5_ctloutput(so, sopt)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_ISCONNECTED) {
|
if (so->so_state & SS_ISCONNECTED) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
ATM_RETERR(EISCONN);
|
ATM_RETERR(EISCONN);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -181,7 +181,6 @@ atm_sock_detach(so)
|
|||||||
* Break links and free control blocks
|
* Break links and free control blocks
|
||||||
*/
|
*/
|
||||||
so->so_pcb = NULL;
|
so->so_pcb = NULL;
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
|
|
||||||
uma_zfree(atm_pcb_zone, atp);
|
uma_zfree(atm_pcb_zone, atp);
|
||||||
@ -494,9 +493,7 @@ atm_sock_connect(so, addr, epp)
|
|||||||
/*
|
/*
|
||||||
* We're finally ready to initiate the ATM connection
|
* We're finally ready to initiate the ATM connection
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnecting(so);
|
soisconnecting(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
atm_sock_stat.as_connreq[atp->atp_type]++;
|
atm_sock_stat.as_connreq[atp->atp_type]++;
|
||||||
err = atm_cm_connect(epp, atp, &atp->atp_attr, &atp->atp_conn);
|
err = atm_cm_connect(epp, atp, &atp->atp_attr, &atp->atp_conn);
|
||||||
if (err == 0) {
|
if (err == 0) {
|
||||||
@ -504,9 +501,7 @@ atm_sock_connect(so, addr, epp)
|
|||||||
* Connection is setup
|
* Connection is setup
|
||||||
*/
|
*/
|
||||||
atm_sock_stat.as_conncomp[atp->atp_type]++;
|
atm_sock_stat.as_conncomp[atp->atp_type]++;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
} else if (err == EINPROGRESS) {
|
} else if (err == EINPROGRESS) {
|
||||||
/*
|
/*
|
||||||
@ -519,9 +514,7 @@ atm_sock_connect(so, addr, epp)
|
|||||||
* Call failed...
|
* Call failed...
|
||||||
*/
|
*/
|
||||||
atm_sock_stat.as_connfail[atp->atp_type]++;
|
atm_sock_stat.as_connfail[atp->atp_type]++;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return (err);
|
return (err);
|
||||||
@ -571,9 +564,7 @@ atm_sock_disconnect(so)
|
|||||||
atp->atp_conn = NULL;
|
atp->atp_conn = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
@ -670,9 +661,7 @@ atm_sock_peeraddr(so, addr)
|
|||||||
satm->satm_family = AF_ATM;
|
satm->satm_family = AF_ATM;
|
||||||
satm->satm_len = sizeof(*satm);
|
satm->satm_len = sizeof(*satm);
|
||||||
saddr = &satm->satm_addr.t_atm_sap_addr;
|
saddr = &satm->satm_addr.t_atm_sap_addr;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_ISCONNECTED) {
|
if (so->so_state & SS_ISCONNECTED) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
cvp = atp->atp_conn->co_connvc;
|
cvp = atp->atp_conn->co_connvc;
|
||||||
saddr->SVE_tag_addr = T_ATM_PRESENT;
|
saddr->SVE_tag_addr = T_ATM_PRESENT;
|
||||||
if (cvp->cvc_flags & CVCF_CALLER) {
|
if (cvp->cvc_flags & CVCF_CALLER) {
|
||||||
@ -691,7 +680,6 @@ atm_sock_peeraddr(so, addr)
|
|||||||
else
|
else
|
||||||
saddr->SVE_tag_selector = T_ATM_ABSENT;
|
saddr->SVE_tag_selector = T_ATM_ABSENT;
|
||||||
} else {
|
} else {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
saddr->SVE_tag_addr = T_ATM_ABSENT;
|
saddr->SVE_tag_addr = T_ATM_ABSENT;
|
||||||
saddr->SVE_tag_selector = T_ATM_ABSENT;
|
saddr->SVE_tag_selector = T_ATM_ABSENT;
|
||||||
saddr->address_format = T_ATM_ABSENT;
|
saddr->address_format = T_ATM_ABSENT;
|
||||||
@ -1099,12 +1087,10 @@ atm_sock_getopt(so, sopt, atp)
|
|||||||
* If socket is connected, return attributes for the VCC in use,
|
* If socket is connected, return attributes for the VCC in use,
|
||||||
* otherwise just return what the user has setup so far.
|
* otherwise just return what the user has setup so far.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_ISCONNECTED)
|
if (so->so_state & SS_ISCONNECTED)
|
||||||
ap = &atp->atp_conn->co_connvc->cvc_attr;
|
ap = &atp->atp_conn->co_connvc->cvc_attr;
|
||||||
else
|
else
|
||||||
ap = &atp->atp_attr;
|
ap = &atp->atp_attr;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
switch (sopt->sopt_name) {
|
switch (sopt->sopt_name) {
|
||||||
|
|
||||||
@ -1282,9 +1268,7 @@ atm_sock_connected(toku)
|
|||||||
* Connection is setup
|
* Connection is setup
|
||||||
*/
|
*/
|
||||||
atm_sock_stat.as_conncomp[atp->atp_type]++;
|
atm_sock_stat.as_conncomp[atp->atp_type]++;
|
||||||
SOCK_LOCK(atp->atp_socket);
|
|
||||||
soisconnected(atp->atp_socket);
|
soisconnected(atp->atp_socket);
|
||||||
SOCK_UNLOCK(atp->atp_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1318,7 +1302,6 @@ atm_sock_cleared(toku, cause)
|
|||||||
/*
|
/*
|
||||||
* Set user error code
|
* Set user error code
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_ISCONNECTED) {
|
if (so->so_state & SS_ISCONNECTED) {
|
||||||
so->so_error = ECONNRESET;
|
so->so_error = ECONNRESET;
|
||||||
atm_sock_stat.as_connclr[atp->atp_type]++;
|
atm_sock_stat.as_connclr[atp->atp_type]++;
|
||||||
@ -1337,9 +1320,7 @@ atm_sock_cleared(toku, cause)
|
|||||||
* Cleanup failed incoming connection setup
|
* Cleanup failed incoming connection setup
|
||||||
*/
|
*/
|
||||||
if (so->so_state & SS_NOFDREF) {
|
if (so->so_state & SS_NOFDREF) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
(void) atm_sock_detach(so);
|
(void) atm_sock_detach(so);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -507,17 +507,7 @@ static struct ng_type ng_ksocket_typestruct = {
|
|||||||
};
|
};
|
||||||
NETGRAPH_INIT(ksocket, &ng_ksocket_typestruct);
|
NETGRAPH_INIT(ksocket, &ng_ksocket_typestruct);
|
||||||
|
|
||||||
#define ERROUT(x) \
|
#define ERROUT(x) do { error = (x); goto done; } while (0)
|
||||||
do { \
|
|
||||||
error = (x); \
|
|
||||||
goto done; \
|
|
||||||
} while (0)
|
|
||||||
#define ERROUT_SOCK(x) \
|
|
||||||
do { \
|
|
||||||
error = (x); \
|
|
||||||
SOCK_UNLOCK(so); \
|
|
||||||
goto done; \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
/************************************************************************
|
/************************************************************************
|
||||||
NETGRAPH NODE STUFF
|
NETGRAPH NODE STUFF
|
||||||
@ -623,9 +613,7 @@ ng_ksocket_connect(hook_p hook)
|
|||||||
priv->so->so_upcall = ng_ksocket_incoming;
|
priv->so->so_upcall = ng_ksocket_incoming;
|
||||||
priv->so->so_rcv.sb_flags |= SB_UPCALL;
|
priv->so->so_rcv.sb_flags |= SB_UPCALL;
|
||||||
priv->so->so_snd.sb_flags |= SB_UPCALL;
|
priv->so->so_snd.sb_flags |= SB_UPCALL;
|
||||||
SOCK_LOCK(priv->so);
|
|
||||||
priv->so->so_state |= SS_NBIO;
|
priv->so->so_state |= SS_NBIO;
|
||||||
SOCK_UNLOCK(priv->so);
|
|
||||||
/*
|
/*
|
||||||
* --Original comment--
|
* --Original comment--
|
||||||
* On a cloned socket we may have already received one or more
|
* On a cloned socket we may have already received one or more
|
||||||
@ -720,10 +708,8 @@ ng_ksocket_rcvmsg(node_p node, item_p item, hook_p lasthook)
|
|||||||
ERROUT(ENXIO);
|
ERROUT(ENXIO);
|
||||||
|
|
||||||
/* Make sure the socket is capable of accepting */
|
/* Make sure the socket is capable of accepting */
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (!(so->so_options & SO_ACCEPTCONN))
|
if (!(so->so_options & SO_ACCEPTCONN))
|
||||||
ERROUT_SOCK(EINVAL);
|
ERROUT(EINVAL);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (priv->flags & KSF_ACCEPTING)
|
if (priv->flags & KSF_ACCEPTING)
|
||||||
ERROUT(EALREADY);
|
ERROUT(EALREADY);
|
||||||
|
|
||||||
@ -758,24 +744,18 @@ ng_ksocket_rcvmsg(node_p node, item_p item, hook_p lasthook)
|
|||||||
ERROUT(ENXIO);
|
ERROUT(ENXIO);
|
||||||
|
|
||||||
/* Do connect */
|
/* Do connect */
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_ISCONNECTING) != 0)
|
if ((so->so_state & SS_ISCONNECTING) != 0)
|
||||||
ERROUT_SOCK(EALREADY);
|
ERROUT(EALREADY);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if ((error = soconnect(so, sa, td)) != 0) {
|
if ((error = soconnect(so, sa, td)) != 0) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_ISCONNECTING;
|
so->so_state &= ~SS_ISCONNECTING;
|
||||||
ERROUT_SOCK(error);
|
ERROUT(error);
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
if ((so->so_state & SS_ISCONNECTING) != 0)
|
||||||
if ((so->so_state & SS_ISCONNECTING) != 0) {
|
|
||||||
/* We will notify the sender when we connect */
|
/* We will notify the sender when we connect */
|
||||||
priv->response_token = msg->header.token;
|
priv->response_token = msg->header.token;
|
||||||
raddr = priv->response_addr;
|
raddr = priv->response_addr;
|
||||||
priv->flags |= KSF_CONNECTING;
|
priv->flags |= KSF_CONNECTING;
|
||||||
ERROUT_SOCK(EINPROGRESS);
|
ERROUT(EINPROGRESS);
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -794,11 +774,9 @@ ng_ksocket_rcvmsg(node_p node, item_p item, hook_p lasthook)
|
|||||||
|
|
||||||
/* Get function */
|
/* Get function */
|
||||||
if (msg->header.cmd == NGM_KSOCKET_GETPEERNAME) {
|
if (msg->header.cmd == NGM_KSOCKET_GETPEERNAME) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state
|
if ((so->so_state
|
||||||
& (SS_ISCONNECTED|SS_ISCONFIRMING)) == 0)
|
& (SS_ISCONNECTED|SS_ISCONFIRMING)) == 0)
|
||||||
ERROUT_SOCK(ENOTCONN);
|
ERROUT(ENOTCONN);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
func = so->so_proto->pr_usrreqs->pru_peeraddr;
|
func = so->so_proto->pr_usrreqs->pru_peeraddr;
|
||||||
} else
|
} else
|
||||||
func = so->so_proto->pr_usrreqs->pru_sockaddr;
|
func = so->so_proto->pr_usrreqs->pru_sockaddr;
|
||||||
@ -1068,13 +1046,11 @@ ng_ksocket_incoming2(node_p node, hook_p hook, void *arg1, int waitflag)
|
|||||||
|
|
||||||
/* Check whether a pending connect operation has completed */
|
/* Check whether a pending connect operation has completed */
|
||||||
if (priv->flags & KSF_CONNECTING) {
|
if (priv->flags & KSF_CONNECTING) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((error = so->so_error) != 0) {
|
if ((error = so->so_error) != 0) {
|
||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
so->so_state &= ~SS_ISCONNECTING;
|
so->so_state &= ~SS_ISCONNECTING;
|
||||||
}
|
}
|
||||||
if (!(so->so_state & SS_ISCONNECTING)) {
|
if (!(so->so_state & SS_ISCONNECTING)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
NG_MKMESSAGE(response, NGM_KSOCKET_COOKIE,
|
NG_MKMESSAGE(response, NGM_KSOCKET_COOKIE,
|
||||||
NGM_KSOCKET_CONNECT, sizeof(int32_t), waitflag);
|
NGM_KSOCKET_CONNECT, sizeof(int32_t), waitflag);
|
||||||
if (response != NULL) {
|
if (response != NULL) {
|
||||||
@ -1090,8 +1066,7 @@ ng_ksocket_incoming2(node_p node, hook_p hook, void *arg1, int waitflag)
|
|||||||
priv->response_addr, NULL);
|
priv->response_addr, NULL);
|
||||||
}
|
}
|
||||||
priv->flags &= ~KSF_CONNECTING;
|
priv->flags &= ~KSF_CONNECTING;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check whether a pending accept operation has completed */
|
/* Check whether a pending accept operation has completed */
|
||||||
@ -1121,14 +1096,10 @@ ng_ksocket_incoming2(node_p node, hook_p hook, void *arg1, int waitflag)
|
|||||||
struct sockaddr *sa = NULL;
|
struct sockaddr *sa = NULL;
|
||||||
meta_p meta = NULL;
|
meta_p meta = NULL;
|
||||||
struct mbuf *n;
|
struct mbuf *n;
|
||||||
int sostate;
|
|
||||||
|
|
||||||
/* Try to get next packet from socket */
|
/* Try to get next packet from socket */
|
||||||
SOCK_LOCK(so);
|
|
||||||
sostate = so->so_state & SS_ISCONNECTED;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if ((error = (*so->so_proto->pr_usrreqs->pru_soreceive)
|
if ((error = (*so->so_proto->pr_usrreqs->pru_soreceive)
|
||||||
(so, sostate ? NULL : &sa,
|
(so, (so->so_state & SS_ISCONNECTED) ? NULL : &sa,
|
||||||
&auio, &m, (struct mbuf **)0, &flags)) != 0)
|
&auio, &m, (struct mbuf **)0, &flags)) != 0)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1175,17 +1146,14 @@ sendit: /* Forward data with optional peer sockaddr as meta info */
|
|||||||
* If the peer has closed the connection, forward a 0-length mbuf
|
* If the peer has closed the connection, forward a 0-length mbuf
|
||||||
* to indicate end-of-file.
|
* to indicate end-of-file.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTRCVMORE && !(priv->flags & KSF_EOFSEEN)) {
|
if (so->so_state & SS_CANTRCVMORE && !(priv->flags & KSF_EOFSEEN)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
MGETHDR(m, waitflag, MT_DATA);
|
MGETHDR(m, waitflag, MT_DATA);
|
||||||
if (m != NULL) {
|
if (m != NULL) {
|
||||||
m->m_len = m->m_pkthdr.len = 0;
|
m->m_len = m->m_pkthdr.len = 0;
|
||||||
NG_SEND_DATA_ONLY(error, priv->hook, m);
|
NG_SEND_DATA_ONLY(error, priv->hook, m);
|
||||||
}
|
}
|
||||||
priv->flags |= KSF_EOFSEEN;
|
priv->flags |= KSF_EOFSEEN;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1204,12 +1172,8 @@ ng_ksocket_check_accept(priv_p priv)
|
|||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
if (TAILQ_EMPTY(&head->so_comp)) {
|
if (TAILQ_EMPTY(&head->so_comp)) {
|
||||||
SOCK_LOCK(head);
|
if (head->so_state & SS_CANTRCVMORE)
|
||||||
if (head->so_state & SS_CANTRCVMORE) {
|
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
return ECONNABORTED;
|
return ECONNABORTED;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(head);
|
|
||||||
return EWOULDBLOCK;
|
return EWOULDBLOCK;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
@ -1240,10 +1204,8 @@ ng_ksocket_finish_accept(priv_p priv)
|
|||||||
|
|
||||||
/* XXX KNOTE(&head->so_rcv.sb_sel.si_note, 0); */
|
/* XXX KNOTE(&head->so_rcv.sb_sel.si_note, 0); */
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_COMP;
|
so->so_state &= ~SS_COMP;
|
||||||
so->so_state |= SS_NBIO;
|
so->so_state |= SS_NBIO;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
so->so_head = NULL;
|
so->so_head = NULL;
|
||||||
|
|
||||||
soaccept(so, &sa);
|
soaccept(so, &sa);
|
||||||
|
@ -734,9 +734,7 @@ ship_msg(struct ngpcb *pcbp, struct ng_mesg *msg, struct sockaddr_ng *addr)
|
|||||||
m_freem(mdata);
|
m_freem(mdata);
|
||||||
return (ENOBUFS);
|
return (ENOBUFS);
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -772,13 +770,9 @@ ngs_connect(hook_p hook)
|
|||||||
if ((priv->datasock)
|
if ((priv->datasock)
|
||||||
&& (priv->datasock->ng_socket)) {
|
&& (priv->datasock->ng_socket)) {
|
||||||
if (NG_NODE_NUMHOOKS(node) == 1) {
|
if (NG_NODE_NUMHOOKS(node) == 1) {
|
||||||
SOCK_LOCK(priv->datasock->ng_socket);
|
|
||||||
priv->datasock->ng_socket->so_state |= SS_ISCONNECTED;
|
priv->datasock->ng_socket->so_state |= SS_ISCONNECTED;
|
||||||
SOCK_UNLOCK(priv->datasock->ng_socket);
|
|
||||||
} else {
|
} else {
|
||||||
SOCK_LOCK(priv->datasock->ng_socket);
|
|
||||||
priv->datasock->ng_socket->so_state &= ~SS_ISCONNECTED;
|
priv->datasock->ng_socket->so_state &= ~SS_ISCONNECTED;
|
||||||
SOCK_UNLOCK(priv->datasock->ng_socket);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (0);
|
return (0);
|
||||||
@ -892,9 +886,7 @@ ngs_rcvdata(hook_p hook, item_p item)
|
|||||||
TRAP_ERROR;
|
TRAP_ERROR;
|
||||||
return (ENOBUFS);
|
return (ENOBUFS);
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -913,13 +905,9 @@ ngs_disconnect(hook_p hook)
|
|||||||
if ((priv->datasock)
|
if ((priv->datasock)
|
||||||
&& (priv->datasock->ng_socket)) {
|
&& (priv->datasock->ng_socket)) {
|
||||||
if (NG_NODE_NUMHOOKS(node) == 1) {
|
if (NG_NODE_NUMHOOKS(node) == 1) {
|
||||||
SOCK_LOCK(priv->datasock->ng_socket);
|
|
||||||
priv->datasock->ng_socket->so_state |= SS_ISCONNECTED;
|
priv->datasock->ng_socket->so_state |= SS_ISCONNECTED;
|
||||||
SOCK_UNLOCK(priv->datasock->ng_socket);
|
|
||||||
} else {
|
} else {
|
||||||
SOCK_LOCK(priv->datasock->ng_socket);
|
|
||||||
priv->datasock->ng_socket->so_state &= ~SS_ISCONNECTED;
|
priv->datasock->ng_socket->so_state &= ~SS_ISCONNECTED;
|
||||||
SOCK_UNLOCK(priv->datasock->ng_socket);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -944,17 +932,13 @@ ngs_shutdown(node_p node)
|
|||||||
struct ngpcb *const pcbp = priv->ctlsock;
|
struct ngpcb *const pcbp = priv->ctlsock;
|
||||||
|
|
||||||
if (dpcbp != NULL) {
|
if (dpcbp != NULL) {
|
||||||
SOCK_LOCK(dpcbp->ng_socket);
|
|
||||||
soisdisconnected(dpcbp->ng_socket);
|
soisdisconnected(dpcbp->ng_socket);
|
||||||
SOCK_UNLOCK(dpcbp->ng_socket);
|
|
||||||
dpcbp->sockdata = NULL;
|
dpcbp->sockdata = NULL;
|
||||||
priv->datasock = NULL;
|
priv->datasock = NULL;
|
||||||
priv->refs--;
|
priv->refs--;
|
||||||
}
|
}
|
||||||
if (pcbp != NULL) {
|
if (pcbp != NULL) {
|
||||||
SOCK_LOCK(pcbp->ng_socket);
|
|
||||||
soisdisconnected(pcbp->ng_socket);
|
soisdisconnected(pcbp->ng_socket);
|
||||||
SOCK_UNLOCK(pcbp->ng_socket);
|
|
||||||
pcbp->sockdata = NULL;
|
pcbp->sockdata = NULL;
|
||||||
priv->ctlsock = NULL;
|
priv->ctlsock = NULL;
|
||||||
priv->refs--;
|
priv->refs--;
|
||||||
|
@ -30,8 +30,6 @@
|
|||||||
|
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/kernel.h>
|
#include <sys/kernel.h>
|
||||||
#include <sys/lock.h>
|
|
||||||
#include <sys/mutex.h>
|
|
||||||
#include <sys/sysctl.h>
|
#include <sys/sysctl.h>
|
||||||
#include <sys/signalvar.h>
|
#include <sys/signalvar.h>
|
||||||
#include <sys/socketvar.h>
|
#include <sys/socketvar.h>
|
||||||
@ -59,15 +57,11 @@ static void
|
|||||||
sohasdata(struct socket *so, void *arg, int waitflag)
|
sohasdata(struct socket *so, void *arg, int waitflag)
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
if (!soreadable(so))
|
||||||
if (!soreadable(so)) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
so->so_upcall = NULL;
|
so->so_upcall = NULL;
|
||||||
so->so_rcv.sb_flags &= ~SB_UPCALL;
|
so->so_rcv.sb_flags &= ~SB_UPCALL;
|
||||||
soisconnected(so);
|
soisconnected_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -31,9 +31,7 @@
|
|||||||
|
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#include <sys/kernel.h>
|
#include <sys/kernel.h>
|
||||||
#include <sys/lock.h>
|
|
||||||
#include <sys/mbuf.h>
|
#include <sys/mbuf.h>
|
||||||
#include <sys/mutex.h>
|
|
||||||
#include <sys/signalvar.h>
|
#include <sys/signalvar.h>
|
||||||
#include <sys/sysctl.h>
|
#include <sys/sysctl.h>
|
||||||
#include <sys/socketvar.h>
|
#include <sys/socketvar.h>
|
||||||
@ -162,13 +160,11 @@ static void
|
|||||||
sohashttpget(struct socket *so, void *arg, int waitflag)
|
sohashttpget(struct socket *so, void *arg, int waitflag)
|
||||||
{
|
{
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_CANTRCVMORE) == 0 && !sbfull(&so->so_rcv)) {
|
if ((so->so_state & SS_CANTRCVMORE) == 0 && !sbfull(&so->so_rcv)) {
|
||||||
struct mbuf *m;
|
struct mbuf *m;
|
||||||
char *cmp;
|
char *cmp;
|
||||||
int cmplen, cc;
|
int cmplen, cc;
|
||||||
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
m = so->so_rcv.sb_mb;
|
m = so->so_rcv.sb_mb;
|
||||||
cc = so->so_rcv.sb_cc - 1;
|
cc = so->so_rcv.sb_cc - 1;
|
||||||
if (cc < 1)
|
if (cc < 1)
|
||||||
@ -201,16 +197,13 @@ sohashttpget(struct socket *so, void *arg, int waitflag)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
DPRINT("mbufstrcmp bad");
|
DPRINT("mbufstrcmp bad");
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
fallout:
|
fallout:
|
||||||
DPRINT("fallout");
|
DPRINT("fallout");
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_upcall = NULL;
|
so->so_upcall = NULL;
|
||||||
so->so_rcv.sb_flags &= ~SB_UPCALL;
|
so->so_rcv.sb_flags &= ~SB_UPCALL;
|
||||||
soisconnected(so);
|
soisconnected_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -220,12 +213,8 @@ soparsehttpvers(struct socket *so, void *arg, int waitflag)
|
|||||||
struct mbuf *m, *n;
|
struct mbuf *m, *n;
|
||||||
int i, cc, spaces, inspaces;
|
int i, cc, spaces, inspaces;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
if ((so->so_state & SS_CANTRCVMORE) != 0 || sbfull(&so->so_rcv))
|
||||||
if ((so->so_state & SS_CANTRCVMORE) != 0 || sbfull(&so->so_rcv)) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto fallout;
|
goto fallout;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
m = so->so_rcv.sb_mb;
|
m = so->so_rcv.sb_mb;
|
||||||
cc = so->so_rcv.sb_cc;
|
cc = so->so_rcv.sb_cc;
|
||||||
@ -294,11 +283,9 @@ readmore:
|
|||||||
|
|
||||||
fallout:
|
fallout:
|
||||||
DPRINT("fallout");
|
DPRINT("fallout");
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_upcall = NULL;
|
so->so_upcall = NULL;
|
||||||
so->so_rcv.sb_flags &= ~SB_UPCALL;
|
so->so_rcv.sb_flags &= ~SB_UPCALL;
|
||||||
soisconnected(so);
|
soisconnected_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -313,12 +300,8 @@ soishttpconnected(struct socket *so, void *arg, int waitflag)
|
|||||||
int ccleft, copied;
|
int ccleft, copied;
|
||||||
|
|
||||||
DPRINT("start");
|
DPRINT("start");
|
||||||
SOCK_LOCK(so);
|
if ((so->so_state & SS_CANTRCVMORE) != 0 || sbfull(&so->so_rcv))
|
||||||
if ((so->so_state & SS_CANTRCVMORE) != 0 || sbfull(&so->so_rcv)) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto gotit;
|
goto gotit;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Walk the socketbuffer and copy the last NCHRS (3) into a, b, and c
|
* Walk the socketbuffer and copy the last NCHRS (3) into a, b, and c
|
||||||
@ -370,10 +353,8 @@ readmore:
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
gotit:
|
gotit:
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_upcall = NULL;
|
so->so_upcall = NULL;
|
||||||
so->so_rcv.sb_flags &= ~SB_UPCALL;
|
so->so_rcv.sb_flags &= ~SB_UPCALL;
|
||||||
soisconnected(so);
|
soisconnected_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -183,20 +183,15 @@ in_pcbbind(inp, nam, td)
|
|||||||
struct sockaddr_in *sin;
|
struct sockaddr_in *sin;
|
||||||
struct inpcbinfo *pcbinfo = inp->inp_pcbinfo;
|
struct inpcbinfo *pcbinfo = inp->inp_pcbinfo;
|
||||||
u_short lport = 0;
|
u_short lport = 0;
|
||||||
int wild = 0, reuseport;
|
int wild = 0, reuseport = (so->so_options & SO_REUSEPORT);
|
||||||
int error, prison = 0;
|
int error, prison = 0;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
reuseport = (so->so_options & SO_REUSEPORT);
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (TAILQ_EMPTY(&in_ifaddrhead)) /* XXX broken! */
|
if (TAILQ_EMPTY(&in_ifaddrhead)) /* XXX broken! */
|
||||||
return (EADDRNOTAVAIL);
|
return (EADDRNOTAVAIL);
|
||||||
if (inp->inp_lport || inp->inp_laddr.s_addr != INADDR_ANY)
|
if (inp->inp_lport || inp->inp_laddr.s_addr != INADDR_ANY)
|
||||||
return (EINVAL);
|
return (EINVAL);
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_options & (SO_REUSEADDR|SO_REUSEPORT)) == 0)
|
if ((so->so_options & (SO_REUSEADDR|SO_REUSEPORT)) == 0)
|
||||||
wild = 1;
|
wild = 1;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (nam) {
|
if (nam) {
|
||||||
sin = (struct sockaddr_in *)nam;
|
sin = (struct sockaddr_in *)nam;
|
||||||
if (nam->sa_len != sizeof (*sin))
|
if (nam->sa_len != sizeof (*sin))
|
||||||
@ -221,10 +216,8 @@ in_pcbbind(inp, nam, td)
|
|||||||
* and a multicast address is bound on both
|
* and a multicast address is bound on both
|
||||||
* new and duplicated sockets.
|
* new and duplicated sockets.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_options & SO_REUSEADDR)
|
if (so->so_options & SO_REUSEADDR)
|
||||||
reuseport = SO_REUSEADDR|SO_REUSEPORT;
|
reuseport = SO_REUSEADDR|SO_REUSEPORT;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
} else if (sin->sin_addr.s_addr != INADDR_ANY) {
|
} else if (sin->sin_addr.s_addr != INADDR_ANY) {
|
||||||
sin->sin_port = 0; /* yech... */
|
sin->sin_port = 0; /* yech... */
|
||||||
bzero(&sin->sin_zero, sizeof(sin->sin_zero));
|
bzero(&sin->sin_zero, sizeof(sin->sin_zero));
|
||||||
@ -244,37 +237,13 @@ in_pcbbind(inp, nam, td)
|
|||||||
t = in_pcblookup_local(inp->inp_pcbinfo,
|
t = in_pcblookup_local(inp->inp_pcbinfo,
|
||||||
sin->sin_addr, lport,
|
sin->sin_addr, lport,
|
||||||
prison ? 0 : INPLOOKUP_WILDCARD);
|
prison ? 0 : INPLOOKUP_WILDCARD);
|
||||||
if (t != NULL) {
|
if (t &&
|
||||||
SOCK_LOCK(t->inp_socket);
|
(ntohl(sin->sin_addr.s_addr) != INADDR_ANY ||
|
||||||
if ((ntohl(sin->sin_addr.s_addr) != INADDR_ANY ||
|
ntohl(t->inp_laddr.s_addr) != INADDR_ANY ||
|
||||||
ntohl(t->inp_laddr.s_addr) != INADDR_ANY ||
|
(t->inp_socket->so_options &
|
||||||
(t->inp_socket->so_options &
|
SO_REUSEPORT) == 0) &&
|
||||||
SO_REUSEPORT) == 0) &&
|
(so->so_cred->cr_uid !=
|
||||||
(so->so_cred->cr_uid !=
|
t->inp_socket->so_cred->cr_uid)) {
|
||||||
t->inp_socket->so_cred->cr_uid)) {
|
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
#if defined(INET6)
|
|
||||||
if (ntohl(sin->sin_addr.s_addr) !=
|
|
||||||
INADDR_ANY ||
|
|
||||||
ntohl(t->inp_laddr.s_addr) !=
|
|
||||||
INADDR_ANY ||
|
|
||||||
INP_SOCKAF(so) ==
|
|
||||||
INP_SOCKAF(t->inp_socket))
|
|
||||||
#endif /* defined(INET6) */
|
|
||||||
return (EADDRINUSE);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (prison &&
|
|
||||||
prison_ip(td->td_ucred, 0, &sin->sin_addr.s_addr))
|
|
||||||
return (EADDRNOTAVAIL);
|
|
||||||
t = in_pcblookup_local(pcbinfo, sin->sin_addr,
|
|
||||||
lport, prison ? 0 : wild);
|
|
||||||
if (t != NULL) {
|
|
||||||
SOCK_LOCK(t->inp_socket);
|
|
||||||
if ((reuseport & t->inp_socket->so_options) == 0) {
|
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
#if defined(INET6)
|
#if defined(INET6)
|
||||||
if (ntohl(sin->sin_addr.s_addr) !=
|
if (ntohl(sin->sin_addr.s_addr) !=
|
||||||
INADDR_ANY ||
|
INADDR_ANY ||
|
||||||
@ -284,8 +253,24 @@ in_pcbbind(inp, nam, td)
|
|||||||
INP_SOCKAF(t->inp_socket))
|
INP_SOCKAF(t->inp_socket))
|
||||||
#endif /* defined(INET6) */
|
#endif /* defined(INET6) */
|
||||||
return (EADDRINUSE);
|
return (EADDRINUSE);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
}
|
||||||
|
if (prison &&
|
||||||
|
prison_ip(td->td_ucred, 0, &sin->sin_addr.s_addr))
|
||||||
|
return (EADDRNOTAVAIL);
|
||||||
|
t = in_pcblookup_local(pcbinfo, sin->sin_addr,
|
||||||
|
lport, prison ? 0 : wild);
|
||||||
|
if (t &&
|
||||||
|
(reuseport & t->inp_socket->so_options) == 0) {
|
||||||
|
#if defined(INET6)
|
||||||
|
if (ntohl(sin->sin_addr.s_addr) !=
|
||||||
|
INADDR_ANY ||
|
||||||
|
ntohl(t->inp_laddr.s_addr) !=
|
||||||
|
INADDR_ANY ||
|
||||||
|
INP_SOCKAF(so) ==
|
||||||
|
INP_SOCKAF(t->inp_socket))
|
||||||
|
#endif /* defined(INET6) */
|
||||||
|
return (EADDRINUSE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
inp->inp_laddr = sin->sin_addr;
|
inp->inp_laddr = sin->sin_addr;
|
||||||
@ -431,21 +416,17 @@ in_pcbladdr(inp, nam, plocal_sin)
|
|||||||
* destination, in case of sharing the cache with IPv6.
|
* destination, in case of sharing the cache with IPv6.
|
||||||
*/
|
*/
|
||||||
ro = &inp->inp_route;
|
ro = &inp->inp_route;
|
||||||
SOCK_LOCK(inp->inp_socket);
|
|
||||||
if (ro->ro_rt &&
|
if (ro->ro_rt &&
|
||||||
(ro->ro_dst.sa_family != AF_INET ||
|
(ro->ro_dst.sa_family != AF_INET ||
|
||||||
satosin(&ro->ro_dst)->sin_addr.s_addr !=
|
satosin(&ro->ro_dst)->sin_addr.s_addr !=
|
||||||
sin->sin_addr.s_addr ||
|
sin->sin_addr.s_addr ||
|
||||||
inp->inp_socket->so_options & SO_DONTROUTE)) {
|
inp->inp_socket->so_options & SO_DONTROUTE)) {
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
RTFREE(ro->ro_rt);
|
RTFREE(ro->ro_rt);
|
||||||
ro->ro_rt = (struct rtentry *)0;
|
ro->ro_rt = (struct rtentry *)0;
|
||||||
SOCK_LOCK(inp->inp_socket);
|
|
||||||
}
|
}
|
||||||
if ((inp->inp_socket->so_options & SO_DONTROUTE) == 0 && /*XXX*/
|
if ((inp->inp_socket->so_options & SO_DONTROUTE) == 0 && /*XXX*/
|
||||||
(ro->ro_rt == (struct rtentry *)0 ||
|
(ro->ro_rt == (struct rtentry *)0 ||
|
||||||
ro->ro_rt->rt_ifp == (struct ifnet *)0)) {
|
ro->ro_rt->rt_ifp == (struct ifnet *)0)) {
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
/* No route yet, so try to acquire one */
|
/* No route yet, so try to acquire one */
|
||||||
bzero(&ro->ro_dst, sizeof(struct sockaddr_in));
|
bzero(&ro->ro_dst, sizeof(struct sockaddr_in));
|
||||||
ro->ro_dst.sa_family = AF_INET;
|
ro->ro_dst.sa_family = AF_INET;
|
||||||
@ -453,8 +434,7 @@ in_pcbladdr(inp, nam, plocal_sin)
|
|||||||
((struct sockaddr_in *) &ro->ro_dst)->sin_addr =
|
((struct sockaddr_in *) &ro->ro_dst)->sin_addr =
|
||||||
sin->sin_addr;
|
sin->sin_addr;
|
||||||
rtalloc(ro);
|
rtalloc(ro);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
/*
|
/*
|
||||||
* If we found a route, use the address
|
* If we found a route, use the address
|
||||||
* corresponding to the outgoing interface
|
* corresponding to the outgoing interface
|
||||||
@ -568,12 +548,8 @@ in_pcbdisconnect(inp)
|
|||||||
inp->inp_faddr.s_addr = INADDR_ANY;
|
inp->inp_faddr.s_addr = INADDR_ANY;
|
||||||
inp->inp_fport = 0;
|
inp->inp_fport = 0;
|
||||||
in_pcbrehash(inp);
|
in_pcbrehash(inp);
|
||||||
SOCK_LOCK(inp->inp_socket);
|
if (inp->inp_socket->so_state & SS_NOFDREF)
|
||||||
if (inp->inp_socket->so_state & SS_NOFDREF) {
|
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
in_pcbdetach(inp);
|
in_pcbdetach(inp);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -589,7 +565,6 @@ in_pcbdetach(inp)
|
|||||||
inp->inp_gencnt = ++ipi->ipi_gencnt;
|
inp->inp_gencnt = ++ipi->ipi_gencnt;
|
||||||
in_pcbremlists(inp);
|
in_pcbremlists(inp);
|
||||||
so->so_pcb = 0;
|
so->so_pcb = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
if (inp->inp_options)
|
if (inp->inp_options)
|
||||||
(void)m_free(inp->inp_options);
|
(void)m_free(inp->inp_options);
|
||||||
|
@ -228,11 +228,8 @@ divert_packet(struct mbuf *m, int incoming, int port)
|
|||||||
if (sbappendaddr(&sa->so_rcv, (struct sockaddr *)&divsrc,
|
if (sbappendaddr(&sa->so_rcv, (struct sockaddr *)&divsrc,
|
||||||
m, (struct mbuf *)0) == 0)
|
m, (struct mbuf *)0) == 0)
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
else {
|
else
|
||||||
SOCK_LOCK(sa);
|
|
||||||
sorwakeup(sa);
|
sorwakeup(sa);
|
||||||
SOCK_UNLOCK(sa);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
ipstat.ips_noproto++;
|
ipstat.ips_noproto++;
|
||||||
@ -258,7 +255,6 @@ div_output(so, m, addr, control)
|
|||||||
register struct ip *const ip = mtod(m, struct ip *);
|
register struct ip *const ip = mtod(m, struct ip *);
|
||||||
struct sockaddr_in *sin = (struct sockaddr_in *)addr;
|
struct sockaddr_in *sin = (struct sockaddr_in *)addr;
|
||||||
int error = 0;
|
int error = 0;
|
||||||
int soopts;
|
|
||||||
|
|
||||||
if (control)
|
if (control)
|
||||||
m_freem(control); /* XXX */
|
m_freem(control); /* XXX */
|
||||||
@ -304,11 +300,8 @@ div_output(so, m, addr, control)
|
|||||||
|
|
||||||
/* Send packet to output processing */
|
/* Send packet to output processing */
|
||||||
ipstat.ips_rawout++; /* XXX */
|
ipstat.ips_rawout++; /* XXX */
|
||||||
SOCK_LOCK(so);
|
|
||||||
soopts = so->so_options & SO_DONTROUTE;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ip_output(m, inp->inp_options, &inp->inp_route,
|
error = ip_output(m, inp->inp_options, &inp->inp_route,
|
||||||
soopts |
|
(so->so_options & SO_DONTROUTE) |
|
||||||
IP_ALLOWBROADCAST | IP_RAWOUTPUT,
|
IP_ALLOWBROADCAST | IP_RAWOUTPUT,
|
||||||
inp->inp_moptions);
|
inp->inp_moptions);
|
||||||
} else {
|
} else {
|
||||||
@ -372,9 +365,7 @@ div_attach(struct socket *so, int proto, struct thread *td)
|
|||||||
inp->inp_flags |= INP_HDRINCL;
|
inp->inp_flags |= INP_HDRINCL;
|
||||||
/* The socket is always "connected" because
|
/* The socket is always "connected" because
|
||||||
we always know "where" to send the packet */
|
we always know "where" to send the packet */
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_ISCONNECTED;
|
so->so_state |= SS_ISCONNECTED;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -393,21 +384,15 @@ div_detach(struct socket *so)
|
|||||||
static int
|
static int
|
||||||
div_abort(struct socket *so)
|
div_abort(struct socket *so)
|
||||||
{
|
{
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return div_detach(so);
|
return div_detach(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
div_disconnect(struct socket *so)
|
div_disconnect(struct socket *so)
|
||||||
{
|
{
|
||||||
SOCK_LOCK(so);
|
if ((so->so_state & SS_ISCONNECTED) == 0)
|
||||||
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return ENOTCONN;
|
return ENOTCONN;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return div_abort(so);
|
return div_abort(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1836,18 +1836,15 @@ ip_savecontrol(inp, mp, ip, m)
|
|||||||
register struct ip *ip;
|
register struct ip *ip;
|
||||||
register struct mbuf *m;
|
register struct mbuf *m;
|
||||||
{
|
{
|
||||||
SOCK_LOCK(inp->inp_socket);
|
|
||||||
if (inp->inp_socket->so_options & SO_TIMESTAMP) {
|
if (inp->inp_socket->so_options & SO_TIMESTAMP) {
|
||||||
struct timeval tv;
|
struct timeval tv;
|
||||||
|
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
microtime(&tv);
|
microtime(&tv);
|
||||||
*mp = sbcreatecontrol((caddr_t) &tv, sizeof(tv),
|
*mp = sbcreatecontrol((caddr_t) &tv, sizeof(tv),
|
||||||
SCM_TIMESTAMP, SOL_SOCKET);
|
SCM_TIMESTAMP, SOL_SOCKET);
|
||||||
if (*mp)
|
if (*mp)
|
||||||
mp = &(*mp)->m_next;
|
mp = &(*mp)->m_next;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
if (inp->inp_flags & INP_RECVDSTADDR) {
|
if (inp->inp_flags & INP_RECVDSTADDR) {
|
||||||
*mp = sbcreatecontrol((caddr_t) &ip->ip_dst,
|
*mp = sbcreatecontrol((caddr_t) &ip->ip_dst,
|
||||||
sizeof(struct in_addr), IP_RECVDSTADDR, IPPROTO_IP);
|
sizeof(struct in_addr), IP_RECVDSTADDR, IPPROTO_IP);
|
||||||
|
@ -1133,9 +1133,7 @@ socket_send(s, mm, src)
|
|||||||
if (sbappendaddr(&s->so_rcv,
|
if (sbappendaddr(&s->so_rcv,
|
||||||
(struct sockaddr *)src,
|
(struct sockaddr *)src,
|
||||||
mm, (struct mbuf *)0) != 0) {
|
mm, (struct mbuf *)0) != 0) {
|
||||||
SOCK_LOCK(s);
|
|
||||||
sorwakeup(s);
|
sorwakeup(s);
|
||||||
SOCK_UNLOCK(s);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -153,16 +153,9 @@ rip_input(m, off)
|
|||||||
} else
|
} else
|
||||||
#endif /*IPSEC*/
|
#endif /*IPSEC*/
|
||||||
if (n) {
|
if (n) {
|
||||||
if (last->inp_flags & INP_CONTROLOPTS)
|
if (last->inp_flags & INP_CONTROLOPTS ||
|
||||||
ip_savecontrol(last, &opts, ip, n);
|
last->inp_socket->so_options & SO_TIMESTAMP)
|
||||||
else {
|
ip_savecontrol(last, &opts, ip, n);
|
||||||
SOCK_LOCK(last->inp_socket);
|
|
||||||
if(last->inp_socket->so_options & SO_TIMESTAMP) {
|
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
ip_savecontrol(last, &opts, ip, n);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
}
|
|
||||||
if (sbappendaddr(&last->inp_socket->so_rcv,
|
if (sbappendaddr(&last->inp_socket->so_rcv,
|
||||||
(struct sockaddr *)&ripsrc, n,
|
(struct sockaddr *)&ripsrc, n,
|
||||||
opts) == 0) {
|
opts) == 0) {
|
||||||
@ -170,11 +163,8 @@ rip_input(m, off)
|
|||||||
m_freem(n);
|
m_freem(n);
|
||||||
if (opts)
|
if (opts)
|
||||||
m_freem(opts);
|
m_freem(opts);
|
||||||
} else {
|
} else
|
||||||
SOCK_LOCK(last->inp_socket);
|
|
||||||
sorwakeup(last->inp_socket);
|
sorwakeup(last->inp_socket);
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
}
|
|
||||||
opts = 0;
|
opts = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -190,26 +180,16 @@ rip_input(m, off)
|
|||||||
} else
|
} else
|
||||||
#endif /*IPSEC*/
|
#endif /*IPSEC*/
|
||||||
if (last) {
|
if (last) {
|
||||||
if (last->inp_flags & INP_CONTROLOPTS)
|
if (last->inp_flags & INP_CONTROLOPTS ||
|
||||||
|
last->inp_socket->so_options & SO_TIMESTAMP)
|
||||||
ip_savecontrol(last, &opts, ip, m);
|
ip_savecontrol(last, &opts, ip, m);
|
||||||
else {
|
|
||||||
SOCK_LOCK(last->inp_socket);
|
|
||||||
if (last->inp_socket->so_options & SO_TIMESTAMP) {
|
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
ip_savecontrol(last, &opts, ip, m);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
}
|
|
||||||
if (sbappendaddr(&last->inp_socket->so_rcv,
|
if (sbappendaddr(&last->inp_socket->so_rcv,
|
||||||
(struct sockaddr *)&ripsrc, m, opts) == 0) {
|
(struct sockaddr *)&ripsrc, m, opts) == 0) {
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
if (opts)
|
if (opts)
|
||||||
m_freem(opts);
|
m_freem(opts);
|
||||||
} else {
|
} else
|
||||||
SOCK_LOCK(last->inp_socket);
|
|
||||||
sorwakeup(last->inp_socket);
|
sorwakeup(last->inp_socket);
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
ipstat.ips_noproto++;
|
ipstat.ips_noproto++;
|
||||||
@ -229,11 +209,8 @@ rip_output(m, so, dst)
|
|||||||
{
|
{
|
||||||
register struct ip *ip;
|
register struct ip *ip;
|
||||||
register struct inpcb *inp = sotoinpcb(so);
|
register struct inpcb *inp = sotoinpcb(so);
|
||||||
int flags;
|
int flags = (so->so_options & SO_DONTROUTE) | IP_ALLOWBROADCAST;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
flags = (so->so_options & SO_DONTROUTE) | IP_ALLOWBROADCAST;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/*
|
/*
|
||||||
* If the user handed us a complete IP packet, use it.
|
* If the user handed us a complete IP packet, use it.
|
||||||
* Otherwise, allocate an mbuf for a header and fill it in.
|
* Otherwise, allocate an mbuf for a header and fill it in.
|
||||||
@ -531,21 +508,15 @@ rip_detach(struct socket *so)
|
|||||||
static int
|
static int
|
||||||
rip_abort(struct socket *so)
|
rip_abort(struct socket *so)
|
||||||
{
|
{
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return rip_detach(so);
|
return rip_detach(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
rip_disconnect(struct socket *so)
|
rip_disconnect(struct socket *so)
|
||||||
{
|
{
|
||||||
SOCK_LOCK(so);
|
if ((so->so_state & SS_ISCONNECTED) == 0)
|
||||||
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return ENOTCONN;
|
return ENOTCONN;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return rip_abort(so);
|
return rip_abort(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -581,9 +552,7 @@ rip_connect(struct socket *so, struct sockaddr *nam, struct thread *td)
|
|||||||
(addr->sin_family != AF_IMPLINK))
|
(addr->sin_family != AF_IMPLINK))
|
||||||
return EAFNOSUPPORT;
|
return EAFNOSUPPORT;
|
||||||
inp->inp_faddr = addr->sin_addr;
|
inp->inp_faddr = addr->sin_addr;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -601,16 +570,13 @@ rip_send(struct socket *so, int flags, struct mbuf *m, struct sockaddr *nam,
|
|||||||
struct inpcb *inp = sotoinpcb(so);
|
struct inpcb *inp = sotoinpcb(so);
|
||||||
register u_long dst;
|
register u_long dst;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_ISCONNECTED) {
|
if (so->so_state & SS_ISCONNECTED) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (nam) {
|
if (nam) {
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
return EISCONN;
|
return EISCONN;
|
||||||
}
|
}
|
||||||
dst = inp->inp_faddr.s_addr;
|
dst = inp->inp_faddr.s_addr;
|
||||||
} else {
|
} else {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (nam == NULL) {
|
if (nam == NULL) {
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
return ENOTCONN;
|
return ENOTCONN;
|
||||||
|
@ -281,21 +281,15 @@ present:
|
|||||||
flags = q->tqe_th->th_flags & TH_FIN;
|
flags = q->tqe_th->th_flags & TH_FIN;
|
||||||
nq = LIST_NEXT(q, tqe_q);
|
nq = LIST_NEXT(q, tqe_q);
|
||||||
LIST_REMOVE(q, tqe_q);
|
LIST_REMOVE(q, tqe_q);
|
||||||
SOCK_LOCK(so);
|
if (so->so_state & SS_CANTRCVMORE)
|
||||||
if (so->so_state & SS_CANTRCVMORE) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
m_freem(q->tqe_m);
|
m_freem(q->tqe_m);
|
||||||
} else {
|
else
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbappend(&so->so_rcv, q->tqe_m);
|
sbappend(&so->so_rcv, q->tqe_m);
|
||||||
}
|
|
||||||
FREE(q, M_TSEGQ);
|
FREE(q, M_TSEGQ);
|
||||||
q = nq;
|
q = nq;
|
||||||
} while (q && q->tqe_th->th_seq == tp->rcv_nxt);
|
} while (q && q->tqe_th->th_seq == tp->rcv_nxt);
|
||||||
ND6_HINT(tp);
|
ND6_HINT(tp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (flags);
|
return (flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -644,7 +638,6 @@ findpcb:
|
|||||||
tiwin = th->th_win;
|
tiwin = th->th_win;
|
||||||
|
|
||||||
so = inp->inp_socket;
|
so = inp->inp_socket;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_options & (SO_DEBUG|SO_ACCEPTCONN)) {
|
if (so->so_options & (SO_DEBUG|SO_ACCEPTCONN)) {
|
||||||
struct in_conninfo inc;
|
struct in_conninfo inc;
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
@ -661,11 +654,8 @@ findpcb:
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
/* skip if this isn't a listen socket */
|
/* skip if this isn't a listen socket */
|
||||||
if ((so->so_options & SO_ACCEPTCONN) == 0) {
|
if ((so->so_options & SO_ACCEPTCONN) == 0)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto after_listen;
|
goto after_listen;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
inc.inc_isipv6 = isipv6;
|
inc.inc_isipv6 = isipv6;
|
||||||
if (isipv6) {
|
if (isipv6) {
|
||||||
@ -878,14 +868,11 @@ findpcb:
|
|||||||
tp->t_flags |= (TF_ACKNOW | TF_NEEDSYN);
|
tp->t_flags |= (TF_ACKNOW | TF_NEEDSYN);
|
||||||
|
|
||||||
tcpstat.tcps_connects++;
|
tcpstat.tcps_connects++;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto trimthenstep6;
|
goto trimthenstep6;
|
||||||
}
|
}
|
||||||
goto drop;
|
goto drop;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
after_listen:
|
after_listen:
|
||||||
|
|
||||||
/* XXX temp debugging */
|
/* XXX temp debugging */
|
||||||
@ -1017,9 +1004,7 @@ after_listen:
|
|||||||
tp->t_rxtcur,
|
tp->t_rxtcur,
|
||||||
tcp_timer_rexmt, tp);
|
tcp_timer_rexmt, tp);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
sowwakeup(so);
|
sowwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (so->so_snd.sb_cc)
|
if (so->so_snd.sb_cc)
|
||||||
(void) tcp_output(tp);
|
(void) tcp_output(tp);
|
||||||
return;
|
return;
|
||||||
@ -1042,9 +1027,7 @@ after_listen:
|
|||||||
*/
|
*/
|
||||||
m_adj(m, drop_hdrlen); /* delayed header drop */
|
m_adj(m, drop_hdrlen); /* delayed header drop */
|
||||||
sbappend(&so->so_rcv, m);
|
sbappend(&so->so_rcv, m);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (DELAY_ACK(tp)) {
|
if (DELAY_ACK(tp)) {
|
||||||
callout_reset(tp->tt_delack, tcp_delacktime,
|
callout_reset(tp->tt_delack, tcp_delacktime,
|
||||||
tcp_timer_delack, tp);
|
tcp_timer_delack, tp);
|
||||||
@ -1154,9 +1137,7 @@ after_listen:
|
|||||||
} else
|
} else
|
||||||
tp->t_flags &= ~TF_RCVD_CC;
|
tp->t_flags &= ~TF_RCVD_CC;
|
||||||
tcpstat.tcps_connects++;
|
tcpstat.tcps_connects++;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/* Do window scaling on this connection? */
|
/* Do window scaling on this connection? */
|
||||||
if ((tp->t_flags & (TF_RCVD_SCALE|TF_REQ_SCALE)) ==
|
if ((tp->t_flags & (TF_RCVD_SCALE|TF_REQ_SCALE)) ==
|
||||||
(TF_RCVD_SCALE|TF_REQ_SCALE)) {
|
(TF_RCVD_SCALE|TF_REQ_SCALE)) {
|
||||||
@ -1486,16 +1467,13 @@ trimthenstep6:
|
|||||||
* If new data are received on a connection after the
|
* If new data are received on a connection after the
|
||||||
* user processes are gone, then RST the other end.
|
* user processes are gone, then RST the other end.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_NOFDREF) &&
|
if ((so->so_state & SS_NOFDREF) &&
|
||||||
tp->t_state > TCPS_CLOSE_WAIT && tlen) {
|
tp->t_state > TCPS_CLOSE_WAIT && tlen) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tp = tcp_close(tp);
|
tp = tcp_close(tp);
|
||||||
tcpstat.tcps_rcvafterclose++;
|
tcpstat.tcps_rcvafterclose++;
|
||||||
rstreason = BANDLIM_UNLIMITED;
|
rstreason = BANDLIM_UNLIMITED;
|
||||||
goto dropwithreset;
|
goto dropwithreset;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If segment ends after window, drop trailing data
|
* If segment ends after window, drop trailing data
|
||||||
@ -1585,9 +1563,7 @@ trimthenstep6:
|
|||||||
case TCPS_SYN_RECEIVED:
|
case TCPS_SYN_RECEIVED:
|
||||||
|
|
||||||
tcpstat.tcps_connects++;
|
tcpstat.tcps_connects++;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/* Do window scaling? */
|
/* Do window scaling? */
|
||||||
if ((tp->t_flags & (TF_RCVD_SCALE|TF_REQ_SCALE)) ==
|
if ((tp->t_flags & (TF_RCVD_SCALE|TF_REQ_SCALE)) ==
|
||||||
(TF_RCVD_SCALE|TF_REQ_SCALE)) {
|
(TF_RCVD_SCALE|TF_REQ_SCALE)) {
|
||||||
@ -1846,9 +1822,7 @@ process_ACK:
|
|||||||
tp->snd_wnd -= acked;
|
tp->snd_wnd -= acked;
|
||||||
ourfinisacked = 0;
|
ourfinisacked = 0;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
sowwakeup(so);
|
sowwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tp->snd_una = th->th_ack;
|
tp->snd_una = th->th_ack;
|
||||||
if (SEQ_LT(tp->snd_nxt, tp->snd_una))
|
if (SEQ_LT(tp->snd_nxt, tp->snd_una))
|
||||||
tp->snd_nxt = tp->snd_una;
|
tp->snd_nxt = tp->snd_una;
|
||||||
@ -1869,14 +1843,11 @@ process_ACK:
|
|||||||
* specification, but if we don't get a FIN
|
* specification, but if we don't get a FIN
|
||||||
* we'll hang forever.
|
* we'll hang forever.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTRCVMORE) {
|
if (so->so_state & SS_CANTRCVMORE) {
|
||||||
soisdisconnected(so);
|
soisdisconnected_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
callout_reset(tp->tt_2msl, tcp_maxidle,
|
callout_reset(tp->tt_2msl, tcp_maxidle,
|
||||||
tcp_timer_2msl, tp);
|
tcp_timer_2msl, tp);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tp->t_state = TCPS_FIN_WAIT_2;
|
tp->t_state = TCPS_FIN_WAIT_2;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -1901,9 +1872,7 @@ process_ACK:
|
|||||||
else
|
else
|
||||||
callout_reset(tp->tt_2msl, 2 * tcp_msl,
|
callout_reset(tp->tt_2msl, 2 * tcp_msl,
|
||||||
tcp_timer_2msl, tp);
|
tcp_timer_2msl, tp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1987,11 +1956,8 @@ step6:
|
|||||||
tp->rcv_up = th->th_seq + th->th_urp;
|
tp->rcv_up = th->th_seq + th->th_urp;
|
||||||
so->so_oobmark = so->so_rcv.sb_cc +
|
so->so_oobmark = so->so_rcv.sb_cc +
|
||||||
(tp->rcv_up - tp->rcv_nxt) - 1;
|
(tp->rcv_up - tp->rcv_nxt) - 1;
|
||||||
if (so->so_oobmark == 0) {
|
if (so->so_oobmark == 0)
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_RCVATMARK;
|
so->so_state |= SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
sohasoutofband(so);
|
sohasoutofband(so);
|
||||||
tp->t_oobflags &= ~(TCPOOB_HAVEDATA | TCPOOB_HADDATA);
|
tp->t_oobflags &= ~(TCPOOB_HAVEDATA | TCPOOB_HADDATA);
|
||||||
}
|
}
|
||||||
@ -2001,19 +1967,13 @@ step6:
|
|||||||
* but if two URG's are pending at once, some out-of-band
|
* but if two URG's are pending at once, some out-of-band
|
||||||
* data may creep in... ick.
|
* data may creep in... ick.
|
||||||
*/
|
*/
|
||||||
if (th->th_urp <= (u_long)tlen) {
|
if (th->th_urp <= (u_long)tlen
|
||||||
#ifdef SO_OOBINLINE
|
#ifdef SO_OOBINLINE
|
||||||
SOCK_LOCK(so);
|
&& (so->so_options & SO_OOBINLINE) == 0
|
||||||
if ((so->so_options & SO_OOBINLINE) == 0) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
tcp_pulloutofband(so, th, m,
|
)
|
||||||
drop_hdrlen); /* hdr drop is delayed */
|
tcp_pulloutofband(so, th, m,
|
||||||
#ifdef SO_OOBINLINE
|
drop_hdrlen); /* hdr drop is delayed */
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
} else
|
} else
|
||||||
/*
|
/*
|
||||||
* If no out of band data is expected,
|
* If no out of band data is expected,
|
||||||
@ -2059,9 +2019,7 @@ dodata: /* XXX */
|
|||||||
tcpstat.tcps_rcvbyte += tlen;
|
tcpstat.tcps_rcvbyte += tlen;
|
||||||
ND6_HINT(tp);
|
ND6_HINT(tp);
|
||||||
sbappend(&so->so_rcv, m);
|
sbappend(&so->so_rcv, m);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
} else {
|
} else {
|
||||||
thflags = tcp_reass(tp, th, &tlen, m);
|
thflags = tcp_reass(tp, th, &tlen, m);
|
||||||
tp->t_flags |= TF_ACKNOW;
|
tp->t_flags |= TF_ACKNOW;
|
||||||
@ -2140,9 +2098,7 @@ dodata: /* XXX */
|
|||||||
else
|
else
|
||||||
callout_reset(tp->tt_2msl, 2 * tcp_msl,
|
callout_reset(tp->tt_2msl, 2 * tcp_msl,
|
||||||
tcp_timer_2msl, tp);
|
tcp_timer_2msl, tp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2155,13 +2111,9 @@ dodata: /* XXX */
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG)
|
||||||
if (so->so_options & SO_DEBUG) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tcp_trace(TA_INPUT, ostate, tp, (void *)tcp_saveipgen,
|
tcp_trace(TA_INPUT, ostate, tp, (void *)tcp_saveipgen,
|
||||||
&tcp_savetcp, 0);
|
&tcp_savetcp, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2194,13 +2146,9 @@ dropafterack:
|
|||||||
goto dropwithreset;
|
goto dropwithreset;
|
||||||
}
|
}
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG)
|
||||||
if (so->so_options & SO_DEBUG) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
||||||
&tcp_savetcp, 0);
|
&tcp_savetcp, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
tp->t_flags |= TF_ACKNOW;
|
tp->t_flags |= TF_ACKNOW;
|
||||||
@ -2236,18 +2184,9 @@ dropwithreset:
|
|||||||
goto drop;
|
goto drop;
|
||||||
|
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp == 0)
|
if (tp == 0 || (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
||||||
&tcp_savetcp, 0);
|
&tcp_savetcp, 0);
|
||||||
else {
|
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
|
||||||
&tcp_savetcp, 0);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
if (thflags & TH_ACK)
|
if (thflags & TH_ACK)
|
||||||
/* mtod() below is safe as long as hdr dropping is delayed */
|
/* mtod() below is safe as long as hdr dropping is delayed */
|
||||||
@ -2267,18 +2206,9 @@ drop:
|
|||||||
* Drop space held by incoming segment and return.
|
* Drop space held by incoming segment and return.
|
||||||
*/
|
*/
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp == 0)
|
if (tp == 0 || (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
||||||
&tcp_savetcp, 0);
|
&tcp_savetcp, 0);
|
||||||
else {
|
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
|
||||||
&tcp_savetcp, 0);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
return;
|
return;
|
||||||
|
@ -126,7 +126,6 @@ tcp_output(tp)
|
|||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
int isipv6;
|
int isipv6;
|
||||||
#endif
|
#endif
|
||||||
int soopts;
|
|
||||||
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
isipv6 = (tp->t_inpcb->inp_vflag & INP_IPV6) != 0;
|
isipv6 = (tp->t_inpcb->inp_vflag & INP_IPV6) != 0;
|
||||||
@ -820,12 +819,8 @@ send:
|
|||||||
/*
|
/*
|
||||||
* Trace.
|
* Trace.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG)
|
||||||
if (so->so_options & SO_DEBUG) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tcp_trace(TA_OUTPUT, tp->t_state, tp, mtod(m, void *), th, 0);
|
tcp_trace(TA_OUTPUT, tp->t_state, tp, mtod(m, void *), th, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -859,13 +854,10 @@ send:
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
#endif /*IPSEC*/
|
#endif /*IPSEC*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
soopts = (so->so_options & SO_DONTROUTE);
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ip6_output(m,
|
error = ip6_output(m,
|
||||||
tp->t_inpcb->in6p_outputopts,
|
tp->t_inpcb->in6p_outputopts,
|
||||||
&tp->t_inpcb->in6p_route,
|
&tp->t_inpcb->in6p_route,
|
||||||
soopts, NULL, NULL);
|
(so->so_options & SO_DONTROUTE), NULL, NULL);
|
||||||
} else
|
} else
|
||||||
#endif /* INET6 */
|
#endif /* INET6 */
|
||||||
{
|
{
|
||||||
@ -897,11 +889,8 @@ send:
|
|||||||
#ifdef IPSEC
|
#ifdef IPSEC
|
||||||
ipsec_setsocket(m, so);
|
ipsec_setsocket(m, so);
|
||||||
#endif /*IPSEC*/
|
#endif /*IPSEC*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
soopts = (so->so_options & SO_DONTROUTE);
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ip_output(m, tp->t_inpcb->inp_options, &tp->t_inpcb->inp_route,
|
error = ip_output(m, tp->t_inpcb->inp_options, &tp->t_inpcb->inp_route,
|
||||||
soopts, 0);
|
(so->so_options & SO_DONTROUTE), 0);
|
||||||
}
|
}
|
||||||
if (error) {
|
if (error) {
|
||||||
|
|
||||||
|
@ -281,21 +281,15 @@ present:
|
|||||||
flags = q->tqe_th->th_flags & TH_FIN;
|
flags = q->tqe_th->th_flags & TH_FIN;
|
||||||
nq = LIST_NEXT(q, tqe_q);
|
nq = LIST_NEXT(q, tqe_q);
|
||||||
LIST_REMOVE(q, tqe_q);
|
LIST_REMOVE(q, tqe_q);
|
||||||
SOCK_LOCK(so);
|
if (so->so_state & SS_CANTRCVMORE)
|
||||||
if (so->so_state & SS_CANTRCVMORE) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
m_freem(q->tqe_m);
|
m_freem(q->tqe_m);
|
||||||
} else {
|
else
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbappend(&so->so_rcv, q->tqe_m);
|
sbappend(&so->so_rcv, q->tqe_m);
|
||||||
}
|
|
||||||
FREE(q, M_TSEGQ);
|
FREE(q, M_TSEGQ);
|
||||||
q = nq;
|
q = nq;
|
||||||
} while (q && q->tqe_th->th_seq == tp->rcv_nxt);
|
} while (q && q->tqe_th->th_seq == tp->rcv_nxt);
|
||||||
ND6_HINT(tp);
|
ND6_HINT(tp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (flags);
|
return (flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -644,7 +638,6 @@ findpcb:
|
|||||||
tiwin = th->th_win;
|
tiwin = th->th_win;
|
||||||
|
|
||||||
so = inp->inp_socket;
|
so = inp->inp_socket;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_options & (SO_DEBUG|SO_ACCEPTCONN)) {
|
if (so->so_options & (SO_DEBUG|SO_ACCEPTCONN)) {
|
||||||
struct in_conninfo inc;
|
struct in_conninfo inc;
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
@ -661,11 +654,8 @@ findpcb:
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
/* skip if this isn't a listen socket */
|
/* skip if this isn't a listen socket */
|
||||||
if ((so->so_options & SO_ACCEPTCONN) == 0) {
|
if ((so->so_options & SO_ACCEPTCONN) == 0)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto after_listen;
|
goto after_listen;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
inc.inc_isipv6 = isipv6;
|
inc.inc_isipv6 = isipv6;
|
||||||
if (isipv6) {
|
if (isipv6) {
|
||||||
@ -878,14 +868,11 @@ findpcb:
|
|||||||
tp->t_flags |= (TF_ACKNOW | TF_NEEDSYN);
|
tp->t_flags |= (TF_ACKNOW | TF_NEEDSYN);
|
||||||
|
|
||||||
tcpstat.tcps_connects++;
|
tcpstat.tcps_connects++;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto trimthenstep6;
|
goto trimthenstep6;
|
||||||
}
|
}
|
||||||
goto drop;
|
goto drop;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
after_listen:
|
after_listen:
|
||||||
|
|
||||||
/* XXX temp debugging */
|
/* XXX temp debugging */
|
||||||
@ -1017,9 +1004,7 @@ after_listen:
|
|||||||
tp->t_rxtcur,
|
tp->t_rxtcur,
|
||||||
tcp_timer_rexmt, tp);
|
tcp_timer_rexmt, tp);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
sowwakeup(so);
|
sowwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (so->so_snd.sb_cc)
|
if (so->so_snd.sb_cc)
|
||||||
(void) tcp_output(tp);
|
(void) tcp_output(tp);
|
||||||
return;
|
return;
|
||||||
@ -1042,9 +1027,7 @@ after_listen:
|
|||||||
*/
|
*/
|
||||||
m_adj(m, drop_hdrlen); /* delayed header drop */
|
m_adj(m, drop_hdrlen); /* delayed header drop */
|
||||||
sbappend(&so->so_rcv, m);
|
sbappend(&so->so_rcv, m);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (DELAY_ACK(tp)) {
|
if (DELAY_ACK(tp)) {
|
||||||
callout_reset(tp->tt_delack, tcp_delacktime,
|
callout_reset(tp->tt_delack, tcp_delacktime,
|
||||||
tcp_timer_delack, tp);
|
tcp_timer_delack, tp);
|
||||||
@ -1154,9 +1137,7 @@ after_listen:
|
|||||||
} else
|
} else
|
||||||
tp->t_flags &= ~TF_RCVD_CC;
|
tp->t_flags &= ~TF_RCVD_CC;
|
||||||
tcpstat.tcps_connects++;
|
tcpstat.tcps_connects++;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/* Do window scaling on this connection? */
|
/* Do window scaling on this connection? */
|
||||||
if ((tp->t_flags & (TF_RCVD_SCALE|TF_REQ_SCALE)) ==
|
if ((tp->t_flags & (TF_RCVD_SCALE|TF_REQ_SCALE)) ==
|
||||||
(TF_RCVD_SCALE|TF_REQ_SCALE)) {
|
(TF_RCVD_SCALE|TF_REQ_SCALE)) {
|
||||||
@ -1486,16 +1467,13 @@ trimthenstep6:
|
|||||||
* If new data are received on a connection after the
|
* If new data are received on a connection after the
|
||||||
* user processes are gone, then RST the other end.
|
* user processes are gone, then RST the other end.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_state & SS_NOFDREF) &&
|
if ((so->so_state & SS_NOFDREF) &&
|
||||||
tp->t_state > TCPS_CLOSE_WAIT && tlen) {
|
tp->t_state > TCPS_CLOSE_WAIT && tlen) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tp = tcp_close(tp);
|
tp = tcp_close(tp);
|
||||||
tcpstat.tcps_rcvafterclose++;
|
tcpstat.tcps_rcvafterclose++;
|
||||||
rstreason = BANDLIM_UNLIMITED;
|
rstreason = BANDLIM_UNLIMITED;
|
||||||
goto dropwithreset;
|
goto dropwithreset;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If segment ends after window, drop trailing data
|
* If segment ends after window, drop trailing data
|
||||||
@ -1585,9 +1563,7 @@ trimthenstep6:
|
|||||||
case TCPS_SYN_RECEIVED:
|
case TCPS_SYN_RECEIVED:
|
||||||
|
|
||||||
tcpstat.tcps_connects++;
|
tcpstat.tcps_connects++;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/* Do window scaling? */
|
/* Do window scaling? */
|
||||||
if ((tp->t_flags & (TF_RCVD_SCALE|TF_REQ_SCALE)) ==
|
if ((tp->t_flags & (TF_RCVD_SCALE|TF_REQ_SCALE)) ==
|
||||||
(TF_RCVD_SCALE|TF_REQ_SCALE)) {
|
(TF_RCVD_SCALE|TF_REQ_SCALE)) {
|
||||||
@ -1846,9 +1822,7 @@ process_ACK:
|
|||||||
tp->snd_wnd -= acked;
|
tp->snd_wnd -= acked;
|
||||||
ourfinisacked = 0;
|
ourfinisacked = 0;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
sowwakeup(so);
|
sowwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tp->snd_una = th->th_ack;
|
tp->snd_una = th->th_ack;
|
||||||
if (SEQ_LT(tp->snd_nxt, tp->snd_una))
|
if (SEQ_LT(tp->snd_nxt, tp->snd_una))
|
||||||
tp->snd_nxt = tp->snd_una;
|
tp->snd_nxt = tp->snd_una;
|
||||||
@ -1869,14 +1843,11 @@ process_ACK:
|
|||||||
* specification, but if we don't get a FIN
|
* specification, but if we don't get a FIN
|
||||||
* we'll hang forever.
|
* we'll hang forever.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTRCVMORE) {
|
if (so->so_state & SS_CANTRCVMORE) {
|
||||||
soisdisconnected(so);
|
soisdisconnected_locked(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
callout_reset(tp->tt_2msl, tcp_maxidle,
|
callout_reset(tp->tt_2msl, tcp_maxidle,
|
||||||
tcp_timer_2msl, tp);
|
tcp_timer_2msl, tp);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tp->t_state = TCPS_FIN_WAIT_2;
|
tp->t_state = TCPS_FIN_WAIT_2;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -1901,9 +1872,7 @@ process_ACK:
|
|||||||
else
|
else
|
||||||
callout_reset(tp->tt_2msl, 2 * tcp_msl,
|
callout_reset(tp->tt_2msl, 2 * tcp_msl,
|
||||||
tcp_timer_2msl, tp);
|
tcp_timer_2msl, tp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1987,11 +1956,8 @@ step6:
|
|||||||
tp->rcv_up = th->th_seq + th->th_urp;
|
tp->rcv_up = th->th_seq + th->th_urp;
|
||||||
so->so_oobmark = so->so_rcv.sb_cc +
|
so->so_oobmark = so->so_rcv.sb_cc +
|
||||||
(tp->rcv_up - tp->rcv_nxt) - 1;
|
(tp->rcv_up - tp->rcv_nxt) - 1;
|
||||||
if (so->so_oobmark == 0) {
|
if (so->so_oobmark == 0)
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_RCVATMARK;
|
so->so_state |= SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
sohasoutofband(so);
|
sohasoutofband(so);
|
||||||
tp->t_oobflags &= ~(TCPOOB_HAVEDATA | TCPOOB_HADDATA);
|
tp->t_oobflags &= ~(TCPOOB_HAVEDATA | TCPOOB_HADDATA);
|
||||||
}
|
}
|
||||||
@ -2001,19 +1967,13 @@ step6:
|
|||||||
* but if two URG's are pending at once, some out-of-band
|
* but if two URG's are pending at once, some out-of-band
|
||||||
* data may creep in... ick.
|
* data may creep in... ick.
|
||||||
*/
|
*/
|
||||||
if (th->th_urp <= (u_long)tlen) {
|
if (th->th_urp <= (u_long)tlen
|
||||||
#ifdef SO_OOBINLINE
|
#ifdef SO_OOBINLINE
|
||||||
SOCK_LOCK(so);
|
&& (so->so_options & SO_OOBINLINE) == 0
|
||||||
if ((so->so_options & SO_OOBINLINE) == 0) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
tcp_pulloutofband(so, th, m,
|
)
|
||||||
drop_hdrlen); /* hdr drop is delayed */
|
tcp_pulloutofband(so, th, m,
|
||||||
#ifdef SO_OOBINLINE
|
drop_hdrlen); /* hdr drop is delayed */
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
} else
|
} else
|
||||||
/*
|
/*
|
||||||
* If no out of band data is expected,
|
* If no out of band data is expected,
|
||||||
@ -2059,9 +2019,7 @@ dodata: /* XXX */
|
|||||||
tcpstat.tcps_rcvbyte += tlen;
|
tcpstat.tcps_rcvbyte += tlen;
|
||||||
ND6_HINT(tp);
|
ND6_HINT(tp);
|
||||||
sbappend(&so->so_rcv, m);
|
sbappend(&so->so_rcv, m);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
} else {
|
} else {
|
||||||
thflags = tcp_reass(tp, th, &tlen, m);
|
thflags = tcp_reass(tp, th, &tlen, m);
|
||||||
tp->t_flags |= TF_ACKNOW;
|
tp->t_flags |= TF_ACKNOW;
|
||||||
@ -2140,9 +2098,7 @@ dodata: /* XXX */
|
|||||||
else
|
else
|
||||||
callout_reset(tp->tt_2msl, 2 * tcp_msl,
|
callout_reset(tp->tt_2msl, 2 * tcp_msl,
|
||||||
tcp_timer_2msl, tp);
|
tcp_timer_2msl, tp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2155,13 +2111,9 @@ dodata: /* XXX */
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG)
|
||||||
if (so->so_options & SO_DEBUG) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tcp_trace(TA_INPUT, ostate, tp, (void *)tcp_saveipgen,
|
tcp_trace(TA_INPUT, ostate, tp, (void *)tcp_saveipgen,
|
||||||
&tcp_savetcp, 0);
|
&tcp_savetcp, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2194,13 +2146,9 @@ dropafterack:
|
|||||||
goto dropwithreset;
|
goto dropwithreset;
|
||||||
}
|
}
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG)
|
||||||
if (so->so_options & SO_DEBUG) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
||||||
&tcp_savetcp, 0);
|
&tcp_savetcp, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
tp->t_flags |= TF_ACKNOW;
|
tp->t_flags |= TF_ACKNOW;
|
||||||
@ -2236,18 +2184,9 @@ dropwithreset:
|
|||||||
goto drop;
|
goto drop;
|
||||||
|
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp == 0)
|
if (tp == 0 || (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
||||||
&tcp_savetcp, 0);
|
&tcp_savetcp, 0);
|
||||||
else {
|
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
|
||||||
&tcp_savetcp, 0);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
if (thflags & TH_ACK)
|
if (thflags & TH_ACK)
|
||||||
/* mtod() below is safe as long as hdr dropping is delayed */
|
/* mtod() below is safe as long as hdr dropping is delayed */
|
||||||
@ -2267,18 +2206,9 @@ drop:
|
|||||||
* Drop space held by incoming segment and return.
|
* Drop space held by incoming segment and return.
|
||||||
*/
|
*/
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp == 0)
|
if (tp == 0 || (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
||||||
&tcp_savetcp, 0);
|
&tcp_savetcp, 0);
|
||||||
else {
|
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_DROP, ostate, tp, (void *)tcp_saveipgen,
|
|
||||||
&tcp_savetcp, 0);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
return;
|
return;
|
||||||
|
@ -469,16 +469,8 @@ tcp_respond(tp, ipgen, th, m, ack, seq, flags)
|
|||||||
m->m_pkthdr.csum_data = offsetof(struct tcphdr, th_sum);
|
m->m_pkthdr.csum_data = offsetof(struct tcphdr, th_sum);
|
||||||
}
|
}
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp == NULL)
|
if (tp == NULL || (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
tcp_trace(TA_OUTPUT, 0, tp, mtod(m, void *), th, 0);
|
tcp_trace(TA_OUTPUT, 0, tp, mtod(m, void *), th, 0);
|
||||||
else {
|
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_OUTPUT, 0, tp, mtod(m, void *), th, 0);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef IPSEC
|
#ifdef IPSEC
|
||||||
if (ipsec_setsocket(m, tp ? tp->t_inpcb->inp_socket : NULL) != 0) {
|
if (ipsec_setsocket(m, tp ? tp->t_inpcb->inp_socket : NULL) != 0) {
|
||||||
@ -728,9 +720,7 @@ tcp_close(tp)
|
|||||||
FREE(q, M_TSEGQ);
|
FREE(q, M_TSEGQ);
|
||||||
}
|
}
|
||||||
inp->inp_ppcb = NULL;
|
inp->inp_ppcb = NULL;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
if (INP_CHECK_SOCKAF(so, AF_INET6))
|
if (INP_CHECK_SOCKAF(so, AF_INET6))
|
||||||
in6_pcbdetach(inp);
|
in6_pcbdetach(inp);
|
||||||
@ -803,11 +793,9 @@ tcp_notify(inp, error)
|
|||||||
else
|
else
|
||||||
tp->t_softerror = error;
|
tp->t_softerror = error;
|
||||||
#if 0
|
#if 0
|
||||||
SOCK_LOCK(so);
|
|
||||||
wakeup((caddr_t) &so->so_timeo);
|
wakeup((caddr_t) &so->so_timeo);
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
sowwakeup(so);
|
sowwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,15 +39,13 @@
|
|||||||
#include "opt_tcpdebug.h"
|
#include "opt_tcpdebug.h"
|
||||||
|
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
|
#include <sys/systm.h>
|
||||||
#include <sys/kernel.h>
|
#include <sys/kernel.h>
|
||||||
#include <sys/lock.h>
|
|
||||||
#include <sys/mbuf.h>
|
#include <sys/mbuf.h>
|
||||||
#include <sys/mutex.h>
|
#include <sys/sysctl.h>
|
||||||
#include <sys/protosw.h>
|
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#include <sys/socketvar.h>
|
#include <sys/socketvar.h>
|
||||||
#include <sys/sysctl.h>
|
#include <sys/protosw.h>
|
||||||
#include <sys/systm.h>
|
|
||||||
|
|
||||||
#include <machine/cpu.h> /* before tcp_seq.h, for tcp_random18() */
|
#include <machine/cpu.h> /* before tcp_seq.h, for tcp_random18() */
|
||||||
|
|
||||||
@ -213,15 +211,9 @@ tcp_timer_2msl(xtp)
|
|||||||
tp = tcp_close(tp);
|
tp = tcp_close(tp);
|
||||||
|
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp != 0) {
|
if (tp && (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
PRU_SLOWTIMO);
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
|
||||||
PRU_SLOWTIMO);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
splx(s);
|
splx(s);
|
||||||
}
|
}
|
||||||
@ -251,11 +243,9 @@ tcp_timer_keep(xtp)
|
|||||||
tcpstat.tcps_keeptimeo++;
|
tcpstat.tcps_keeptimeo++;
|
||||||
if (tp->t_state < TCPS_ESTABLISHED)
|
if (tp->t_state < TCPS_ESTABLISHED)
|
||||||
goto dropit;
|
goto dropit;
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
|
||||||
if ((always_keepalive ||
|
if ((always_keepalive ||
|
||||||
tp->t_inpcb->inp_socket->so_options & SO_KEEPALIVE) &&
|
tp->t_inpcb->inp_socket->so_options & SO_KEEPALIVE) &&
|
||||||
tp->t_state <= TCPS_CLOSING) {
|
tp->t_state <= TCPS_CLOSING) {
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
if ((ticks - tp->t_rcvtime) >= tcp_keepidle + tcp_maxidle)
|
if ((ticks - tp->t_rcvtime) >= tcp_keepidle + tcp_maxidle)
|
||||||
goto dropit;
|
goto dropit;
|
||||||
/*
|
/*
|
||||||
@ -279,19 +269,13 @@ tcp_timer_keep(xtp)
|
|||||||
(void) m_free(dtom(t_template));
|
(void) m_free(dtom(t_template));
|
||||||
}
|
}
|
||||||
callout_reset(tp->tt_keep, tcp_keepintvl, tcp_timer_keep, tp);
|
callout_reset(tp->tt_keep, tcp_keepintvl, tcp_timer_keep, tp);
|
||||||
} else {
|
} else
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
callout_reset(tp->tt_keep, tcp_keepidle, tcp_timer_keep, tp);
|
callout_reset(tp->tt_keep, tcp_keepidle, tcp_timer_keep, tp);
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
if (tp->t_inpcb->inp_socket->so_options & SO_DEBUG)
|
||||||
if (tp->t_inpcb->inp_socket->so_options & SO_DEBUG) {
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
||||||
PRU_SLOWTIMO);
|
PRU_SLOWTIMO);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
#endif
|
#endif
|
||||||
splx(s);
|
splx(s);
|
||||||
return;
|
return;
|
||||||
@ -301,15 +285,9 @@ dropit:
|
|||||||
tp = tcp_drop(tp, ETIMEDOUT);
|
tp = tcp_drop(tp, ETIMEDOUT);
|
||||||
|
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp != 0) {
|
if (tp && (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
PRU_SLOWTIMO);
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
|
||||||
PRU_SLOWTIMO);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
splx(s);
|
splx(s);
|
||||||
}
|
}
|
||||||
@ -357,15 +335,9 @@ tcp_timer_persist(xtp)
|
|||||||
|
|
||||||
out:
|
out:
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp != 0) {
|
if (tp && tp->t_inpcb->inp_socket->so_options & SO_DEBUG)
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
||||||
if (tp->t_inpcb->inp_socket->so_options & SO_DEBUG) {
|
PRU_SLOWTIMO);
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
|
||||||
PRU_SLOWTIMO);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
splx(s);
|
splx(s);
|
||||||
}
|
}
|
||||||
@ -498,15 +470,9 @@ tcp_timer_rexmt(xtp)
|
|||||||
|
|
||||||
out:
|
out:
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp != 0) {
|
if (tp && (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
PRU_SLOWTIMO);
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_USER, ostate, tp, (void *)0, (struct tcphdr *)0,
|
|
||||||
PRU_SLOWTIMO);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
splx(s);
|
splx(s);
|
||||||
}
|
}
|
||||||
|
@ -469,16 +469,8 @@ tcp_respond(tp, ipgen, th, m, ack, seq, flags)
|
|||||||
m->m_pkthdr.csum_data = offsetof(struct tcphdr, th_sum);
|
m->m_pkthdr.csum_data = offsetof(struct tcphdr, th_sum);
|
||||||
}
|
}
|
||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
if (tp == NULL)
|
if (tp == NULL || (tp->t_inpcb->inp_socket->so_options & SO_DEBUG))
|
||||||
tcp_trace(TA_OUTPUT, 0, tp, mtod(m, void *), th, 0);
|
tcp_trace(TA_OUTPUT, 0, tp, mtod(m, void *), th, 0);
|
||||||
else {
|
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
|
||||||
if ((tp->t_inpcb->inp_socket->so_options & SO_DEBUG)) {
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
tcp_trace(TA_OUTPUT, 0, tp, mtod(m, void *), th, 0);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef IPSEC
|
#ifdef IPSEC
|
||||||
if (ipsec_setsocket(m, tp ? tp->t_inpcb->inp_socket : NULL) != 0) {
|
if (ipsec_setsocket(m, tp ? tp->t_inpcb->inp_socket : NULL) != 0) {
|
||||||
@ -728,9 +720,7 @@ tcp_close(tp)
|
|||||||
FREE(q, M_TSEGQ);
|
FREE(q, M_TSEGQ);
|
||||||
}
|
}
|
||||||
inp->inp_ppcb = NULL;
|
inp->inp_ppcb = NULL;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
if (INP_CHECK_SOCKAF(so, AF_INET6))
|
if (INP_CHECK_SOCKAF(so, AF_INET6))
|
||||||
in6_pcbdetach(inp);
|
in6_pcbdetach(inp);
|
||||||
@ -803,11 +793,9 @@ tcp_notify(inp, error)
|
|||||||
else
|
else
|
||||||
tp->t_softerror = error;
|
tp->t_softerror = error;
|
||||||
#if 0
|
#if 0
|
||||||
SOCK_LOCK(so);
|
|
||||||
wakeup((caddr_t) &so->so_timeo);
|
wakeup((caddr_t) &so->so_timeo);
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
sowwakeup(so);
|
sowwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -103,17 +103,8 @@ static struct tcpcb *
|
|||||||
#ifdef TCPDEBUG
|
#ifdef TCPDEBUG
|
||||||
#define TCPDEBUG0 int ostate = 0
|
#define TCPDEBUG0 int ostate = 0
|
||||||
#define TCPDEBUG1() ostate = tp ? tp->t_state : 0
|
#define TCPDEBUG1() ostate = tp ? tp->t_state : 0
|
||||||
#define TCPDEBUG2(req) \
|
#define TCPDEBUG2(req) if (tp && (so->so_options & SO_DEBUG)) \
|
||||||
do { \
|
tcp_trace(TA_USER, ostate, tp, 0, 0, req)
|
||||||
if (tp != 0) { \
|
|
||||||
SOCK_LOCK(so); \
|
|
||||||
if (so->so_options & SO_DEBUG) { \
|
|
||||||
SOCK_UNLOCK(so); \
|
|
||||||
tcp_trace(TA_USER, ostate, tp, 0, 0, req); \
|
|
||||||
} else \
|
|
||||||
SOCK_UNLOCK(so); \
|
|
||||||
} \
|
|
||||||
} while(0)
|
|
||||||
#else
|
#else
|
||||||
#define TCPDEBUG0
|
#define TCPDEBUG0
|
||||||
#define TCPDEBUG1()
|
#define TCPDEBUG1()
|
||||||
@ -143,10 +134,8 @@ tcp_usr_attach(struct socket *so, int proto, struct thread *td)
|
|||||||
if (error)
|
if (error)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_options & SO_LINGER) && so->so_linger == 0)
|
if ((so->so_options & SO_LINGER) && so->so_linger == 0)
|
||||||
so->so_linger = TCP_LINGERTIME;
|
so->so_linger = TCP_LINGERTIME;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tp = sototcpcb(so);
|
tp = sototcpcb(so);
|
||||||
out:
|
out:
|
||||||
TCPDEBUG2(PRU_ATTACH);
|
TCPDEBUG2(PRU_ATTACH);
|
||||||
@ -435,13 +424,10 @@ tcp_usr_accept(struct socket *so, struct sockaddr **nam)
|
|||||||
struct tcpcb *tp = NULL;
|
struct tcpcb *tp = NULL;
|
||||||
TCPDEBUG0;
|
TCPDEBUG0;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_ISDISCONNECTED) {
|
if (so->so_state & SS_ISDISCONNECTED) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ECONNABORTED;
|
error = ECONNABORTED;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (inp == 0) {
|
if (inp == 0) {
|
||||||
splx(s);
|
splx(s);
|
||||||
return (EINVAL);
|
return (EINVAL);
|
||||||
@ -462,13 +448,10 @@ tcp6_usr_accept(struct socket *so, struct sockaddr **nam)
|
|||||||
struct tcpcb *tp = NULL;
|
struct tcpcb *tp = NULL;
|
||||||
TCPDEBUG0;
|
TCPDEBUG0;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_ISDISCONNECTED) {
|
if (so->so_state & SS_ISDISCONNECTED) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ECONNABORTED;
|
error = ECONNABORTED;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (inp == 0) {
|
if (inp == 0) {
|
||||||
splx(s);
|
splx(s);
|
||||||
return (EINVAL);
|
return (EINVAL);
|
||||||
@ -671,16 +654,10 @@ tcp_usr_rcvoob(struct socket *so, struct mbuf *m, int flags)
|
|||||||
struct tcpcb *tp;
|
struct tcpcb *tp;
|
||||||
|
|
||||||
COMMON_START();
|
COMMON_START();
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_oobmark == 0 &&
|
if ((so->so_oobmark == 0 &&
|
||||||
(so->so_state & SS_RCVATMARK) == 0) ||
|
(so->so_state & SS_RCVATMARK) == 0) ||
|
||||||
so->so_options & SO_OOBINLINE) {
|
so->so_options & SO_OOBINLINE ||
|
||||||
SOCK_UNLOCK(so);
|
tp->t_oobflags & TCPOOB_HADDATA) {
|
||||||
error = EINVAL;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (tp->t_oobflags & TCPOOB_HADDATA) {
|
|
||||||
error = EINVAL;
|
error = EINVAL;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
@ -778,9 +755,7 @@ tcp_connect(tp, nam, td)
|
|||||||
(TCP_MAXWIN << tp->request_r_scale) < so->so_rcv.sb_hiwat)
|
(TCP_MAXWIN << tp->request_r_scale) < so->so_rcv.sb_hiwat)
|
||||||
tp->request_r_scale++;
|
tp->request_r_scale++;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnecting(so);
|
soisconnecting(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tcpstat.tcps_connattempt++;
|
tcpstat.tcps_connattempt++;
|
||||||
tp->t_state = TCPS_SYN_SENT;
|
tp->t_state = TCPS_SYN_SENT;
|
||||||
callout_reset(tp->tt_keep, tcp_keepinit, tcp_timer_keep, tp);
|
callout_reset(tp->tt_keep, tcp_keepinit, tcp_timer_keep, tp);
|
||||||
@ -866,9 +841,7 @@ tcp6_connect(tp, nam, td)
|
|||||||
(TCP_MAXWIN << tp->request_r_scale) < so->so_rcv.sb_hiwat)
|
(TCP_MAXWIN << tp->request_r_scale) < so->so_rcv.sb_hiwat)
|
||||||
tp->request_r_scale++;
|
tp->request_r_scale++;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnecting(so);
|
soisconnecting(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
tcpstat.tcps_connattempt++;
|
tcpstat.tcps_connattempt++;
|
||||||
tp->t_state = TCPS_SYN_SENT;
|
tp->t_state = TCPS_SYN_SENT;
|
||||||
callout_reset(tp->tt_keep, tcp_keepinit, tcp_timer_keep, tp);
|
callout_reset(tp->tt_keep, tcp_keepinit, tcp_timer_keep, tp);
|
||||||
@ -1066,21 +1039,16 @@ tcp_attach(so, td)
|
|||||||
inp->inp_vflag |= INP_IPV4;
|
inp->inp_vflag |= INP_IPV4;
|
||||||
tp = tcp_newtcpcb(inp);
|
tp = tcp_newtcpcb(inp);
|
||||||
if (tp == 0) {
|
if (tp == 0) {
|
||||||
int nofd;
|
int nofd = so->so_state & SS_NOFDREF; /* XXX */
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
nofd = so->so_state & SS_NOFDREF; /* XXX */
|
|
||||||
so->so_state &= ~SS_NOFDREF; /* don't free the socket yet */
|
so->so_state &= ~SS_NOFDREF; /* don't free the socket yet */
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
if (isipv6)
|
if (isipv6)
|
||||||
in6_pcbdetach(inp);
|
in6_pcbdetach(inp);
|
||||||
else
|
else
|
||||||
#endif
|
#endif
|
||||||
in_pcbdetach(inp);
|
in_pcbdetach(inp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= nofd;
|
so->so_state |= nofd;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (ENOBUFS);
|
return (ENOBUFS);
|
||||||
}
|
}
|
||||||
tp->t_state = TCPS_CLOSED;
|
tp->t_state = TCPS_CLOSED;
|
||||||
@ -1103,19 +1071,14 @@ tcp_disconnect(tp)
|
|||||||
|
|
||||||
if (tp->t_state < TCPS_ESTABLISHED)
|
if (tp->t_state < TCPS_ESTABLISHED)
|
||||||
tp = tcp_close(tp);
|
tp = tcp_close(tp);
|
||||||
|
else if ((so->so_options & SO_LINGER) && so->so_linger == 0)
|
||||||
|
tp = tcp_drop(tp, 0);
|
||||||
else {
|
else {
|
||||||
SOCK_LOCK(so);
|
soisdisconnecting(so);
|
||||||
if ((so->so_options & SO_LINGER) && so->so_linger == 0) {
|
sbflush(&so->so_rcv);
|
||||||
SOCK_UNLOCK(so);
|
tp = tcp_usrclosed(tp);
|
||||||
tp = tcp_drop(tp, 0);
|
if (tp)
|
||||||
} else {
|
(void) tcp_output(tp);
|
||||||
soisdisconnecting(so);
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbflush(&so->so_rcv);
|
|
||||||
tp = tcp_usrclosed(tp);
|
|
||||||
if (tp)
|
|
||||||
(void) tcp_output(tp);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return (tp);
|
return (tp);
|
||||||
}
|
}
|
||||||
@ -1157,9 +1120,7 @@ tcp_usrclosed(tp)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (tp && tp->t_state >= TCPS_FIN_WAIT_2) {
|
if (tp && tp->t_state >= TCPS_FIN_WAIT_2) {
|
||||||
SOCK_LOCK(tp->t_inpcb->inp_socket);
|
|
||||||
soisdisconnected(tp->t_inpcb->inp_socket);
|
soisdisconnected(tp->t_inpcb->inp_socket);
|
||||||
SOCK_UNLOCK(tp->t_inpcb->inp_socket);
|
|
||||||
/* To prevent the connection hanging in FIN_WAIT_2 forever. */
|
/* To prevent the connection hanging in FIN_WAIT_2 forever. */
|
||||||
if (tp->t_state == TCPS_FIN_WAIT_2)
|
if (tp->t_state == TCPS_FIN_WAIT_2)
|
||||||
callout_reset(tp->tt_2msl, tcp_maxidle,
|
callout_reset(tp->tt_2msl, tcp_maxidle,
|
||||||
|
@ -319,12 +319,8 @@ udp_input(m, off)
|
|||||||
* port. It * assumes that an application will never
|
* port. It * assumes that an application will never
|
||||||
* clear these options after setting them.
|
* clear these options after setting them.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(last->inp_socket);
|
if ((last->inp_socket->so_options&(SO_REUSEPORT|SO_REUSEADDR)) == 0)
|
||||||
if ((last->inp_socket->so_options&(SO_REUSEPORT|SO_REUSEADDR)) == 0) {
|
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
break;
|
break;
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (last == NULL) {
|
if (last == NULL) {
|
||||||
@ -388,10 +384,8 @@ udp_input(m, off)
|
|||||||
*/
|
*/
|
||||||
udp_in.sin_port = uh->uh_sport;
|
udp_in.sin_port = uh->uh_sport;
|
||||||
udp_in.sin_addr = ip->ip_src;
|
udp_in.sin_addr = ip->ip_src;
|
||||||
SOCK_LOCK(inp->inp_socket);
|
|
||||||
if (inp->inp_flags & INP_CONTROLOPTS
|
if (inp->inp_flags & INP_CONTROLOPTS
|
||||||
|| inp->inp_socket->so_options & SO_TIMESTAMP) {
|
|| inp->inp_socket->so_options & SO_TIMESTAMP) {
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
if (inp->inp_vflag & INP_IPV6) {
|
if (inp->inp_vflag & INP_IPV6) {
|
||||||
int savedflags;
|
int savedflags;
|
||||||
@ -404,8 +398,7 @@ udp_input(m, off)
|
|||||||
} else
|
} else
|
||||||
#endif
|
#endif
|
||||||
ip_savecontrol(inp, &opts, ip, m);
|
ip_savecontrol(inp, &opts, ip, m);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
m_adj(m, iphlen + sizeof(struct udphdr));
|
m_adj(m, iphlen + sizeof(struct udphdr));
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
if (inp->inp_vflag & INP_IPV6) {
|
if (inp->inp_vflag & INP_IPV6) {
|
||||||
@ -418,9 +411,7 @@ udp_input(m, off)
|
|||||||
udpstat.udps_fullsock++;
|
udpstat.udps_fullsock++;
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(inp->inp_socket);
|
|
||||||
sorwakeup(inp->inp_socket);
|
sorwakeup(inp->inp_socket);
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
return;
|
return;
|
||||||
bad:
|
bad:
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
@ -462,10 +453,8 @@ udp_append(last, ip, n, off)
|
|||||||
struct sockaddr *append_sa;
|
struct sockaddr *append_sa;
|
||||||
struct mbuf *opts = 0;
|
struct mbuf *opts = 0;
|
||||||
|
|
||||||
SOCK_LOCK(last->inp_socket);
|
|
||||||
if (last->inp_flags & INP_CONTROLOPTS ||
|
if (last->inp_flags & INP_CONTROLOPTS ||
|
||||||
last->inp_socket->so_options & SO_TIMESTAMP) {
|
last->inp_socket->so_options & SO_TIMESTAMP) {
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
if (last->inp_vflag & INP_IPV6) {
|
if (last->inp_vflag & INP_IPV6) {
|
||||||
int savedflags;
|
int savedflags;
|
||||||
@ -481,8 +470,7 @@ udp_append(last, ip, n, off)
|
|||||||
} else
|
} else
|
||||||
#endif
|
#endif
|
||||||
ip_savecontrol(last, &opts, ip, n);
|
ip_savecontrol(last, &opts, ip, n);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
#ifdef INET6
|
#ifdef INET6
|
||||||
if (last->inp_vflag & INP_IPV6) {
|
if (last->inp_vflag & INP_IPV6) {
|
||||||
if (udp_in6.uin6_init_done == 0) {
|
if (udp_in6.uin6_init_done == 0) {
|
||||||
@ -499,11 +487,8 @@ udp_append(last, ip, n, off)
|
|||||||
if (opts)
|
if (opts)
|
||||||
m_freem(opts);
|
m_freem(opts);
|
||||||
udpstat.udps_fullsock++;
|
udpstat.udps_fullsock++;
|
||||||
} else {
|
} else
|
||||||
SOCK_LOCK(last->inp_socket);
|
|
||||||
sorwakeup(last->inp_socket);
|
sorwakeup(last->inp_socket);
|
||||||
SOCK_UNLOCK(last->inp_socket);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -516,10 +501,8 @@ udp_notify(inp, errno)
|
|||||||
int errno;
|
int errno;
|
||||||
{
|
{
|
||||||
inp->inp_socket->so_error = errno;
|
inp->inp_socket->so_error = errno;
|
||||||
SOCK_LOCK(inp->inp_socket);
|
|
||||||
sorwakeup(inp->inp_socket);
|
sorwakeup(inp->inp_socket);
|
||||||
sowwakeup(inp->inp_socket);
|
sowwakeup(inp->inp_socket);
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -695,7 +678,7 @@ udp_output(inp, m, addr, control, td)
|
|||||||
register int len = m->m_pkthdr.len;
|
register int len = m->m_pkthdr.len;
|
||||||
struct in_addr laddr;
|
struct in_addr laddr;
|
||||||
struct sockaddr_in *sin;
|
struct sockaddr_in *sin;
|
||||||
int s = 0, error = 0, soopts;
|
int s = 0, error = 0;
|
||||||
|
|
||||||
if (control)
|
if (control)
|
||||||
m_freem(control); /* XXX */
|
m_freem(control); /* XXX */
|
||||||
@ -776,11 +759,8 @@ udp_output(inp, m, addr, control, td)
|
|||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
#endif /*IPSEC*/
|
#endif /*IPSEC*/
|
||||||
SOCK_LOCK(inp->inp_socket);
|
|
||||||
soopts = inp->inp_socket->so_options & (SO_DONTROUTE | SO_BROADCAST);
|
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
error = ip_output(m, inp->inp_options, &inp->inp_route,
|
error = ip_output(m, inp->inp_options, &inp->inp_route,
|
||||||
soopts,
|
(inp->inp_socket->so_options & (SO_DONTROUTE | SO_BROADCAST)),
|
||||||
inp->inp_moptions);
|
inp->inp_moptions);
|
||||||
|
|
||||||
if (addr) {
|
if (addr) {
|
||||||
@ -819,9 +799,7 @@ udp_abort(struct socket *so)
|
|||||||
inp = sotoinpcb(so);
|
inp = sotoinpcb(so);
|
||||||
if (inp == 0)
|
if (inp == 0)
|
||||||
return EINVAL; /* ??? possible? panic instead? */
|
return EINVAL; /* ??? possible? panic instead? */
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
s = splnet();
|
s = splnet();
|
||||||
in_pcbdetach(inp);
|
in_pcbdetach(inp);
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -886,11 +864,8 @@ udp_connect(struct socket *so, struct sockaddr *nam, struct thread *td)
|
|||||||
prison_remote_ip(td->td_ucred, 0, &sin->sin_addr.s_addr);
|
prison_remote_ip(td->td_ucred, 0, &sin->sin_addr.s_addr);
|
||||||
error = in_pcbconnect(inp, nam, td);
|
error = in_pcbconnect(inp, nam, td);
|
||||||
splx(s);
|
splx(s);
|
||||||
if (error == 0) {
|
if (error == 0)
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -925,9 +900,7 @@ udp_disconnect(struct socket *so)
|
|||||||
in_pcbdisconnect(inp);
|
in_pcbdisconnect(inp);
|
||||||
inp->inp_laddr.s_addr = INADDR_ANY;
|
inp->inp_laddr.s_addr = INADDR_ANY;
|
||||||
splx(s);
|
splx(s);
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_ISCONNECTED; /* XXX */
|
so->so_state &= ~SS_ISCONNECTED; /* XXX */
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1941,11 +1941,8 @@ icmp6_rip6_input(mp, off)
|
|||||||
if (opts) {
|
if (opts) {
|
||||||
m_freem(opts);
|
m_freem(opts);
|
||||||
}
|
}
|
||||||
} else {
|
} else
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
sorwakeup(last->in6p_socket);
|
sorwakeup(last->in6p_socket);
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
opts = NULL;
|
opts = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1961,11 +1958,8 @@ icmp6_rip6_input(mp, off)
|
|||||||
m_freem(m);
|
m_freem(m);
|
||||||
if (opts)
|
if (opts)
|
||||||
m_freem(opts);
|
m_freem(opts);
|
||||||
} else {
|
} else
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
sorwakeup(last->in6p_socket);
|
sorwakeup(last->in6p_socket);
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
ip6stat.ip6s_delivered--;
|
ip6stat.ip6s_delivered--;
|
||||||
|
@ -124,19 +124,14 @@ in6_pcbbind(inp, nam, td)
|
|||||||
struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)NULL;
|
struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)NULL;
|
||||||
struct inpcbinfo *pcbinfo = inp->inp_pcbinfo;
|
struct inpcbinfo *pcbinfo = inp->inp_pcbinfo;
|
||||||
u_short lport = 0;
|
u_short lport = 0;
|
||||||
int wild = 0, reuseport;
|
int wild = 0, reuseport = (so->so_options & SO_REUSEPORT);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
reuseport = (so->so_options & SO_REUSEPORT);
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (!in6_ifaddr) /* XXX broken! */
|
if (!in6_ifaddr) /* XXX broken! */
|
||||||
return (EADDRNOTAVAIL);
|
return (EADDRNOTAVAIL);
|
||||||
if (inp->inp_lport || !IN6_IS_ADDR_UNSPECIFIED(&inp->in6p_laddr))
|
if (inp->inp_lport || !IN6_IS_ADDR_UNSPECIFIED(&inp->in6p_laddr))
|
||||||
return(EINVAL);
|
return(EINVAL);
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((so->so_options & (SO_REUSEADDR|SO_REUSEPORT)) == 0)
|
if ((so->so_options & (SO_REUSEADDR|SO_REUSEPORT)) == 0)
|
||||||
wild = 1;
|
wild = 1;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (nam) {
|
if (nam) {
|
||||||
sin6 = (struct sockaddr_in6 *)nam;
|
sin6 = (struct sockaddr_in6 *)nam;
|
||||||
if (nam->sa_len != sizeof(*sin6))
|
if (nam->sa_len != sizeof(*sin6))
|
||||||
@ -162,10 +157,8 @@ in6_pcbbind(inp, nam, td)
|
|||||||
* and a multicast address is bound on both
|
* and a multicast address is bound on both
|
||||||
* new and duplicated sockets.
|
* new and duplicated sockets.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_options & SO_REUSEADDR)
|
if (so->so_options & SO_REUSEADDR)
|
||||||
reuseport = SO_REUSEADDR|SO_REUSEPORT;
|
reuseport = SO_REUSEADDR|SO_REUSEPORT;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
} else if (!IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr)) {
|
} else if (!IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr)) {
|
||||||
struct ifaddr *ia = NULL;
|
struct ifaddr *ia = NULL;
|
||||||
|
|
||||||
@ -197,19 +190,14 @@ in6_pcbbind(inp, nam, td)
|
|||||||
t = in6_pcblookup_local(pcbinfo,
|
t = in6_pcblookup_local(pcbinfo,
|
||||||
&sin6->sin6_addr, lport,
|
&sin6->sin6_addr, lport,
|
||||||
INPLOOKUP_WILDCARD);
|
INPLOOKUP_WILDCARD);
|
||||||
if (t != NULL) {
|
if (t &&
|
||||||
SOCK_LOCK(t->inp_socket);
|
(!IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) ||
|
||||||
if ((!IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr) ||
|
!IN6_IS_ADDR_UNSPECIFIED(&t->in6p_laddr) ||
|
||||||
!IN6_IS_ADDR_UNSPECIFIED(&t->in6p_laddr) ||
|
(t->inp_socket->so_options &
|
||||||
(t->inp_socket->so_options &
|
SO_REUSEPORT) == 0) &&
|
||||||
SO_REUSEPORT) == 0) &&
|
(so->so_cred->cr_uid !=
|
||||||
(so->so_cred->cr_uid !=
|
t->inp_socket->so_cred->cr_uid))
|
||||||
t->inp_socket->so_cred->cr_uid)) {
|
return (EADDRINUSE);
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
return (EADDRINUSE);
|
|
||||||
}
|
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
}
|
|
||||||
if ((inp->inp_flags & IN6P_IPV6_V6ONLY) == 0 &&
|
if ((inp->inp_flags & IN6P_IPV6_V6ONLY) == 0 &&
|
||||||
IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr)) {
|
IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr)) {
|
||||||
struct sockaddr_in sin;
|
struct sockaddr_in sin;
|
||||||
@ -230,14 +218,8 @@ in6_pcbbind(inp, nam, td)
|
|||||||
}
|
}
|
||||||
t = in6_pcblookup_local(pcbinfo, &sin6->sin6_addr,
|
t = in6_pcblookup_local(pcbinfo, &sin6->sin6_addr,
|
||||||
lport, wild);
|
lport, wild);
|
||||||
if (t != NULL) {
|
if (t && (reuseport & t->inp_socket->so_options) == 0)
|
||||||
SOCK_LOCK(t->inp_socket);
|
return(EADDRINUSE);
|
||||||
if ((reuseport & t->inp_socket->so_options) == 0) {
|
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
return(EADDRINUSE);
|
|
||||||
}
|
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
}
|
|
||||||
if ((inp->inp_flags & IN6P_IPV6_V6ONLY) == 0 &&
|
if ((inp->inp_flags & IN6P_IPV6_V6ONLY) == 0 &&
|
||||||
IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr)) {
|
IN6_IS_ADDR_UNSPECIFIED(&sin6->sin6_addr)) {
|
||||||
struct sockaddr_in sin;
|
struct sockaddr_in sin;
|
||||||
@ -245,19 +227,14 @@ in6_pcbbind(inp, nam, td)
|
|||||||
in6_sin6_2_sin(&sin, sin6);
|
in6_sin6_2_sin(&sin, sin6);
|
||||||
t = in_pcblookup_local(pcbinfo, sin.sin_addr,
|
t = in_pcblookup_local(pcbinfo, sin.sin_addr,
|
||||||
lport, wild);
|
lport, wild);
|
||||||
if (t != NULL) {
|
if (t &&
|
||||||
SOCK_LOCK(t->inp_socket);
|
(reuseport & t->inp_socket->so_options)
|
||||||
if ((reuseport & t->inp_socket->so_options)
|
== 0 &&
|
||||||
== 0 &&
|
(ntohl(t->inp_laddr.s_addr)
|
||||||
(ntohl(t->inp_laddr.s_addr)
|
!= INADDR_ANY ||
|
||||||
!= INADDR_ANY ||
|
INP_SOCKAF(so) ==
|
||||||
INP_SOCKAF(so) ==
|
INP_SOCKAF(t->inp_socket)))
|
||||||
INP_SOCKAF(t->inp_socket))) {
|
return (EADDRINUSE);
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
return (EADDRINUSE);
|
|
||||||
}
|
|
||||||
SOCK_UNLOCK(t->inp_socket);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
inp->in6p_laddr = sin6->sin6_addr;
|
inp->in6p_laddr = sin6->sin6_addr;
|
||||||
@ -612,12 +589,8 @@ in6_pcbdisconnect(inp)
|
|||||||
/* clear flowinfo - draft-itojun-ipv6-flowlabel-api-00 */
|
/* clear flowinfo - draft-itojun-ipv6-flowlabel-api-00 */
|
||||||
inp->in6p_flowinfo &= ~IPV6_FLOWLABEL_MASK;
|
inp->in6p_flowinfo &= ~IPV6_FLOWLABEL_MASK;
|
||||||
in_pcbrehash(inp);
|
in_pcbrehash(inp);
|
||||||
SOCK_LOCK(inp->inp_socket);
|
if (inp->inp_socket->so_state & SS_NOFDREF)
|
||||||
if (inp->inp_socket->so_state & SS_NOFDREF) {
|
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
in6_pcbdetach(inp);
|
in6_pcbdetach(inp);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(inp->inp_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -634,7 +607,6 @@ in6_pcbdetach(inp)
|
|||||||
inp->inp_gencnt = ++ipi->ipi_gencnt;
|
inp->inp_gencnt = ++ipi->ipi_gencnt;
|
||||||
in_pcbremlists(inp);
|
in_pcbremlists(inp);
|
||||||
sotoinpcb(so) = 0;
|
sotoinpcb(so) = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
|
|
||||||
if (inp->in6p_options)
|
if (inp->in6p_options)
|
||||||
|
@ -338,11 +338,9 @@ in6_pcbsetport(laddr, inp, td)
|
|||||||
int count, error = 0, wild = 0;
|
int count, error = 0, wild = 0;
|
||||||
struct inpcbinfo *pcbinfo = inp->inp_pcbinfo;
|
struct inpcbinfo *pcbinfo = inp->inp_pcbinfo;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
/* XXX: this is redundant when called from in6_pcbbind */
|
/* XXX: this is redundant when called from in6_pcbbind */
|
||||||
if ((so->so_options & (SO_REUSEADDR|SO_REUSEPORT)) == 0)
|
if ((so->so_options & (SO_REUSEADDR|SO_REUSEPORT)) == 0)
|
||||||
wild = INPLOOKUP_WILDCARD;
|
wild = INPLOOKUP_WILDCARD;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
inp->inp_flags |= INP_ANONPORT;
|
inp->inp_flags |= INP_ANONPORT;
|
||||||
|
|
||||||
|
@ -1147,19 +1147,16 @@ ip6_savecontrol(in6p, mp, ip6, m)
|
|||||||
privileged++;
|
privileged++;
|
||||||
|
|
||||||
#ifdef SO_TIMESTAMP
|
#ifdef SO_TIMESTAMP
|
||||||
SOCK_LOCK(in6p->in6p_socket);
|
|
||||||
if ((in6p->in6p_socket->so_options & SO_TIMESTAMP) != 0) {
|
if ((in6p->in6p_socket->so_options & SO_TIMESTAMP) != 0) {
|
||||||
struct timeval tv;
|
struct timeval tv;
|
||||||
|
|
||||||
SOCK_UNLOCK(in6p->in6p_socket);
|
|
||||||
microtime(&tv);
|
microtime(&tv);
|
||||||
*mp = sbcreatecontrol((caddr_t) &tv, sizeof(tv),
|
*mp = sbcreatecontrol((caddr_t) &tv, sizeof(tv),
|
||||||
SCM_TIMESTAMP, SOL_SOCKET);
|
SCM_TIMESTAMP, SOL_SOCKET);
|
||||||
if (*mp) {
|
if (*mp) {
|
||||||
mp = &(*mp)->m_next;
|
mp = &(*mp)->m_next;
|
||||||
}
|
}
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(in6p->in6p_socket);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* RFC 2292 sec. 5 */
|
/* RFC 2292 sec. 5 */
|
||||||
|
@ -892,9 +892,7 @@ socket_send(s, mm, src)
|
|||||||
if (sbappendaddr(&s->so_rcv,
|
if (sbappendaddr(&s->so_rcv,
|
||||||
(struct sockaddr *)src,
|
(struct sockaddr *)src,
|
||||||
mm, (struct mbuf *)0) != 0) {
|
mm, (struct mbuf *)0) != 0) {
|
||||||
SOCK_LOCK(s);
|
|
||||||
sorwakeup(s);
|
sorwakeup(s);
|
||||||
SOCK_UNLOCK(s);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -182,16 +182,9 @@ rip6_input(mp, offp, proto)
|
|||||||
} else
|
} else
|
||||||
#endif /*IPSEC*/
|
#endif /*IPSEC*/
|
||||||
if (n) {
|
if (n) {
|
||||||
if (last->in6p_flags & IN6P_CONTROLOPTS)
|
if (last->in6p_flags & IN6P_CONTROLOPTS ||
|
||||||
|
last->in6p_socket->so_options & SO_TIMESTAMP)
|
||||||
ip6_savecontrol(last, &opts, ip6, n);
|
ip6_savecontrol(last, &opts, ip6, n);
|
||||||
else {
|
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
if (last->in6p_socket->so_options & SO_TIMESTAMP) {
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
ip6_savecontrol(last, &opts, ip6, n);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
/* strip intermediate headers */
|
/* strip intermediate headers */
|
||||||
m_adj(n, *offp);
|
m_adj(n, *offp);
|
||||||
if (sbappendaddr(&last->in6p_socket->so_rcv,
|
if (sbappendaddr(&last->in6p_socket->so_rcv,
|
||||||
@ -201,11 +194,8 @@ rip6_input(mp, offp, proto)
|
|||||||
if (opts)
|
if (opts)
|
||||||
m_freem(opts);
|
m_freem(opts);
|
||||||
rip6stat.rip6s_fullsock++;
|
rip6stat.rip6s_fullsock++;
|
||||||
} else {
|
} else
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
sorwakeup(last->in6p_socket);
|
sorwakeup(last->in6p_socket);
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
opts = NULL;
|
opts = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -223,16 +213,9 @@ rip6_input(mp, offp, proto)
|
|||||||
} else
|
} else
|
||||||
#endif /*IPSEC*/
|
#endif /*IPSEC*/
|
||||||
if (last) {
|
if (last) {
|
||||||
if (last->in6p_flags & IN6P_CONTROLOPTS)
|
if (last->in6p_flags & IN6P_CONTROLOPTS ||
|
||||||
|
last->in6p_socket->so_options & SO_TIMESTAMP)
|
||||||
ip6_savecontrol(last, &opts, ip6, m);
|
ip6_savecontrol(last, &opts, ip6, m);
|
||||||
else {
|
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
if (last->in6p_socket->so_options & SO_TIMESTAMP) {
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
ip6_savecontrol(last, &opts, ip6, m);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
/* strip intermediate headers */
|
/* strip intermediate headers */
|
||||||
m_adj(m, *offp);
|
m_adj(m, *offp);
|
||||||
if (sbappendaddr(&last->in6p_socket->so_rcv,
|
if (sbappendaddr(&last->in6p_socket->so_rcv,
|
||||||
@ -241,11 +224,8 @@ rip6_input(mp, offp, proto)
|
|||||||
if (opts)
|
if (opts)
|
||||||
m_freem(opts);
|
m_freem(opts);
|
||||||
rip6stat.rip6s_fullsock++;
|
rip6stat.rip6s_fullsock++;
|
||||||
} else {
|
} else
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
sorwakeup(last->in6p_socket);
|
sorwakeup(last->in6p_socket);
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
rip6stat.rip6s_nosock++;
|
rip6stat.rip6s_nosock++;
|
||||||
if (m->m_flags & M_MCAST)
|
if (m->m_flags & M_MCAST)
|
||||||
@ -611,9 +591,7 @@ rip6_detach(struct socket *so)
|
|||||||
static int
|
static int
|
||||||
rip6_abort(struct socket *so)
|
rip6_abort(struct socket *so)
|
||||||
{
|
{
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return rip6_detach(so);
|
return rip6_detach(so);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -622,12 +600,8 @@ rip6_disconnect(struct socket *so)
|
|||||||
{
|
{
|
||||||
struct inpcb *inp = sotoinpcb(so);
|
struct inpcb *inp = sotoinpcb(so);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
if ((so->so_state & SS_ISCONNECTED) == 0)
|
||||||
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return ENOTCONN;
|
return ENOTCONN;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
inp->in6p_faddr = in6addr_any;
|
inp->in6p_faddr = in6addr_any;
|
||||||
return rip6_abort(so);
|
return rip6_abort(so);
|
||||||
}
|
}
|
||||||
@ -695,9 +669,7 @@ rip6_connect(struct socket *so, struct sockaddr *nam, struct thread *td)
|
|||||||
return (error ? error : EADDRNOTAVAIL);
|
return (error ? error : EADDRNOTAVAIL);
|
||||||
inp->in6p_laddr = *in6a;
|
inp->in6p_laddr = *in6a;
|
||||||
inp->in6p_faddr = addr->sin6_addr;
|
inp->in6p_faddr = addr->sin6_addr;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -717,9 +689,7 @@ rip6_send(struct socket *so, int flags, struct mbuf *m, struct sockaddr *nam,
|
|||||||
struct sockaddr_in6 *dst;
|
struct sockaddr_in6 *dst;
|
||||||
|
|
||||||
/* always copy sockaddr to avoid overwrites */
|
/* always copy sockaddr to avoid overwrites */
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_ISCONNECTED) {
|
if (so->so_state & SS_ISCONNECTED) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (nam) {
|
if (nam) {
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
return EISCONN;
|
return EISCONN;
|
||||||
@ -732,7 +702,6 @@ rip6_send(struct socket *so, int flags, struct mbuf *m, struct sockaddr *nam,
|
|||||||
sizeof(struct in6_addr));
|
sizeof(struct in6_addr));
|
||||||
dst = &tmp;
|
dst = &tmp;
|
||||||
} else {
|
} else {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (nam == NULL) {
|
if (nam == NULL) {
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
return ENOTCONN;
|
return ENOTCONN;
|
||||||
|
@ -269,18 +269,10 @@ udp6_input(mp, offp, proto)
|
|||||||
* and m_copy() will copy M_PKTHDR
|
* and m_copy() will copy M_PKTHDR
|
||||||
* only if offset is 0.
|
* only if offset is 0.
|
||||||
*/
|
*/
|
||||||
if (last->in6p_flags & IN6P_CONTROLOPTS)
|
if (last->in6p_flags & IN6P_CONTROLOPTS
|
||||||
|
|| last->in6p_socket->so_options & SO_TIMESTAMP)
|
||||||
ip6_savecontrol(last, &opts,
|
ip6_savecontrol(last, &opts,
|
||||||
ip6, n);
|
ip6, n);
|
||||||
else {
|
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
if (last->in6p_socket->so_options & SO_TIMESTAMP) {
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
ip6_savecontrol(last, &opts,
|
|
||||||
ip6, n);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
|
|
||||||
m_adj(n, off + sizeof(struct udphdr));
|
m_adj(n, off + sizeof(struct udphdr));
|
||||||
if (sbappendaddr(&last->in6p_socket->so_rcv,
|
if (sbappendaddr(&last->in6p_socket->so_rcv,
|
||||||
@ -290,11 +282,8 @@ udp6_input(mp, offp, proto)
|
|||||||
if (opts)
|
if (opts)
|
||||||
m_freem(opts);
|
m_freem(opts);
|
||||||
udpstat.udps_fullsock++;
|
udpstat.udps_fullsock++;
|
||||||
} else {
|
} else
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
sorwakeup(last->in6p_socket);
|
sorwakeup(last->in6p_socket);
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
opts = NULL;
|
opts = NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -307,13 +296,9 @@ udp6_input(mp, offp, proto)
|
|||||||
* port. It assumes that an application will never
|
* port. It assumes that an application will never
|
||||||
* clear these options after setting them.
|
* clear these options after setting them.
|
||||||
*/
|
*/
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
if ((last->in6p_socket->so_options &
|
if ((last->in6p_socket->so_options &
|
||||||
(SO_REUSEPORT|SO_REUSEADDR)) == 0) {
|
(SO_REUSEPORT|SO_REUSEADDR)) == 0)
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
break;
|
break;
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (last == NULL) {
|
if (last == NULL) {
|
||||||
@ -335,16 +320,9 @@ udp6_input(mp, offp, proto)
|
|||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
#endif /* IPSEC */
|
#endif /* IPSEC */
|
||||||
if (last->in6p_flags & IN6P_CONTROLOPTS)
|
if (last->in6p_flags & IN6P_CONTROLOPTS
|
||||||
|
|| last->in6p_socket->so_options & SO_TIMESTAMP)
|
||||||
ip6_savecontrol(last, &opts, ip6, m);
|
ip6_savecontrol(last, &opts, ip6, m);
|
||||||
else {
|
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
if (last->in6p_socket->so_options & SO_TIMESTAMP) {
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
ip6_savecontrol(last, &opts, ip6, m);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
}
|
|
||||||
|
|
||||||
m_adj(m, off + sizeof(struct udphdr));
|
m_adj(m, off + sizeof(struct udphdr));
|
||||||
if (sbappendaddr(&last->in6p_socket->so_rcv,
|
if (sbappendaddr(&last->in6p_socket->so_rcv,
|
||||||
@ -353,9 +331,7 @@ udp6_input(mp, offp, proto)
|
|||||||
udpstat.udps_fullsock++;
|
udpstat.udps_fullsock++;
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(last->in6p_socket);
|
|
||||||
sorwakeup(last->in6p_socket);
|
sorwakeup(last->in6p_socket);
|
||||||
SOCK_UNLOCK(last->in6p_socket);
|
|
||||||
return IPPROTO_DONE;
|
return IPPROTO_DONE;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
@ -399,16 +375,9 @@ udp6_input(mp, offp, proto)
|
|||||||
*/
|
*/
|
||||||
init_sin6(&udp_in6, m); /* general init */
|
init_sin6(&udp_in6, m); /* general init */
|
||||||
udp_in6.sin6_port = uh->uh_sport;
|
udp_in6.sin6_port = uh->uh_sport;
|
||||||
if (in6p->in6p_flags & IN6P_CONTROLOPTS)
|
if (in6p->in6p_flags & IN6P_CONTROLOPTS
|
||||||
|
|| in6p->in6p_socket->so_options & SO_TIMESTAMP)
|
||||||
ip6_savecontrol(in6p, &opts, ip6, m);
|
ip6_savecontrol(in6p, &opts, ip6, m);
|
||||||
else {
|
|
||||||
SOCK_LOCK(in6p->in6p_socket);
|
|
||||||
if (in6p->in6p_socket->so_options & SO_TIMESTAMP) {
|
|
||||||
SOCK_UNLOCK(in6p->in6p_socket);
|
|
||||||
ip6_savecontrol(in6p, &opts, ip6, m);
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(in6p->in6p_socket);
|
|
||||||
}
|
|
||||||
m_adj(m, off + sizeof(struct udphdr));
|
m_adj(m, off + sizeof(struct udphdr));
|
||||||
if (sbappendaddr(&in6p->in6p_socket->so_rcv,
|
if (sbappendaddr(&in6p->in6p_socket->so_rcv,
|
||||||
(struct sockaddr *)&udp_in6,
|
(struct sockaddr *)&udp_in6,
|
||||||
@ -416,9 +385,7 @@ udp6_input(mp, offp, proto)
|
|||||||
udpstat.udps_fullsock++;
|
udpstat.udps_fullsock++;
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(in6p->in6p_socket);
|
|
||||||
sorwakeup(in6p->in6p_socket);
|
sorwakeup(in6p->in6p_socket);
|
||||||
SOCK_UNLOCK(in6p->in6p_socket);
|
|
||||||
return IPPROTO_DONE;
|
return IPPROTO_DONE;
|
||||||
bad:
|
bad:
|
||||||
if (m)
|
if (m)
|
||||||
@ -542,9 +509,7 @@ udp6_abort(struct socket *so)
|
|||||||
inp = sotoinpcb(so);
|
inp = sotoinpcb(so);
|
||||||
if (inp == 0)
|
if (inp == 0)
|
||||||
return EINVAL; /* ??? possible? panic instead? */
|
return EINVAL; /* ??? possible? panic instead? */
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
s = splnet();
|
s = splnet();
|
||||||
in6_pcbdetach(inp);
|
in6_pcbdetach(inp);
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -649,9 +614,7 @@ udp6_connect(struct socket *so, struct sockaddr *nam, struct thread *td)
|
|||||||
if (error == 0) {
|
if (error == 0) {
|
||||||
inp->inp_vflag |= INP_IPV4;
|
inp->inp_vflag |= INP_IPV4;
|
||||||
inp->inp_vflag &= ~INP_IPV6;
|
inp->inp_vflag &= ~INP_IPV6;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
@ -666,9 +629,7 @@ udp6_connect(struct socket *so, struct sockaddr *nam, struct thread *td)
|
|||||||
inp->inp_vflag &= ~INP_IPV4;
|
inp->inp_vflag &= ~INP_IPV4;
|
||||||
inp->inp_vflag |= INP_IPV6;
|
inp->inp_vflag |= INP_IPV6;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
@ -712,9 +673,7 @@ udp6_disconnect(struct socket *so)
|
|||||||
in6_pcbdisconnect(inp);
|
in6_pcbdisconnect(inp);
|
||||||
inp->in6p_laddr = in6addr_any;
|
inp->in6p_laddr = in6addr_any;
|
||||||
splx(s);
|
splx(s);
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_ISCONNECTED; /* XXX */
|
so->so_state &= ~SS_ISCONNECTED; /* XXX */
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -156,12 +156,8 @@ ipx_pcbconnect(ipxp, nam, td)
|
|||||||
*/
|
*/
|
||||||
ro = &ipxp->ipxp_route;
|
ro = &ipxp->ipxp_route;
|
||||||
dst = &satoipx_addr(ro->ro_dst);
|
dst = &satoipx_addr(ro->ro_dst);
|
||||||
SOCK_LOCK(ipxp->ipxp_socket);
|
if (ipxp->ipxp_socket->so_options & SO_DONTROUTE)
|
||||||
if (ipxp->ipxp_socket->so_options & SO_DONTROUTE) {
|
|
||||||
SOCK_UNLOCK(ipxp->ipxp_socket);
|
|
||||||
goto flush;
|
goto flush;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(ipxp->ipxp_socket);
|
|
||||||
if (!ipx_neteq(ipxp->ipxp_lastdst, sipx->sipx_addr))
|
if (!ipx_neteq(ipxp->ipxp_lastdst, sipx->sipx_addr))
|
||||||
goto flush;
|
goto flush;
|
||||||
if (!ipx_hosteq(ipxp->ipxp_lastdst, sipx->sipx_addr)) {
|
if (!ipx_hosteq(ipxp->ipxp_lastdst, sipx->sipx_addr)) {
|
||||||
@ -176,18 +172,15 @@ ipx_pcbconnect(ipxp, nam, td)
|
|||||||
}
|
}
|
||||||
}/* else cached route is ok; do nothing */
|
}/* else cached route is ok; do nothing */
|
||||||
ipxp->ipxp_lastdst = sipx->sipx_addr;
|
ipxp->ipxp_lastdst = sipx->sipx_addr;
|
||||||
SOCK_LOCK(ipxp->ipxp_socket);
|
|
||||||
if ((ipxp->ipxp_socket->so_options & SO_DONTROUTE) == 0 && /*XXX*/
|
if ((ipxp->ipxp_socket->so_options & SO_DONTROUTE) == 0 && /*XXX*/
|
||||||
(ro->ro_rt == NULL || ro->ro_rt->rt_ifp == NULL)) {
|
(ro->ro_rt == NULL || ro->ro_rt->rt_ifp == NULL)) {
|
||||||
SOCK_UNLOCK(ipxp->ipxp_socket);
|
|
||||||
/* No route yet, so try to acquire one */
|
/* No route yet, so try to acquire one */
|
||||||
ro->ro_dst.sa_family = AF_IPX;
|
ro->ro_dst.sa_family = AF_IPX;
|
||||||
ro->ro_dst.sa_len = sizeof(ro->ro_dst);
|
ro->ro_dst.sa_len = sizeof(ro->ro_dst);
|
||||||
*dst = sipx->sipx_addr;
|
*dst = sipx->sipx_addr;
|
||||||
dst->x_port = 0;
|
dst->x_port = 0;
|
||||||
rtalloc(ro);
|
rtalloc(ro);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(ipxp->ipxp_socket);
|
|
||||||
if (ipx_neteqnn(ipxp->ipxp_laddr.x_net, ipx_zeronet)) {
|
if (ipx_neteqnn(ipxp->ipxp_laddr.x_net, ipx_zeronet)) {
|
||||||
/*
|
/*
|
||||||
* If route is known or can be allocated now,
|
* If route is known or can be allocated now,
|
||||||
@ -264,12 +257,8 @@ ipx_pcbdisconnect(ipxp)
|
|||||||
{
|
{
|
||||||
|
|
||||||
ipxp->ipxp_faddr = zeroipx_addr;
|
ipxp->ipxp_faddr = zeroipx_addr;
|
||||||
SOCK_LOCK(ipxp->ipxp_socket);
|
if (ipxp->ipxp_socket->so_state & SS_NOFDREF)
|
||||||
if (ipxp->ipxp_socket->so_state & SS_NOFDREF) {
|
|
||||||
SOCK_UNLOCK(ipxp->ipxp_socket);
|
|
||||||
ipx_pcbdetach(ipxp);
|
ipx_pcbdetach(ipxp);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(ipxp->ipxp_socket);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -279,7 +268,6 @@ ipx_pcbdetach(ipxp)
|
|||||||
struct socket *so = ipxp->ipxp_socket;
|
struct socket *so = ipxp->ipxp_socket;
|
||||||
|
|
||||||
so->so_pcb = 0;
|
so->so_pcb = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
if (ipxp->ipxp_route.ro_rt != NULL)
|
if (ipxp->ipxp_route.ro_rt != NULL)
|
||||||
rtfree(ipxp->ipxp_route.ro_rt);
|
rtfree(ipxp->ipxp_route.ro_rt);
|
||||||
|
@ -146,9 +146,7 @@ ipx_input(m, ipxp)
|
|||||||
if (sbappendaddr(&ipxp->ipxp_socket->so_rcv, (struct sockaddr *)&ipx_ipx,
|
if (sbappendaddr(&ipxp->ipxp_socket->so_rcv, (struct sockaddr *)&ipx_ipx,
|
||||||
m, (struct mbuf *)NULL) == 0)
|
m, (struct mbuf *)NULL) == 0)
|
||||||
goto bad;
|
goto bad;
|
||||||
SOCK_LOCK(ipxp->ipxp_socket);
|
|
||||||
sorwakeup(ipxp->ipxp_socket);
|
sorwakeup(ipxp->ipxp_socket);
|
||||||
SOCK_UNLOCK(ipxp->ipxp_socket);
|
|
||||||
return;
|
return;
|
||||||
bad:
|
bad:
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
@ -161,9 +159,7 @@ ipx_abort(ipxp)
|
|||||||
struct socket *so = ipxp->ipxp_socket;
|
struct socket *so = ipxp->ipxp_socket;
|
||||||
|
|
||||||
ipx_pcbdisconnect(ipxp);
|
ipx_pcbdisconnect(ipxp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -190,9 +186,7 @@ ipx_drop(ipxp, errno)
|
|||||||
}*/
|
}*/
|
||||||
so->so_error = errno;
|
so->so_error = errno;
|
||||||
ipx_pcbdisconnect(ipxp);
|
ipx_pcbdisconnect(ipxp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
@ -206,7 +200,6 @@ ipx_output(ipxp, m0)
|
|||||||
register struct route *ro;
|
register struct route *ro;
|
||||||
struct mbuf *m;
|
struct mbuf *m;
|
||||||
struct mbuf *mprev = NULL;
|
struct mbuf *mprev = NULL;
|
||||||
int soopts;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Calculate data length.
|
* Calculate data length.
|
||||||
@ -268,14 +261,9 @@ ipx_output(ipxp, m0)
|
|||||||
* Output datagram.
|
* Output datagram.
|
||||||
*/
|
*/
|
||||||
so = ipxp->ipxp_socket;
|
so = ipxp->ipxp_socket;
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DONTROUTE)
|
||||||
if (so->so_options & SO_DONTROUTE) {
|
|
||||||
soopts = so->so_options & SO_BROADCAST;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (ipx_outputfl(m, (struct route *)NULL,
|
return (ipx_outputfl(m, (struct route *)NULL,
|
||||||
soopts | IPX_ROUTETOIF));
|
(so->so_options & SO_BROADCAST) | IPX_ROUTETOIF));
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/*
|
/*
|
||||||
* Use cached route for previous datagram if
|
* Use cached route for previous datagram if
|
||||||
* possible. If the previous net was the same
|
* possible. If the previous net was the same
|
||||||
@ -318,10 +306,7 @@ ipx_output(ipxp, m0)
|
|||||||
}
|
}
|
||||||
ipxp->ipxp_lastdst = ipx->ipx_dna;
|
ipxp->ipxp_lastdst = ipx->ipx_dna;
|
||||||
#endif /* ancient_history */
|
#endif /* ancient_history */
|
||||||
SOCK_LOCK(so);
|
return (ipx_outputfl(m, ro, so->so_options & SO_BROADCAST));
|
||||||
soopts = so->so_options & SO_BROADCAST;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (ipx_outputfl(m, ro, soopts));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
@ -444,11 +429,8 @@ ipx_usr_abort(so)
|
|||||||
s = splnet();
|
s = splnet();
|
||||||
ipx_pcbdetach(ipxp);
|
ipx_pcbdetach(ipxp);
|
||||||
splx(s);
|
splx(s);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -498,11 +480,8 @@ ipx_connect(so, nam, td)
|
|||||||
s = splnet();
|
s = splnet();
|
||||||
error = ipx_pcbconnect(ipxp, nam, td);
|
error = ipx_pcbconnect(ipxp, nam, td);
|
||||||
splx(s);
|
splx(s);
|
||||||
if (error == 0) {
|
if (error == 0)
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
return (error);
|
return (error);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -533,9 +512,7 @@ ipx_disconnect(so)
|
|||||||
s = splnet();
|
s = splnet();
|
||||||
ipx_pcbdisconnect(ipxp);
|
ipx_pcbdisconnect(ipxp);
|
||||||
splx(s);
|
splx(s);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -163,7 +163,6 @@ spx_input(m, ipxp)
|
|||||||
|
|
||||||
so = ipxp->ipxp_socket;
|
so = ipxp->ipxp_socket;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_options & SO_DEBUG || traceallspxs) {
|
if (so->so_options & SO_DEBUG || traceallspxs) {
|
||||||
ostate = cb->s_state;
|
ostate = cb->s_state;
|
||||||
spx_savesi = *si;
|
spx_savesi = *si;
|
||||||
@ -171,7 +170,6 @@ spx_input(m, ipxp)
|
|||||||
if (so->so_options & SO_ACCEPTCONN) {
|
if (so->so_options & SO_ACCEPTCONN) {
|
||||||
struct spxpcb *ocb = cb;
|
struct spxpcb *ocb = cb;
|
||||||
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
so = sonewconn(so, 0);
|
so = sonewconn(so, 0);
|
||||||
if (so == NULL) {
|
if (so == NULL) {
|
||||||
goto drop;
|
goto drop;
|
||||||
@ -195,8 +193,7 @@ spx_input(m, ipxp)
|
|||||||
cb->s_flags = ocb->s_flags; /* preserve sockopts */
|
cb->s_flags = ocb->s_flags; /* preserve sockopts */
|
||||||
cb->s_flags2 = ocb->s_flags2; /* preserve sockopts */
|
cb->s_flags2 = ocb->s_flags2; /* preserve sockopts */
|
||||||
cb->s_state = TCPS_LISTEN;
|
cb->s_state = TCPS_LISTEN;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Packet received on connection.
|
* Packet received on connection.
|
||||||
@ -261,9 +258,7 @@ spx_input(m, ipxp)
|
|||||||
ipxp->ipxp_fport = si->si_sport;
|
ipxp->ipxp_fport = si->si_sport;
|
||||||
cb->s_timer[SPXT_REXMT] = 0;
|
cb->s_timer[SPXT_REXMT] = 0;
|
||||||
cb->s_timer[SPXT_KEEP] = SPXTV_KEEP;
|
cb->s_timer[SPXT_KEEP] = SPXTV_KEEP;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
cb->s_state = TCPS_ESTABLISHED;
|
cb->s_state = TCPS_ESTABLISHED;
|
||||||
spxstat.spxs_accepts++;
|
spxstat.spxs_accepts++;
|
||||||
}
|
}
|
||||||
@ -290,9 +285,7 @@ spx_input(m, ipxp)
|
|||||||
cb->s_dport = ipxp->ipxp_fport = si->si_sport;
|
cb->s_dport = ipxp->ipxp_fport = si->si_sport;
|
||||||
cb->s_timer[SPXT_REXMT] = 0;
|
cb->s_timer[SPXT_REXMT] = 0;
|
||||||
cb->s_flags |= SF_ACKNOW;
|
cb->s_flags |= SF_ACKNOW;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
cb->s_state = TCPS_ESTABLISHED;
|
cb->s_state = TCPS_ESTABLISHED;
|
||||||
/* Use roundtrip time of connection request for initial rtt */
|
/* Use roundtrip time of connection request for initial rtt */
|
||||||
if (cb->s_rtt) {
|
if (cb->s_rtt) {
|
||||||
@ -304,12 +297,8 @@ spx_input(m, ipxp)
|
|||||||
cb->s_rtt = 0;
|
cb->s_rtt = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG || traceallspxs)
|
||||||
if (so->so_options & SO_DEBUG || traceallspxs) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spx_trace(SA_INPUT, (u_char)ostate, cb, &spx_savesi, 0);
|
spx_trace(SA_INPUT, (u_char)ostate, cb, &spx_savesi, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
m->m_len -= sizeof(struct ipx);
|
m->m_len -= sizeof(struct ipx);
|
||||||
m->m_pkthdr.len -= sizeof(struct ipx);
|
m->m_pkthdr.len -= sizeof(struct ipx);
|
||||||
@ -330,23 +319,15 @@ dropwithreset:
|
|||||||
si->si_ack = ntohs(si->si_ack);
|
si->si_ack = ntohs(si->si_ack);
|
||||||
si->si_alo = ntohs(si->si_alo);
|
si->si_alo = ntohs(si->si_alo);
|
||||||
m_freem(dtom(si));
|
m_freem(dtom(si));
|
||||||
SOCK_LOCK(cb->s_ipxpcb->ipxp_socket);
|
if (cb->s_ipxpcb->ipxp_socket->so_options & SO_DEBUG || traceallspxs)
|
||||||
if (cb->s_ipxpcb->ipxp_socket->so_options & SO_DEBUG || traceallspxs) {
|
|
||||||
SOCK_UNLOCK(cb->s_ipxpcb->ipxp_socket);
|
|
||||||
spx_trace(SA_DROP, (u_char)ostate, cb, &spx_savesi, 0);
|
spx_trace(SA_DROP, (u_char)ostate, cb, &spx_savesi, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(cb->s_ipxpcb->ipxp_socket);
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
drop:
|
drop:
|
||||||
bad:
|
bad:
|
||||||
SOCK_LOCK(cb->s_ipxpcb->ipxp_socket);
|
|
||||||
if (cb == 0 || cb->s_ipxpcb->ipxp_socket->so_options & SO_DEBUG ||
|
if (cb == 0 || cb->s_ipxpcb->ipxp_socket->so_options & SO_DEBUG ||
|
||||||
traceallspxs) {
|
traceallspxs)
|
||||||
SOCK_UNLOCK(cb->s_ipxpcb->ipxp_socket);
|
|
||||||
spx_trace(SA_DROP, (u_char)ostate, cb, &spx_savesi, 0);
|
spx_trace(SA_DROP, (u_char)ostate, cb, &spx_savesi, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(cb->s_ipxpcb->ipxp_socket);
|
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -481,9 +462,7 @@ register struct spx *si;
|
|||||||
else
|
else
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
sowwakeup(so);
|
sowwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
cb->s_rack = si->si_ack;
|
cb->s_rack = si->si_ack;
|
||||||
update_window:
|
update_window:
|
||||||
if (SSEQ_LT(cb->s_snxt, cb->s_rack))
|
if (SSEQ_LT(cb->s_snxt, cb->s_rack))
|
||||||
@ -522,13 +501,10 @@ update_window:
|
|||||||
} /* else queue this packet; */
|
} /* else queue this packet; */
|
||||||
} else {
|
} else {
|
||||||
/*register struct socket *so = cb->s_ipxpcb->ipxp_socket;
|
/*register struct socket *so = cb->s_ipxpcb->ipxp_socket;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state && SS_NOFDREF) {
|
if (so->so_state && SS_NOFDREF) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spx_close(cb);
|
spx_close(cb);
|
||||||
} else {
|
} else
|
||||||
SOCK_UNLOCK(so);
|
would crash system*/
|
||||||
would crash system } */
|
|
||||||
spx_istat.notyet++;
|
spx_istat.notyet++;
|
||||||
m_freem(dtom(si));
|
m_freem(dtom(si));
|
||||||
return (0);
|
return (0);
|
||||||
@ -589,11 +565,8 @@ present:
|
|||||||
cb->s_oobflags &= ~SF_IOOB;
|
cb->s_oobflags &= ~SF_IOOB;
|
||||||
if (so->so_rcv.sb_cc)
|
if (so->so_rcv.sb_cc)
|
||||||
so->so_oobmark = so->so_rcv.sb_cc;
|
so->so_oobmark = so->so_rcv.sb_cc;
|
||||||
else {
|
else
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_RCVATMARK;
|
so->so_state |= SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
q = q->si_prev;
|
q = q->si_prev;
|
||||||
remque(q->si_next);
|
remque(q->si_next);
|
||||||
@ -623,9 +596,7 @@ present:
|
|||||||
MCHTYPE(m, MT_OOBDATA);
|
MCHTYPE(m, MT_OOBDATA);
|
||||||
spx_newchecks[1]++;
|
spx_newchecks[1]++;
|
||||||
so->so_oobmark = 0;
|
so->so_oobmark = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_RCVATMARK;
|
so->so_state &= ~SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
if (packetp == 0) {
|
if (packetp == 0) {
|
||||||
m->m_data += SPINC;
|
m->m_data += SPINC;
|
||||||
@ -651,11 +622,8 @@ present:
|
|||||||
} else
|
} else
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (wakeup) {
|
if (wakeup)
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1059,12 +1027,8 @@ send:
|
|||||||
si->si_cc |= SPX_SP;
|
si->si_cc |= SPX_SP;
|
||||||
} else {
|
} else {
|
||||||
cb->s_outx = 3;
|
cb->s_outx = 3;
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG || traceallspxs)
|
||||||
if (so->so_options & SO_DEBUG || traceallspxs) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spx_trace(SA_OUTPUT, cb->s_state, cb, si, 0);
|
spx_trace(SA_OUTPUT, cb->s_state, cb, si, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
@ -1126,20 +1090,13 @@ send:
|
|||||||
si->si_sum = 0xffff;
|
si->si_sum = 0xffff;
|
||||||
|
|
||||||
cb->s_outx = 4;
|
cb->s_outx = 4;
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG || traceallspxs)
|
||||||
if (so->so_options & SO_DEBUG || traceallspxs) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spx_trace(SA_OUTPUT, cb->s_state, cb, si, 0);
|
spx_trace(SA_OUTPUT, cb->s_state, cb, si, 0);
|
||||||
SOCK_LOCK(so);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (so->so_options & SO_DONTROUTE) {
|
if (so->so_options & SO_DONTROUTE)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ipx_outputfl(m, (struct route *)NULL, IPX_ROUTETOIF);
|
error = ipx_outputfl(m, (struct route *)NULL, IPX_ROUTETOIF);
|
||||||
} else {
|
else
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ipx_outputfl(m, &cb->s_ipxpcb->ipxp_route, 0);
|
error = ipx_outputfl(m, &cb->s_ipxpcb->ipxp_route, 0);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (error) {
|
if (error) {
|
||||||
return (error);
|
return (error);
|
||||||
@ -1456,9 +1413,7 @@ spx_connect(so, nam, td)
|
|||||||
error = ipx_pcbconnect(ipxp, nam, td);
|
error = ipx_pcbconnect(ipxp, nam, td);
|
||||||
if (error)
|
if (error)
|
||||||
goto spx_connect_end;
|
goto spx_connect_end;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnecting(so);
|
soisconnecting(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spxstat.spxs_connattempt++;
|
spxstat.spxs_connattempt++;
|
||||||
cb->s_state = TCPS_SYN_SENT;
|
cb->s_state = TCPS_SYN_SENT;
|
||||||
cb->s_did = 0;
|
cb->s_did = 0;
|
||||||
@ -1580,15 +1535,12 @@ spx_rcvoob(so, m, flags)
|
|||||||
ipxp = sotoipxpcb(so);
|
ipxp = sotoipxpcb(so);
|
||||||
cb = ipxtospxpcb(ipxp);
|
cb = ipxtospxpcb(ipxp);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((cb->s_oobflags & SF_IOOB) || so->so_oobmark ||
|
if ((cb->s_oobflags & SF_IOOB) || so->so_oobmark ||
|
||||||
(so->so_state & SS_RCVATMARK)) {
|
(so->so_state & SS_RCVATMARK)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
m->m_len = 1;
|
m->m_len = 1;
|
||||||
*mtod(m, caddr_t) = cb->s_iobc;
|
*mtod(m, caddr_t) = cb->s_iobc;
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (EINVAL);
|
return (EINVAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1732,9 +1684,7 @@ spx_close(cb)
|
|||||||
m_free(dtom(cb->s_ipx));
|
m_free(dtom(cb->s_ipx));
|
||||||
FREE(cb, M_PCB);
|
FREE(cb, M_PCB);
|
||||||
ipxp->ipxp_pcb = 0;
|
ipxp->ipxp_pcb = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
ipx_pcbdetach(ipxp);
|
ipx_pcbdetach(ipxp);
|
||||||
spxstat.spxs_closed++;
|
spxstat.spxs_closed++;
|
||||||
return ((struct spxpcb *)NULL);
|
return ((struct spxpcb *)NULL);
|
||||||
@ -1937,17 +1887,13 @@ spx_timers(cb, timer)
|
|||||||
spxstat.spxs_keeptimeo++;
|
spxstat.spxs_keeptimeo++;
|
||||||
if (cb->s_state < TCPS_ESTABLISHED)
|
if (cb->s_state < TCPS_ESTABLISHED)
|
||||||
goto dropit;
|
goto dropit;
|
||||||
SOCK_LOCK(cb->s_ipxpcb->ipxp_socket);
|
|
||||||
if (cb->s_ipxpcb->ipxp_socket->so_options & SO_KEEPALIVE) {
|
if (cb->s_ipxpcb->ipxp_socket->so_options & SO_KEEPALIVE) {
|
||||||
SOCK_UNLOCK(cb->s_ipxpcb->ipxp_socket);
|
|
||||||
if (cb->s_idle >= SPXTV_MAXIDLE)
|
if (cb->s_idle >= SPXTV_MAXIDLE)
|
||||||
goto dropit;
|
goto dropit;
|
||||||
spxstat.spxs_keepprobe++;
|
spxstat.spxs_keepprobe++;
|
||||||
spx_output(cb, (struct mbuf *)NULL);
|
spx_output(cb, (struct mbuf *)NULL);
|
||||||
} else {
|
} else
|
||||||
SOCK_UNLOCK(cb->s_ipxpcb->ipxp_socket);
|
|
||||||
cb->s_idle = 0;
|
cb->s_idle = 0;
|
||||||
}
|
|
||||||
cb->s_timer[SPXT_KEEP] = SPXTV_KEEP;
|
cb->s_timer[SPXT_KEEP] = SPXTV_KEEP;
|
||||||
break;
|
break;
|
||||||
dropit:
|
dropit:
|
||||||
|
@ -174,9 +174,7 @@ key_sendup0(rp, m, promisc)
|
|||||||
error = ENOBUFS;
|
error = ENOBUFS;
|
||||||
} else
|
} else
|
||||||
error = 0;
|
error = 0;
|
||||||
SOCK_LOCK(rp->rcb_socket);
|
|
||||||
sorwakeup(rp->rcb_socket);
|
sorwakeup(rp->rcb_socket);
|
||||||
SOCK_UNLOCK(rp->rcb_socket);
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -429,10 +427,8 @@ key_attach(struct socket *so, int proto, struct thread *td)
|
|||||||
key_cb.any_count++;
|
key_cb.any_count++;
|
||||||
kp->kp_raw.rcb_laddr = &key_src;
|
kp->kp_raw.rcb_laddr = &key_src;
|
||||||
kp->kp_raw.rcb_faddr = &key_dst;
|
kp->kp_raw.rcb_faddr = &key_dst;
|
||||||
SOCK_LOCK(so);
|
soisconnected_locked(so);
|
||||||
soisconnected(so);
|
|
||||||
so->so_options |= SO_USELOOPBACK;
|
so->so_options |= SO_USELOOPBACK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
splx(s);
|
splx(s);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -134,7 +134,6 @@ natm_usr_detach(struct socket *so)
|
|||||||
*/
|
*/
|
||||||
npcb_free(npcb, NPCB_DESTROY); /* drain */
|
npcb_free(npcb, NPCB_DESTROY); /* drain */
|
||||||
so->so_pcb = NULL;
|
so->so_pcb = NULL;
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
out:
|
out:
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -217,9 +216,7 @@ natm_usr_connect(struct socket *so, struct sockaddr *nam, d_thread_t *p)
|
|||||||
}
|
}
|
||||||
splx(s2);
|
splx(s2);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
out:
|
out:
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -262,9 +259,7 @@ natm_usr_disconnect(struct socket *so)
|
|||||||
splx(s2);
|
splx(s2);
|
||||||
|
|
||||||
npcb_free(npcb, NPCB_REMOVE);
|
npcb_free(npcb, NPCB_REMOVE);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
out:
|
out:
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -487,7 +482,6 @@ struct proc *p;
|
|||||||
|
|
||||||
npcb_free(npcb, NPCB_DESTROY); /* drain */
|
npcb_free(npcb, NPCB_DESTROY); /* drain */
|
||||||
so->so_pcb = NULL;
|
so->so_pcb = NULL;
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -558,9 +552,7 @@ struct proc *p;
|
|||||||
}
|
}
|
||||||
splx(s2);
|
splx(s2);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -587,9 +579,7 @@ struct proc *p;
|
|||||||
splx(s2);
|
splx(s2);
|
||||||
|
|
||||||
npcb_free(npcb, NPCB_REMOVE);
|
npcb_free(npcb, NPCB_REMOVE);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -762,9 +752,7 @@ m->m_pkthdr.rcvif = NULL; /* null it out to be safe */
|
|||||||
natm_sookbytes += m->m_pkthdr.len;
|
natm_sookbytes += m->m_pkthdr.len;
|
||||||
#endif
|
#endif
|
||||||
sbappendrecord(&so->so_rcv, m);
|
sbappendrecord(&so->so_rcv, m);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
sorwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
} else {
|
} else {
|
||||||
#ifdef NATM_STAT
|
#ifdef NATM_STAT
|
||||||
natm_sodropcnt++;
|
natm_sodropcnt++;
|
||||||
|
@ -86,18 +86,15 @@ ncp_soconnect(struct socket *so,struct sockaddr *target, struct proc *p) {
|
|||||||
*/
|
*/
|
||||||
error = EIO;
|
error = EIO;
|
||||||
s = splnet();
|
s = splnet();
|
||||||
SOCK_LOCK(so);
|
|
||||||
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
||||||
(void) msleep((caddr_t)&so->so_timeo, SOCK_MTX(so), PSOCK, "ncpcon", 2 * hz);
|
(void) tsleep((caddr_t)&so->so_timeo, PSOCK, "ncpcon", 2 * hz);
|
||||||
if ((so->so_state & SS_ISCONNECTING) &&
|
if ((so->so_state & SS_ISCONNECTING) &&
|
||||||
so->so_error == 0 /*&& rep &&*/) {
|
so->so_error == 0 /*&& rep &&*/) {
|
||||||
so->so_state &= ~SS_ISCONNECTING;
|
so->so_state &= ~SS_ISCONNECTING;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (so->so_error) {
|
if (so->so_error) {
|
||||||
error = so->so_error;
|
error = so->so_error;
|
||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
|
@ -97,9 +97,7 @@ idp_input(m, nsp)
|
|||||||
if (sbappendaddr(&nsp->nsp_socket->so_rcv, (struct sockaddr *)&idp_ns,
|
if (sbappendaddr(&nsp->nsp_socket->so_rcv, (struct sockaddr *)&idp_ns,
|
||||||
m, (struct mbuf *)0) == 0)
|
m, (struct mbuf *)0) == 0)
|
||||||
goto bad;
|
goto bad;
|
||||||
SOCK_LOCK(nsp->nsp_socket);
|
|
||||||
sorwakeup(nsp->nsp_socket);
|
sorwakeup(nsp->nsp_socket);
|
||||||
SOCK_UNLOCK(nsp->nsp_socket);
|
|
||||||
return;
|
return;
|
||||||
bad:
|
bad:
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
@ -111,9 +109,7 @@ idp_abort(nsp)
|
|||||||
struct socket *so = nsp->nsp_socket;
|
struct socket *so = nsp->nsp_socket;
|
||||||
|
|
||||||
ns_pcbdisconnect(nsp);
|
ns_pcbdisconnect(nsp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* Drop connection, reporting
|
* Drop connection, reporting
|
||||||
@ -137,9 +133,7 @@ idp_drop(nsp, errno)
|
|||||||
}*/
|
}*/
|
||||||
so->so_error = errno;
|
so->so_error = errno;
|
||||||
ns_pcbdisconnect(nsp);
|
ns_pcbdisconnect(nsp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int noIdpRoute;
|
int noIdpRoute;
|
||||||
@ -154,7 +148,6 @@ idp_output(nsp, m0)
|
|||||||
register struct route *ro;
|
register struct route *ro;
|
||||||
struct mbuf *mprev;
|
struct mbuf *mprev;
|
||||||
extern int idpcksum;
|
extern int idpcksum;
|
||||||
int soopts;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Calculate data length.
|
* Calculate data length.
|
||||||
@ -218,14 +211,9 @@ idp_output(nsp, m0)
|
|||||||
* Output datagram.
|
* Output datagram.
|
||||||
*/
|
*/
|
||||||
so = nsp->nsp_socket;
|
so = nsp->nsp_socket;
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DONTROUTE)
|
||||||
if (so->so_options & SO_DONTROUTE) {
|
|
||||||
soopts = so->so_options & SO_BROADCAST;
|
|
||||||
SO_UNLOCK(so);
|
|
||||||
return (ns_output(m, (struct route *)0,
|
return (ns_output(m, (struct route *)0,
|
||||||
soopts | NS_ROUTETOIF));
|
(so->so_options & SO_BROADCAST) | NS_ROUTETOIF));
|
||||||
}
|
|
||||||
SO_UNLOCK(so);
|
|
||||||
/*
|
/*
|
||||||
* Use cached route for previous datagram if
|
* Use cached route for previous datagram if
|
||||||
* possible. If the previous net was the same
|
* possible. If the previous net was the same
|
||||||
@ -269,10 +257,7 @@ idp_output(nsp, m0)
|
|||||||
nsp->nsp_lastdst = idp->idp_dna;
|
nsp->nsp_lastdst = idp->idp_dna;
|
||||||
#endif /* ancient_history */
|
#endif /* ancient_history */
|
||||||
if (noIdpRoute) ro = 0;
|
if (noIdpRoute) ro = 0;
|
||||||
SOCK_LOCK(so);
|
return (ns_output(m, ro, so->so_options & SO_BROADCAST));
|
||||||
soopts = so->so_options & SO_BROADCAST;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (ns_output(m, ro, soopts));
|
|
||||||
}
|
}
|
||||||
/* ARGSUSED */
|
/* ARGSUSED */
|
||||||
idp_ctloutput(req, so, level, name, value)
|
idp_ctloutput(req, so, level, name, value)
|
||||||
@ -442,11 +427,8 @@ idp_usrreq(so, req, m, nam, control)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
error = ns_pcbconnect(nsp, nam);
|
error = ns_pcbconnect(nsp, nam);
|
||||||
if (error == 0) {
|
if (error == 0)
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PRU_CONNECT2:
|
case PRU_CONNECT2:
|
||||||
@ -463,9 +445,7 @@ idp_usrreq(so, req, m, nam, control)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
ns_pcbdisconnect(nsp);
|
ns_pcbdisconnect(nsp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PRU_SHUTDOWN:
|
case PRU_SHUTDOWN:
|
||||||
@ -511,11 +491,8 @@ idp_usrreq(so, req, m, nam, control)
|
|||||||
|
|
||||||
case PRU_ABORT:
|
case PRU_ABORT:
|
||||||
ns_pcbdetach(nsp);
|
ns_pcbdetach(nsp);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so); /* XXX huh, called after sofree()? */
|
soisdisconnected(so); /* XXX huh, called after sofree()? */
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PRU_SOCKADDR:
|
case PRU_SOCKADDR:
|
||||||
@ -569,13 +546,10 @@ idp_raw_usrreq(so, req, m, nam, control)
|
|||||||
|
|
||||||
case PRU_ATTACH:
|
case PRU_ATTACH:
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (!(so->so_state & SS_PRIV) || (nsp != NULL)) {
|
if (!(so->so_state & SS_PRIV) || (nsp != NULL)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = EINVAL;
|
error = EINVAL;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ns_pcballoc(so, &nsrawpcb);
|
error = ns_pcballoc(so, &nsrawpcb);
|
||||||
if (error)
|
if (error)
|
||||||
break;
|
break;
|
||||||
|
@ -107,12 +107,8 @@ ns_control(so, cmd, data, ifp)
|
|||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
if ((so->so_state & SS_PRIV) == 0)
|
||||||
if ((so->so_state & SS_PRIV) == 0) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (EPERM);
|
return (EPERM);
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
switch (cmd) {
|
switch (cmd) {
|
||||||
case SIOCAIFADDR:
|
case SIOCAIFADDR:
|
||||||
|
@ -94,13 +94,9 @@ ns_pcbbind(nsp, nam)
|
|||||||
if (lport) {
|
if (lport) {
|
||||||
u_short aport = ntohs(lport);
|
u_short aport = ntohs(lport);
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (aport < NSPORT_RESERVED &&
|
if (aport < NSPORT_RESERVED &&
|
||||||
(nsp->nsp_socket->so_state & SS_PRIV) == 0) {
|
(nsp->nsp_socket->so_state & SS_PRIV) == 0)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (EACCES);
|
return (EACCES);
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (ns_pcblookup(&zerons_addr, lport, 0))
|
if (ns_pcblookup(&zerons_addr, lport, 0))
|
||||||
return (EADDRINUSE);
|
return (EADDRINUSE);
|
||||||
}
|
}
|
||||||
@ -152,12 +148,8 @@ ns_pcbconnect(nsp, nam)
|
|||||||
*/
|
*/
|
||||||
ro = &nsp->nsp_route;
|
ro = &nsp->nsp_route;
|
||||||
dst = &satons_addr(ro->ro_dst);
|
dst = &satons_addr(ro->ro_dst);
|
||||||
SOCK_LOCK(nsp->nsp_socket);
|
if (nsp->nsp_socket->so_options & SO_DONTROUTE)
|
||||||
if (nsp->nsp_socket->so_options & SO_DONTROUTE) {
|
|
||||||
SOCK_UNLOCK(nsp->nsp_socket);
|
|
||||||
goto flush;
|
goto flush;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(nsp->nsp_socket);
|
|
||||||
if (!ns_neteq(nsp->nsp_lastdst, sns->sns_addr))
|
if (!ns_neteq(nsp->nsp_lastdst, sns->sns_addr))
|
||||||
goto flush;
|
goto flush;
|
||||||
if (!ns_hosteq(nsp->nsp_lastdst, sns->sns_addr)) {
|
if (!ns_hosteq(nsp->nsp_lastdst, sns->sns_addr)) {
|
||||||
@ -173,19 +165,16 @@ ns_pcbconnect(nsp, nam)
|
|||||||
}
|
}
|
||||||
}/* else cached route is ok; do nothing */
|
}/* else cached route is ok; do nothing */
|
||||||
nsp->nsp_lastdst = sns->sns_addr;
|
nsp->nsp_lastdst = sns->sns_addr;
|
||||||
SOCK_LOCK(nsp->nsp_socket);
|
|
||||||
if ((nsp->nsp_socket->so_options & SO_DONTROUTE) == 0 && /*XXX*/
|
if ((nsp->nsp_socket->so_options & SO_DONTROUTE) == 0 && /*XXX*/
|
||||||
(ro->ro_rt == (struct rtentry *)0 ||
|
(ro->ro_rt == (struct rtentry *)0 ||
|
||||||
ro->ro_rt->rt_ifp == (struct ifnet *)0)) {
|
ro->ro_rt->rt_ifp == (struct ifnet *)0)) {
|
||||||
SOCK_UNLOCK(nsp->nsp_socket);
|
|
||||||
/* No route yet, so try to acquire one */
|
/* No route yet, so try to acquire one */
|
||||||
ro->ro_dst.sa_family = AF_NS;
|
ro->ro_dst.sa_family = AF_NS;
|
||||||
ro->ro_dst.sa_len = sizeof(ro->ro_dst);
|
ro->ro_dst.sa_len = sizeof(ro->ro_dst);
|
||||||
*dst = sns->sns_addr;
|
*dst = sns->sns_addr;
|
||||||
dst->x_port = 0;
|
dst->x_port = 0;
|
||||||
rtalloc(ro);
|
rtalloc(ro);
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(nsp->nsp_socket);
|
|
||||||
if (ns_neteqnn(nsp->nsp_laddr.x_net, ns_zeronet)) {
|
if (ns_neteqnn(nsp->nsp_laddr.x_net, ns_zeronet)) {
|
||||||
/*
|
/*
|
||||||
* If route is known or can be allocated now,
|
* If route is known or can be allocated now,
|
||||||
@ -233,12 +222,8 @@ ns_pcbdisconnect(nsp)
|
|||||||
{
|
{
|
||||||
|
|
||||||
nsp->nsp_faddr = zerons_addr;
|
nsp->nsp_faddr = zerons_addr;
|
||||||
SOCK_LOCK(so);
|
if (nsp->nsp_socket->so_state & SS_NOFDREF)
|
||||||
if (nsp->nsp_socket->so_state & SS_NOFDREF) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
ns_pcbdetach(nsp);
|
ns_pcbdetach(nsp);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ns_pcbdetach(nsp)
|
ns_pcbdetach(nsp)
|
||||||
@ -247,7 +232,6 @@ ns_pcbdetach(nsp)
|
|||||||
struct socket *so = nsp->nsp_socket;
|
struct socket *so = nsp->nsp_socket;
|
||||||
|
|
||||||
so->so_pcb = 0;
|
so->so_pcb = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
sotryfree(so);
|
sotryfree(so);
|
||||||
if (nsp->nsp_route.ro_rt)
|
if (nsp->nsp_route.ro_rt)
|
||||||
rtfree(nsp->nsp_route.ro_rt);
|
rtfree(nsp->nsp_route.ro_rt);
|
||||||
|
@ -106,7 +106,6 @@ spp_input(m, nsp)
|
|||||||
si->si_alo = ntohs(si->si_alo);
|
si->si_alo = ntohs(si->si_alo);
|
||||||
|
|
||||||
so = nsp->nsp_socket;
|
so = nsp->nsp_socket;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_options & SO_DEBUG || traceallspps) {
|
if (so->so_options & SO_DEBUG || traceallspps) {
|
||||||
ostate = cb->s_state;
|
ostate = cb->s_state;
|
||||||
spp_savesi = *si;
|
spp_savesi = *si;
|
||||||
@ -114,7 +113,6 @@ spp_input(m, nsp)
|
|||||||
if (so->so_options & SO_ACCEPTCONN) {
|
if (so->so_options & SO_ACCEPTCONN) {
|
||||||
struct sppcb *ocb = cb;
|
struct sppcb *ocb = cb;
|
||||||
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
so = sonewconn(so, 0);
|
so = sonewconn(so, 0);
|
||||||
if (so == 0) {
|
if (so == 0) {
|
||||||
goto drop;
|
goto drop;
|
||||||
@ -138,8 +136,7 @@ spp_input(m, nsp)
|
|||||||
cb->s_flags = ocb->s_flags; /* preserve sockopts */
|
cb->s_flags = ocb->s_flags; /* preserve sockopts */
|
||||||
cb->s_flags2 = ocb->s_flags2; /* preserve sockopts */
|
cb->s_flags2 = ocb->s_flags2; /* preserve sockopts */
|
||||||
cb->s_state = TCPS_LISTEN;
|
cb->s_state = TCPS_LISTEN;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Packet received on connection.
|
* Packet received on connection.
|
||||||
@ -210,9 +207,7 @@ spp_input(m, nsp)
|
|||||||
nsp->nsp_fport = si->si_sport;
|
nsp->nsp_fport = si->si_sport;
|
||||||
cb->s_timer[SPPT_REXMT] = 0;
|
cb->s_timer[SPPT_REXMT] = 0;
|
||||||
cb->s_timer[SPPT_KEEP] = SPPTV_KEEP;
|
cb->s_timer[SPPT_KEEP] = SPPTV_KEEP;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
cb->s_state = TCPS_ESTABLISHED;
|
cb->s_state = TCPS_ESTABLISHED;
|
||||||
sppstat.spps_accepts++;
|
sppstat.spps_accepts++;
|
||||||
}
|
}
|
||||||
@ -239,9 +234,7 @@ spp_input(m, nsp)
|
|||||||
cb->s_dport = nsp->nsp_fport = si->si_sport;
|
cb->s_dport = nsp->nsp_fport = si->si_sport;
|
||||||
cb->s_timer[SPPT_REXMT] = 0;
|
cb->s_timer[SPPT_REXMT] = 0;
|
||||||
cb->s_flags |= SF_ACKNOW;
|
cb->s_flags |= SF_ACKNOW;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnected(so);
|
soisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
cb->s_state = TCPS_ESTABLISHED;
|
cb->s_state = TCPS_ESTABLISHED;
|
||||||
/* Use roundtrip time of connection request for initial rtt */
|
/* Use roundtrip time of connection request for initial rtt */
|
||||||
if (cb->s_rtt) {
|
if (cb->s_rtt) {
|
||||||
@ -253,12 +246,8 @@ spp_input(m, nsp)
|
|||||||
cb->s_rtt = 0;
|
cb->s_rtt = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG || traceallspps)
|
||||||
if (so->so_options & SO_DEBUG || traceallspps) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spp_trace(SA_INPUT, (u_char)ostate, cb, &spp_savesi, 0);
|
spp_trace(SA_INPUT, (u_char)ostate, cb, &spp_savesi, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
m->m_len -= sizeof (struct idp);
|
m->m_len -= sizeof (struct idp);
|
||||||
m->m_pkthdr.len -= sizeof (struct idp);
|
m->m_pkthdr.len -= sizeof (struct idp);
|
||||||
@ -279,23 +268,15 @@ dropwithreset:
|
|||||||
si->si_ack = ntohs(si->si_ack);
|
si->si_ack = ntohs(si->si_ack);
|
||||||
si->si_alo = ntohs(si->si_alo);
|
si->si_alo = ntohs(si->si_alo);
|
||||||
ns_error(dtom(si), NS_ERR_NOSOCK, 0);
|
ns_error(dtom(si), NS_ERR_NOSOCK, 0);
|
||||||
SOCK_LOCK(cb->s_nspcb->nsp_socket);
|
if (cb->s_nspcb->nsp_socket->so_options & SO_DEBUG || traceallspps)
|
||||||
if (cb->s_nspcb->nsp_socket->so_options & SO_DEBUG || traceallspps) {
|
|
||||||
SOCK_UNLOCK(cb->s_nspcb->nsp_socket);
|
|
||||||
spp_trace(SA_DROP, (u_char)ostate, cb, &spp_savesi, 0);
|
spp_trace(SA_DROP, (u_char)ostate, cb, &spp_savesi, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(cb->s_nspcb->nsp_socket);
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
drop:
|
drop:
|
||||||
bad:
|
bad:
|
||||||
SOCK_LOCK(cb->s_nspcb->nsp_socket);
|
|
||||||
if (cb == 0 || cb->s_nspcb->nsp_socket->so_options & SO_DEBUG ||
|
if (cb == 0 || cb->s_nspcb->nsp_socket->so_options & SO_DEBUG ||
|
||||||
traceallspps) {
|
traceallspps)
|
||||||
SOCK_UNLOCK(cb->s_nspcb->nsp_socket);
|
|
||||||
spp_trace(SA_DROP, (u_char)ostate, cb, &spp_savesi, 0);
|
spp_trace(SA_DROP, (u_char)ostate, cb, &spp_savesi, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(cb->s_nspcb->nsp_socket);
|
|
||||||
m_freem(m);
|
m_freem(m);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -429,9 +410,7 @@ register struct spidp *si;
|
|||||||
else
|
else
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
sowwakeup(so);
|
sowwakeup(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
cb->s_rack = si->si_ack;
|
cb->s_rack = si->si_ack;
|
||||||
update_window:
|
update_window:
|
||||||
if (SSEQ_LT(cb->s_snxt, cb->s_rack))
|
if (SSEQ_LT(cb->s_snxt, cb->s_rack))
|
||||||
@ -470,14 +449,11 @@ update_window:
|
|||||||
} /* else queue this packet; */
|
} /* else queue this packet; */
|
||||||
} else {
|
} else {
|
||||||
/*register struct socket *so = cb->s_nspcb->nsp_socket;
|
/*register struct socket *so = cb->s_nspcb->nsp_socket;
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state && SS_NOFDREF) {
|
if (so->so_state && SS_NOFDREF) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
ns_error(dtom(si), NS_ERR_NOSOCK, 0);
|
ns_error(dtom(si), NS_ERR_NOSOCK, 0);
|
||||||
(void)spp_close(cb);
|
(void)spp_close(cb);
|
||||||
} else {
|
} else
|
||||||
SOCK_UNLOCK(so);
|
would crash system*/
|
||||||
would crash system }*/
|
|
||||||
spp_istat.notyet++;
|
spp_istat.notyet++;
|
||||||
ns_error(dtom(si), NS_ERR_FULLUP, 0);
|
ns_error(dtom(si), NS_ERR_FULLUP, 0);
|
||||||
return (0);
|
return (0);
|
||||||
@ -538,11 +514,8 @@ present:
|
|||||||
cb->s_oobflags &= ~SF_IOOB;
|
cb->s_oobflags &= ~SF_IOOB;
|
||||||
if (so->so_rcv.sb_cc)
|
if (so->so_rcv.sb_cc)
|
||||||
so->so_oobmark = so->so_rcv.sb_cc;
|
so->so_oobmark = so->so_rcv.sb_cc;
|
||||||
else {
|
else
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_RCVATMARK;
|
so->so_state |= SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
q = q->si_prev;
|
q = q->si_prev;
|
||||||
remque(q->si_next);
|
remque(q->si_next);
|
||||||
@ -572,9 +545,7 @@ present:
|
|||||||
MCHTYPE(m, MT_OOBDATA);
|
MCHTYPE(m, MT_OOBDATA);
|
||||||
spp_newchecks[1]++;
|
spp_newchecks[1]++;
|
||||||
so->so_oobmark = 0;
|
so->so_oobmark = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_RCVATMARK;
|
so->so_state &= ~SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
if (packetp == 0) {
|
if (packetp == 0) {
|
||||||
m->m_data += SPINC;
|
m->m_data += SPINC;
|
||||||
@ -600,11 +571,7 @@ present:
|
|||||||
} else
|
} else
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (wakeup) {
|
if (wakeup) sorwakeup(so);
|
||||||
SOCK_LOCK(so);
|
|
||||||
sorwakeup(so);
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1048,12 +1015,8 @@ send:
|
|||||||
si->si_cc |= SP_SP;
|
si->si_cc |= SP_SP;
|
||||||
} else {
|
} else {
|
||||||
cb->s_outx = 3;
|
cb->s_outx = 3;
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG || traceallspps)
|
||||||
if (so->so_options & SO_DEBUG || traceallspps) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spp_trace(SA_OUTPUT, cb->s_state, cb, si, 0);
|
spp_trace(SA_OUTPUT, cb->s_state, cb, si, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
@ -1119,20 +1082,13 @@ send:
|
|||||||
si->si_sum = 0xffff;
|
si->si_sum = 0xffff;
|
||||||
|
|
||||||
cb->s_outx = 4;
|
cb->s_outx = 4;
|
||||||
SOCK_LOCK(so);
|
if (so->so_options & SO_DEBUG || traceallspps)
|
||||||
if (so->so_options & SO_DEBUG || traceallspps) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spp_trace(SA_OUTPUT, cb->s_state, cb, si, 0);
|
spp_trace(SA_OUTPUT, cb->s_state, cb, si, 0);
|
||||||
SOCK_LOCK(so);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (so->so_options & SO_DONTROUTE) {
|
if (so->so_options & SO_DONTROUTE)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ns_output(m, (struct route *)0, NS_ROUTETOIF);
|
error = ns_output(m, (struct route *)0, NS_ROUTETOIF);
|
||||||
} else {
|
else
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ns_output(m, &cb->s_nspcb->nsp_route, 0);
|
error = ns_output(m, &cb->s_nspcb->nsp_route, 0);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (error) {
|
if (error) {
|
||||||
return (error);
|
return (error);
|
||||||
@ -1413,9 +1369,7 @@ spp_usrreq(so, req, m, nam, controlp)
|
|||||||
error = ns_pcbconnect(nsp, nam);
|
error = ns_pcbconnect(nsp, nam);
|
||||||
if (error)
|
if (error)
|
||||||
break;
|
break;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisconnecting(so);
|
soisconnecting(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sppstat.spps_connattempt++;
|
sppstat.spps_connattempt++;
|
||||||
cb->s_state = TCPS_SYN_SENT;
|
cb->s_state = TCPS_SYN_SENT;
|
||||||
cb->s_did = 0;
|
cb->s_did = 0;
|
||||||
@ -1489,15 +1443,12 @@ spp_usrreq(so, req, m, nam, controlp)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case PRU_RCVOOB:
|
case PRU_RCVOOB:
|
||||||
SOCK_LOCK(so);
|
|
||||||
if ((cb->s_oobflags & SF_IOOB) || so->so_oobmark ||
|
if ((cb->s_oobflags & SF_IOOB) || so->so_oobmark ||
|
||||||
(so->so_state & SS_RCVATMARK)) {
|
(so->so_state & SS_RCVATMARK)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
m->m_len = 1;
|
m->m_len = 1;
|
||||||
*mtod(m, caddr_t) = cb->s_iobc;
|
*mtod(m, caddr_t) = cb->s_iobc;
|
||||||
break;
|
break;
|
||||||
} else
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = EINVAL;
|
error = EINVAL;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -1545,12 +1496,8 @@ spp_usrreq(so, req, m, nam, controlp)
|
|||||||
default:
|
default:
|
||||||
panic("sp_usrreq");
|
panic("sp_usrreq");
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
if (cb && (so->so_options & SO_DEBUG || traceallspps))
|
||||||
if (cb && (so->so_options & SO_DEBUG || traceallspps)) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
spp_trace(SA_USER, (u_char)ostate, cb, (struct spidp *)0, req);
|
spp_trace(SA_USER, (u_char)ostate, cb, (struct spidp *)0, req);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
release:
|
release:
|
||||||
if (controlp != NULL)
|
if (controlp != NULL)
|
||||||
m_freem(controlp);
|
m_freem(controlp);
|
||||||
@ -1627,9 +1574,7 @@ spp_close(cb)
|
|||||||
(void) m_free(dtom(cb->s_idp));
|
(void) m_free(dtom(cb->s_idp));
|
||||||
(void) m_free(dtom(cb));
|
(void) m_free(dtom(cb));
|
||||||
nsp->nsp_pcb = 0;
|
nsp->nsp_pcb = 0;
|
||||||
SOCK_LOCK(so);
|
|
||||||
soisdisconnected(so);
|
soisdisconnected(so);
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
ns_pcbdetach(nsp);
|
ns_pcbdetach(nsp);
|
||||||
sppstat.spps_closed++;
|
sppstat.spps_closed++;
|
||||||
return ((struct sppcb *)0);
|
return ((struct sppcb *)0);
|
||||||
@ -1838,17 +1783,13 @@ spp_timers(cb, timer)
|
|||||||
sppstat.spps_keeptimeo++;
|
sppstat.spps_keeptimeo++;
|
||||||
if (cb->s_state < TCPS_ESTABLISHED)
|
if (cb->s_state < TCPS_ESTABLISHED)
|
||||||
goto dropit;
|
goto dropit;
|
||||||
SOCK_LOCK(cb->s_nspcb->nsp_socket);
|
|
||||||
if (cb->s_nspcb->nsp_socket->so_options & SO_KEEPALIVE) {
|
if (cb->s_nspcb->nsp_socket->so_options & SO_KEEPALIVE) {
|
||||||
SOCK_UNLOCK(cb->s_nspcb->nsp_socket);
|
|
||||||
if (cb->s_idle >= SPPTV_MAXIDLE)
|
if (cb->s_idle >= SPPTV_MAXIDLE)
|
||||||
goto dropit;
|
goto dropit;
|
||||||
sppstat.spps_keepprobe++;
|
sppstat.spps_keepprobe++;
|
||||||
(void) spp_output(cb, (struct mbuf *) 0);
|
(void) spp_output(cb, (struct mbuf *) 0);
|
||||||
} else {
|
} else
|
||||||
SOCK_UNLOCK(cb->s_nspcb->nsp_socket);
|
|
||||||
cb->s_idle = 0;
|
cb->s_idle = 0;
|
||||||
}
|
|
||||||
cb->s_timer[SPPT_KEEP] = SPPTV_KEEP;
|
cb->s_timer[SPPT_KEEP] = SPPTV_KEEP;
|
||||||
break;
|
break;
|
||||||
dropit:
|
dropit:
|
||||||
|
@ -250,18 +250,15 @@ nb_connect_in(struct nbpcb *nbp, struct sockaddr_in *to, struct thread *td)
|
|||||||
if (error)
|
if (error)
|
||||||
goto bad;
|
goto bad;
|
||||||
s = splnet();
|
s = splnet();
|
||||||
SOCK_LOCK(so);
|
|
||||||
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
||||||
msleep(&so->so_timeo, SOCK_MTX(so), PSOCK, "nbcon", 2 * hz);
|
tsleep(&so->so_timeo, PSOCK, "nbcon", 2 * hz);
|
||||||
if ((so->so_state & SS_ISCONNECTING) && so->so_error == 0 &&
|
if ((so->so_state & SS_ISCONNECTING) && so->so_error == 0 &&
|
||||||
(error = nb_intr(nbp, td->td_proc)) != 0) {
|
(error = nb_intr(nbp, td->td_proc)) != 0) {
|
||||||
so->so_state &= ~SS_ISCONNECTING;
|
so->so_state &= ~SS_ISCONNECTING;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (so->so_error) {
|
if (so->so_error) {
|
||||||
error = so->so_error;
|
error = so->so_error;
|
||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
@ -411,15 +408,12 @@ nbssn_recv(struct nbpcb *nbp, struct mbuf **mpp, int *lenp,
|
|||||||
for(;;) {
|
for(;;) {
|
||||||
m = NULL;
|
m = NULL;
|
||||||
error = nbssn_recvhdr(nbp, &len, &rpcode, MSG_DONTWAIT, td);
|
error = nbssn_recvhdr(nbp, &len, &rpcode, MSG_DONTWAIT, td);
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state &
|
if (so->so_state &
|
||||||
(SS_ISDISCONNECTING | SS_ISDISCONNECTED | SS_CANTRCVMORE)) {
|
(SS_ISDISCONNECTING | SS_ISDISCONNECTED | SS_CANTRCVMORE)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
nbp->nbp_state = NBST_CLOSED;
|
nbp->nbp_state = NBST_CLOSED;
|
||||||
NBDEBUG("session closed by peer\n");
|
NBDEBUG("session closed by peer\n");
|
||||||
return ECONNRESET;
|
return ECONNRESET;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (error)
|
if (error)
|
||||||
return error;
|
return error;
|
||||||
if (len == 0 && nbp->nbp_state != NBST_SESSION)
|
if (len == 0 && nbp->nbp_state != NBST_SESSION)
|
||||||
@ -645,10 +639,8 @@ smb_nbst_intr(struct smb_vc *vcp)
|
|||||||
|
|
||||||
if (nbp == NULL || nbp->nbp_tso == NULL)
|
if (nbp == NULL || nbp->nbp_tso == NULL)
|
||||||
return;
|
return;
|
||||||
SOCK_LOCK(nbp->nbp_tso);
|
|
||||||
sorwakeup(nbp->nbp_tso);
|
sorwakeup(nbp->nbp_tso);
|
||||||
sowwakeup(nbp->nbp_tso);
|
sowwakeup(nbp->nbp_tso);
|
||||||
SOCK_UNLOCK(nbp->nbp_tso);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
@ -752,13 +752,8 @@ bootpc_call(struct bootpc_globalcontext *gctx, struct thread *td)
|
|||||||
error = sosend(so, (struct sockaddr *) &dst,
|
error = sosend(so, (struct sockaddr *) &dst,
|
||||||
&auio, NULL, NULL, 0, td);
|
&auio, NULL, NULL, 0, td);
|
||||||
if (error != 0) {
|
if (error != 0) {
|
||||||
int sostate;
|
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
|
||||||
sostate = (int)so->so_state;
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
printf("bootpc_call: sosend: %d state %08x\n",
|
printf("bootpc_call: sosend: %d state %08x\n",
|
||||||
error, sostate);
|
error, (int) so->so_state);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* XXX: Is this needed ? */
|
/* XXX: Is this needed ? */
|
||||||
|
@ -230,21 +230,18 @@ nfs_connect(struct nfsmount *nmp, struct nfsreq *rep)
|
|||||||
* that interruptible mounts don't hang here for a long time.
|
* that interruptible mounts don't hang here for a long time.
|
||||||
*/
|
*/
|
||||||
s = splnet();
|
s = splnet();
|
||||||
SOCK_LOCK(so);
|
|
||||||
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0) {
|
||||||
(void) msleep((caddr_t)&so->so_timeo, SOCK_MTX(so),
|
(void) tsleep((caddr_t)&so->so_timeo,
|
||||||
PSOCK, "nfscon", 2 * hz);
|
PSOCK, "nfscon", 2 * hz);
|
||||||
if ((so->so_state & SS_ISCONNECTING) &&
|
if ((so->so_state & SS_ISCONNECTING) &&
|
||||||
so->so_error == 0 && rep &&
|
so->so_error == 0 && rep &&
|
||||||
(error = nfs_sigintr(nmp, rep,
|
(error = nfs_sigintr(nmp, rep,
|
||||||
(rep->r_td ? rep->r_td->td_proc : NULL))) != 0){
|
(rep->r_td ? rep->r_td->td_proc : NULL))) != 0){
|
||||||
so->so_state &= ~SS_ISCONNECTING;
|
so->so_state &= ~SS_ISCONNECTING;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
splx(s);
|
splx(s);
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (so->so_error) {
|
if (so->so_error) {
|
||||||
error = so->so_error;
|
error = so->so_error;
|
||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
@ -417,14 +414,10 @@ nfs_send(struct socket *so, struct sockaddr *nam, struct mbuf *top,
|
|||||||
rep->r_flags &= ~R_MUSTRESEND;
|
rep->r_flags &= ~R_MUSTRESEND;
|
||||||
soflags = rep->r_nmp->nm_soflags;
|
soflags = rep->r_nmp->nm_soflags;
|
||||||
|
|
||||||
SOCK_LOCK(so);
|
if ((soflags & PR_CONNREQUIRED) || (so->so_state & SS_ISCONNECTED))
|
||||||
if ((soflags & PR_CONNREQUIRED) || (so->so_state & SS_ISCONNECTED)) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sendnam = (struct sockaddr *)0;
|
sendnam = (struct sockaddr *)0;
|
||||||
} else {
|
else
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sendnam = nam;
|
sendnam = nam;
|
||||||
}
|
|
||||||
if (so->so_type == SOCK_SEQPACKET)
|
if (so->so_type == SOCK_SEQPACKET)
|
||||||
flags = MSG_EOR;
|
flags = MSG_EOR;
|
||||||
else
|
else
|
||||||
@ -653,14 +646,10 @@ errout:
|
|||||||
} else {
|
} else {
|
||||||
if ((so = rep->r_nmp->nm_so) == NULL)
|
if ((so = rep->r_nmp->nm_so) == NULL)
|
||||||
return (EACCES);
|
return (EACCES);
|
||||||
SOCK_LOCK(so);
|
if (so->so_state & SS_ISCONNECTED)
|
||||||
if (so->so_state & SS_ISCONNECTED) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
getnam = (struct sockaddr **)0;
|
getnam = (struct sockaddr **)0;
|
||||||
} else {
|
else
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
getnam = aname;
|
getnam = aname;
|
||||||
}
|
|
||||||
auio.uio_resid = len = 1000000;
|
auio.uio_resid = len = 1000000;
|
||||||
auio.uio_td = td;
|
auio.uio_td = td;
|
||||||
do {
|
do {
|
||||||
|
@ -714,14 +714,10 @@ nfsrv_send(struct socket *so, struct sockaddr *nam, struct mbuf *top)
|
|||||||
int error, soflags, flags;
|
int error, soflags, flags;
|
||||||
|
|
||||||
soflags = so->so_proto->pr_flags;
|
soflags = so->so_proto->pr_flags;
|
||||||
SOCK_LOCK(so);
|
if ((soflags & PR_CONNREQUIRED) || (so->so_state & SS_ISCONNECTED))
|
||||||
if ((soflags & PR_CONNREQUIRED) || (so->so_state & SS_ISCONNECTED)) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sendnam = (struct sockaddr *)0;
|
sendnam = (struct sockaddr *)0;
|
||||||
} else {
|
else
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sendnam = nam;
|
sendnam = nam;
|
||||||
}
|
|
||||||
if (so->so_type == SOCK_SEQPACKET)
|
if (so->so_type == SOCK_SEQPACKET)
|
||||||
flags = MSG_EOR;
|
flags = MSG_EOR;
|
||||||
else
|
else
|
||||||
|
@ -413,15 +413,10 @@ set_object_lattr(lomac_object_t *obj, lattr_t lattr) {
|
|||||||
case LO_TYPE_SOCKETPAIR:
|
case LO_TYPE_SOCKETPAIR:
|
||||||
socket = obj->lo_object.socket;
|
socket = obj->lo_object.socket;
|
||||||
/* KASSERT that socket peer levels are synchronized */
|
/* KASSERT that socket peer levels are synchronized */
|
||||||
if (lattr.level == LOMAC_HIGHEST_LEVEL) {
|
if (lattr.level == LOMAC_HIGHEST_LEVEL)
|
||||||
SOCK_LOCK(socket);
|
|
||||||
socket->so_state &= ~SOCKET_LEVEL_LOWEST;
|
socket->so_state &= ~SOCKET_LEVEL_LOWEST;
|
||||||
SOCK_UNLOCK(socket);
|
else
|
||||||
} else {
|
|
||||||
SOCK_LOCK(socket);
|
|
||||||
socket->so_state |= SOCKET_LEVEL_LOWEST;
|
socket->so_state |= SOCKET_LEVEL_LOWEST;
|
||||||
SOCK_UNLOCK(socket);
|
|
||||||
}
|
|
||||||
#ifdef NOT_YET
|
#ifdef NOT_YET
|
||||||
pipe = pipe->pipe_peer;
|
pipe = pipe->pipe_peer;
|
||||||
if (pipe != NULL) {
|
if (pipe != NULL) {
|
||||||
@ -479,10 +474,8 @@ get_object_lattr(const lomac_object_t *obj, lattr_t *lattr) {
|
|||||||
break;
|
break;
|
||||||
case LO_TYPE_SOCKETPAIR:
|
case LO_TYPE_SOCKETPAIR:
|
||||||
socket = obj->lo_object.socket;
|
socket = obj->lo_object.socket;
|
||||||
SOCK_LOCK(socket);
|
|
||||||
lattr->level = (socket->so_state & SOCKET_LEVEL_LOWEST) ?
|
lattr->level = (socket->so_state & SOCKET_LEVEL_LOWEST) ?
|
||||||
LOMAC_LOWEST_LEVEL : LOMAC_HIGHEST_LEVEL;
|
LOMAC_LOWEST_LEVEL : LOMAC_HIGHEST_LEVEL;
|
||||||
SOCK_UNLOCK(socket);
|
|
||||||
lattr->flags = 0;
|
lattr->flags = 0;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -245,14 +245,8 @@ lomac_local_connect(struct socket *so, struct sockaddr *nam, struct thread *td)
|
|||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
if (so->so_proto->pr_flags & PR_CONNREQUIRED) {
|
if (so->so_proto->pr_flags & PR_CONNREQUIRED) {
|
||||||
SOCK_LOCK(so2);
|
if ((so2->so_options & SO_ACCEPTCONN) == 0 ||
|
||||||
if ((so2->so_options & SO_ACCEPTCONN) == 0) {
|
(so3 = sonewconn(so2, 0)) == 0) {
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
error = ECONNREFUSED;
|
|
||||||
goto bad;
|
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so2);
|
|
||||||
if ((so3 = sonewconn(so2, 0)) == 0) {
|
|
||||||
error = ECONNREFUSED;
|
error = ECONNREFUSED;
|
||||||
goto bad;
|
goto bad;
|
||||||
}
|
}
|
||||||
@ -351,20 +345,15 @@ lomac_local_send( struct socket *so, int flags, struct mbuf *m,
|
|||||||
error = ENOTCONN;
|
error = ENOTCONN;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
} else {
|
} else if ((so->so_state & SS_ISCONNECTED) == 0) {
|
||||||
SOCK_LOCK(so);
|
if (addr != NULL) {
|
||||||
if ((so->so_state & SS_ISCONNECTED) == 0) {
|
error = lomac_local_connect(so, addr, td);
|
||||||
SOCK_UNLOCK(so);
|
if (error)
|
||||||
if (addr != NULL) {
|
goto out; /* XXX */
|
||||||
error = lomac_local_connect(so, addr, td);
|
} else {
|
||||||
if (error)
|
error = ENOTCONN;
|
||||||
goto out; /* XXX */
|
goto out;
|
||||||
} else {
|
}
|
||||||
error = ENOTCONN;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
}
|
}
|
||||||
vp = unp->unp_vnode;
|
vp = unp->unp_vnode;
|
||||||
if (vp != NULL) {
|
if (vp != NULL) {
|
||||||
@ -571,12 +560,8 @@ bad:
|
|||||||
}
|
}
|
||||||
if (mp)
|
if (mp)
|
||||||
*mp = (struct mbuf *)0;
|
*mp = (struct mbuf *)0;
|
||||||
SOCK_LOCK(so);
|
if (so->so_state & SS_ISCONFIRMING && uio->uio_resid)
|
||||||
if (so->so_state & SS_ISCONFIRMING && uio->uio_resid) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
(*pr->pr_usrreqs->pru_rcvd)(so, 0);
|
(*pr->pr_usrreqs->pru_rcvd)(so, 0);
|
||||||
} else
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
restart:
|
restart:
|
||||||
error = sblock(&so->so_rcv, SBLOCKWAIT(flags));
|
error = sblock(&so->so_rcv, SBLOCKWAIT(flags));
|
||||||
@ -612,9 +597,7 @@ restart:
|
|||||||
so->so_error = 0;
|
so->so_error = 0;
|
||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (so->so_state & SS_CANTRCVMORE) {
|
if (so->so_state & SS_CANTRCVMORE) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
if (m)
|
if (m)
|
||||||
goto dontblock;
|
goto dontblock;
|
||||||
else
|
else
|
||||||
@ -623,25 +606,19 @@ restart:
|
|||||||
for (; m; m = m->m_next)
|
for (; m; m = m->m_next)
|
||||||
if (m->m_type == MT_OOBDATA || (m->m_flags & M_EOR)) {
|
if (m->m_type == MT_OOBDATA || (m->m_flags & M_EOR)) {
|
||||||
m = so->so_rcv.sb_mb;
|
m = so->so_rcv.sb_mb;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto dontblock;
|
goto dontblock;
|
||||||
}
|
}
|
||||||
if ((so->so_state & (SS_ISCONNECTED|SS_ISCONNECTING)) == 0 &&
|
if ((so->so_state & (SS_ISCONNECTED|SS_ISCONNECTING)) == 0 &&
|
||||||
(so->so_proto->pr_flags & PR_CONNREQUIRED)) {
|
(so->so_proto->pr_flags & PR_CONNREQUIRED)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = ENOTCONN;
|
error = ENOTCONN;
|
||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
if (uio->uio_resid == 0) {
|
if (uio->uio_resid == 0)
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
goto release;
|
goto release;
|
||||||
}
|
|
||||||
if ((so->so_state & SS_NBIO) || (flags & MSG_DONTWAIT)) {
|
if ((so->so_state & SS_NBIO) || (flags & MSG_DONTWAIT)) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
error = EWOULDBLOCK;
|
error = EWOULDBLOCK;
|
||||||
goto release;
|
goto release;
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
sbunlock(&so->so_rcv);
|
sbunlock(&so->so_rcv);
|
||||||
error = sbwait(&so->so_rcv);
|
error = sbwait(&so->so_rcv);
|
||||||
splx(s);
|
splx(s);
|
||||||
@ -710,9 +687,7 @@ dontblock:
|
|||||||
else
|
else
|
||||||
KASSERT(m->m_type == MT_DATA || m->m_type == MT_HEADER,
|
KASSERT(m->m_type == MT_DATA || m->m_type == MT_HEADER,
|
||||||
("receive 3"));
|
("receive 3"));
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state &= ~SS_RCVATMARK;
|
so->so_state &= ~SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
len = uio->uio_resid;
|
len = uio->uio_resid;
|
||||||
if (so->so_oobmark && len > so->so_oobmark - offset)
|
if (so->so_oobmark && len > so->so_oobmark - offset)
|
||||||
len = so->so_oobmark - offset;
|
len = so->so_oobmark - offset;
|
||||||
@ -771,9 +746,7 @@ dontblock:
|
|||||||
if ((flags & MSG_PEEK) == 0) {
|
if ((flags & MSG_PEEK) == 0) {
|
||||||
so->so_oobmark -= len;
|
so->so_oobmark -= len;
|
||||||
if (so->so_oobmark == 0) {
|
if (so->so_oobmark == 0) {
|
||||||
SOCK_LOCK(so);
|
|
||||||
so->so_state |= SS_RCVATMARK;
|
so->so_state |= SS_RCVATMARK;
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -793,12 +766,8 @@ dontblock:
|
|||||||
*/
|
*/
|
||||||
while (flags & MSG_WAITALL && m == 0 && uio->uio_resid > 0 &&
|
while (flags & MSG_WAITALL && m == 0 && uio->uio_resid > 0 &&
|
||||||
!sosendallatonce(so) && !nextrecord) {
|
!sosendallatonce(so) && !nextrecord) {
|
||||||
SOCK_LOCK(so);
|
if (so->so_error || so->so_state & SS_CANTRCVMORE)
|
||||||
if (so->so_error || so->so_state & SS_CANTRCVMORE) {
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
/*
|
/*
|
||||||
* Notify the protocol that some data has been
|
* Notify the protocol that some data has been
|
||||||
* drained before blocking.
|
* drained before blocking.
|
||||||
@ -828,15 +797,12 @@ dontblock:
|
|||||||
if (pr->pr_flags & PR_WANTRCVD && so->so_pcb)
|
if (pr->pr_flags & PR_WANTRCVD && so->so_pcb)
|
||||||
(*pr->pr_usrreqs->pru_rcvd)(so, flags);
|
(*pr->pr_usrreqs->pru_rcvd)(so, flags);
|
||||||
}
|
}
|
||||||
SOCK_LOCK(so);
|
|
||||||
if (orig_resid == uio->uio_resid && orig_resid &&
|
if (orig_resid == uio->uio_resid && orig_resid &&
|
||||||
(flags & MSG_EOR) == 0 && (so->so_state & SS_CANTRCVMORE) == 0) {
|
(flags & MSG_EOR) == 0 && (so->so_state & SS_CANTRCVMORE) == 0) {
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
sbunlock(&so->so_rcv);
|
sbunlock(&so->so_rcv);
|
||||||
splx(s);
|
splx(s);
|
||||||
goto restart;
|
goto restart;
|
||||||
}
|
}
|
||||||
SOCK_UNLOCK(so);
|
|
||||||
|
|
||||||
if (flagsp)
|
if (flagsp)
|
||||||
*flagsp |= flags;
|
*flagsp |= flags;
|
||||||
|
@ -38,8 +38,6 @@
|
|||||||
#define _SYS_SOCKETVAR_H_
|
#define _SYS_SOCKETVAR_H_
|
||||||
|
|
||||||
#include <sys/queue.h> /* for TAILQ macros */
|
#include <sys/queue.h> /* for TAILQ macros */
|
||||||
#include <sys/_lock.h>
|
|
||||||
#include <sys/_mutex.h>
|
|
||||||
#include <sys/selinfo.h> /* for struct selinfo */
|
#include <sys/selinfo.h> /* for struct selinfo */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -50,27 +48,27 @@
|
|||||||
*/
|
*/
|
||||||
typedef u_quad_t so_gen_t;
|
typedef u_quad_t so_gen_t;
|
||||||
|
|
||||||
struct socket;
|
|
||||||
typedef void so_upcall_t(struct socket *, void *, int);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* List of locks:
|
* List of locks:
|
||||||
* (c) const, inited in either socreate() or sonewconn()
|
* (c) const, inited in either socreate() or sonewconn()
|
||||||
* (m) sb_mtx mutex
|
* (m) sb_mtx mutex
|
||||||
* (mh) the mutex of so_head
|
|
||||||
* (mr) so_rcv.sb_mtx mutex
|
* (mr) so_rcv.sb_mtx mutex
|
||||||
* (sg) sigio_lock mutex
|
* (sg) sigio_lock sx
|
||||||
|
* (sh) sohead_lock sx
|
||||||
*
|
*
|
||||||
* Members marked by brackets are not locked yet.
|
* Lock of so_rcv.sb_mtx can duplicate, provided that sohead_lock
|
||||||
|
* is exclusively locked.
|
||||||
|
*
|
||||||
|
* Brackets mean that this data is not protected yet.
|
||||||
*/
|
*/
|
||||||
struct socket {
|
struct socket {
|
||||||
int so_count; /* (mr) reference count */
|
int so_count; /* reference count */
|
||||||
short so_type; /* (c) generic type, see socket.h */
|
short so_type; /* generic type, see socket.h */
|
||||||
short so_options; /* (mr) from socket call, see socket.h */
|
short so_options; /* from socket call, see socket.h */
|
||||||
short so_linger; /* (mr) time to linger while closing */
|
short so_linger; /* time to linger while closing */
|
||||||
short so_state; /* (mr) internal state flags SS_*, below */
|
short so_state; /* internal state flags SS_*, below */
|
||||||
caddr_t so_pcb; /* [mr] protocol control block */
|
caddr_t so_pcb; /* protocol control block */
|
||||||
struct protosw *so_proto; /* (c) protocol handle */
|
struct protosw *so_proto; /* protocol handle */
|
||||||
/*
|
/*
|
||||||
* Variables for connection queuing.
|
* Variables for connection queuing.
|
||||||
* Socket where accepts occur is so_head in all subsidiary sockets.
|
* Socket where accepts occur is so_head in all subsidiary sockets.
|
||||||
@ -82,36 +80,33 @@ struct socket {
|
|||||||
* We allow connections to queue up based on current queue lengths
|
* We allow connections to queue up based on current queue lengths
|
||||||
* and limit on number of queued connections for this socket.
|
* and limit on number of queued connections for this socket.
|
||||||
*/
|
*/
|
||||||
struct socket *so_head; /* [mr] back pointer to accept socket */
|
struct socket *so_head; /* back pointer to accept socket */
|
||||||
TAILQ_HEAD(, socket) so_incomp; /* [mr] queue of partial unaccepted connections */
|
TAILQ_HEAD(, socket) so_incomp; /* queue of partial unaccepted connections */
|
||||||
TAILQ_HEAD(, socket) so_comp; /* [mr] queue of complete unaccepted connections */
|
TAILQ_HEAD(, socket) so_comp; /* queue of complete unaccepted connections */
|
||||||
TAILQ_ENTRY(socket) so_list; /* [mh] list of unaccepted connections */
|
TAILQ_ENTRY(socket) so_list; /* list of unaccepted connections */
|
||||||
short so_qlen; /* [mr] number of unaccepted connections */
|
short so_qlen; /* number of unaccepted connections */
|
||||||
short so_incqlen; /* [mr] number of unaccepted incomplete
|
short so_incqlen; /* number of unaccepted incomplete
|
||||||
connections */
|
connections */
|
||||||
short so_qlimit; /* [mr] max number queued connections */
|
short so_qlimit; /* max number queued connections */
|
||||||
short so_timeo; /* [mr] connection timeout */
|
short so_timeo; /* connection timeout */
|
||||||
u_short so_error; /* [mr] error affecting connection */
|
u_short so_error; /* error affecting connection */
|
||||||
struct sigio *so_sigio; /* (sg) information for async I/O or
|
struct sigio *so_sigio; /* [sg] information for async I/O or
|
||||||
out of band data (SIGURG) */
|
out of band data (SIGURG) */
|
||||||
u_long so_oobmark; /* [mr] chars to oob mark */
|
u_long so_oobmark; /* chars to oob mark */
|
||||||
TAILQ_HEAD(, aiocblist) so_aiojobq; /* [mr] AIO ops waiting on socket */
|
TAILQ_HEAD(, aiocblist) so_aiojobq; /* AIO ops waiting on socket */
|
||||||
/*
|
/*
|
||||||
* Variables for socket buffering.
|
* Variables for socket buffering.
|
||||||
*/
|
*/
|
||||||
struct sockbuf {
|
struct sockbuf {
|
||||||
#define sb_startzero sb_cc
|
u_long sb_cc; /* actual chars in buffer */
|
||||||
u_long sb_cc; /* [m] actual chars in buffer */
|
u_long sb_hiwat; /* max actual char count */
|
||||||
u_long sb_hiwat; /* [m] max actual char count */
|
u_long sb_mbcnt; /* chars of mbufs used */
|
||||||
u_long sb_mbcnt; /* [m] chars of mbufs used */
|
u_long sb_mbmax; /* max chars of mbufs to use */
|
||||||
u_long sb_mbmax; /* [m] max chars of mbufs to use */
|
long sb_lowat; /* low water mark */
|
||||||
long sb_lowat; /* [m] low water mark */
|
struct mbuf *sb_mb; /* the mbuf chain */
|
||||||
struct mbuf *sb_mb; /* [m] the mbuf chain */
|
struct selinfo sb_sel; /* process selecting read/write */
|
||||||
struct selinfo sb_sel; /* [m] process selecting read/write */
|
short sb_flags; /* flags, see below */
|
||||||
short sb_flags; /* [m] flags, see below */
|
short sb_timeo; /* timeout for read/write */
|
||||||
short sb_timeo; /* [m] timeout for read/write */
|
|
||||||
#define sb_endzero sb_timeo
|
|
||||||
struct mtx sb_mtx; /* mutex of this socket buffer */
|
|
||||||
} so_rcv, so_snd;
|
} so_rcv, so_snd;
|
||||||
#define SB_MAX (256*1024) /* default for max chars in sockbuf */
|
#define SB_MAX (256*1024) /* default for max chars in sockbuf */
|
||||||
#define SB_LOCK 0x01 /* lock on data queue */
|
#define SB_LOCK 0x01 /* lock on data queue */
|
||||||
@ -124,35 +119,19 @@ struct socket {
|
|||||||
#define SB_AIO 0x80 /* AIO operations queued */
|
#define SB_AIO 0x80 /* AIO operations queued */
|
||||||
#define SB_KNOTE 0x100 /* kernel note attached */
|
#define SB_KNOTE 0x100 /* kernel note attached */
|
||||||
|
|
||||||
so_upcall_t *so_upcall; /* [mr] */
|
void (*so_upcall)(struct socket *, void *, int);
|
||||||
void *so_upcallarg; /* [mr] */
|
void *so_upcallarg;
|
||||||
struct ucred *so_cred; /* (c) user credentials */
|
struct ucred *so_cred; /* user credentials */
|
||||||
/* NB: generation count must not be first; easiest to make it last. */
|
/* NB: generation count must not be first; easiest to make it last. */
|
||||||
so_gen_t so_gencnt; /* [mr] generation count */
|
so_gen_t so_gencnt; /* generation count */
|
||||||
void *so_emuldata; /* [mr] private data for emulators */
|
void *so_emuldata; /* private data for emulators */
|
||||||
struct so_accf {
|
struct so_accf {
|
||||||
struct accept_filter *so_accept_filter; /* [mr] */
|
struct accept_filter *so_accept_filter;
|
||||||
void *so_accept_filter_arg; /* [mr] saved filter args */
|
void *so_accept_filter_arg; /* saved filter args */
|
||||||
char *so_accept_filter_str; /* [mr] saved user args */
|
char *so_accept_filter_str; /* saved user args */
|
||||||
} *so_accf; /* [mr] */
|
} *so_accf;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
|
||||||
* Macros to lock a socket.
|
|
||||||
*/
|
|
||||||
#define SOCKBUF_LOCK(sb) mtx_lock(&(sb)->sb_mtx)
|
|
||||||
#define SOCKBUF_TRYLOCK(sb) mtx_trylock(&(sb)->sb_mtx)
|
|
||||||
#define SOCKBUF_UNLOCK(sb) mtx_unlock(&(sb)->sb_mtx)
|
|
||||||
#define SOCKBUF_LOCKED(sb) mtx_owned(&(sb)->sb_mtx)
|
|
||||||
#define SOCKBUF_ASSERT(sb, type) mtx_assert(&(sb)->sb_mtx, type)
|
|
||||||
|
|
||||||
#define SOCK_MTX(so) (&(so)->so_rcv.sb_mtx)
|
|
||||||
#define SOCK_LOCK(so) SOCKBUF_LOCK(&(so)->so_rcv)
|
|
||||||
#define SOCK_TRYLOCK(so) SOCKBUF_TRYLOCK(&(so)->so_rcv)
|
|
||||||
#define SOCK_UNLOCK(so) SOCKBUF_UNLOCK(&(so)->so_rcv)
|
|
||||||
#define SOCK_LOCKED(so) SOCKBUF_LOCKED(&(so)->so_rcv)
|
|
||||||
#define SOCK_ASSERT(so, type) SOCKBUF_ASSERT(&(so)->so_rcv, type)
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Socket state bits.
|
* Socket state bits.
|
||||||
*/
|
*/
|
||||||
@ -281,41 +260,39 @@ struct xsocket {
|
|||||||
* still explicitly close the socket, but the last ref count will free
|
* still explicitly close the socket, but the last ref count will free
|
||||||
* the structure.
|
* the structure.
|
||||||
*/
|
*/
|
||||||
|
#define soref(so) do { \
|
||||||
#define soref(so) do { \
|
++(so)->so_count; \
|
||||||
SOCK_ASSERT(so, MA_OWNED); \
|
|
||||||
++(so)->so_count; \
|
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define sorele(so) do { \
|
#define sorele(so) do { \
|
||||||
SOCK_ASSERT(so, MA_OWNED); \
|
|
||||||
if ((so)->so_count <= 0) \
|
if ((so)->so_count <= 0) \
|
||||||
panic("sorele"); \
|
panic("sorele");\
|
||||||
if (--(so)->so_count == 0) \
|
if (--(so)->so_count == 0)\
|
||||||
sofree(so); \
|
sofree(so); \
|
||||||
else \
|
|
||||||
SOCK_UNLOCK(so); \
|
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define sotryfree(so) do { \
|
#define sotryfree(so) do { \
|
||||||
SOCK_ASSERT(so, MA_OWNED); \
|
|
||||||
if ((so)->so_count == 0) \
|
if ((so)->so_count == 0) \
|
||||||
sofree(so); \
|
sofree(so); \
|
||||||
else \
|
|
||||||
SOCK_UNLOCK(so); \
|
|
||||||
} while(0)
|
} while(0)
|
||||||
|
|
||||||
#define sorwakeup(so) do { \
|
#define sorwakeup_locked(so) do { \
|
||||||
SOCK_ASSERT(so, MA_OWNED); \
|
if (sb_notify(&(so)->so_rcv)) \
|
||||||
if (sb_notify(&(so)->so_rcv)) \
|
sowakeup((so), &(so)->so_rcv); \
|
||||||
sowakeup((so), &(so)->so_rcv); \
|
} while (0)
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define sowwakeup(so) do { \
|
#define sorwakeup(so) do { \
|
||||||
SOCK_ASSERT(so, MA_OWNED); \
|
sorwakeup_locked(so); \
|
||||||
if (sb_notify(&(so)->so_snd)) \
|
} while (0)
|
||||||
sowakeup((so), &(so)->so_snd); \
|
|
||||||
} while (0)
|
#define sowwakeup_locked(so) do { \
|
||||||
|
if (sb_notify(&(so)->so_snd)) \
|
||||||
|
sowakeup((so), &(so)->so_snd); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#define sowwakeup(so) do { \
|
||||||
|
sowwakeup_locked(so); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
#ifdef _KERNEL
|
#ifdef _KERNEL
|
||||||
|
|
||||||
@ -425,12 +402,14 @@ int soconnect2(struct socket *so1, struct socket *so2);
|
|||||||
int socreate(int dom, struct socket **aso, int type, int proto,
|
int socreate(int dom, struct socket **aso, int type, int proto,
|
||||||
struct ucred *cred, struct thread *td);
|
struct ucred *cred, struct thread *td);
|
||||||
int sodisconnect(struct socket *so);
|
int sodisconnect(struct socket *so);
|
||||||
|
void soisconnected_locked(struct socket *so);
|
||||||
void sofree(struct socket *so);
|
void sofree(struct socket *so);
|
||||||
int sogetopt(struct socket *so, struct sockopt *sopt);
|
int sogetopt(struct socket *so, struct sockopt *sopt);
|
||||||
void sohasoutofband(struct socket *so);
|
void sohasoutofband(struct socket *so);
|
||||||
void soisconnected(struct socket *so);
|
void soisconnected(struct socket *so);
|
||||||
void soisconnecting(struct socket *so);
|
void soisconnecting(struct socket *so);
|
||||||
void soisdisconnected(struct socket *so);
|
void soisdisconnected(struct socket *so);
|
||||||
|
void soisdisconnected_locked(struct socket *so);
|
||||||
void soisdisconnecting(struct socket *so);
|
void soisdisconnecting(struct socket *so);
|
||||||
int solisten(struct socket *so, int backlog, struct thread *td);
|
int solisten(struct socket *so, int backlog, struct thread *td);
|
||||||
struct socket *
|
struct socket *
|
||||||
|
Loading…
x
Reference in New Issue
Block a user