diff options
Diffstat (limited to 'net/sctp/socket.c')
| -rw-r--r-- | net/sctp/socket.c | 246 | 
1 files changed, 217 insertions, 29 deletions
| diff --git a/net/sctp/socket.c b/net/sctp/socket.c index 6f45d1713452..3204a9b29407 100644 --- a/net/sctp/socket.c +++ b/net/sctp/socket.c @@ -79,12 +79,13 @@  #include <net/sock.h>  #include <net/sctp/sctp.h>  #include <net/sctp/sm.h> +#include <net/sctp/stream_sched.h>  /* 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); @@ -1957,14 +1958,28 @@ static int sctp_sendmsg(struct sock *sk, struct msghdr *msg, size_t msg_len)  		goto out_free;  	} +	/* Allocate sctp_stream_out_ext if not already done */ +	if (unlikely(!asoc->stream.out[sinfo->sinfo_stream].ext)) { +		err = sctp_stream_init_ext(&asoc->stream, sinfo->sinfo_stream); +		if (err) +			goto out_free; +	} +  	if (sctp_wspace(asoc) < msg_len)  		sctp_prsctp_prune(asoc, sinfo, msg_len - sctp_wspace(asoc));  	timeo = sock_sndtimeo(sk, msg->msg_flags & MSG_DONTWAIT);  	if (!sctp_wspace(asoc)) { -		err = sctp_wait_for_sndbuf(asoc, &timeo, msg_len); -		if (err) +		/* 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. */ +				new_asoc = NULL; +				err = -EPIPE; +			}  			goto out_free; +		}  	}  	/* If an address is passed with the sendto/sendmsg call, it is used @@ -3125,9 +3140,9 @@ static int sctp_setsockopt_mappedv4(struct sock *sk, char __user *optval, unsign   */  static int sctp_setsockopt_maxseg(struct sock *sk, char __user *optval, unsigned int optlen)  { +	struct sctp_sock *sp = sctp_sk(sk);  	struct sctp_assoc_value params;  	struct sctp_association *asoc; -	struct sctp_sock *sp = sctp_sk(sk);  	int val;  	if (optlen == sizeof(int)) { @@ -3143,26 +3158,35 @@ static int sctp_setsockopt_maxseg(struct sock *sk, char __user *optval, unsigned  		if (copy_from_user(¶ms, optval, optlen))  			return -EFAULT;  		val = params.assoc_value; -	} else +	} else {  		return -EINVAL; +	} -	if ((val != 0) && ((val < 8) || (val > SCTP_MAX_CHUNK_LEN))) -		return -EINVAL; +	if (val) { +		int min_len, max_len; -	asoc = sctp_id2assoc(sk, params.assoc_id); -	if (!asoc && params.assoc_id && sctp_style(sk, UDP)) -		return -EINVAL; +		min_len = SCTP_DEFAULT_MINSEGMENT - sp->pf->af->net_header_len; +		min_len -= sizeof(struct sctphdr) + +			   sizeof(struct sctp_data_chunk); + +		max_len = SCTP_MAX_CHUNK_LEN - sizeof(struct sctp_data_chunk); + +		if (val < min_len || val > max_len) +			return -EINVAL; +	} +	asoc = sctp_id2assoc(sk, params.assoc_id);  	if (asoc) {  		if (val == 0) { -			val = asoc->pathmtu; -			val -= sp->pf->af->net_header_len; +			val = asoc->pathmtu - sp->pf->af->net_header_len;  			val -= sizeof(struct sctphdr) + -					sizeof(struct sctp_data_chunk); +			       sizeof(struct sctp_data_chunk);  		}  		asoc->user_frag = val;  		asoc->frag_point = sctp_frag_point(asoc, asoc->pathmtu);  	} else { +		if (params.assoc_id && sctp_style(sk, UDP)) +			return -EINVAL;  		sp->user_frag = val;  	} @@ -3937,6 +3961,64 @@ out:  	return retval;  } +static int sctp_setsockopt_scheduler(struct sock *sk, +				     char __user *optval, +				     unsigned int optlen) +{ +	struct sctp_association *asoc; +	struct sctp_assoc_value params; +	int retval = -EINVAL; + +	if (optlen < sizeof(params)) +		goto out; + +	optlen = sizeof(params); +	if (copy_from_user(¶ms, optval, optlen)) { +		retval = -EFAULT; +		goto out; +	} + +	if (params.assoc_value > SCTP_SS_MAX) +		goto out; + +	asoc = sctp_id2assoc(sk, params.assoc_id); +	if (!asoc) +		goto out; + +	retval = sctp_sched_set_sched(asoc, params.assoc_value); + +out: +	return retval; +} + +static int sctp_setsockopt_scheduler_value(struct sock *sk, +					   char __user *optval, +					   unsigned int optlen) +{ +	struct sctp_association *asoc; +	struct sctp_stream_value params; +	int retval = -EINVAL; + +	if (optlen < sizeof(params)) +		goto out; + +	optlen = sizeof(params); +	if (copy_from_user(¶ms, optval, optlen)) { +		retval = -EFAULT; +		goto out; +	} + +	asoc = sctp_id2assoc(sk, params.assoc_id); +	if (!asoc) +		goto out; + +	retval = sctp_sched_set_value(asoc, params.stream_id, +				      params.stream_value, GFP_KERNEL); + +out: +	return retval; +} +  /* API 6.2 setsockopt(), getsockopt()   *   * Applications use setsockopt() and getsockopt() to set or retrieve @@ -4118,6 +4200,12 @@ static int sctp_setsockopt(struct sock *sk, int level, int optname,  	case SCTP_ADD_STREAMS:  		retval = sctp_setsockopt_add_streams(sk, optval, optlen);  		break; +	case SCTP_STREAM_SCHEDULER: +		retval = sctp_setsockopt_scheduler(sk, optval, optlen); +		break; +	case SCTP_STREAM_SCHEDULER_VALUE: +		retval = sctp_setsockopt_scheduler_value(sk, optval, optlen); +		break;  	default:  		retval = -ENOPROTOOPT;  		break; @@ -4943,12 +5031,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.  	 */ @@ -6679,7 +6761,7 @@ static int sctp_getsockopt_pr_streamstatus(struct sock *sk, int len,  					   char __user *optval,  					   int __user *optlen)  { -	struct sctp_stream_out *streamout; +	struct sctp_stream_out_ext *streamoute;  	struct sctp_association *asoc;  	struct sctp_prstatus params;  	int retval = -EINVAL; @@ -6702,21 +6784,29 @@ static int sctp_getsockopt_pr_streamstatus(struct sock *sk, int len,  	if (!asoc || params.sprstat_sid >= asoc->stream.outcnt)  		goto out; -	streamout = &asoc->stream.out[params.sprstat_sid]; +	streamoute = asoc->stream.out[params.sprstat_sid].ext; +	if (!streamoute) { +		/* Not allocated yet, means all stats are 0 */ +		params.sprstat_abandoned_unsent = 0; +		params.sprstat_abandoned_sent = 0; +		retval = 0; +		goto out; +	} +  	if (policy == SCTP_PR_SCTP_NONE) {  		params.sprstat_abandoned_unsent = 0;  		params.sprstat_abandoned_sent = 0;  		for (policy = 0; policy <= SCTP_PR_INDEX(MAX); policy++) {  			params.sprstat_abandoned_unsent += -				streamout->abandoned_unsent[policy]; +				streamoute->abandoned_unsent[policy];  			params.sprstat_abandoned_sent += -				streamout->abandoned_sent[policy]; +				streamoute->abandoned_sent[policy];  		}  	} else {  		params.sprstat_abandoned_unsent = -			streamout->abandoned_unsent[__SCTP_PR_INDEX(policy)]; +			streamoute->abandoned_unsent[__SCTP_PR_INDEX(policy)];  		params.sprstat_abandoned_sent = -			streamout->abandoned_sent[__SCTP_PR_INDEX(policy)]; +			streamoute->abandoned_sent[__SCTP_PR_INDEX(policy)];  	}  	if (put_user(len, optlen) || copy_to_user(optval, ¶ms, len)) { @@ -6812,6 +6902,85 @@ out:  	return retval;  } +static int sctp_getsockopt_scheduler(struct sock *sk, int len, +				     char __user *optval, +				     int __user *optlen) +{ +	struct sctp_assoc_value params; +	struct sctp_association *asoc; +	int retval = -EFAULT; + +	if (len < sizeof(params)) { +		retval = -EINVAL; +		goto out; +	} + +	len = sizeof(params); +	if (copy_from_user(¶ms, optval, len)) +		goto out; + +	asoc = sctp_id2assoc(sk, params.assoc_id); +	if (!asoc) { +		retval = -EINVAL; +		goto out; +	} + +	params.assoc_value = sctp_sched_get_sched(asoc); + +	if (put_user(len, optlen)) +		goto out; + +	if (copy_to_user(optval, ¶ms, len)) +		goto out; + +	retval = 0; + +out: +	return retval; +} + +static int sctp_getsockopt_scheduler_value(struct sock *sk, int len, +					   char __user *optval, +					   int __user *optlen) +{ +	struct sctp_stream_value params; +	struct sctp_association *asoc; +	int retval = -EFAULT; + +	if (len < sizeof(params)) { +		retval = -EINVAL; +		goto out; +	} + +	len = sizeof(params); +	if (copy_from_user(¶ms, optval, len)) +		goto out; + +	asoc = sctp_id2assoc(sk, params.assoc_id); +	if (!asoc) { +		retval = -EINVAL; +		goto out; +	} + +	retval = sctp_sched_get_value(asoc, params.stream_id, +				      ¶ms.stream_value); +	if (retval) +		goto out; + +	if (put_user(len, optlen)) { +		retval = -EFAULT; +		goto out; +	} + +	if (copy_to_user(optval, ¶ms, len)) { +		retval = -EFAULT; +		goto out; +	} + +out: +	return retval; +} +  static int sctp_getsockopt(struct sock *sk, int level, int optname,  			   char __user *optval, int __user *optlen)  { @@ -6994,6 +7163,14 @@ static int sctp_getsockopt(struct sock *sk, int level, int optname,  		retval = sctp_getsockopt_enable_strreset(sk, len, optval,  							 optlen);  		break; +	case SCTP_STREAM_SCHEDULER: +		retval = sctp_getsockopt_scheduler(sk, len, optval, +						   optlen); +		break; +	case SCTP_STREAM_SCHEDULER_VALUE: +		retval = sctp_getsockopt_scheduler_value(sk, len, optval, +							 optlen); +		break;  	default:  		retval = -ENOPROTOOPT;  		break; @@ -7822,7 +7999,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; @@ -7839,10 +8016,11 @@ static int sctp_wait_for_sndbuf(struct sctp_association *asoc, long *timeo_p,  	for (;;) {  		prepare_to_wait_exclusive(&asoc->wait, &wait,  					  TASK_INTERRUPTIBLE); +		if (asoc->base.dead) +			goto do_dead;  		if (!*timeo_p)  			goto do_nonblock; -		if (sk->sk_err || asoc->state >= SCTP_STATE_SHUTDOWN_PENDING || -		    asoc->base.dead) +		if (sk->sk_err || asoc->state >= SCTP_STATE_SHUTDOWN_PENDING)  			goto do_error;  		if (signal_pending(current))  			goto do_interrupted; @@ -7855,11 +8033,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.  */ @@ -7867,6 +8051,10 @@ out:  	return err; +do_dead: +	err = -ESRCH; +	goto out; +  do_error:  	err = -EPIPE;  	goto out; | 
