diff options
author | Gerrit Renker <gerrit@erg.abdn.ac.uk> | 2008-11-23 19:02:31 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2008-11-23 19:02:31 -0500 |
commit | b20a9c24d5c5d466d7e4a25c6f1bedbd2d16ad4f (patch) | |
tree | 17789215657f693caf36e22d3e724cc1b1e07ba6 /net/dccp | |
parent | 2c62ad7b56fa8e2658253c0256ef4c4de228a0b9 (diff) |
dccp: Set per-connection CCIDs via socket options
With this patch, TX/RX CCIDs can now be changed on a per-connection
basis, which overrides the defaults set by the global sysctl variables
for TX/RX CCIDs.
To make full use of this facility, the remaining patches of this patch
set are needed, which track dependencies and activate negotiated
feature values.
Signed-off-by: Gerrit Renker <gerrit@erg.abdn.ac.uk>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/dccp')
-rw-r--r-- | net/dccp/ackvec.c | 9 | ||||
-rw-r--r-- | net/dccp/ackvec.h | 5 | ||||
-rw-r--r-- | net/dccp/feat.h | 2 | ||||
-rw-r--r-- | net/dccp/proto.c | 34 |
4 files changed, 42 insertions, 8 deletions
diff --git a/net/dccp/ackvec.c b/net/dccp/ackvec.c index 1e8be246ad15..01e4d39fa232 100644 --- a/net/dccp/ackvec.c +++ b/net/dccp/ackvec.c | |||
@@ -12,7 +12,6 @@ | |||
12 | #include "ackvec.h" | 12 | #include "ackvec.h" |
13 | #include "dccp.h" | 13 | #include "dccp.h" |
14 | 14 | ||
15 | #include <linux/dccp.h> | ||
16 | #include <linux/init.h> | 15 | #include <linux/init.h> |
17 | #include <linux/errno.h> | 16 | #include <linux/errno.h> |
18 | #include <linux/kernel.h> | 17 | #include <linux/kernel.h> |
@@ -68,7 +67,7 @@ int dccp_insert_option_ackvec(struct sock *sk, struct sk_buff *skb) | |||
68 | struct dccp_sock *dp = dccp_sk(sk); | 67 | struct dccp_sock *dp = dccp_sk(sk); |
69 | struct dccp_ackvec *av = dp->dccps_hc_rx_ackvec; | 68 | struct dccp_ackvec *av = dp->dccps_hc_rx_ackvec; |
70 | /* Figure out how many options do we need to represent the ackvec */ | 69 | /* Figure out how many options do we need to represent the ackvec */ |
71 | const u16 nr_opts = DIV_ROUND_UP(av->av_vec_len, DCCP_MAX_ACKVEC_OPT_LEN); | 70 | const u8 nr_opts = DIV_ROUND_UP(av->av_vec_len, DCCP_SINGLE_OPT_MAXLEN); |
72 | u16 len = av->av_vec_len + 2 * nr_opts, i; | 71 | u16 len = av->av_vec_len + 2 * nr_opts, i; |
73 | u32 elapsed_time; | 72 | u32 elapsed_time; |
74 | const unsigned char *tail, *from; | 73 | const unsigned char *tail, *from; |
@@ -100,8 +99,8 @@ int dccp_insert_option_ackvec(struct sock *sk, struct sk_buff *skb) | |||
100 | for (i = 0; i < nr_opts; ++i) { | 99 | for (i = 0; i < nr_opts; ++i) { |
101 | int copylen = len; | 100 | int copylen = len; |
102 | 101 | ||
103 | if (len > DCCP_MAX_ACKVEC_OPT_LEN) | 102 | if (len > DCCP_SINGLE_OPT_MAXLEN) |
104 | copylen = DCCP_MAX_ACKVEC_OPT_LEN; | 103 | copylen = DCCP_SINGLE_OPT_MAXLEN; |
105 | 104 | ||
106 | *to++ = DCCPO_ACK_VECTOR_0; | 105 | *to++ = DCCPO_ACK_VECTOR_0; |
107 | *to++ = copylen + 2; | 106 | *to++ = copylen + 2; |
@@ -432,7 +431,7 @@ found: | |||
432 | int dccp_ackvec_parse(struct sock *sk, const struct sk_buff *skb, | 431 | int dccp_ackvec_parse(struct sock *sk, const struct sk_buff *skb, |
433 | u64 *ackno, const u8 opt, const u8 *value, const u8 len) | 432 | u64 *ackno, const u8 opt, const u8 *value, const u8 len) |
434 | { | 433 | { |
435 | if (len > DCCP_MAX_ACKVEC_OPT_LEN) | 434 | if (len > DCCP_SINGLE_OPT_MAXLEN) |
436 | return -1; | 435 | return -1; |
437 | 436 | ||
438 | /* dccp_ackvector_print(DCCP_SKB_CB(skb)->dccpd_ack_seq, value, len); */ | 437 | /* dccp_ackvector_print(DCCP_SKB_CB(skb)->dccpd_ack_seq, value, len); */ |
diff --git a/net/dccp/ackvec.h b/net/dccp/ackvec.h index bcb64fb4acef..4ccee030524e 100644 --- a/net/dccp/ackvec.h +++ b/net/dccp/ackvec.h | |||
@@ -11,15 +11,14 @@ | |||
11 | * published by the Free Software Foundation. | 11 | * published by the Free Software Foundation. |
12 | */ | 12 | */ |
13 | 13 | ||
14 | #include <linux/dccp.h> | ||
14 | #include <linux/compiler.h> | 15 | #include <linux/compiler.h> |
15 | #include <linux/ktime.h> | 16 | #include <linux/ktime.h> |
16 | #include <linux/list.h> | 17 | #include <linux/list.h> |
17 | #include <linux/types.h> | 18 | #include <linux/types.h> |
18 | 19 | ||
19 | /* Read about the ECN nonce to see why it is 253 */ | ||
20 | #define DCCP_MAX_ACKVEC_OPT_LEN 253 | ||
21 | /* We can spread an ack vector across multiple options */ | 20 | /* We can spread an ack vector across multiple options */ |
22 | #define DCCP_MAX_ACKVEC_LEN (DCCP_MAX_ACKVEC_OPT_LEN * 2) | 21 | #define DCCP_MAX_ACKVEC_LEN (DCCP_SINGLE_OPT_MAXLEN * 2) |
23 | 22 | ||
24 | #define DCCP_ACKVEC_STATE_RECEIVED 0 | 23 | #define DCCP_ACKVEC_STATE_RECEIVED 0 |
25 | #define DCCP_ACKVEC_STATE_ECN_MARKED (1 << 6) | 24 | #define DCCP_ACKVEC_STATE_ECN_MARKED (1 << 6) |
diff --git a/net/dccp/feat.h b/net/dccp/feat.h index 4d172822df17..093af1610d11 100644 --- a/net/dccp/feat.h +++ b/net/dccp/feat.h | |||
@@ -22,6 +22,8 @@ | |||
22 | /* Wmin=32 and Wmax=2^46-1 from 7.5.2 */ | 22 | /* Wmin=32 and Wmax=2^46-1 from 7.5.2 */ |
23 | #define DCCPF_SEQ_WMIN 32 | 23 | #define DCCPF_SEQ_WMIN 32 |
24 | #define DCCPF_SEQ_WMAX 0x3FFFFFFFFFFFull | 24 | #define DCCPF_SEQ_WMAX 0x3FFFFFFFFFFFull |
25 | /* Maximum number of SP values that fit in a single (Confirm) option */ | ||
26 | #define DCCP_FEAT_MAX_SP_VALS (DCCP_SINGLE_OPT_MAXLEN - 2) | ||
25 | 27 | ||
26 | enum dccp_feat_type { | 28 | enum dccp_feat_type { |
27 | FEAT_AT_RX = 1, /* located at RX side of half-connection */ | 29 | FEAT_AT_RX = 1, /* located at RX side of half-connection */ |
diff --git a/net/dccp/proto.c b/net/dccp/proto.c index 8b63394ec24c..445884cf1c29 100644 --- a/net/dccp/proto.c +++ b/net/dccp/proto.c | |||
@@ -501,6 +501,36 @@ static int dccp_setsockopt_cscov(struct sock *sk, int cscov, bool rx) | |||
501 | return rc; | 501 | return rc; |
502 | } | 502 | } |
503 | 503 | ||
504 | static int dccp_setsockopt_ccid(struct sock *sk, int type, | ||
505 | char __user *optval, int optlen) | ||
506 | { | ||
507 | u8 *val; | ||
508 | int rc = 0; | ||
509 | |||
510 | if (optlen < 1 || optlen > DCCP_FEAT_MAX_SP_VALS) | ||
511 | return -EINVAL; | ||
512 | |||
513 | val = kmalloc(optlen, GFP_KERNEL); | ||
514 | if (val == NULL) | ||
515 | return -ENOMEM; | ||
516 | |||
517 | if (copy_from_user(val, optval, optlen)) { | ||
518 | kfree(val); | ||
519 | return -EFAULT; | ||
520 | } | ||
521 | |||
522 | lock_sock(sk); | ||
523 | if (type == DCCP_SOCKOPT_TX_CCID || type == DCCP_SOCKOPT_CCID) | ||
524 | rc = dccp_feat_register_sp(sk, DCCPF_CCID, 1, val, optlen); | ||
525 | |||
526 | if (!rc && (type == DCCP_SOCKOPT_RX_CCID || type == DCCP_SOCKOPT_CCID)) | ||
527 | rc = dccp_feat_register_sp(sk, DCCPF_CCID, 0, val, optlen); | ||
528 | release_sock(sk); | ||
529 | |||
530 | kfree(val); | ||
531 | return rc; | ||
532 | } | ||
533 | |||
504 | static int do_dccp_setsockopt(struct sock *sk, int level, int optname, | 534 | static int do_dccp_setsockopt(struct sock *sk, int level, int optname, |
505 | char __user *optval, int optlen) | 535 | char __user *optval, int optlen) |
506 | { | 536 | { |
@@ -515,6 +545,10 @@ static int do_dccp_setsockopt(struct sock *sk, int level, int optname, | |||
515 | case DCCP_SOCKOPT_CHANGE_R: | 545 | case DCCP_SOCKOPT_CHANGE_R: |
516 | DCCP_WARN("sockopt(CHANGE_L/R) is deprecated: fix your app\n"); | 546 | DCCP_WARN("sockopt(CHANGE_L/R) is deprecated: fix your app\n"); |
517 | return 0; | 547 | return 0; |
548 | case DCCP_SOCKOPT_CCID: | ||
549 | case DCCP_SOCKOPT_RX_CCID: | ||
550 | case DCCP_SOCKOPT_TX_CCID: | ||
551 | return dccp_setsockopt_ccid(sk, optname, optval, optlen); | ||
518 | } | 552 | } |
519 | 553 | ||
520 | if (optlen < (int)sizeof(int)) | 554 | if (optlen < (int)sizeof(int)) |