[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:
parent
16d00fb776
commit
ce5325c133
1 changed files with 22 additions and 10 deletions
|
@ -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;
|
||||||
|
|
Loading…
Reference in a new issue