2005-06-23 23:24:58 +04:00
/*
* Sally Floyd ' s High Speed TCP ( RFC 3649 ) congestion control
*
* See http : //www.icir.org/floyd/hstcp.html
*
* John Heffner < jheffner @ psc . edu >
*/
# include <linux/module.h>
# include <net/tcp.h>
/* From AIMD tables from RFC 3649 appendix B,
* with fixed - point MD scaled < < 8.
*/
static const struct hstcp_aimd_val {
2007-02-09 17:24:47 +03:00
unsigned int cwnd ;
unsigned int md ;
2005-06-23 23:24:58 +04:00
} hstcp_aimd_vals [ ] = {
{ 38 , 128 , /* 0.50 */ } ,
{ 118 , 112 , /* 0.44 */ } ,
{ 221 , 104 , /* 0.41 */ } ,
{ 347 , 98 , /* 0.38 */ } ,
{ 495 , 93 , /* 0.37 */ } ,
{ 663 , 89 , /* 0.35 */ } ,
{ 851 , 86 , /* 0.34 */ } ,
{ 1058 , 83 , /* 0.33 */ } ,
{ 1284 , 81 , /* 0.32 */ } ,
{ 1529 , 78 , /* 0.31 */ } ,
{ 1793 , 76 , /* 0.30 */ } ,
{ 2076 , 74 , /* 0.29 */ } ,
{ 2378 , 72 , /* 0.28 */ } ,
{ 2699 , 71 , /* 0.28 */ } ,
{ 3039 , 69 , /* 0.27 */ } ,
{ 3399 , 68 , /* 0.27 */ } ,
{ 3778 , 66 , /* 0.26 */ } ,
{ 4177 , 65 , /* 0.26 */ } ,
{ 4596 , 64 , /* 0.25 */ } ,
{ 5036 , 62 , /* 0.25 */ } ,
{ 5497 , 61 , /* 0.24 */ } ,
{ 5979 , 60 , /* 0.24 */ } ,
{ 6483 , 59 , /* 0.23 */ } ,
{ 7009 , 58 , /* 0.23 */ } ,
{ 7558 , 57 , /* 0.22 */ } ,
{ 8130 , 56 , /* 0.22 */ } ,
{ 8726 , 55 , /* 0.22 */ } ,
{ 9346 , 54 , /* 0.21 */ } ,
{ 9991 , 53 , /* 0.21 */ } ,
{ 10661 , 52 , /* 0.21 */ } ,
{ 11358 , 52 , /* 0.20 */ } ,
{ 12082 , 51 , /* 0.20 */ } ,
{ 12834 , 50 , /* 0.20 */ } ,
{ 13614 , 49 , /* 0.19 */ } ,
{ 14424 , 48 , /* 0.19 */ } ,
{ 15265 , 48 , /* 0.19 */ } ,
{ 16137 , 47 , /* 0.19 */ } ,
{ 17042 , 46 , /* 0.18 */ } ,
{ 17981 , 45 , /* 0.18 */ } ,
{ 18955 , 45 , /* 0.18 */ } ,
{ 19965 , 44 , /* 0.17 */ } ,
{ 21013 , 43 , /* 0.17 */ } ,
{ 22101 , 43 , /* 0.17 */ } ,
{ 23230 , 42 , /* 0.17 */ } ,
{ 24402 , 41 , /* 0.16 */ } ,
{ 25618 , 41 , /* 0.16 */ } ,
{ 26881 , 40 , /* 0.16 */ } ,
{ 28193 , 39 , /* 0.16 */ } ,
{ 29557 , 39 , /* 0.15 */ } ,
{ 30975 , 38 , /* 0.15 */ } ,
{ 32450 , 38 , /* 0.15 */ } ,
{ 33986 , 37 , /* 0.15 */ } ,
{ 35586 , 36 , /* 0.14 */ } ,
{ 37253 , 36 , /* 0.14 */ } ,
{ 38992 , 35 , /* 0.14 */ } ,
{ 40808 , 35 , /* 0.14 */ } ,
{ 42707 , 34 , /* 0.13 */ } ,
{ 44694 , 33 , /* 0.13 */ } ,
{ 46776 , 33 , /* 0.13 */ } ,
{ 48961 , 32 , /* 0.13 */ } ,
{ 51258 , 32 , /* 0.13 */ } ,
{ 53677 , 31 , /* 0.12 */ } ,
{ 56230 , 30 , /* 0.12 */ } ,
{ 58932 , 30 , /* 0.12 */ } ,
{ 61799 , 29 , /* 0.12 */ } ,
{ 64851 , 28 , /* 0.11 */ } ,
{ 68113 , 28 , /* 0.11 */ } ,
{ 71617 , 27 , /* 0.11 */ } ,
{ 75401 , 26 , /* 0.10 */ } ,
{ 79517 , 26 , /* 0.10 */ } ,
{ 84035 , 25 , /* 0.10 */ } ,
{ 89053 , 24 , /* 0.10 */ } ,
} ;
# define HSTCP_AIMD_MAX ARRAY_SIZE(hstcp_aimd_vals)
struct hstcp {
u32 ai ;
} ;
2005-08-10 11:03:31 +04:00
static void hstcp_init ( struct sock * sk )
2005-06-23 23:24:58 +04:00
{
2005-08-10 11:03:31 +04:00
struct tcp_sock * tp = tcp_sk ( sk ) ;
struct hstcp * ca = inet_csk_ca ( sk ) ;
2005-06-23 23:24:58 +04:00
ca - > ai = 0 ;
/* Ensure the MD arithmetic works. This is somewhat pedantic,
* since I don ' t think we will see a cwnd this large . : ) */
tp - > snd_cwnd_clamp = min_t ( u32 , tp - > snd_cwnd_clamp , 0xffffffff / 128 ) ;
}
2007-12-02 01:47:59 +03:00
static void hstcp_cong_avoid ( struct sock * sk , u32 adk , u32 in_flight )
2005-06-23 23:24:58 +04:00
{
2005-08-10 11:03:31 +04:00
struct tcp_sock * tp = tcp_sk ( sk ) ;
struct hstcp * ca = inet_csk_ca ( sk ) ;
2005-06-23 23:24:58 +04:00
2005-11-11 03:53:30 +03:00
if ( ! tcp_is_cwnd_limited ( sk , in_flight ) )
2005-06-23 23:24:58 +04:00
return ;
2007-05-04 00:28:35 +04:00
if ( tp - > snd_cwnd < = tp - > snd_ssthresh )
tcp_slow_start ( tp ) ;
else {
2006-07-12 00:03:28 +04:00
/* Update AIMD parameters.
*
* We want to guarantee that :
* hstcp_aimd_vals [ ca - > ai - 1 ] . cwnd <
* snd_cwnd < =
* hstcp_aimd_vals [ ca - > ai ] . cwnd
*/
2005-06-23 23:24:58 +04:00
if ( tp - > snd_cwnd > hstcp_aimd_vals [ ca - > ai ] . cwnd ) {
while ( tp - > snd_cwnd > hstcp_aimd_vals [ ca - > ai ] . cwnd & &
2006-03-13 07:34:53 +03:00
ca - > ai < HSTCP_AIMD_MAX - 1 )
2005-06-23 23:24:58 +04:00
ca - > ai + + ;
2006-07-12 00:03:28 +04:00
} else if ( ca - > ai & & tp - > snd_cwnd < = hstcp_aimd_vals [ ca - > ai - 1 ] . cwnd ) {
while ( ca - > ai & & tp - > snd_cwnd < = hstcp_aimd_vals [ ca - > ai - 1 ] . cwnd )
2005-06-23 23:24:58 +04:00
ca - > ai - - ;
}
/* Do additive increase */
if ( tp - > snd_cwnd < tp - > snd_cwnd_clamp ) {
2006-06-03 04:51:08 +04:00
/* cwnd = cwnd + a(w) / cwnd */
tp - > snd_cwnd_cnt + = ca - > ai + 1 ;
2005-06-23 23:24:58 +04:00
if ( tp - > snd_cwnd_cnt > = tp - > snd_cwnd ) {
tp - > snd_cwnd_cnt - = tp - > snd_cwnd ;
2006-05-06 04:41:44 +04:00
tp - > snd_cwnd + + ;
2005-06-23 23:24:58 +04:00
}
}
}
}
2005-08-10 11:03:31 +04:00
static u32 hstcp_ssthresh ( struct sock * sk )
2005-06-23 23:24:58 +04:00
{
2005-08-10 11:03:31 +04:00
const struct tcp_sock * tp = tcp_sk ( sk ) ;
const struct hstcp * ca = inet_csk_ca ( sk ) ;
2005-06-23 23:24:58 +04:00
/* Do multiplicative decrease */
return max ( tp - > snd_cwnd - ( ( tp - > snd_cwnd * hstcp_aimd_vals [ ca - > ai ] . md ) > > 8 ) , 2U ) ;
}
static struct tcp_congestion_ops tcp_highspeed = {
. init = hstcp_init ,
. ssthresh = hstcp_ssthresh ,
. cong_avoid = hstcp_cong_avoid ,
. min_cwnd = tcp_reno_min_cwnd ,
. owner = THIS_MODULE ,
. name = " highspeed "
} ;
static int __init hstcp_register ( void )
{
2006-08-26 04:10:33 +04:00
BUILD_BUG_ON ( sizeof ( struct hstcp ) > ICSK_CA_PRIV_SIZE ) ;
2005-06-23 23:24:58 +04:00
return tcp_register_congestion_control ( & tcp_highspeed ) ;
}
static void __exit hstcp_unregister ( void )
{
tcp_unregister_congestion_control ( & tcp_highspeed ) ;
}
module_init ( hstcp_register ) ;
module_exit ( hstcp_unregister ) ;
MODULE_AUTHOR ( " John Heffner " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " High Speed TCP " ) ;