|
@@ -3350,7 +3350,10 @@ static int sctp_setsockopt_fragment_interleave(struct sock *sk,
|
|
|
if (get_user(val, (int __user *)optval))
|
|
|
return -EFAULT;
|
|
|
|
|
|
- sctp_sk(sk)->frag_interleave = (val == 0) ? 0 : 1;
|
|
|
+ sctp_sk(sk)->frag_interleave = !!val;
|
|
|
+
|
|
|
+ if (!sctp_sk(sk)->frag_interleave)
|
|
|
+ sctp_sk(sk)->strm_interleave = 0;
|
|
|
|
|
|
return 0;
|
|
|
}
|
|
@@ -4019,6 +4022,40 @@ out:
|
|
|
return retval;
|
|
|
}
|
|
|
|
|
|
+static int sctp_setsockopt_interleaving_supported(struct sock *sk,
|
|
|
+ char __user *optval,
|
|
|
+ unsigned int optlen)
|
|
|
+{
|
|
|
+ struct sctp_sock *sp = sctp_sk(sk);
|
|
|
+ struct net *net = sock_net(sk);
|
|
|
+ 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_id)
|
|
|
+ goto out;
|
|
|
+
|
|
|
+ if (!net->sctp.intl_enable || !sp->frag_interleave) {
|
|
|
+ retval = -EPERM;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+
|
|
|
+ sp->strm_interleave = !!params.assoc_value;
|
|
|
+
|
|
|
+ retval = 0;
|
|
|
+
|
|
|
+out:
|
|
|
+ return retval;
|
|
|
+}
|
|
|
+
|
|
|
/* API 6.2 setsockopt(), getsockopt()
|
|
|
*
|
|
|
* Applications use setsockopt() and getsockopt() to set or retrieve
|
|
@@ -4206,6 +4243,10 @@ static int sctp_setsockopt(struct sock *sk, int level, int optname,
|
|
|
case SCTP_STREAM_SCHEDULER_VALUE:
|
|
|
retval = sctp_setsockopt_scheduler_value(sk, optval, optlen);
|
|
|
break;
|
|
|
+ case SCTP_INTERLEAVING_SUPPORTED:
|
|
|
+ retval = sctp_setsockopt_interleaving_supported(sk, optval,
|
|
|
+ optlen);
|
|
|
+ break;
|
|
|
default:
|
|
|
retval = -ENOPROTOOPT;
|
|
|
break;
|
|
@@ -6969,6 +7010,47 @@ out:
|
|
|
return retval;
|
|
|
}
|
|
|
|
|
|
+static int sctp_getsockopt_interleaving_supported(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) {
|
|
|
+ params.assoc_value = asoc->intl_enable;
|
|
|
+ } else if (!params.assoc_id) {
|
|
|
+ struct sctp_sock *sp = sctp_sk(sk);
|
|
|
+
|
|
|
+ params.assoc_value = sp->strm_interleave;
|
|
|
+ } else {
|
|
|
+ retval = -EINVAL;
|
|
|
+ goto out;
|
|
|
+ }
|
|
|
+
|
|
|
+ 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(struct sock *sk, int level, int optname,
|
|
|
char __user *optval, int __user *optlen)
|
|
|
{
|
|
@@ -7159,6 +7241,10 @@ static int sctp_getsockopt(struct sock *sk, int level, int optname,
|
|
|
retval = sctp_getsockopt_scheduler_value(sk, len, optval,
|
|
|
optlen);
|
|
|
break;
|
|
|
+ case SCTP_INTERLEAVING_SUPPORTED:
|
|
|
+ retval = sctp_getsockopt_interleaving_supported(sk, len, optval,
|
|
|
+ optlen);
|
|
|
+ break;
|
|
|
default:
|
|
|
retval = -ENOPROTOOPT;
|
|
|
break;
|