diff options
Diffstat (limited to 'net/ipv4')
-rw-r--r-- | net/ipv4/tcp_input.c | 50 | ||||
-rw-r--r-- | net/ipv4/tcp_minisocks.c | 6 | ||||
-rw-r--r-- | net/ipv4/tcp_output.c | 50 |
3 files changed, 106 insertions, 0 deletions
diff --git a/net/ipv4/tcp_input.c b/net/ipv4/tcp_input.c index 29999ef73b43..ea690afa592a 100644 --- a/net/ipv4/tcp_input.c +++ b/net/ipv4/tcp_input.c | |||
@@ -118,6 +118,7 @@ int sysctl_tcp_abc __read_mostly; | |||
118 | #define IsSackFrto() (sysctl_tcp_frto == 0x2) | 118 | #define IsSackFrto() (sysctl_tcp_frto == 0x2) |
119 | 119 | ||
120 | #define TCP_REMNANT (TCP_FLAG_FIN|TCP_FLAG_URG|TCP_FLAG_SYN|TCP_FLAG_PSH) | 120 | #define TCP_REMNANT (TCP_FLAG_FIN|TCP_FLAG_URG|TCP_FLAG_SYN|TCP_FLAG_PSH) |
121 | #define TCP_HP_BITS (~(TCP_RESERVED_BITS|TCP_FLAG_PSH)) | ||
121 | 122 | ||
122 | /* Adapt the MSS value used to make delayed ack decision to the | 123 | /* Adapt the MSS value used to make delayed ack decision to the |
123 | * real world. | 124 | * real world. |
@@ -198,6 +199,55 @@ static inline int tcp_in_quickack_mode(const struct sock *sk) | |||
198 | return icsk->icsk_ack.quick && !icsk->icsk_ack.pingpong; | 199 | return icsk->icsk_ack.quick && !icsk->icsk_ack.pingpong; |
199 | } | 200 | } |
200 | 201 | ||
202 | static inline void TCP_ECN_queue_cwr(struct tcp_sock *tp) | ||
203 | { | ||
204 | if (tp->ecn_flags&TCP_ECN_OK) | ||
205 | tp->ecn_flags |= TCP_ECN_QUEUE_CWR; | ||
206 | } | ||
207 | |||
208 | static inline void TCP_ECN_accept_cwr(struct tcp_sock *tp, struct sk_buff *skb) | ||
209 | { | ||
210 | if (tcp_hdr(skb)->cwr) | ||
211 | tp->ecn_flags &= ~TCP_ECN_DEMAND_CWR; | ||
212 | } | ||
213 | |||
214 | static inline void TCP_ECN_withdraw_cwr(struct tcp_sock *tp) | ||
215 | { | ||
216 | tp->ecn_flags &= ~TCP_ECN_DEMAND_CWR; | ||
217 | } | ||
218 | |||
219 | static inline void TCP_ECN_check_ce(struct tcp_sock *tp, struct sk_buff *skb) | ||
220 | { | ||
221 | if (tp->ecn_flags&TCP_ECN_OK) { | ||
222 | if (INET_ECN_is_ce(TCP_SKB_CB(skb)->flags)) | ||
223 | tp->ecn_flags |= TCP_ECN_DEMAND_CWR; | ||
224 | /* Funny extension: if ECT is not set on a segment, | ||
225 | * it is surely retransmit. It is not in ECN RFC, | ||
226 | * but Linux follows this rule. */ | ||
227 | else if (INET_ECN_is_not_ect((TCP_SKB_CB(skb)->flags))) | ||
228 | tcp_enter_quickack_mode((struct sock *)tp); | ||
229 | } | ||
230 | } | ||
231 | |||
232 | static inline void TCP_ECN_rcv_synack(struct tcp_sock *tp, struct tcphdr *th) | ||
233 | { | ||
234 | if ((tp->ecn_flags&TCP_ECN_OK) && (!th->ece || th->cwr)) | ||
235 | tp->ecn_flags &= ~TCP_ECN_OK; | ||
236 | } | ||
237 | |||
238 | static inline void TCP_ECN_rcv_syn(struct tcp_sock *tp, struct tcphdr *th) | ||
239 | { | ||
240 | if ((tp->ecn_flags&TCP_ECN_OK) && (!th->ece || !th->cwr)) | ||
241 | tp->ecn_flags &= ~TCP_ECN_OK; | ||
242 | } | ||
243 | |||
244 | static inline int TCP_ECN_rcv_ecn_echo(struct tcp_sock *tp, struct tcphdr *th) | ||
245 | { | ||
246 | if (th->ece && !th->syn && (tp->ecn_flags&TCP_ECN_OK)) | ||
247 | return 1; | ||
248 | return 0; | ||
249 | } | ||
250 | |||
201 | /* Buffer size and advertised window tuning. | 251 | /* Buffer size and advertised window tuning. |
202 | * | 252 | * |
203 | * 1. Tuning sk->sk_sndbuf, when connection enters established state. | 253 | * 1. Tuning sk->sk_sndbuf, when connection enters established state. |
diff --git a/net/ipv4/tcp_minisocks.c b/net/ipv4/tcp_minisocks.c index a12b08fca5ad..36a8fbd0e64e 100644 --- a/net/ipv4/tcp_minisocks.c +++ b/net/ipv4/tcp_minisocks.c | |||
@@ -368,6 +368,12 @@ void tcp_twsk_destructor(struct sock *sk) | |||
368 | 368 | ||
369 | EXPORT_SYMBOL_GPL(tcp_twsk_destructor); | 369 | EXPORT_SYMBOL_GPL(tcp_twsk_destructor); |
370 | 370 | ||
371 | static inline void TCP_ECN_openreq_child(struct tcp_sock *tp, | ||
372 | struct request_sock *req) | ||
373 | { | ||
374 | tp->ecn_flags = inet_rsk(req)->ecn_ok ? TCP_ECN_OK : 0; | ||
375 | } | ||
376 | |||
371 | /* This is not only more efficient than what we used to do, it eliminates | 377 | /* This is not only more efficient than what we used to do, it eliminates |
372 | * a lot of code duplication between IPv4/IPv6 SYN recv processing. -DaveM | 378 | * a lot of code duplication between IPv4/IPv6 SYN recv processing. -DaveM |
373 | * | 379 | * |
diff --git a/net/ipv4/tcp_output.c b/net/ipv4/tcp_output.c index b11025e8a49e..3abe22e4b576 100644 --- a/net/ipv4/tcp_output.c +++ b/net/ipv4/tcp_output.c | |||
@@ -269,6 +269,56 @@ static u16 tcp_select_window(struct sock *sk) | |||
269 | return new_win; | 269 | return new_win; |
270 | } | 270 | } |
271 | 271 | ||
272 | static inline void TCP_ECN_send_synack(struct tcp_sock *tp, | ||
273 | struct sk_buff *skb) | ||
274 | { | ||
275 | TCP_SKB_CB(skb)->flags &= ~TCPCB_FLAG_CWR; | ||
276 | if (!(tp->ecn_flags&TCP_ECN_OK)) | ||
277 | TCP_SKB_CB(skb)->flags &= ~TCPCB_FLAG_ECE; | ||
278 | } | ||
279 | |||
280 | static inline void TCP_ECN_send_syn(struct sock *sk, struct sk_buff *skb) | ||
281 | { | ||
282 | struct tcp_sock *tp = tcp_sk(sk); | ||
283 | |||
284 | tp->ecn_flags = 0; | ||
285 | if (sysctl_tcp_ecn) { | ||
286 | TCP_SKB_CB(skb)->flags |= TCPCB_FLAG_ECE|TCPCB_FLAG_CWR; | ||
287 | tp->ecn_flags = TCP_ECN_OK; | ||
288 | } | ||
289 | } | ||
290 | |||
291 | static __inline__ void | ||
292 | TCP_ECN_make_synack(struct request_sock *req, struct tcphdr *th) | ||
293 | { | ||
294 | if (inet_rsk(req)->ecn_ok) | ||
295 | th->ece = 1; | ||
296 | } | ||
297 | |||
298 | static inline void TCP_ECN_send(struct sock *sk, struct sk_buff *skb, | ||
299 | int tcp_header_len) | ||
300 | { | ||
301 | struct tcp_sock *tp = tcp_sk(sk); | ||
302 | |||
303 | if (tp->ecn_flags & TCP_ECN_OK) { | ||
304 | /* Not-retransmitted data segment: set ECT and inject CWR. */ | ||
305 | if (skb->len != tcp_header_len && | ||
306 | !before(TCP_SKB_CB(skb)->seq, tp->snd_nxt)) { | ||
307 | INET_ECN_xmit(sk); | ||
308 | if (tp->ecn_flags&TCP_ECN_QUEUE_CWR) { | ||
309 | tp->ecn_flags &= ~TCP_ECN_QUEUE_CWR; | ||
310 | tcp_hdr(skb)->cwr = 1; | ||
311 | skb_shinfo(skb)->gso_type |= SKB_GSO_TCP_ECN; | ||
312 | } | ||
313 | } else { | ||
314 | /* ACK or retransmitted segment: clear ECT|CE */ | ||
315 | INET_ECN_dontxmit(sk); | ||
316 | } | ||
317 | if (tp->ecn_flags & TCP_ECN_DEMAND_CWR) | ||
318 | tcp_hdr(skb)->ece = 1; | ||
319 | } | ||
320 | } | ||
321 | |||
272 | static void tcp_build_and_update_options(__be32 *ptr, struct tcp_sock *tp, | 322 | static void tcp_build_and_update_options(__be32 *ptr, struct tcp_sock *tp, |
273 | __u32 tstamp, __u8 **md5_hash) | 323 | __u32 tstamp, __u8 **md5_hash) |
274 | { | 324 | { |