[SCTP]: Fix the SO_REUSEADDR handling to be similar to TCP.

Update the SO_REUSEADDR handling to also check for listen state.  This
was muliple listening server sockets can't be created and they will
not steal packets from each other.

Reported by Paolo Galtieri <pgaltieri@mvista.com>

Signed-off-by: Vlad Yasevich <vladislav.yasevich@hp.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Vlad Yasevich 2007-05-04 13:34:49 -07:00 committed by David S. Miller
parent 16d00fb776
commit ce5325c133

View file

@ -5020,7 +5020,8 @@ static long sctp_get_port_local(struct sock *sk, union sctp_addr *addr)
struct hlist_node *node; struct hlist_node *node;
SCTP_DEBUG_PRINTK("sctp_get_port() found a possible match\n"); SCTP_DEBUG_PRINTK("sctp_get_port() found a possible match\n");
if (pp->fastreuse && sk->sk_reuse) if (pp->fastreuse && sk->sk_reuse &&
sk->sk_state != SCTP_SS_LISTENING)
goto success; goto success;
/* Run through the list of sockets bound to the port /* Run through the list of sockets bound to the port
@ -5037,7 +5038,8 @@ static long sctp_get_port_local(struct sock *sk, union sctp_addr *addr)
struct sctp_endpoint *ep2; struct sctp_endpoint *ep2;
ep2 = sctp_sk(sk2)->ep; ep2 = sctp_sk(sk2)->ep;
if (reuse && sk2->sk_reuse) if (reuse && sk2->sk_reuse &&
sk2->sk_state != SCTP_SS_LISTENING)
continue; continue;
if (sctp_bind_addr_match(&ep2->base.bind_addr, addr, if (sctp_bind_addr_match(&ep2->base.bind_addr, addr,
@ -5058,9 +5060,13 @@ static long sctp_get_port_local(struct sock *sk, union sctp_addr *addr)
* if sk->sk_reuse is too (that is, if the caller requested * if sk->sk_reuse is too (that is, if the caller requested
* SO_REUSEADDR on this socket -sk-). * SO_REUSEADDR on this socket -sk-).
*/ */
if (hlist_empty(&pp->owner)) if (hlist_empty(&pp->owner)) {
pp->fastreuse = sk->sk_reuse ? 1 : 0; if (sk->sk_reuse && sk->sk_state != SCTP_SS_LISTENING)
else if (pp->fastreuse && !sk->sk_reuse) pp->fastreuse = 1;
else
pp->fastreuse = 0;
} else if (pp->fastreuse &&
(!sk->sk_reuse || sk->sk_state == SCTP_SS_LISTENING))
pp->fastreuse = 0; pp->fastreuse = 0;
/* We are set, so fill up all the data in the hash table /* We are set, so fill up all the data in the hash table
@ -5068,8 +5074,8 @@ static long sctp_get_port_local(struct sock *sk, union sctp_addr *addr)
* sockets FIXME: Blurry, NPI (ipg). * sockets FIXME: Blurry, NPI (ipg).
*/ */
success: success:
inet_sk(sk)->num = snum;
if (!sctp_sk(sk)->bind_hash) { if (!sctp_sk(sk)->bind_hash) {
inet_sk(sk)->num = snum;
sk_add_bind_node(sk, &pp->owner); sk_add_bind_node(sk, &pp->owner);
sctp_sk(sk)->bind_hash = pp; sctp_sk(sk)->bind_hash = pp;
} }
@ -5142,12 +5148,16 @@ SCTP_STATIC int sctp_seqpacket_listen(struct sock *sk, int backlog)
* This is not currently spelled out in the SCTP sockets * This is not currently spelled out in the SCTP sockets
* extensions draft, but follows the practice as seen in TCP * extensions draft, but follows the practice as seen in TCP
* sockets. * sockets.
*
* Additionally, turn off fastreuse flag since we are not listening
*/ */
sk->sk_state = SCTP_SS_LISTENING;
if (!ep->base.bind_addr.port) { if (!ep->base.bind_addr.port) {
if (sctp_autobind(sk)) if (sctp_autobind(sk))
return -EAGAIN; return -EAGAIN;
} } else
sk->sk_state = SCTP_SS_LISTENING; sctp_sk(sk)->bind_hash->fastreuse = 0;
sctp_hash_endpoint(ep); sctp_hash_endpoint(ep);
return 0; return 0;
} }
@ -5185,11 +5195,13 @@ SCTP_STATIC int sctp_stream_listen(struct sock *sk, int backlog)
* extensions draft, but follows the practice as seen in TCP * extensions draft, but follows the practice as seen in TCP
* sockets. * sockets.
*/ */
sk->sk_state = SCTP_SS_LISTENING;
if (!ep->base.bind_addr.port) { if (!ep->base.bind_addr.port) {
if (sctp_autobind(sk)) if (sctp_autobind(sk))
return -EAGAIN; return -EAGAIN;
} } else
sk->sk_state = SCTP_SS_LISTENING; sctp_sk(sk)->bind_hash->fastreuse = 0;
sk->sk_max_ack_backlog = backlog; sk->sk_max_ack_backlog = backlog;
sctp_hash_endpoint(ep); sctp_hash_endpoint(ep);
return 0; return 0;