aboutsummaryrefslogtreecommitdiffstats
path: root/net
diff options
context:
space:
mode:
authorStephen Hemminger <shemminger@osdl.org>2005-11-10 20:09:53 -0500
committerDavid S. Miller <davem@davemloft.net>2005-11-10 20:09:53 -0500
commit9772efb970780aeed488c19d8b4afd46c3b484af (patch)
treede016aaa29c8a95e98c7abaa70c8b590160e2886 /net
parent7faffa1c7fb9b8e8917e3225d4e2638270c0a48b (diff)
[TCP]: Appropriate Byte Count support
This is an updated version of the RFC3465 ABC patch originally for Linux 2.6.11-rc4 by Yee-Ting Li. ABC is a way of counting bytes ack'd rather than packets when updating congestion control. The orignal ABC described in the RFC applied to a Reno style algorithm. For advanced congestion control there is little change after leaving slow start. Signed-off-by: Stephen Hemminger <shemminger@osdl.org> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net')
-rw-r--r--net/ipv4/sysctl_net_ipv4.c8
-rw-r--r--net/ipv4/tcp.c1
-rw-r--r--net/ipv4/tcp_cong.c31
-rw-r--r--net/ipv4/tcp_input.c7
-rw-r--r--net/ipv4/tcp_minisocks.c1
5 files changed, 37 insertions, 11 deletions
diff --git a/net/ipv4/sysctl_net_ipv4.c b/net/ipv4/sysctl_net_ipv4.c
index 652685623519..01444a02b48b 100644
--- a/net/ipv4/sysctl_net_ipv4.c
+++ b/net/ipv4/sysctl_net_ipv4.c
@@ -645,6 +645,14 @@ ctl_table ipv4_table[] = {
645 .proc_handler = &proc_tcp_congestion_control, 645 .proc_handler = &proc_tcp_congestion_control,
646 .strategy = &sysctl_tcp_congestion_control, 646 .strategy = &sysctl_tcp_congestion_control,
647 }, 647 },
648 {
649 .ctl_name = NET_TCP_ABC,
650 .procname = "tcp_abc",
651 .data = &sysctl_tcp_abc,
652 .maxlen = sizeof(int),
653 .mode = 0644,
654 .proc_handler = &proc_dointvec,
655 },
648 656
649 { .ctl_name = 0 } 657 { .ctl_name = 0 }
650}; 658};
diff --git a/net/ipv4/tcp.c b/net/ipv4/tcp.c
index 72b7c22e1ea5..cfaf76133759 100644
--- a/net/ipv4/tcp.c
+++ b/net/ipv4/tcp.c
@@ -1669,6 +1669,7 @@ int tcp_disconnect(struct sock *sk, int flags)
1669 tp->packets_out = 0; 1669 tp->packets_out = 0;
1670 tp->snd_ssthresh = 0x7fffffff; 1670 tp->snd_ssthresh = 0x7fffffff;
1671 tp->snd_cwnd_cnt = 0; 1671 tp->snd_cwnd_cnt = 0;
1672 tp->bytes_acked = 0;
1672 tcp_set_ca_state(sk, TCP_CA_Open); 1673 tcp_set_ca_state(sk, TCP_CA_Open);
1673 tcp_clear_retrans(tp); 1674 tcp_clear_retrans(tp);
1674 inet_csk_delack_init(sk); 1675 inet_csk_delack_init(sk);
diff --git a/net/ipv4/tcp_cong.c b/net/ipv4/tcp_cong.c
index 6d3e883b48f6..c7cc62c8dc12 100644
--- a/net/ipv4/tcp_cong.c
+++ b/net/ipv4/tcp_cong.c
@@ -192,17 +192,26 @@ void tcp_reno_cong_avoid(struct sock *sk, u32 ack, u32 rtt, u32 in_flight,
192 /* In "safe" area, increase. */ 192 /* In "safe" area, increase. */
193 if (tp->snd_cwnd <= tp->snd_ssthresh) 193 if (tp->snd_cwnd <= tp->snd_ssthresh)
194 tcp_slow_start(tp); 194 tcp_slow_start(tp);
195 else { 195
196 /* In dangerous area, increase slowly. 196 /* In dangerous area, increase slowly. */
197 * In theory this is tp->snd_cwnd += 1 / tp->snd_cwnd 197 else if (sysctl_tcp_abc) {
198 */ 198 /* RFC3465: Apppriate Byte Count
199 if (tp->snd_cwnd_cnt >= tp->snd_cwnd) { 199 * increase once for each full cwnd acked
200 if (tp->snd_cwnd < tp->snd_cwnd_clamp) 200 */
201 tp->snd_cwnd++; 201 if (tp->bytes_acked >= tp->snd_cwnd*tp->mss_cache) {
202 tp->snd_cwnd_cnt = 0; 202 tp->bytes_acked -= tp->snd_cwnd*tp->mss_cache;
203 } else 203 if (tp->snd_cwnd < tp->snd_cwnd_clamp)
204 tp->snd_cwnd_cnt++; 204 tp->snd_cwnd++;
205 } 205 }
206 } else {
207 /* In theory this is tp->snd_cwnd += 1 / tp->snd_cwnd */
208 if (tp->snd_cwnd_cnt >= tp->snd_cwnd) {
209 if (tp->snd_cwnd < tp->snd_cwnd_clamp)
210 tp->snd_cwnd++;
211 tp->snd_cwnd_cnt = 0;
212 } else
213 tp->snd_cwnd_cnt++;
214 }
206} 215}
207EXPORT_SYMBOL_GPL(tcp_reno_cong_avoid); 216EXPORT_SYMBOL_GPL(tcp_reno_cong_avoid);
208 217
diff --git a/net/ipv4/tcp_input.c b/net/ipv4/tcp_input.c
index e43065654930..4cb5e6f408dc 100644
--- a/net/ipv4/tcp_input.c
+++ b/net/ipv4/tcp_input.c
@@ -89,6 +89,7 @@ int sysctl_tcp_frto;
89int sysctl_tcp_nometrics_save; 89int sysctl_tcp_nometrics_save;
90 90
91int sysctl_tcp_moderate_rcvbuf = 1; 91int sysctl_tcp_moderate_rcvbuf = 1;
92int sysctl_tcp_abc = 1;
92 93
93#define FLAG_DATA 0x01 /* Incoming frame contained data. */ 94#define FLAG_DATA 0x01 /* Incoming frame contained data. */
94#define FLAG_WIN_UPDATE 0x02 /* Incoming ACK was a window update. */ 95#define FLAG_WIN_UPDATE 0x02 /* Incoming ACK was a window update. */
@@ -1247,6 +1248,7 @@ void tcp_enter_loss(struct sock *sk, int how)
1247 tp->snd_cwnd_cnt = 0; 1248 tp->snd_cwnd_cnt = 0;
1248 tp->snd_cwnd_stamp = tcp_time_stamp; 1249 tp->snd_cwnd_stamp = tcp_time_stamp;
1249 1250
1251 tp->bytes_acked = 0;
1250 tcp_clear_retrans(tp); 1252 tcp_clear_retrans(tp);
1251 1253
1252 /* Push undo marker, if it was plain RTO and nothing 1254 /* Push undo marker, if it was plain RTO and nothing
@@ -1904,6 +1906,7 @@ tcp_fastretrans_alert(struct sock *sk, u32 prior_snd_una,
1904 TCP_ECN_queue_cwr(tp); 1906 TCP_ECN_queue_cwr(tp);
1905 } 1907 }
1906 1908
1909 tp->bytes_acked = 0;
1907 tp->snd_cwnd_cnt = 0; 1910 tp->snd_cwnd_cnt = 0;
1908 tcp_set_ca_state(sk, TCP_CA_Recovery); 1911 tcp_set_ca_state(sk, TCP_CA_Recovery);
1909 } 1912 }
@@ -2310,6 +2313,9 @@ static int tcp_ack(struct sock *sk, struct sk_buff *skb, int flag)
2310 if (before(ack, prior_snd_una)) 2313 if (before(ack, prior_snd_una))
2311 goto old_ack; 2314 goto old_ack;
2312 2315
2316 if (sysctl_tcp_abc && icsk->icsk_ca_state < TCP_CA_CWR)
2317 tp->bytes_acked += ack - prior_snd_una;
2318
2313 if (!(flag&FLAG_SLOWPATH) && after(ack, prior_snd_una)) { 2319 if (!(flag&FLAG_SLOWPATH) && after(ack, prior_snd_una)) {
2314 /* Window is constant, pure forward advance. 2320 /* Window is constant, pure forward advance.
2315 * No more checks are required. 2321 * No more checks are required.
@@ -4370,6 +4376,7 @@ discard:
4370 4376
4371EXPORT_SYMBOL(sysctl_tcp_ecn); 4377EXPORT_SYMBOL(sysctl_tcp_ecn);
4372EXPORT_SYMBOL(sysctl_tcp_reordering); 4378EXPORT_SYMBOL(sysctl_tcp_reordering);
4379EXPORT_SYMBOL(sysctl_tcp_abc);
4373EXPORT_SYMBOL(tcp_parse_options); 4380EXPORT_SYMBOL(tcp_parse_options);
4374EXPORT_SYMBOL(tcp_rcv_established); 4381EXPORT_SYMBOL(tcp_rcv_established);
4375EXPORT_SYMBOL(tcp_rcv_state_process); 4382EXPORT_SYMBOL(tcp_rcv_state_process);
diff --git a/net/ipv4/tcp_minisocks.c b/net/ipv4/tcp_minisocks.c
index b1a63b2c6b4a..9203a21e299f 100644
--- a/net/ipv4/tcp_minisocks.c
+++ b/net/ipv4/tcp_minisocks.c
@@ -380,6 +380,7 @@ struct sock *tcp_create_openreq_child(struct sock *sk, struct request_sock *req,
380 */ 380 */
381 newtp->snd_cwnd = 2; 381 newtp->snd_cwnd = 2;
382 newtp->snd_cwnd_cnt = 0; 382 newtp->snd_cwnd_cnt = 0;
383 newtp->bytes_acked = 0;
383 384
384 newtp->frto_counter = 0; 385 newtp->frto_counter = 0;
385 newtp->frto_highmark = 0; 386 newtp->frto_highmark = 0;