|
@@ -84,8 +84,8 @@
|
|
|
/* Forward declarations for internal helper functions. */
|
|
|
static int sctp_writeable(struct sock *sk);
|
|
|
static void sctp_wfree(struct sk_buff *skb);
|
|
|
-static int sctp_wait_for_sndbuf(struct sctp_association *, long *timeo_p,
|
|
|
- size_t msg_len);
|
|
|
+static int sctp_wait_for_sndbuf(struct sctp_association *asoc, long *timeo_p,
|
|
|
+ size_t msg_len, struct sock **orig_sk);
|
|
|
static int sctp_wait_for_packet(struct sock *sk, int *err, long *timeo_p);
|
|
|
static int sctp_wait_for_connect(struct sctp_association *, long *timeo_p);
|
|
|
static int sctp_wait_for_accept(struct sock *sk, long timeo);
|
|
@@ -1970,7 +1970,8 @@ static int sctp_sendmsg(struct sock *sk, struct msghdr *msg, size_t msg_len)
|
|
|
|
|
|
timeo = sock_sndtimeo(sk, msg->msg_flags & MSG_DONTWAIT);
|
|
|
if (!sctp_wspace(asoc)) {
|
|
|
- err = sctp_wait_for_sndbuf(asoc, &timeo, msg_len);
|
|
|
+ /* sk can be changed by peel off when waiting for buf. */
|
|
|
+ err = sctp_wait_for_sndbuf(asoc, &timeo, msg_len, &sk);
|
|
|
if (err) {
|
|
|
if (err == -ESRCH) {
|
|
|
/* asoc is already dead. */
|
|
@@ -5021,12 +5022,6 @@ int sctp_do_peeloff(struct sock *sk, sctp_assoc_t id, struct socket **sockp)
|
|
|
if (!asoc)
|
|
|
return -EINVAL;
|
|
|
|
|
|
- /* If there is a thread waiting on more sndbuf space for
|
|
|
- * sending on this asoc, it cannot be peeled.
|
|
|
- */
|
|
|
- if (waitqueue_active(&asoc->wait))
|
|
|
- return -EBUSY;
|
|
|
-
|
|
|
/* An association cannot be branched off from an already peeled-off
|
|
|
* socket, nor is this supported for tcp style sockets.
|
|
|
*/
|
|
@@ -7995,7 +7990,7 @@ void sctp_sock_rfree(struct sk_buff *skb)
|
|
|
|
|
|
/* Helper function to wait for space in the sndbuf. */
|
|
|
static int sctp_wait_for_sndbuf(struct sctp_association *asoc, long *timeo_p,
|
|
|
- size_t msg_len)
|
|
|
+ size_t msg_len, struct sock **orig_sk)
|
|
|
{
|
|
|
struct sock *sk = asoc->base.sk;
|
|
|
int err = 0;
|
|
@@ -8029,11 +8024,17 @@ static int sctp_wait_for_sndbuf(struct sctp_association *asoc, long *timeo_p,
|
|
|
release_sock(sk);
|
|
|
current_timeo = schedule_timeout(current_timeo);
|
|
|
lock_sock(sk);
|
|
|
+ if (sk != asoc->base.sk) {
|
|
|
+ release_sock(sk);
|
|
|
+ sk = asoc->base.sk;
|
|
|
+ lock_sock(sk);
|
|
|
+ }
|
|
|
|
|
|
*timeo_p = current_timeo;
|
|
|
}
|
|
|
|
|
|
out:
|
|
|
+ *orig_sk = sk;
|
|
|
finish_wait(&asoc->wait, &wait);
|
|
|
|
|
|
/* Release the association's refcnt. */
|