| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675 |
- /* SPDX-License-Identifier: GPL-2.0-or-later */
- #ifndef _TCP_ECN_H
- #define _TCP_ECN_H
- #include <linux/tcp.h>
- #include <linux/skbuff.h>
- #include <linux/bitfield.h>
- #include <net/inet_connection_sock.h>
- #include <net/sock.h>
- #include <net/tcp.h>
- #include <net/inet_ecn.h>
- /* The highest ECN variant (Accurate ECN, ECN, or no ECN) that is
- * attemped to be negotiated and requested for incoming connection
- * and outgoing connection, respectively.
- */
- enum tcp_ecn_mode {
- TCP_ECN_IN_NOECN_OUT_NOECN = 0,
- TCP_ECN_IN_ECN_OUT_ECN = 1,
- TCP_ECN_IN_ECN_OUT_NOECN = 2,
- TCP_ECN_IN_ACCECN_OUT_ACCECN = 3,
- TCP_ECN_IN_ACCECN_OUT_ECN = 4,
- TCP_ECN_IN_ACCECN_OUT_NOECN = 5,
- };
- /* AccECN option sending when AccECN has been successfully negotiated */
- enum tcp_accecn_option {
- TCP_ACCECN_OPTION_DISABLED = 0,
- TCP_ACCECN_OPTION_MINIMUM = 1,
- TCP_ACCECN_OPTION_FULL = 2,
- TCP_ACCECN_OPTION_PERSIST = 3,
- };
- /* Apply either ECT(0) or ECT(1) based on TCP_CONG_ECT_1_NEGOTIATION flag */
- static inline void INET_ECN_xmit_ect_1_negotiation(struct sock *sk)
- {
- __INET_ECN_xmit(sk, tcp_ca_ect_1_negotiation(sk));
- }
- static inline void tcp_ecn_queue_cwr(struct tcp_sock *tp)
- {
- /* Do not set CWR if in AccECN mode! */
- if (tcp_ecn_mode_rfc3168(tp))
- tp->ecn_flags |= TCP_ECN_QUEUE_CWR;
- }
- static inline void tcp_ecn_accept_cwr(struct sock *sk,
- const struct sk_buff *skb)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- if (tcp_ecn_mode_rfc3168(tp) && tcp_hdr(skb)->cwr) {
- tp->ecn_flags &= ~TCP_ECN_DEMAND_CWR;
- /* If the sender is telling us it has entered CWR, then its
- * cwnd may be very low (even just 1 packet), so we should ACK
- * immediately.
- */
- if (TCP_SKB_CB(skb)->seq != TCP_SKB_CB(skb)->end_seq)
- inet_csk(sk)->icsk_ack.pending |= ICSK_ACK_NOW;
- }
- }
- static inline void tcp_ecn_withdraw_cwr(struct tcp_sock *tp)
- {
- tp->ecn_flags &= ~TCP_ECN_QUEUE_CWR;
- }
- static inline bool tcp_accecn_ace_fail_send(const struct tcp_sock *tp)
- {
- return tp->accecn_fail_mode & TCP_ACCECN_ACE_FAIL_SEND;
- }
- static inline bool tcp_accecn_ace_fail_recv(const struct tcp_sock *tp)
- {
- return tp->accecn_fail_mode & TCP_ACCECN_ACE_FAIL_RECV;
- }
- static inline bool tcp_accecn_opt_fail_send(const struct tcp_sock *tp)
- {
- return tp->accecn_fail_mode & TCP_ACCECN_OPT_FAIL_SEND;
- }
- static inline bool tcp_accecn_opt_fail_recv(const struct tcp_sock *tp)
- {
- return tp->accecn_fail_mode & TCP_ACCECN_OPT_FAIL_RECV;
- }
- static inline void tcp_accecn_fail_mode_set(struct tcp_sock *tp, u8 mode)
- {
- tp->accecn_fail_mode |= mode;
- }
- static inline u8 tcp_accecn_ace(const struct tcphdr *th)
- {
- return (th->ae << 2) | (th->cwr << 1) | th->ece;
- }
- /* Infer the ECT value our SYN arrived with from the echoed ACE field */
- static inline int tcp_accecn_extract_syn_ect(u8 ace)
- {
- /* Below is an excerpt from the 1st block of Table 2 of AccECN spec */
- static const int ace_to_ecn[8] = {
- INET_ECN_ECT_0, /* 0b000 (Undefined) */
- INET_ECN_ECT_1, /* 0b001 (Undefined) */
- INET_ECN_NOT_ECT, /* 0b010 (Not-ECT is received) */
- INET_ECN_ECT_1, /* 0b011 (ECT-1 is received) */
- INET_ECN_ECT_0, /* 0b100 (ECT-0 is received) */
- INET_ECN_ECT_1, /* 0b101 (Reserved) */
- INET_ECN_CE, /* 0b110 (CE is received) */
- INET_ECN_ECT_1 /* 0b111 (Undefined) */
- };
- return ace_to_ecn[ace & 0x7];
- }
- /* Check ECN field transition to detect invalid transitions */
- static inline bool tcp_ect_transition_valid(u8 snt, u8 rcv)
- {
- if (rcv == snt)
- return true;
- /* Non-ECT altered to something or something became non-ECT */
- if (snt == INET_ECN_NOT_ECT || rcv == INET_ECN_NOT_ECT)
- return false;
- /* CE -> ECT(0/1)? */
- if (snt == INET_ECN_CE)
- return false;
- return true;
- }
- static inline bool tcp_accecn_validate_syn_feedback(struct sock *sk, u8 ace,
- u8 sent_ect)
- {
- u8 ect = tcp_accecn_extract_syn_ect(ace);
- struct tcp_sock *tp = tcp_sk(sk);
- if (!READ_ONCE(sock_net(sk)->ipv4.sysctl_tcp_ecn_fallback))
- return true;
- if (!tcp_ect_transition_valid(sent_ect, ect)) {
- tcp_accecn_fail_mode_set(tp, TCP_ACCECN_ACE_FAIL_RECV);
- return false;
- }
- return true;
- }
- static inline void tcp_accecn_saw_opt_fail_recv(struct tcp_sock *tp,
- u8 saw_opt)
- {
- tp->saw_accecn_opt = saw_opt;
- if (tp->saw_accecn_opt == TCP_ACCECN_OPT_FAIL_SEEN)
- tcp_accecn_fail_mode_set(tp, TCP_ACCECN_OPT_FAIL_RECV);
- }
- /* Validate the 3rd ACK based on the ACE field, see Table 4 of AccECN spec */
- static inline void tcp_accecn_third_ack(struct sock *sk,
- const struct sk_buff *skb, u8 sent_ect)
- {
- u8 ace = tcp_accecn_ace(tcp_hdr(skb));
- struct tcp_sock *tp = tcp_sk(sk);
- switch (ace) {
- case 0x0:
- /* Invalid value */
- if (!TCP_SKB_CB(skb)->sacked)
- tcp_accecn_fail_mode_set(tp, TCP_ACCECN_ACE_FAIL_RECV |
- TCP_ACCECN_OPT_FAIL_RECV);
- break;
- case 0x7:
- case 0x5:
- case 0x1:
- /* Unused but legal values */
- break;
- default:
- /* Validation only applies to first non-data packet */
- if (TCP_SKB_CB(skb)->seq == TCP_SKB_CB(skb)->end_seq &&
- !TCP_SKB_CB(skb)->sacked &&
- tcp_accecn_validate_syn_feedback(sk, ace, sent_ect)) {
- if ((tcp_accecn_extract_syn_ect(ace) == INET_ECN_CE) &&
- !tp->delivered_ce)
- tp->delivered_ce++;
- }
- break;
- }
- }
- /* Demand the minimum # to send AccECN optnio */
- static inline void tcp_accecn_opt_demand_min(struct sock *sk,
- u8 opt_demand_min)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- u8 opt_demand;
- opt_demand = max_t(u8, opt_demand_min, tp->accecn_opt_demand);
- tp->accecn_opt_demand = opt_demand;
- }
- /* Maps IP ECN field ECT/CE code point to AccECN option field number, given
- * we are sending fields with Accurate ECN Order 1: ECT(1), CE, ECT(0).
- */
- static inline u8 tcp_ecnfield_to_accecn_optfield(u8 ecnfield)
- {
- switch (ecnfield & INET_ECN_MASK) {
- case INET_ECN_NOT_ECT:
- return 0; /* AccECN does not send counts of NOT_ECT */
- case INET_ECN_ECT_1:
- return 1;
- case INET_ECN_CE:
- return 2;
- case INET_ECN_ECT_0:
- return 3;
- }
- return 0;
- }
- /* Maps IP ECN field ECT/CE code point to AccECN option field value offset.
- * Some fields do not start from zero, to detect zeroing by middleboxes.
- */
- static inline u32 tcp_accecn_field_init_offset(u8 ecnfield)
- {
- switch (ecnfield & INET_ECN_MASK) {
- case INET_ECN_NOT_ECT:
- return 0; /* AccECN does not send counts of NOT_ECT */
- case INET_ECN_ECT_1:
- return TCP_ACCECN_E1B_INIT_OFFSET;
- case INET_ECN_CE:
- return TCP_ACCECN_CEB_INIT_OFFSET;
- case INET_ECN_ECT_0:
- return TCP_ACCECN_E0B_INIT_OFFSET;
- }
- return 0;
- }
- /* Maps AccECN option field #nr to IP ECN field ECT/CE bits */
- static inline unsigned int tcp_accecn_optfield_to_ecnfield(unsigned int option,
- bool order)
- {
- /* Based on Table 5 of the AccECN spec to map (option, order) to
- * the corresponding ECN conuters (ECT-1, ECT-0, or CE).
- */
- static const u8 optfield_lookup[2][3] = {
- /* order = 0: 1st field ECT-0, 2nd field CE, 3rd field ECT-1 */
- { INET_ECN_ECT_0, INET_ECN_CE, INET_ECN_ECT_1 },
- /* order = 1: 1st field ECT-1, 2nd field CE, 3rd field ECT-0 */
- { INET_ECN_ECT_1, INET_ECN_CE, INET_ECN_ECT_0 }
- };
- return optfield_lookup[order][option % 3];
- }
- /* Handles AccECN option ECT and CE 24-bit byte counters update into
- * the u32 value in tcp_sock. As we're processing TCP options, it is
- * safe to access from - 1.
- */
- static inline s32 tcp_update_ecn_bytes(u32 *cnt, const char *from,
- u32 init_offset)
- {
- u32 truncated = (get_unaligned_be32(from - 1) - init_offset) &
- 0xFFFFFFU;
- u32 delta = (truncated - *cnt) & 0xFFFFFFU;
- /* If delta has the highest bit set (24th bit) indicating
- * negative, sign extend to correct an estimation using
- * sign_extend32(delta, 24 - 1)
- */
- delta = sign_extend32(delta, 23);
- *cnt += delta;
- return (s32)delta;
- }
- /* Updates Accurate ECN received counters from the received IP ECN field */
- static inline void tcp_ecn_received_counters(struct sock *sk,
- const struct sk_buff *skb, u32 len)
- {
- u8 ecnfield = TCP_SKB_CB(skb)->ip_dsfield & INET_ECN_MASK;
- u8 is_ce = INET_ECN_is_ce(ecnfield);
- struct tcp_sock *tp = tcp_sk(sk);
- bool ecn_edge;
- if (!INET_ECN_is_not_ect(ecnfield)) {
- u32 pcount = is_ce * max_t(u16, 1, skb_shinfo(skb)->gso_segs);
- /* As for accurate ECN, the TCP_ECN_SEEN flag is set by
- * tcp_ecn_received_counters() when the ECN codepoint of
- * received TCP data or ACK contains ECT(0), ECT(1), or CE.
- */
- if (!tcp_ecn_mode_rfc3168(tp))
- tp->ecn_flags |= TCP_ECN_SEEN;
- /* ACE counter tracks *all* segments including pure ACKs */
- tp->received_ce += pcount;
- tp->received_ce_pending = min(tp->received_ce_pending + pcount,
- 0xfU);
- if (len > 0) {
- u8 minlen = tcp_ecnfield_to_accecn_optfield(ecnfield);
- u32 oldbytes = tp->received_ecn_bytes[ecnfield - 1];
- u32 bytes_mask = GENMASK_U32(31, 22);
- tp->received_ecn_bytes[ecnfield - 1] += len;
- tp->accecn_minlen = max_t(u8, tp->accecn_minlen,
- minlen);
- /* Send AccECN option at least once per 2^22-byte
- * increase in any ECN byte counter.
- */
- if ((tp->received_ecn_bytes[ecnfield - 1] ^ oldbytes) &
- bytes_mask) {
- tcp_accecn_opt_demand_min(sk, 1);
- }
- }
- }
- ecn_edge = tp->prev_ecnfield != ecnfield;
- if (ecn_edge || is_ce) {
- tp->prev_ecnfield = ecnfield;
- /* Demand Accurate ECN change-triggered ACKs. Two ACK are
- * demanded to indicate unambiguously the ecnfield value
- * in the latter ACK.
- */
- if (tcp_ecn_mode_accecn(tp)) {
- if (ecn_edge)
- inet_csk(sk)->icsk_ack.pending |= ICSK_ACK_NOW;
- tp->accecn_opt_demand = 2;
- }
- }
- }
- /* AccECN specification, 2.2: [...] A Data Receiver maintains four counters
- * initialized at the start of the half-connection. [...] These byte counters
- * reflect only the TCP payload length, excluding TCP header and TCP options.
- */
- static inline void tcp_ecn_received_counters_payload(struct sock *sk,
- const struct sk_buff *skb)
- {
- const struct tcphdr *th = (const struct tcphdr *)skb->data;
- tcp_ecn_received_counters(sk, skb, skb->len - th->doff * 4);
- }
- /* AccECN specification, 5.1: [...] a server can determine that it
- * negotiated AccECN as [...] if the ACK contains an ACE field with
- * the value 0b010 to 0b111 (decimal 2 to 7).
- */
- static inline bool cookie_accecn_ok(const struct tcphdr *th)
- {
- return tcp_accecn_ace(th) > 0x1;
- }
- /* Used to form the ACE flags for SYN/ACK */
- static inline u16 tcp_accecn_reflector_flags(u8 ect)
- {
- /* TCP ACE flags of SYN/ACK are set based on IP-ECN received from SYN.
- * Below is an excerpt from the 1st block of Table 2 of AccECN spec,
- * in which TCP ACE flags are encoded as: (AE << 2) | (CWR << 1) | ECE
- */
- static const u8 ecn_to_ace_flags[4] = {
- 0b010, /* Not-ECT is received */
- 0b011, /* ECT(1) is received */
- 0b100, /* ECT(0) is received */
- 0b110 /* CE is received */
- };
- return FIELD_PREP(TCPHDR_ACE, ecn_to_ace_flags[ect & 0x3]);
- }
- /* AccECN specification, 3.1.2: If a TCP server that implements AccECN
- * receives a SYN with the three TCP header flags (AE, CWR and ECE) set
- * to any combination other than 000, 011 or 111, it MUST negotiate the
- * use of AccECN as if they had been set to 111.
- */
- static inline bool tcp_accecn_syn_requested(const struct tcphdr *th)
- {
- u8 ace = tcp_accecn_ace(th);
- return ace && ace != 0x3;
- }
- static inline void __tcp_accecn_init_bytes_counters(int *counter_array)
- {
- BUILD_BUG_ON(INET_ECN_ECT_1 != 0x1);
- BUILD_BUG_ON(INET_ECN_ECT_0 != 0x2);
- BUILD_BUG_ON(INET_ECN_CE != 0x3);
- counter_array[INET_ECN_ECT_1 - 1] = 0;
- counter_array[INET_ECN_ECT_0 - 1] = 0;
- counter_array[INET_ECN_CE - 1] = 0;
- }
- static inline void tcp_accecn_init_counters(struct tcp_sock *tp)
- {
- tp->received_ce = 0;
- tp->received_ce_pending = 0;
- __tcp_accecn_init_bytes_counters(tp->received_ecn_bytes);
- __tcp_accecn_init_bytes_counters(tp->delivered_ecn_bytes);
- tp->accecn_opt_sent_w_dsack = 0;
- tp->accecn_minlen = 0;
- tp->accecn_opt_demand = 0;
- tp->est_ecnfield = 0;
- }
- /* Used for make_synack to form the ACE flags */
- static inline void tcp_accecn_echo_syn_ect(struct tcphdr *th, u8 ect)
- {
- /* TCP ACE flags of SYN/ACK are set based on IP-ECN codepoint received
- * from SYN. Below is an excerpt from Table 2 of the AccECN spec:
- * +====================+====================================+
- * | IP-ECN codepoint | Respective ACE falgs on SYN/ACK |
- * | received on SYN | AE CWR ECE |
- * +====================+====================================+
- * | Not-ECT | 0 1 0 |
- * | ECT(1) | 0 1 1 |
- * | ECT(0) | 1 0 0 |
- * | CE | 1 1 0 |
- * +====================+====================================+
- */
- th->ae = !!(ect & INET_ECN_ECT_0);
- th->cwr = ect != INET_ECN_ECT_0;
- th->ece = ect == INET_ECN_ECT_1;
- }
- static inline void tcp_accecn_set_ace(struct tcp_sock *tp, struct sk_buff *skb,
- struct tcphdr *th)
- {
- u32 wire_ace;
- /* The final packet of the 3WHS or anything like it must reflect
- * the SYN/ACK ECT instead of putting CEP into ACE field, such
- * case show up in tcp_flags.
- */
- if (likely(!(TCP_SKB_CB(skb)->tcp_flags & TCPHDR_ACE))) {
- wire_ace = tp->received_ce + TCP_ACCECN_CEP_INIT_OFFSET;
- th->ece = !!(wire_ace & 0x1);
- th->cwr = !!(wire_ace & 0x2);
- th->ae = !!(wire_ace & 0x4);
- tp->received_ce_pending = 0;
- }
- }
- static inline u8 tcp_accecn_option_init(const struct sk_buff *skb,
- u8 opt_offset)
- {
- u8 *ptr = skb_transport_header(skb) + opt_offset;
- unsigned int optlen = ptr[1] - 2;
- if (WARN_ON_ONCE(ptr[0] != TCPOPT_ACCECN0 && ptr[0] != TCPOPT_ACCECN1))
- return TCP_ACCECN_OPT_FAIL_SEEN;
- ptr += 2;
- /* Detect option zeroing: an AccECN connection "MAY check that the
- * initial value of the EE0B field or the EE1B field is non-zero"
- */
- if (optlen < TCPOLEN_ACCECN_PERFIELD)
- return TCP_ACCECN_OPT_EMPTY_SEEN;
- if (get_unaligned_be24(ptr) == 0)
- return TCP_ACCECN_OPT_FAIL_SEEN;
- if (optlen < TCPOLEN_ACCECN_PERFIELD * 3)
- return TCP_ACCECN_OPT_COUNTER_SEEN;
- ptr += TCPOLEN_ACCECN_PERFIELD * 2;
- if (get_unaligned_be24(ptr) == 0)
- return TCP_ACCECN_OPT_FAIL_SEEN;
- return TCP_ACCECN_OPT_COUNTER_SEEN;
- }
- static inline void tcp_ecn_rcv_synack_accecn(struct sock *sk,
- const struct sk_buff *skb, u8 dsf)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- tcp_ecn_mode_set(tp, TCP_ECN_MODE_ACCECN);
- tp->syn_ect_rcv = dsf & INET_ECN_MASK;
- /* Demand Accurate ECN option in response to the SYN on the SYN/ACK
- * and the TCP server will try to send one more packet with an AccECN
- * Option at a later point during the connection.
- */
- if (tp->rx_opt.accecn &&
- tp->saw_accecn_opt < TCP_ACCECN_OPT_COUNTER_SEEN) {
- u8 saw_opt = tcp_accecn_option_init(skb, tp->rx_opt.accecn);
- tcp_accecn_saw_opt_fail_recv(tp, saw_opt);
- tp->accecn_opt_demand = 2;
- }
- }
- /* See Table 2 of the AccECN draft */
- static inline void tcp_ecn_rcv_synack(struct sock *sk, const struct sk_buff *skb,
- const struct tcphdr *th, u8 ip_dsfield)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- u8 ace = tcp_accecn_ace(th);
- switch (ace) {
- case 0x0:
- case 0x7:
- /* +========+========+============+=============+
- * | A | B | SYN/ACK | Feedback |
- * | | | B->A | Mode of A |
- * | | | AE CWR ECE | |
- * +========+========+============+=============+
- * | AccECN | No ECN | 0 0 0 | Not ECN |
- * | AccECN | Broken | 1 1 1 | Not ECN |
- * +========+========+============+=============+
- */
- tcp_ecn_mode_set(tp, TCP_ECN_DISABLED);
- break;
- case 0x1:
- /* +========+========+============+=============+
- * | A | B | SYN/ACK | Feedback |
- * | | | B->A | Mode of A |
- * | | | AE CWR ECE | |
- * +========+========+============+=============+
- * | AccECN | ECN | 0 0 1 | Classic ECN |
- * | Nonce | AccECN | 0 0 1 | Classic ECN |
- * | ECN | AccECN | 0 0 1 | Classic ECN |
- * +========+========+============+=============+
- */
- if (tcp_ca_no_fallback_rfc3168(sk))
- tcp_ecn_mode_set(tp, TCP_ECN_DISABLED);
- else
- tcp_ecn_mode_set(tp, TCP_ECN_MODE_RFC3168);
- break;
- case 0x5:
- if (tcp_ecn_mode_pending(tp)) {
- tcp_ecn_rcv_synack_accecn(sk, skb, ip_dsfield);
- if (INET_ECN_is_ce(ip_dsfield)) {
- tp->received_ce++;
- tp->received_ce_pending++;
- }
- }
- break;
- default:
- tcp_ecn_rcv_synack_accecn(sk, skb, ip_dsfield);
- if (INET_ECN_is_ce(ip_dsfield) &&
- tcp_accecn_validate_syn_feedback(sk, ace,
- tp->syn_ect_snt)) {
- tp->received_ce++;
- tp->received_ce_pending++;
- }
- break;
- }
- }
- static inline void tcp_ecn_rcv_syn(struct sock *sk, const struct tcphdr *th,
- const struct sk_buff *skb)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- if (tcp_ecn_mode_pending(tp)) {
- if (!tcp_accecn_syn_requested(th)) {
- /* Downgrade to classic ECN feedback */
- tcp_ecn_mode_set(tp, TCP_ECN_MODE_RFC3168);
- } else {
- tp->syn_ect_rcv = TCP_SKB_CB(skb)->ip_dsfield &
- INET_ECN_MASK;
- tp->prev_ecnfield = tp->syn_ect_rcv;
- tcp_ecn_mode_set(tp, TCP_ECN_MODE_ACCECN);
- }
- }
- if (tcp_ecn_mode_rfc3168(tp) &&
- (!th->ece || !th->cwr || tcp_ca_no_fallback_rfc3168(sk)))
- tcp_ecn_mode_set(tp, TCP_ECN_DISABLED);
- }
- static inline bool tcp_ecn_rcv_ecn_echo(const struct tcp_sock *tp,
- const struct tcphdr *th)
- {
- if (th->ece && !th->syn && tcp_ecn_mode_rfc3168(tp))
- return true;
- return false;
- }
- /* Packet ECN state for a SYN-ACK */
- static inline void tcp_ecn_send_synack(struct sock *sk, struct sk_buff *skb)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- TCP_SKB_CB(skb)->tcp_flags &= ~TCPHDR_CWR;
- if (tcp_ecn_disabled(tp))
- TCP_SKB_CB(skb)->tcp_flags &= ~TCPHDR_ECE;
- else if (tcp_ca_needs_ecn(sk) ||
- tcp_bpf_ca_needs_ecn(sk))
- INET_ECN_xmit_ect_1_negotiation(sk);
- if (tp->ecn_flags & TCP_ECN_MODE_ACCECN) {
- TCP_SKB_CB(skb)->tcp_flags &= ~TCPHDR_ACE;
- TCP_SKB_CB(skb)->tcp_flags |=
- tcp_accecn_reflector_flags(tp->syn_ect_rcv);
- tp->syn_ect_snt = inet_sk(sk)->tos & INET_ECN_MASK;
- }
- }
- /* Packet ECN state for a SYN. */
- static inline void tcp_ecn_send_syn(struct sock *sk, struct sk_buff *skb)
- {
- struct tcp_sock *tp = tcp_sk(sk);
- bool bpf_needs_ecn = tcp_bpf_ca_needs_ecn(sk);
- bool use_ecn, use_accecn;
- u8 tcp_ecn = READ_ONCE(sock_net(sk)->ipv4.sysctl_tcp_ecn);
- use_accecn = tcp_ecn == TCP_ECN_IN_ACCECN_OUT_ACCECN ||
- tcp_ca_needs_accecn(sk);
- use_ecn = tcp_ecn == TCP_ECN_IN_ECN_OUT_ECN ||
- tcp_ecn == TCP_ECN_IN_ACCECN_OUT_ECN ||
- tcp_ca_needs_ecn(sk) || bpf_needs_ecn || use_accecn;
- if (!use_ecn) {
- const struct dst_entry *dst = __sk_dst_get(sk);
- if (dst && dst_feature(dst, RTAX_FEATURE_ECN))
- use_ecn = true;
- }
- tp->ecn_flags = 0;
- if (use_ecn) {
- if (tcp_ca_needs_ecn(sk) || bpf_needs_ecn)
- INET_ECN_xmit_ect_1_negotiation(sk);
- TCP_SKB_CB(skb)->tcp_flags |= TCPHDR_ECE | TCPHDR_CWR;
- if (use_accecn) {
- TCP_SKB_CB(skb)->tcp_flags |= TCPHDR_AE;
- tcp_ecn_mode_set(tp, TCP_ECN_MODE_PENDING);
- tp->syn_ect_snt = inet_sk(sk)->tos & INET_ECN_MASK;
- } else {
- tcp_ecn_mode_set(tp, TCP_ECN_MODE_RFC3168);
- }
- }
- }
- static inline void tcp_ecn_clear_syn(struct sock *sk, struct sk_buff *skb)
- {
- if (READ_ONCE(sock_net(sk)->ipv4.sysctl_tcp_ecn_fallback)) {
- /* tp->ecn_flags are cleared at a later point in time when
- * SYN ACK is ultimatively being received.
- */
- TCP_SKB_CB(skb)->tcp_flags &= ~TCPHDR_ACE;
- }
- }
- static inline void
- tcp_ecn_make_synack(const struct request_sock *req, struct tcphdr *th,
- enum tcp_synack_type synack_type)
- {
- /* Accurate ECN shall retransmit SYN/ACK with ACE=0 if the
- * previously retransmitted SYN/ACK also times out.
- */
- if (!req->num_timeout || synack_type != TCP_SYNACK_RETRANS) {
- if (tcp_rsk(req)->accecn_ok)
- tcp_accecn_echo_syn_ect(th, tcp_rsk(req)->syn_ect_rcv);
- else if (inet_rsk(req)->ecn_ok)
- th->ece = 1;
- } else if (tcp_rsk(req)->accecn_ok) {
- th->ae = 0;
- th->cwr = 0;
- th->ece = 0;
- }
- }
- static inline bool tcp_accecn_option_beacon_check(const struct sock *sk)
- {
- u32 ecn_beacon = READ_ONCE(sock_net(sk)->ipv4.sysctl_tcp_ecn_option_beacon);
- const struct tcp_sock *tp = tcp_sk(sk);
- if (!ecn_beacon)
- return false;
- return tcp_stamp_us_delta(tp->tcp_mstamp, tp->accecn_opt_tstamp) * ecn_beacon >=
- (tp->srtt_us >> 3);
- }
- #endif /* _LINUX_TCP_ECN_H */
|