diff options
author | Andrea Bittau <a.bittau@cs.ucl.ac.uk> | 2006-09-19 16:13:37 -0400 |
---|---|---|
committer | David S. Miller <davem@sunset.davemloft.net> | 2006-09-22 18:19:40 -0400 |
commit | 07978aabd52ce67f59971872c80f76d6e3ca18ae (patch) | |
tree | 30b878674d4cf147cc6a0e91ff953298bc07b8e2 /net/dccp/ccids/ccid2.c | |
parent | 8d424f6ca2d02026dadff409770639d720375afb (diff) |
[DCCP] CCID2: Allocate seq records on demand
Allocate more sequence state on demand. Each time a packet is sent
out by CCID2, a record of it needs to be kept. This list of records
grows proportionally to cwnd. Previously, the length of this list was
hardcored and therefore the cwnd could only grow to this value (of
128). Now, records are allocated on demand as necessary---cwnd may
grow as it wishes. The exceptional case of when memory is not
available is not handled gracefully. Perhaps, cwnd should be capped
at that point.
Signed-off-by: Andrea Bittau <a.bittau@cs.ucl.ac.uk>
Signed-off-by: Arnaldo Carvalho de Melo <acme@mandriva.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/dccp/ccids/ccid2.c')
-rw-r--r-- | net/dccp/ccids/ccid2.c | 96 |
1 files changed, 65 insertions, 31 deletions
diff --git a/net/dccp/ccids/ccid2.c b/net/dccp/ccids/ccid2.c index dbcda7e868b7..93a30ae8d07a 100644 --- a/net/dccp/ccids/ccid2.c +++ b/net/dccp/ccids/ccid2.c | |||
@@ -44,8 +44,6 @@ static int ccid2_debug; | |||
44 | #define ccid2_pr_debug(format, a...) | 44 | #define ccid2_pr_debug(format, a...) |
45 | #endif | 45 | #endif |
46 | 46 | ||
47 | static const int ccid2_seq_len = 128; | ||
48 | |||
49 | #ifdef CONFIG_IP_DCCP_CCID2_DEBUG | 47 | #ifdef CONFIG_IP_DCCP_CCID2_DEBUG |
50 | static void ccid2_hc_tx_check_sanity(const struct ccid2_hc_tx_sock *hctx) | 48 | static void ccid2_hc_tx_check_sanity(const struct ccid2_hc_tx_sock *hctx) |
51 | { | 49 | { |
@@ -71,7 +69,6 @@ static void ccid2_hc_tx_check_sanity(const struct ccid2_hc_tx_sock *hctx) | |||
71 | BUG_ON(seqp->ccid2s_seq <= prev->ccid2s_seq); | 69 | BUG_ON(seqp->ccid2s_seq <= prev->ccid2s_seq); |
72 | BUG_ON(time_before(seqp->ccid2s_sent, | 70 | BUG_ON(time_before(seqp->ccid2s_sent, |
73 | prev->ccid2s_sent)); | 71 | prev->ccid2s_sent)); |
74 | BUG_ON(len > ccid2_seq_len); | ||
75 | 72 | ||
76 | seqp = prev; | 73 | seqp = prev; |
77 | } | 74 | } |
@@ -83,16 +80,57 @@ static void ccid2_hc_tx_check_sanity(const struct ccid2_hc_tx_sock *hctx) | |||
83 | do { | 80 | do { |
84 | seqp = seqp->ccid2s_prev; | 81 | seqp = seqp->ccid2s_prev; |
85 | len++; | 82 | len++; |
86 | BUG_ON(len > ccid2_seq_len); | ||
87 | } while (seqp != hctx->ccid2hctx_seqh); | 83 | } while (seqp != hctx->ccid2hctx_seqh); |
88 | 84 | ||
89 | BUG_ON(len != ccid2_seq_len); | ||
90 | ccid2_pr_debug("total len=%d\n", len); | 85 | ccid2_pr_debug("total len=%d\n", len); |
86 | BUG_ON(len != hctx->ccid2hctx_seqbufc * CCID2_SEQBUF_LEN); | ||
91 | } | 87 | } |
92 | #else | 88 | #else |
93 | #define ccid2_hc_tx_check_sanity(hctx) do {} while (0) | 89 | #define ccid2_hc_tx_check_sanity(hctx) do {} while (0) |
94 | #endif | 90 | #endif |
95 | 91 | ||
92 | static int ccid2_hc_tx_alloc_seq(struct ccid2_hc_tx_sock *hctx, int num, | ||
93 | gfp_t gfp) | ||
94 | { | ||
95 | struct ccid2_seq *seqp; | ||
96 | int i; | ||
97 | |||
98 | /* check if we have space to preserve the pointer to the buffer */ | ||
99 | if (hctx->ccid2hctx_seqbufc >= (sizeof(hctx->ccid2hctx_seqbuf) / | ||
100 | sizeof(struct ccid2_seq*))) | ||
101 | return -ENOMEM; | ||
102 | |||
103 | /* allocate buffer and initialize linked list */ | ||
104 | seqp = kmalloc(sizeof(*seqp) * num, gfp); | ||
105 | if (seqp == NULL) | ||
106 | return -ENOMEM; | ||
107 | |||
108 | for (i = 0; i < (num - 1); i++) { | ||
109 | seqp[i].ccid2s_next = &seqp[i + 1]; | ||
110 | seqp[i + 1].ccid2s_prev = &seqp[i]; | ||
111 | } | ||
112 | seqp[num - 1].ccid2s_next = seqp; | ||
113 | seqp->ccid2s_prev = &seqp[num - 1]; | ||
114 | |||
115 | /* This is the first allocation. Initiate the head and tail. */ | ||
116 | if (hctx->ccid2hctx_seqbufc == 0) | ||
117 | hctx->ccid2hctx_seqh = hctx->ccid2hctx_seqt = seqp; | ||
118 | else { | ||
119 | /* link the existing list with the one we just created */ | ||
120 | hctx->ccid2hctx_seqh->ccid2s_next = seqp; | ||
121 | seqp->ccid2s_prev = hctx->ccid2hctx_seqh; | ||
122 | |||
123 | hctx->ccid2hctx_seqt->ccid2s_prev = &seqp[num - 1]; | ||
124 | seqp[num - 1].ccid2s_next = hctx->ccid2hctx_seqt; | ||
125 | } | ||
126 | |||
127 | /* store the original pointer to the buffer so we can free it */ | ||
128 | hctx->ccid2hctx_seqbuf[hctx->ccid2hctx_seqbufc] = seqp; | ||
129 | hctx->ccid2hctx_seqbufc++; | ||
130 | |||
131 | return 0; | ||
132 | } | ||
133 | |||
96 | static int ccid2_hc_tx_send_packet(struct sock *sk, | 134 | static int ccid2_hc_tx_send_packet(struct sock *sk, |
97 | struct sk_buff *skb, int len) | 135 | struct sk_buff *skb, int len) |
98 | { | 136 | { |
@@ -231,6 +269,7 @@ static void ccid2_hc_tx_packet_sent(struct sock *sk, int more, int len) | |||
231 | { | 269 | { |
232 | struct dccp_sock *dp = dccp_sk(sk); | 270 | struct dccp_sock *dp = dccp_sk(sk); |
233 | struct ccid2_hc_tx_sock *hctx = ccid2_hc_tx_sk(sk); | 271 | struct ccid2_hc_tx_sock *hctx = ccid2_hc_tx_sk(sk); |
272 | struct ccid2_seq *next; | ||
234 | u64 seq; | 273 | u64 seq; |
235 | 274 | ||
236 | ccid2_hc_tx_check_sanity(hctx); | 275 | ccid2_hc_tx_check_sanity(hctx); |
@@ -250,15 +289,23 @@ static void ccid2_hc_tx_packet_sent(struct sock *sk, int more, int len) | |||
250 | hctx->ccid2hctx_seqh->ccid2s_seq = seq; | 289 | hctx->ccid2hctx_seqh->ccid2s_seq = seq; |
251 | hctx->ccid2hctx_seqh->ccid2s_acked = 0; | 290 | hctx->ccid2hctx_seqh->ccid2s_acked = 0; |
252 | hctx->ccid2hctx_seqh->ccid2s_sent = jiffies; | 291 | hctx->ccid2hctx_seqh->ccid2s_sent = jiffies; |
253 | hctx->ccid2hctx_seqh = hctx->ccid2hctx_seqh->ccid2s_next; | ||
254 | 292 | ||
255 | ccid2_pr_debug("cwnd=%d pipe=%d\n", hctx->ccid2hctx_cwnd, | 293 | next = hctx->ccid2hctx_seqh->ccid2s_next; |
256 | hctx->ccid2hctx_pipe); | 294 | /* check if we need to alloc more space */ |
295 | if (next == hctx->ccid2hctx_seqt) { | ||
296 | int rc; | ||
257 | 297 | ||
258 | if (hctx->ccid2hctx_seqh == hctx->ccid2hctx_seqt) { | 298 | ccid2_pr_debug("allocating more space in history\n"); |
259 | /* XXX allocate more space */ | 299 | rc = ccid2_hc_tx_alloc_seq(hctx, CCID2_SEQBUF_LEN, GFP_KERNEL); |
260 | WARN_ON(1); | 300 | BUG_ON(rc); /* XXX what do we do? */ |
301 | |||
302 | next = hctx->ccid2hctx_seqh->ccid2s_next; | ||
303 | BUG_ON(next == hctx->ccid2hctx_seqt); | ||
261 | } | 304 | } |
305 | hctx->ccid2hctx_seqh = next; | ||
306 | |||
307 | ccid2_pr_debug("cwnd=%d pipe=%d\n", hctx->ccid2hctx_cwnd, | ||
308 | hctx->ccid2hctx_pipe); | ||
262 | 309 | ||
263 | hctx->ccid2hctx_sent++; | 310 | hctx->ccid2hctx_sent++; |
264 | 311 | ||
@@ -674,8 +721,6 @@ static void ccid2_hc_tx_packet_recv(struct sock *sk, struct sk_buff *skb) | |||
674 | static int ccid2_hc_tx_init(struct ccid *ccid, struct sock *sk) | 721 | static int ccid2_hc_tx_init(struct ccid *ccid, struct sock *sk) |
675 | { | 722 | { |
676 | struct ccid2_hc_tx_sock *hctx = ccid_priv(ccid); | 723 | struct ccid2_hc_tx_sock *hctx = ccid_priv(ccid); |
677 | int seqcount = ccid2_seq_len; | ||
678 | int i; | ||
679 | 724 | ||
680 | hctx->ccid2hctx_cwnd = 1; | 725 | hctx->ccid2hctx_cwnd = 1; |
681 | /* Initialize ssthresh to infinity. This means that we will exit the | 726 | /* Initialize ssthresh to infinity. This means that we will exit the |
@@ -684,26 +729,12 @@ static int ccid2_hc_tx_init(struct ccid *ccid, struct sock *sk) | |||
684 | */ | 729 | */ |
685 | hctx->ccid2hctx_ssthresh = ~0; | 730 | hctx->ccid2hctx_ssthresh = ~0; |
686 | hctx->ccid2hctx_numdupack = 3; | 731 | hctx->ccid2hctx_numdupack = 3; |
732 | hctx->ccid2hctx_seqbufc = 0; | ||
687 | 733 | ||
688 | /* XXX init ~ to window size... */ | 734 | /* XXX init ~ to window size... */ |
689 | hctx->ccid2hctx_seqbuf = kmalloc(sizeof(*hctx->ccid2hctx_seqbuf) * | 735 | if (ccid2_hc_tx_alloc_seq(hctx, CCID2_SEQBUF_LEN, GFP_ATOMIC) != 0) |
690 | seqcount, gfp_any()); | ||
691 | if (hctx->ccid2hctx_seqbuf == NULL) | ||
692 | return -ENOMEM; | 736 | return -ENOMEM; |
693 | 737 | ||
694 | for (i = 0; i < (seqcount - 1); i++) { | ||
695 | hctx->ccid2hctx_seqbuf[i].ccid2s_next = | ||
696 | &hctx->ccid2hctx_seqbuf[i + 1]; | ||
697 | hctx->ccid2hctx_seqbuf[i + 1].ccid2s_prev = | ||
698 | &hctx->ccid2hctx_seqbuf[i]; | ||
699 | } | ||
700 | hctx->ccid2hctx_seqbuf[seqcount - 1].ccid2s_next = | ||
701 | hctx->ccid2hctx_seqbuf; | ||
702 | hctx->ccid2hctx_seqbuf->ccid2s_prev = | ||
703 | &hctx->ccid2hctx_seqbuf[seqcount - 1]; | ||
704 | |||
705 | hctx->ccid2hctx_seqh = hctx->ccid2hctx_seqbuf; | ||
706 | hctx->ccid2hctx_seqt = hctx->ccid2hctx_seqh; | ||
707 | hctx->ccid2hctx_sent = 0; | 738 | hctx->ccid2hctx_sent = 0; |
708 | hctx->ccid2hctx_rto = 3 * HZ; | 739 | hctx->ccid2hctx_rto = 3 * HZ; |
709 | hctx->ccid2hctx_srtt = -1; | 740 | hctx->ccid2hctx_srtt = -1; |
@@ -722,10 +753,13 @@ static int ccid2_hc_tx_init(struct ccid *ccid, struct sock *sk) | |||
722 | static void ccid2_hc_tx_exit(struct sock *sk) | 753 | static void ccid2_hc_tx_exit(struct sock *sk) |
723 | { | 754 | { |
724 | struct ccid2_hc_tx_sock *hctx = ccid2_hc_tx_sk(sk); | 755 | struct ccid2_hc_tx_sock *hctx = ccid2_hc_tx_sk(sk); |
756 | int i; | ||
725 | 757 | ||
726 | ccid2_hc_tx_kill_rto_timer(sk); | 758 | ccid2_hc_tx_kill_rto_timer(sk); |
727 | kfree(hctx->ccid2hctx_seqbuf); | 759 | |
728 | hctx->ccid2hctx_seqbuf = NULL; | 760 | for (i = 0; i < hctx->ccid2hctx_seqbufc; i++) |
761 | kfree(hctx->ccid2hctx_seqbuf[i]); | ||
762 | hctx->ccid2hctx_seqbufc = 0; | ||
729 | } | 763 | } |
730 | 764 | ||
731 | static void ccid2_hc_rx_packet_recv(struct sock *sk, struct sk_buff *skb) | 765 | static void ccid2_hc_rx_packet_recv(struct sock *sk, struct sk_buff *skb) |