Blame view

net/rds/tcp_recv.c 9.16 KB
70041088e   Andy Grover   RDS: Add TCP tran...
1
  /*
eee2fa6ab   Ka-Cheong Poon   rds: Changing IP ...
2
   * Copyright (c) 2006, 2017 Oracle and/or its affiliates. All rights reserved.
70041088e   Andy Grover   RDS: Add TCP tran...
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
   *
   * This software is available to you under a choice of one of two
   * licenses.  You may choose to be licensed under the terms of the GNU
   * General Public License (GPL) Version 2, available from the file
   * COPYING in the main directory of this source tree, or the
   * OpenIB.org BSD license below:
   *
   *     Redistribution and use in source and binary forms, with or
   *     without modification, are permitted provided that the following
   *     conditions are met:
   *
   *      - Redistributions of source code must retain the above
   *        copyright notice, this list of conditions and the following
   *        disclaimer.
   *
   *      - Redistributions in binary form must reproduce the above
   *        copyright notice, this list of conditions and the following
   *        disclaimer in the documentation and/or other materials
   *        provided with the distribution.
   *
   * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
   * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
   * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
   * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
   * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
   * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
   * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
   * SOFTWARE.
   *
   */
  #include <linux/kernel.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
34
  #include <linux/slab.h>
70041088e   Andy Grover   RDS: Add TCP tran...
35
36
37
38
39
40
  #include <net/tcp.h>
  
  #include "rds.h"
  #include "tcp.h"
  
  static struct kmem_cache *rds_tcp_incoming_slab;
809fa148a   Andy Grover   RDS: inc_purge() ...
41
  static void rds_tcp_inc_purge(struct rds_incoming *inc)
70041088e   Andy Grover   RDS: Add TCP tran...
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
  {
  	struct rds_tcp_incoming *tinc;
  	tinc = container_of(inc, struct rds_tcp_incoming, ti_inc);
  	rdsdebug("purging tinc %p inc %p
  ", tinc, inc);
  	skb_queue_purge(&tinc->ti_skb_list);
  }
  
  void rds_tcp_inc_free(struct rds_incoming *inc)
  {
  	struct rds_tcp_incoming *tinc;
  	tinc = container_of(inc, struct rds_tcp_incoming, ti_inc);
  	rds_tcp_inc_purge(inc);
  	rdsdebug("freeing tinc %p inc %p
  ", tinc, inc);
  	kmem_cache_free(rds_tcp_incoming_slab, tinc);
  }
  
  /*
   * this is pretty lame, but, whatever.
   */
c310e72c8   Al Viro   rds: switch ->inc...
63
  int rds_tcp_inc_copy_to_user(struct rds_incoming *inc, struct iov_iter *to)
70041088e   Andy Grover   RDS: Add TCP tran...
64
65
  {
  	struct rds_tcp_incoming *tinc;
70041088e   Andy Grover   RDS: Add TCP tran...
66
  	struct sk_buff *skb;
70041088e   Andy Grover   RDS: Add TCP tran...
67
  	int ret = 0;
c310e72c8   Al Viro   rds: switch ->inc...
68
  	if (!iov_iter_count(to))
70041088e   Andy Grover   RDS: Add TCP tran...
69
70
71
  		goto out;
  
  	tinc = container_of(inc, struct rds_tcp_incoming, ti_inc);
70041088e   Andy Grover   RDS: Add TCP tran...
72
73
  
  	skb_queue_walk(&tinc->ti_skb_list, skb) {
c310e72c8   Al Viro   rds: switch ->inc...
74
75
76
  		unsigned long to_copy, skb_off;
  		for (skb_off = 0; skb_off < skb->len; skb_off += to_copy) {
  			to_copy = iov_iter_count(to);
70041088e   Andy Grover   RDS: Add TCP tran...
77
  			to_copy = min(to_copy, skb->len - skb_off);
c310e72c8   Al Viro   rds: switch ->inc...
78
79
  			if (skb_copy_datagram_iter(skb, skb_off, to, to_copy))
  				return -EFAULT;
70041088e   Andy Grover   RDS: Add TCP tran...
80

b075cfdb6   Andy Grover   RDS: update copy_...
81
  			rds_stats_add(s_copy_to_user, to_copy);
70041088e   Andy Grover   RDS: Add TCP tran...
82
  			ret += to_copy;
c310e72c8   Al Viro   rds: switch ->inc...
83
84
  
  			if (!iov_iter_count(to))
70041088e   Andy Grover   RDS: Add TCP tran...
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
  				goto out;
  		}
  	}
  out:
  	return ret;
  }
  
  /*
   * We have a series of skbs that have fragmented pieces of the congestion
   * bitmap.  They must add up to the exact size of the congestion bitmap.  We
   * use the skb helpers to copy those into the pages that make up the in-memory
   * congestion bitmap for the remote address of this connection.  We then tell
   * the congestion core that the bitmap has been changed so that it can wake up
   * sleepers.
   *
   * This is racing with sending paths which are using test_bit to see if the
   * bitmap indicates that their recipient is congested.
   */
  
  static void rds_tcp_cong_recv(struct rds_connection *conn,
  			      struct rds_tcp_incoming *tinc)
  {
  	struct sk_buff *skb;
  	unsigned int to_copy, skb_off;
  	unsigned int map_off;
  	unsigned int map_page;
  	struct rds_cong_map *map;
  	int ret;
  
  	/* catch completely corrupt packets */
  	if (be32_to_cpu(tinc->ti_inc.i_hdr.h_len) != RDS_CONG_MAP_BYTES)
  		return;
  
  	map_page = 0;
  	map_off = 0;
  	map = conn->c_fcong;
  
  	skb_queue_walk(&tinc->ti_skb_list, skb) {
  		skb_off = 0;
  		while (skb_off < skb->len) {
  			to_copy = min_t(unsigned int, PAGE_SIZE - map_off,
  					skb->len - skb_off);
  
  			BUG_ON(map_page >= RDS_CONG_MAP_PAGES);
  
  			/* only returns 0 or -error */
  			ret = skb_copy_bits(skb, skb_off,
  				(void *)map->m_page_addrs[map_page] + map_off,
  				to_copy);
  			BUG_ON(ret != 0);
  
  			skb_off += to_copy;
  			map_off += to_copy;
  			if (map_off == PAGE_SIZE) {
  				map_off = 0;
  				map_page++;
  			}
  		}
  	}
  
  	rds_cong_map_updated(map, ~(u64) 0);
  }
  
  struct rds_tcp_desc_arg {
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
149
  	struct rds_conn_path *conn_path;
70041088e   Andy Grover   RDS: Add TCP tran...
150
  	gfp_t gfp;
70041088e   Andy Grover   RDS: Add TCP tran...
151
152
153
154
155
156
  };
  
  static int rds_tcp_data_recv(read_descriptor_t *desc, struct sk_buff *skb,
  			     unsigned int offset, size_t len)
  {
  	struct rds_tcp_desc_arg *arg = desc->arg.data;
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
157
158
  	struct rds_conn_path *cp = arg->conn_path;
  	struct rds_tcp_connection *tc = cp->cp_transport_data;
70041088e   Andy Grover   RDS: Add TCP tran...
159
160
161
162
163
164
165
166
167
168
169
170
171
  	struct rds_tcp_incoming *tinc = tc->t_tinc;
  	struct sk_buff *clone;
  	size_t left = len, to_copy;
  
  	rdsdebug("tcp data tc %p skb %p offset %u len %zu
  ", tc, skb, offset,
  		 len);
  
  	/*
  	 * tcp_read_sock() interprets partial progress as an indication to stop
  	 * processing.
  	 */
  	while (left) {
8690bfa17   Andy Grover   RDS: cleanup: rem...
172
  		if (!tinc) {
70041088e   Andy Grover   RDS: Add TCP tran...
173
  			tinc = kmem_cache_alloc(rds_tcp_incoming_slab,
5c3da57d7   Joshua Houghton   net: rds: fix cod...
174
  						arg->gfp);
8690bfa17   Andy Grover   RDS: cleanup: rem...
175
  			if (!tinc) {
70041088e   Andy Grover   RDS: Add TCP tran...
176
177
178
179
180
181
  				desc->error = -ENOMEM;
  				goto out;
  			}
  			tc->t_tinc = tinc;
  			rdsdebug("alloced tinc %p
  ", tinc);
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
182
  			rds_inc_path_init(&tinc->ti_inc, cp,
eee2fa6ab   Ka-Cheong Poon   rds: Changing IP ...
183
  					  &cp->cp_conn->c_faddr);
3289025ae   Santosh Shilimkar   RDS: add receive ...
184
185
  			tinc->ti_inc.i_rx_lat_trace[RDS_MSG_RX_HDR] =
  					local_clock();
70041088e   Andy Grover   RDS: Add TCP tran...
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
  			/*
  			 * XXX * we might be able to use the __ variants when
  			 * we've already serialized at a higher level.
  			 */
  			skb_queue_head_init(&tinc->ti_skb_list);
  		}
  
  		if (left && tc->t_tinc_hdr_rem) {
  			to_copy = min(tc->t_tinc_hdr_rem, left);
  			rdsdebug("copying %zu header from skb %p
  ", to_copy,
  				 skb);
  			skb_copy_bits(skb, offset,
  				      (char *)&tinc->ti_inc.i_hdr +
  						sizeof(struct rds_header) -
  						tc->t_tinc_hdr_rem,
  				      to_copy);
  			tc->t_tinc_hdr_rem -= to_copy;
  			left -= to_copy;
  			offset += to_copy;
  
  			if (tc->t_tinc_hdr_rem == 0) {
  				/* could be 0 for a 0 len message */
  				tc->t_tinc_data_rem =
  					be32_to_cpu(tinc->ti_inc.i_hdr.h_len);
3289025ae   Santosh Shilimkar   RDS: add receive ...
211
212
  				tinc->ti_inc.i_rx_lat_trace[RDS_MSG_RX_START] =
  					local_clock();
70041088e   Andy Grover   RDS: Add TCP tran...
213
214
215
216
  			}
  		}
  
  		if (left && tc->t_tinc_data_rem) {
947d2756c   Sowmini Varadhan   RDS: TCP: Call ps...
217
218
219
  			to_copy = min(tc->t_tinc_data_rem, left);
  
  			clone = pskb_extract(skb, offset, to_copy, arg->gfp);
8690bfa17   Andy Grover   RDS: cleanup: rem...
220
  			if (!clone) {
70041088e   Andy Grover   RDS: Add TCP tran...
221
222
223
  				desc->error = -ENOMEM;
  				goto out;
  			}
70041088e   Andy Grover   RDS: Add TCP tran...
224
225
226
227
228
229
230
231
232
233
234
235
236
237
  			skb_queue_tail(&tinc->ti_skb_list, clone);
  
  			rdsdebug("skb %p data %p len %d off %u to_copy %zu -> "
  				 "clone %p data %p len %d
  ",
  				 skb, skb->data, skb->len, offset, to_copy,
  				 clone, clone->data, clone->len);
  
  			tc->t_tinc_data_rem -= to_copy;
  			left -= to_copy;
  			offset += to_copy;
  		}
  
  		if (tc->t_tinc_hdr_rem == 0 && tc->t_tinc_data_rem == 0) {
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
238
  			struct rds_connection *conn = cp->cp_conn;
70041088e   Andy Grover   RDS: Add TCP tran...
239
240
241
  			if (tinc->ti_inc.i_hdr.h_flags == RDS_FLAG_CONG_BITMAP)
  				rds_tcp_cong_recv(conn, tinc);
  			else
eee2fa6ab   Ka-Cheong Poon   rds: Changing IP ...
242
243
244
  				rds_recv_incoming(conn, &conn->c_faddr,
  						  &conn->c_laddr,
  						  &tinc->ti_inc,
6114eab53   Cong Wang   rds: remove the s...
245
  						  arg->gfp);
70041088e   Andy Grover   RDS: Add TCP tran...
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
  
  			tc->t_tinc_hdr_rem = sizeof(struct rds_header);
  			tc->t_tinc_data_rem = 0;
  			tc->t_tinc = NULL;
  			rds_inc_put(&tinc->ti_inc);
  			tinc = NULL;
  		}
  	}
  out:
  	rdsdebug("returning len %zu left %zu skb len %d rx queue depth %d
  ",
  		 len, left, skb->len,
  		 skb_queue_len(&tc->t_sock->sk->sk_receive_queue));
  	return len - left;
  }
  
  /* the caller has to hold the sock lock */
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
263
  static int rds_tcp_read_sock(struct rds_conn_path *cp, gfp_t gfp)
70041088e   Andy Grover   RDS: Add TCP tran...
264
  {
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
265
  	struct rds_tcp_connection *tc = cp->cp_transport_data;
70041088e   Andy Grover   RDS: Add TCP tran...
266
267
268
269
270
  	struct socket *sock = tc->t_sock;
  	read_descriptor_t desc;
  	struct rds_tcp_desc_arg arg;
  
  	/* It's like glib in the kernel! */
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
271
  	arg.conn_path = cp;
70041088e   Andy Grover   RDS: Add TCP tran...
272
  	arg.gfp = gfp;
70041088e   Andy Grover   RDS: Add TCP tran...
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
  	desc.arg.data = &arg;
  	desc.error = 0;
  	desc.count = 1; /* give more than one skb per call */
  
  	tcp_read_sock(sock->sk, &desc, rds_tcp_data_recv);
  	rdsdebug("tcp_read_sock for tc %p gfp 0x%x returned %d
  ", tc, gfp,
  		 desc.error);
  
  	return desc.error;
  }
  
  /*
   * We hold the sock lock to serialize our rds_tcp_recv->tcp_read_sock from
   * data_ready.
   *
   * if we fail to allocate we're in trouble.. blindly wait some time before
   * trying again to see if the VM can free up something for us.
   */
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
292
  int rds_tcp_recv_path(struct rds_conn_path *cp)
70041088e   Andy Grover   RDS: Add TCP tran...
293
  {
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
294
  	struct rds_tcp_connection *tc = cp->cp_transport_data;
70041088e   Andy Grover   RDS: Add TCP tran...
295
296
  	struct socket *sock = tc->t_sock;
  	int ret = 0;
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
297
298
299
  	rdsdebug("recv worker path [%d] tc %p sock %p
  ",
  		 cp->cp_index, tc, sock);
70041088e   Andy Grover   RDS: Add TCP tran...
300
301
  
  	lock_sock(sock->sk);
2da43c4a1   Sowmini Varadhan   RDS: TCP: make re...
302
  	ret = rds_tcp_read_sock(cp, GFP_KERNEL);
70041088e   Andy Grover   RDS: Add TCP tran...
303
304
305
306
  	release_sock(sock->sk);
  
  	return ret;
  }
676d23690   David S. Miller   net: Fix use afte...
307
  void rds_tcp_data_ready(struct sock *sk)
70041088e   Andy Grover   RDS: Add TCP tran...
308
  {
676d23690   David S. Miller   net: Fix use afte...
309
  	void (*ready)(struct sock *sk);
ea3b1ea53   Sowmini Varadhan   RDS: TCP: make ->...
310
  	struct rds_conn_path *cp;
70041088e   Andy Grover   RDS: Add TCP tran...
311
  	struct rds_tcp_connection *tc;
676d23690   David S. Miller   net: Fix use afte...
312
313
  	rdsdebug("data ready sk %p
  ", sk);
70041088e   Andy Grover   RDS: Add TCP tran...
314

38036629c   Eric Dumazet   rds: tcp: block B...
315
  	read_lock_bh(&sk->sk_callback_lock);
ea3b1ea53   Sowmini Varadhan   RDS: TCP: make ->...
316
317
  	cp = sk->sk_user_data;
  	if (!cp) { /* check for teardown race */
70041088e   Andy Grover   RDS: Add TCP tran...
318
319
320
  		ready = sk->sk_data_ready;
  		goto out;
  	}
ea3b1ea53   Sowmini Varadhan   RDS: TCP: make ->...
321
  	tc = cp->cp_transport_data;
70041088e   Andy Grover   RDS: Add TCP tran...
322
323
  	ready = tc->t_orig_data_ready;
  	rds_tcp_stats_inc(s_tcp_data_ready_calls);
3db6e0d17   Sowmini Varadhan   rds: use RCU to s...
324
325
  	if (rds_tcp_read_sock(cp, GFP_ATOMIC) == -ENOMEM) {
  		rcu_read_lock();
ebeeb1ad9   Sowmini Varadhan   rds: tcp: use rds...
326
  		if (!rds_destroy_pending(cp->cp_conn))
3db6e0d17   Sowmini Varadhan   rds: use RCU to s...
327
328
329
  			queue_delayed_work(rds_wq, &cp->cp_recv_w, 0);
  		rcu_read_unlock();
  	}
70041088e   Andy Grover   RDS: Add TCP tran...
330
  out:
38036629c   Eric Dumazet   rds: tcp: block B...
331
  	read_unlock_bh(&sk->sk_callback_lock);
676d23690   David S. Miller   net: Fix use afte...
332
  	ready(sk);
70041088e   Andy Grover   RDS: Add TCP tran...
333
  }
ef87b7ea3   Zach Brown   RDS: remove __ini...
334
  int rds_tcp_recv_init(void)
70041088e   Andy Grover   RDS: Add TCP tran...
335
336
337
338
  {
  	rds_tcp_incoming_slab = kmem_cache_create("rds_tcp_incoming",
  					sizeof(struct rds_tcp_incoming),
  					0, 0, NULL);
8690bfa17   Andy Grover   RDS: cleanup: rem...
339
  	if (!rds_tcp_incoming_slab)
70041088e   Andy Grover   RDS: Add TCP tran...
340
341
342
343
344
345
346
347
  		return -ENOMEM;
  	return 0;
  }
  
  void rds_tcp_recv_exit(void)
  {
  	kmem_cache_destroy(rds_tcp_incoming_slab);
  }