123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257 |
- #include <linux/mm.h>
- #include <linux/module.h>
- #include <linux/skbuff.h>
- #include <linux/inet_diag.h>
- #include <net/tcp.h>
- #include "tcp_vegas.h"
- #define TCP_YEAH_ALPHA 80
- #define TCP_YEAH_GAMMA 1
- #define TCP_YEAH_DELTA 3
- #define TCP_YEAH_EPSILON 1
- #define TCP_YEAH_PHY 8
- #define TCP_YEAH_RHO 16
- #define TCP_YEAH_ZETA 50
- #define TCP_SCALABLE_AI_CNT 100U
- struct yeah {
- struct vegas vegas;
-
- u32 lastQ;
- u32 doing_reno_now;
- u32 reno_count;
- u32 fast_count;
- u32 pkts_acked;
- };
- static void tcp_yeah_init(struct sock *sk)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- struct yeah *yeah = inet_csk_ca(sk);
- tcp_vegas_init(sk);
- yeah->doing_reno_now = 0;
- yeah->lastQ = 0;
- yeah->reno_count = 2;
-
- tp->snd_cwnd_clamp = min_t(u32, tp->snd_cwnd_clamp, 0xffffffff/128);
- }
- static void tcp_yeah_pkts_acked(struct sock *sk,
- const struct ack_sample *sample)
- {
- const struct inet_connection_sock *icsk = inet_csk(sk);
- struct yeah *yeah = inet_csk_ca(sk);
- if (icsk->icsk_ca_state == TCP_CA_Open)
- yeah->pkts_acked = sample->pkts_acked;
- tcp_vegas_pkts_acked(sk, sample);
- }
- static void tcp_yeah_cong_avoid(struct sock *sk, u32 ack, u32 acked)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- struct yeah *yeah = inet_csk_ca(sk);
- if (!tcp_is_cwnd_limited(sk))
- return;
- if (tcp_in_slow_start(tp))
- tcp_slow_start(tp, acked);
- else if (!yeah->doing_reno_now) {
-
- tp->snd_cwnd_cnt += yeah->pkts_acked;
- if (tp->snd_cwnd_cnt > min(tp->snd_cwnd, TCP_SCALABLE_AI_CNT)) {
- if (tp->snd_cwnd < tp->snd_cwnd_clamp)
- tp->snd_cwnd++;
- tp->snd_cwnd_cnt = 0;
- }
- yeah->pkts_acked = 1;
- } else {
-
- tcp_cong_avoid_ai(tp, tp->snd_cwnd, 1);
- }
-
- if (after(ack, yeah->vegas.beg_snd_nxt)) {
-
- if (yeah->vegas.cntRTT > 2) {
- u32 rtt, queue;
- u64 bw;
-
-
- rtt = yeah->vegas.minRTT;
-
- bw = tp->snd_cwnd;
- bw *= rtt - yeah->vegas.baseRTT;
- do_div(bw, rtt);
- queue = bw;
- if (queue > TCP_YEAH_ALPHA ||
- rtt - yeah->vegas.baseRTT > (yeah->vegas.baseRTT / TCP_YEAH_PHY)) {
- if (queue > TCP_YEAH_ALPHA &&
- tp->snd_cwnd > yeah->reno_count) {
- u32 reduction = min(queue / TCP_YEAH_GAMMA ,
- tp->snd_cwnd >> TCP_YEAH_EPSILON);
- tp->snd_cwnd -= reduction;
- tp->snd_cwnd = max(tp->snd_cwnd,
- yeah->reno_count);
- tp->snd_ssthresh = tp->snd_cwnd;
- }
- if (yeah->reno_count <= 2)
- yeah->reno_count = max(tp->snd_cwnd>>1, 2U);
- else
- yeah->reno_count++;
- yeah->doing_reno_now = min(yeah->doing_reno_now + 1,
- 0xffffffU);
- } else {
- yeah->fast_count++;
- if (yeah->fast_count > TCP_YEAH_ZETA) {
- yeah->reno_count = 2;
- yeah->fast_count = 0;
- }
- yeah->doing_reno_now = 0;
- }
- yeah->lastQ = queue;
- }
-
- yeah->vegas.beg_snd_una = yeah->vegas.beg_snd_nxt;
- yeah->vegas.beg_snd_nxt = tp->snd_nxt;
- yeah->vegas.beg_snd_cwnd = tp->snd_cwnd;
-
- yeah->vegas.cntRTT = 0;
- yeah->vegas.minRTT = 0x7fffffff;
- }
- }
- static u32 tcp_yeah_ssthresh(struct sock *sk)
- {
- const struct tcp_sock *tp = tcp_sk(sk);
- struct yeah *yeah = inet_csk_ca(sk);
- u32 reduction;
- if (yeah->doing_reno_now < TCP_YEAH_RHO) {
- reduction = yeah->lastQ;
- reduction = min(reduction, max(tp->snd_cwnd>>1, 2U));
- reduction = max(reduction, tp->snd_cwnd >> TCP_YEAH_DELTA);
- } else
- reduction = max(tp->snd_cwnd>>1, 2U);
- yeah->fast_count = 0;
- yeah->reno_count = max(yeah->reno_count>>1, 2U);
- return max_t(int, tp->snd_cwnd - reduction, 2);
- }
- static struct tcp_congestion_ops tcp_yeah __read_mostly = {
- .init = tcp_yeah_init,
- .ssthresh = tcp_yeah_ssthresh,
- .cong_avoid = tcp_yeah_cong_avoid,
- .set_state = tcp_vegas_state,
- .cwnd_event = tcp_vegas_cwnd_event,
- .get_info = tcp_vegas_get_info,
- .pkts_acked = tcp_yeah_pkts_acked,
- .owner = THIS_MODULE,
- .name = "yeah",
- };
- static int __init tcp_yeah_register(void)
- {
- BUG_ON(sizeof(struct yeah) > ICSK_CA_PRIV_SIZE);
- tcp_register_congestion_control(&tcp_yeah);
- return 0;
- }
- static void __exit tcp_yeah_unregister(void)
- {
- tcp_unregister_congestion_control(&tcp_yeah);
- }
- module_init(tcp_yeah_register);
- module_exit(tcp_yeah_unregister);
- MODULE_AUTHOR("Angelo P. Castellani");
- MODULE_LICENSE("GPL");
- MODULE_DESCRIPTION("YeAH TCP");
|