diff options
Diffstat (limited to 'net/sctp/socket.c')
| -rw-r--r-- | net/sctp/socket.c | 29 | 
1 files changed, 17 insertions, 12 deletions
| diff --git a/net/sctp/socket.c b/net/sctp/socket.c index b6e4b89539b3..174d4d35e951 100644 --- a/net/sctp/socket.c +++ b/net/sctp/socket.c | |||
| @@ -1057,6 +1057,7 @@ static int __sctp_connect(struct sock* sk, | |||
| 1057 | inet_sk(sk)->dport = htons(asoc->peer.port); | 1057 | inet_sk(sk)->dport = htons(asoc->peer.port); | 
| 1058 | af = sctp_get_af_specific(to.sa.sa_family); | 1058 | af = sctp_get_af_specific(to.sa.sa_family); | 
| 1059 | af->to_sk_daddr(&to, sk); | 1059 | af->to_sk_daddr(&to, sk); | 
| 1060 | sk->sk_err = 0; | ||
| 1060 | 1061 | ||
| 1061 | timeo = sock_sndtimeo(sk, sk->sk_socket->file->f_flags & O_NONBLOCK); | 1062 | timeo = sock_sndtimeo(sk, sk->sk_socket->file->f_flags & O_NONBLOCK); | 
| 1062 | err = sctp_wait_for_connect(asoc, &timeo); | 1063 | err = sctp_wait_for_connect(asoc, &timeo); | 
| @@ -1228,7 +1229,7 @@ SCTP_STATIC void sctp_close(struct sock *sk, long timeout) | |||
| 1228 | 1229 | ||
| 1229 | ep = sctp_sk(sk)->ep; | 1230 | ep = sctp_sk(sk)->ep; | 
| 1230 | 1231 | ||
| 1231 | /* Walk all associations on a socket, not on an endpoint. */ | 1232 | /* Walk all associations on an endpoint. */ | 
| 1232 | list_for_each_safe(pos, temp, &ep->asocs) { | 1233 | list_for_each_safe(pos, temp, &ep->asocs) { | 
| 1233 | asoc = list_entry(pos, struct sctp_association, asocs); | 1234 | asoc = list_entry(pos, struct sctp_association, asocs); | 
| 1234 | 1235 | ||
| @@ -1241,13 +1242,13 @@ SCTP_STATIC void sctp_close(struct sock *sk, long timeout) | |||
| 1241 | if (sctp_state(asoc, CLOSED)) { | 1242 | if (sctp_state(asoc, CLOSED)) { | 
| 1242 | sctp_unhash_established(asoc); | 1243 | sctp_unhash_established(asoc); | 
| 1243 | sctp_association_free(asoc); | 1244 | sctp_association_free(asoc); | 
| 1245 | continue; | ||
| 1246 | } | ||
| 1247 | } | ||
| 1244 | 1248 | ||
| 1245 | } else if (sock_flag(sk, SOCK_LINGER) && | 1249 | if (sock_flag(sk, SOCK_LINGER) && !sk->sk_lingertime) | 
| 1246 | !sk->sk_lingertime) | 1250 | sctp_primitive_ABORT(asoc, NULL); | 
| 1247 | sctp_primitive_ABORT(asoc, NULL); | 1251 | else | 
| 1248 | else | ||
| 1249 | sctp_primitive_SHUTDOWN(asoc, NULL); | ||
| 1250 | } else | ||
| 1251 | sctp_primitive_SHUTDOWN(asoc, NULL); | 1252 | sctp_primitive_SHUTDOWN(asoc, NULL); | 
| 1252 | } | 1253 | } | 
| 1253 | 1254 | ||
| @@ -5317,6 +5318,7 @@ static int sctp_wait_for_sndbuf(struct sctp_association *asoc, long *timeo_p, | |||
| 5317 | */ | 5318 | */ | 
| 5318 | sctp_release_sock(sk); | 5319 | sctp_release_sock(sk); | 
| 5319 | current_timeo = schedule_timeout(current_timeo); | 5320 | current_timeo = schedule_timeout(current_timeo); | 
| 5321 | BUG_ON(sk != asoc->base.sk); | ||
| 5320 | sctp_lock_sock(sk); | 5322 | sctp_lock_sock(sk); | 
| 5321 | 5323 | ||
| 5322 | *timeo_p = current_timeo; | 5324 | *timeo_p = current_timeo; | 
| @@ -5604,12 +5606,14 @@ static void sctp_sock_migrate(struct sock *oldsk, struct sock *newsk, | |||
| 5604 | */ | 5606 | */ | 
| 5605 | newsp->type = type; | 5607 | newsp->type = type; | 
| 5606 | 5608 | ||
| 5607 | spin_lock_bh(&oldsk->sk_lock.slock); | 5609 | /* Mark the new socket "in-use" by the user so that any packets | 
| 5608 | /* Migrate the backlog from oldsk to newsk. */ | 5610 | * that may arrive on the association after we've moved it are | 
| 5609 | sctp_backlog_migrate(assoc, oldsk, newsk); | 5611 | * queued to the backlog. This prevents a potential race between | 
| 5610 | /* Migrate the association to the new socket. */ | 5612 | * backlog processing on the old socket and new-packet processing | 
| 5613 | * on the new socket. | ||
| 5614 | */ | ||
| 5615 | sctp_lock_sock(newsk); | ||
| 5611 | sctp_assoc_migrate(assoc, newsk); | 5616 | sctp_assoc_migrate(assoc, newsk); | 
| 5612 | spin_unlock_bh(&oldsk->sk_lock.slock); | ||
| 5613 | 5617 | ||
| 5614 | /* If the association on the newsk is already closed before accept() | 5618 | /* If the association on the newsk is already closed before accept() | 
| 5615 | * is called, set RCV_SHUTDOWN flag. | 5619 | * is called, set RCV_SHUTDOWN flag. | 
| @@ -5618,6 +5622,7 @@ static void sctp_sock_migrate(struct sock *oldsk, struct sock *newsk, | |||
| 5618 | newsk->sk_shutdown |= RCV_SHUTDOWN; | 5622 | newsk->sk_shutdown |= RCV_SHUTDOWN; | 
| 5619 | 5623 | ||
| 5620 | newsk->sk_state = SCTP_SS_ESTABLISHED; | 5624 | newsk->sk_state = SCTP_SS_ESTABLISHED; | 
| 5625 | sctp_release_sock(newsk); | ||
| 5621 | } | 5626 | } | 
| 5622 | 5627 | ||
| 5623 | /* This proto struct describes the ULP interface for SCTP. */ | 5628 | /* This proto struct describes the ULP interface for SCTP. */ | 
