blob: 99d18107421f92b4ed06113455eeb1d4500bfe46 [file] [log] [blame]
David Howells17926a72007-04-26 15:48:28 -07001/* RxRPC virtual connection handler
2 *
3 * Copyright (C) 2007 Red Hat, Inc. All Rights Reserved.
4 * Written by David Howells (dhowells@redhat.com)
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version
9 * 2 of the License, or (at your option) any later version.
10 */
11
Joe Perches9b6d5392016-06-02 12:08:52 -070012#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
13
David Howells17926a72007-04-26 15:48:28 -070014#include <linux/module.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090015#include <linux/slab.h>
David Howells17926a72007-04-26 15:48:28 -070016#include <linux/net.h>
17#include <linux/skbuff.h>
18#include <linux/crypto.h>
19#include <net/sock.h>
20#include <net/af_rxrpc.h>
21#include "ar-internal.h"
22
David Howells5873c082014-02-07 18:58:44 +000023/*
24 * Time till a connection expires after last use (in seconds).
25 */
David Howellsdad8aff2016-03-09 23:22:56 +000026unsigned int rxrpc_connection_expiry = 10 * 60;
David Howells5873c082014-02-07 18:58:44 +000027
David Howells17926a72007-04-26 15:48:28 -070028static void rxrpc_connection_reaper(struct work_struct *work);
29
30LIST_HEAD(rxrpc_connections);
31DEFINE_RWLOCK(rxrpc_connection_lock);
David Howells17926a72007-04-26 15:48:28 -070032static DECLARE_DELAYED_WORK(rxrpc_connection_reap, rxrpc_connection_reaper);
33
34/*
David Howells17926a72007-04-26 15:48:28 -070035 * allocate a new connection
36 */
37static struct rxrpc_connection *rxrpc_alloc_connection(gfp_t gfp)
38{
39 struct rxrpc_connection *conn;
40
41 _enter("");
42
43 conn = kzalloc(sizeof(struct rxrpc_connection), gfp);
44 if (conn) {
David Howells999b69f2016-06-17 15:42:35 +010045 spin_lock_init(&conn->channel_lock);
46 init_waitqueue_head(&conn->channel_wq);
David Howells17926a72007-04-26 15:48:28 -070047 INIT_WORK(&conn->processor, &rxrpc_process_connection);
David Howells999b69f2016-06-17 15:42:35 +010048 INIT_LIST_HEAD(&conn->link);
David Howells17926a72007-04-26 15:48:28 -070049 conn->calls = RB_ROOT;
50 skb_queue_head_init(&conn->rx_queue);
David Howellse0e4d822016-04-07 17:23:58 +010051 conn->security = &rxrpc_no_security;
David Howells17926a72007-04-26 15:48:28 -070052 rwlock_init(&conn->lock);
53 spin_lock_init(&conn->state_lock);
54 atomic_set(&conn->usage, 1);
55 conn->debug_id = atomic_inc_return(&rxrpc_debug_id);
David Howells999b69f2016-06-17 15:42:35 +010056 atomic_set(&conn->avail_chans, RXRPC_MAXCALLS);
David Howells17926a72007-04-26 15:48:28 -070057 conn->size_align = 4;
David Howells0d12f8a2016-03-04 15:53:46 +000058 conn->header_size = sizeof(struct rxrpc_wire_header);
David Howells17926a72007-04-26 15:48:28 -070059 }
60
Adrian Bunk16c61ad2007-06-15 15:15:43 -070061 _leave(" = %p{%d}", conn, conn ? conn->debug_id : 0);
David Howells17926a72007-04-26 15:48:28 -070062 return conn;
63}
64
65/*
David Howells17926a72007-04-26 15:48:28 -070066 * add a call to a connection's call-by-ID tree
67 */
68static void rxrpc_add_call_ID_to_conn(struct rxrpc_connection *conn,
69 struct rxrpc_call *call)
70{
71 struct rxrpc_call *xcall;
72 struct rb_node *parent, **p;
David Howells88b99d02016-07-01 08:27:42 +010073 u32 call_id;
David Howells17926a72007-04-26 15:48:28 -070074
75 write_lock_bh(&conn->lock);
76
77 call_id = call->call_id;
78 p = &conn->calls.rb_node;
79 parent = NULL;
80 while (*p) {
81 parent = *p;
82 xcall = rb_entry(parent, struct rxrpc_call, conn_node);
83
84 if (call_id < xcall->call_id)
85 p = &(*p)->rb_left;
86 else if (call_id > xcall->call_id)
87 p = &(*p)->rb_right;
88 else
89 BUG();
90 }
91
92 rb_link_node(&call->conn_node, parent, p);
93 rb_insert_color(&call->conn_node, &conn->calls);
94
95 write_unlock_bh(&conn->lock);
96}
97
98/*
David Howells999b69f2016-06-17 15:42:35 +010099 * Allocate a client connection. The caller must take care to clear any
100 * padding bytes in *cp.
David Howells4a3388c2016-04-04 14:00:37 +0100101 */
102static struct rxrpc_connection *
David Howellsaa390bb2016-06-17 10:06:56 +0100103rxrpc_alloc_client_connection(struct rxrpc_conn_parameters *cp, gfp_t gfp)
David Howells4a3388c2016-04-04 14:00:37 +0100104{
105 struct rxrpc_connection *conn;
106 int ret;
107
108 _enter("");
109
110 conn = rxrpc_alloc_connection(gfp);
111 if (!conn) {
112 _leave(" = -ENOMEM");
113 return ERR_PTR(-ENOMEM);
114 }
115
116 conn->params = *cp;
117 conn->proto.local = cp->local;
118 conn->proto.epoch = rxrpc_epoch;
119 conn->proto.cid = 0;
120 conn->proto.in_clientflag = 0;
121 conn->proto.family = cp->peer->srx.transport.family;
122 conn->out_clientflag = RXRPC_CLIENT_INITIATED;
123 conn->state = RXRPC_CONN_CLIENT;
124
125 switch (conn->proto.family) {
126 case AF_INET:
127 conn->proto.addr_size = sizeof(conn->proto.ipv4_addr);
128 conn->proto.ipv4_addr = cp->peer->srx.transport.sin.sin_addr;
129 conn->proto.port = cp->peer->srx.transport.sin.sin_port;
130 break;
131 }
132
David Howells999b69f2016-06-17 15:42:35 +0100133 ret = rxrpc_get_client_connection_id(conn, gfp);
David Howells4a3388c2016-04-04 14:00:37 +0100134 if (ret < 0)
135 goto error_0;
136
137 ret = rxrpc_init_client_conn_security(conn);
138 if (ret < 0)
139 goto error_1;
140
Herbert Xua2636292016-06-26 14:55:24 -0700141 ret = conn->security->prime_packet_security(conn);
142 if (ret < 0)
143 goto error_2;
David Howells4a3388c2016-04-04 14:00:37 +0100144
145 write_lock(&rxrpc_connection_lock);
146 list_add_tail(&conn->link, &rxrpc_connections);
147 write_unlock(&rxrpc_connection_lock);
148
David Howellsaa390bb2016-06-17 10:06:56 +0100149 /* We steal the caller's peer ref. */
150 cp->peer = NULL;
151 rxrpc_get_local(conn->params.local);
David Howells4a3388c2016-04-04 14:00:37 +0100152 key_get(conn->params.key);
153
154 _leave(" = %p", conn);
155 return conn;
156
Herbert Xua2636292016-06-26 14:55:24 -0700157error_2:
158 conn->security->clear(conn);
David Howells4a3388c2016-04-04 14:00:37 +0100159error_1:
160 rxrpc_put_client_connection_id(conn);
161error_0:
162 kfree(conn);
163 _leave(" = %d", ret);
164 return ERR_PTR(ret);
165}
166
167/*
David Howells17926a72007-04-26 15:48:28 -0700168 * find a connection for a call
169 * - called in process context with IRQs enabled
170 */
David Howells999b69f2016-06-17 15:42:35 +0100171int rxrpc_connect_call(struct rxrpc_call *call,
David Howells19ffa012016-04-04 14:00:36 +0100172 struct rxrpc_conn_parameters *cp,
David Howells999b69f2016-06-17 15:42:35 +0100173 struct sockaddr_rxrpc *srx,
David Howells17926a72007-04-26 15:48:28 -0700174 gfp_t gfp)
175{
David Howells999b69f2016-06-17 15:42:35 +0100176 struct rxrpc_connection *conn, *candidate = NULL;
177 struct rxrpc_local *local = cp->local;
178 struct rb_node *p, **pp, *parent;
179 long diff;
David Howells4a3388c2016-04-04 14:00:37 +0100180 int chan;
David Howells17926a72007-04-26 15:48:28 -0700181
182 DECLARE_WAITQUEUE(myself, current);
183
David Howells999b69f2016-06-17 15:42:35 +0100184 _enter("{%d,%lx},", call->debug_id, call->user_call_ID);
David Howells17926a72007-04-26 15:48:28 -0700185
David Howellsaa390bb2016-06-17 10:06:56 +0100186 cp->peer = rxrpc_lookup_peer(cp->local, srx, gfp);
187 if (!cp->peer)
188 return -ENOMEM;
David Howells17926a72007-04-26 15:48:28 -0700189
David Howells999b69f2016-06-17 15:42:35 +0100190 if (!cp->exclusive) {
191 /* Search for a existing client connection unless this is going
192 * to be a connection that's used exclusively for a single call.
193 */
194 _debug("search 1");
195 spin_lock(&local->client_conns_lock);
196 p = local->client_conns.rb_node;
197 while (p) {
198 conn = rb_entry(p, struct rxrpc_connection, client_node);
199
200#define cmp(X) ((long)conn->params.X - (long)cp->X)
201 diff = (cmp(peer) ?:
202 cmp(key) ?:
203 cmp(security_level));
204 if (diff < 0)
205 p = p->rb_left;
206 else if (diff > 0)
207 p = p->rb_right;
208 else
209 goto found_extant_conn;
David Howells17926a72007-04-26 15:48:28 -0700210 }
David Howells999b69f2016-06-17 15:42:35 +0100211 spin_unlock(&local->client_conns_lock);
David Howells17926a72007-04-26 15:48:28 -0700212 }
213
David Howells999b69f2016-06-17 15:42:35 +0100214 /* We didn't find a connection or we want an exclusive one. */
215 _debug("get new conn");
David Howellsaa390bb2016-06-17 10:06:56 +0100216 candidate = rxrpc_alloc_client_connection(cp, gfp);
David Howells999b69f2016-06-17 15:42:35 +0100217 if (!candidate) {
218 _leave(" = -ENOMEM");
219 return -ENOMEM;
220 }
221
222 if (cp->exclusive) {
223 /* Assign the call on an exclusive connection to channel 0 and
224 * don't add the connection to the endpoint's shareable conn
225 * lookup tree.
226 */
227 _debug("exclusive chan 0");
228 conn = candidate;
229 atomic_set(&conn->avail_chans, RXRPC_MAXCALLS - 1);
230 spin_lock(&conn->channel_lock);
231 chan = 0;
232 goto found_channel;
233 }
234
235 /* We need to redo the search before attempting to add a new connection
236 * lest we race with someone else adding a conflicting instance.
David Howells17926a72007-04-26 15:48:28 -0700237 */
David Howells999b69f2016-06-17 15:42:35 +0100238 _debug("search 2");
239 spin_lock(&local->client_conns_lock);
240
241 pp = &local->client_conns.rb_node;
242 parent = NULL;
243 while (*pp) {
244 parent = *pp;
245 conn = rb_entry(parent, struct rxrpc_connection, client_node);
246
247 diff = (cmp(peer) ?:
248 cmp(key) ?:
249 cmp(security_level));
250 if (diff < 0)
251 pp = &(*pp)->rb_left;
252 else if (diff > 0)
253 pp = &(*pp)->rb_right;
254 else
255 goto found_extant_conn;
256 }
257
258 /* The second search also failed; simply add the new connection with
259 * the new call in channel 0. Note that we need to take the channel
260 * lock before dropping the client conn lock.
261 */
262 _debug("new conn");
263 conn = candidate;
264 candidate = NULL;
265
266 rb_link_node(&conn->client_node, parent, pp);
267 rb_insert_color(&conn->client_node, &local->client_conns);
268
269 atomic_set(&conn->avail_chans, RXRPC_MAXCALLS - 1);
270 spin_lock(&conn->channel_lock);
271 spin_unlock(&local->client_conns_lock);
272 chan = 0;
273
274found_channel:
275 _debug("found chan");
276 call->conn = conn;
277 call->channel = chan;
278 call->epoch = conn->proto.epoch;
279 call->cid = conn->proto.cid | chan;
280 call->call_id = ++conn->call_counter;
281 rcu_assign_pointer(conn->channels[chan], call);
282
283 _net("CONNECT call %d on conn %d", call->debug_id, conn->debug_id);
284
285 rxrpc_add_call_ID_to_conn(conn, call);
286 spin_unlock(&conn->channel_lock);
David Howellsaa390bb2016-06-17 10:06:56 +0100287 rxrpc_put_peer(cp->peer);
288 cp->peer = NULL;
David Howells999b69f2016-06-17 15:42:35 +0100289 _leave(" = %p {u=%d}", conn, atomic_read(&conn->usage));
290 return 0;
291
292 /* We found a suitable connection already in existence. Discard any
293 * candidate we may have allocated, and try to get a channel on this
294 * one.
295 */
296found_extant_conn:
297 _debug("found conn");
298 rxrpc_get_connection(conn);
299 spin_unlock(&local->client_conns_lock);
300
301 rxrpc_put_connection(candidate);
302
303 if (!atomic_add_unless(&conn->avail_chans, -1, 0)) {
304 if (!gfpflags_allow_blocking(gfp)) {
305 rxrpc_put_connection(conn);
306 _leave(" = -EAGAIN");
307 return -EAGAIN;
308 }
309
310 add_wait_queue(&conn->channel_wq, &myself);
311 for (;;) {
312 set_current_state(TASK_INTERRUPTIBLE);
313 if (atomic_add_unless(&conn->avail_chans, -1, 0))
314 break;
315 if (signal_pending(current))
316 goto interrupted;
317 schedule();
318 }
319 remove_wait_queue(&conn->channel_wq, &myself);
320 __set_current_state(TASK_RUNNING);
321 }
322
323 /* The connection allegedly now has a free channel and we can now
324 * attach the call to it.
325 */
326 spin_lock(&conn->channel_lock);
327
David Howells17926a72007-04-26 15:48:28 -0700328 for (chan = 0; chan < RXRPC_MAXCALLS; chan++)
329 if (!conn->channels[chan])
330 goto found_channel;
331 BUG();
332
David Howells17926a72007-04-26 15:48:28 -0700333interrupted:
David Howells999b69f2016-06-17 15:42:35 +0100334 remove_wait_queue(&conn->channel_wq, &myself);
335 __set_current_state(TASK_RUNNING);
336 rxrpc_put_connection(conn);
David Howellsaa390bb2016-06-17 10:06:56 +0100337 rxrpc_put_peer(cp->peer);
338 cp->peer = NULL;
David Howells17926a72007-04-26 15:48:28 -0700339 _leave(" = -ERESTARTSYS");
340 return -ERESTARTSYS;
341}
342
343/*
344 * get a record of an incoming connection
345 */
David Howellsaa390bb2016-06-17 10:06:56 +0100346struct rxrpc_connection *rxrpc_incoming_connection(struct rxrpc_local *local,
347 struct rxrpc_peer *peer,
David Howells999b69f2016-06-17 15:42:35 +0100348 struct sk_buff *skb)
David Howells17926a72007-04-26 15:48:28 -0700349{
350 struct rxrpc_connection *conn, *candidate = NULL;
David Howells42886ff2016-06-16 13:31:07 +0100351 struct rxrpc_skb_priv *sp = rxrpc_skb(skb);
David Howells17926a72007-04-26 15:48:28 -0700352 struct rb_node *p, **pp;
353 const char *new = "old";
David Howells88b99d02016-07-01 08:27:42 +0100354 u32 epoch, cid;
David Howells17926a72007-04-26 15:48:28 -0700355
356 _enter("");
357
David Howells42886ff2016-06-16 13:31:07 +0100358 ASSERT(sp->hdr.flags & RXRPC_CLIENT_INITIATED);
David Howells17926a72007-04-26 15:48:28 -0700359
David Howells42886ff2016-06-16 13:31:07 +0100360 epoch = sp->hdr.epoch;
361 cid = sp->hdr.cid & RXRPC_CIDMASK;
David Howells17926a72007-04-26 15:48:28 -0700362
363 /* search the connection list first */
David Howellsaa390bb2016-06-17 10:06:56 +0100364 read_lock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700365
David Howellsaa390bb2016-06-17 10:06:56 +0100366 p = peer->service_conns.rb_node;
David Howells17926a72007-04-26 15:48:28 -0700367 while (p) {
David Howells999b69f2016-06-17 15:42:35 +0100368 conn = rb_entry(p, struct rxrpc_connection, service_node);
David Howells17926a72007-04-26 15:48:28 -0700369
David Howells19ffa012016-04-04 14:00:36 +0100370 _debug("maybe %x", conn->proto.cid);
David Howells17926a72007-04-26 15:48:28 -0700371
David Howells19ffa012016-04-04 14:00:36 +0100372 if (epoch < conn->proto.epoch)
David Howells17926a72007-04-26 15:48:28 -0700373 p = p->rb_left;
David Howells19ffa012016-04-04 14:00:36 +0100374 else if (epoch > conn->proto.epoch)
David Howells17926a72007-04-26 15:48:28 -0700375 p = p->rb_right;
David Howells19ffa012016-04-04 14:00:36 +0100376 else if (cid < conn->proto.cid)
David Howells17926a72007-04-26 15:48:28 -0700377 p = p->rb_left;
David Howells19ffa012016-04-04 14:00:36 +0100378 else if (cid > conn->proto.cid)
David Howells17926a72007-04-26 15:48:28 -0700379 p = p->rb_right;
380 else
381 goto found_extant_connection;
382 }
David Howellsaa390bb2016-06-17 10:06:56 +0100383 read_unlock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700384
385 /* not yet present - create a candidate for a new record and then
386 * redo the search */
David Howells843099c2016-04-07 17:23:37 +0100387 candidate = rxrpc_alloc_connection(GFP_NOIO);
David Howells17926a72007-04-26 15:48:28 -0700388 if (!candidate) {
389 _leave(" = -ENOMEM");
390 return ERR_PTR(-ENOMEM);
391 }
392
David Howellsaa390bb2016-06-17 10:06:56 +0100393 candidate->proto.local = local;
David Howells42886ff2016-06-16 13:31:07 +0100394 candidate->proto.epoch = sp->hdr.epoch;
395 candidate->proto.cid = sp->hdr.cid & RXRPC_CIDMASK;
396 candidate->proto.in_clientflag = RXRPC_CLIENT_INITIATED;
David Howellsaa390bb2016-06-17 10:06:56 +0100397 candidate->params.local = local;
398 candidate->params.peer = peer;
David Howells42886ff2016-06-16 13:31:07 +0100399 candidate->params.service_id = sp->hdr.serviceId;
400 candidate->security_ix = sp->hdr.securityIndex;
401 candidate->out_clientflag = 0;
David Howellsbba304d2016-06-27 10:32:02 +0100402 candidate->state = RXRPC_CONN_SERVICE;
David Howells19ffa012016-04-04 14:00:36 +0100403 if (candidate->params.service_id)
David Howellsbba304d2016-06-27 10:32:02 +0100404 candidate->state = RXRPC_CONN_SERVICE_UNSECURED;
David Howells17926a72007-04-26 15:48:28 -0700405
David Howellsaa390bb2016-06-17 10:06:56 +0100406 write_lock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700407
David Howellsaa390bb2016-06-17 10:06:56 +0100408 pp = &peer->service_conns.rb_node;
David Howells17926a72007-04-26 15:48:28 -0700409 p = NULL;
410 while (*pp) {
411 p = *pp;
David Howells999b69f2016-06-17 15:42:35 +0100412 conn = rb_entry(p, struct rxrpc_connection, service_node);
David Howells17926a72007-04-26 15:48:28 -0700413
David Howells19ffa012016-04-04 14:00:36 +0100414 if (epoch < conn->proto.epoch)
David Howells17926a72007-04-26 15:48:28 -0700415 pp = &(*pp)->rb_left;
David Howells19ffa012016-04-04 14:00:36 +0100416 else if (epoch > conn->proto.epoch)
David Howells17926a72007-04-26 15:48:28 -0700417 pp = &(*pp)->rb_right;
David Howells19ffa012016-04-04 14:00:36 +0100418 else if (cid < conn->proto.cid)
David Howells17926a72007-04-26 15:48:28 -0700419 pp = &(*pp)->rb_left;
David Howells19ffa012016-04-04 14:00:36 +0100420 else if (cid > conn->proto.cid)
David Howells17926a72007-04-26 15:48:28 -0700421 pp = &(*pp)->rb_right;
422 else
423 goto found_extant_second;
424 }
425
426 /* we can now add the new candidate to the list */
427 conn = candidate;
428 candidate = NULL;
David Howells999b69f2016-06-17 15:42:35 +0100429 rb_link_node(&conn->service_node, p, pp);
David Howellsaa390bb2016-06-17 10:06:56 +0100430 rb_insert_color(&conn->service_node, &peer->service_conns);
431 rxrpc_get_peer(peer);
432 rxrpc_get_local(local);
David Howells17926a72007-04-26 15:48:28 -0700433
David Howellsaa390bb2016-06-17 10:06:56 +0100434 write_unlock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700435
David Howellsb3f57502016-06-21 16:10:03 +0100436 write_lock(&rxrpc_connection_lock);
David Howells17926a72007-04-26 15:48:28 -0700437 list_add_tail(&conn->link, &rxrpc_connections);
David Howellsb3f57502016-06-21 16:10:03 +0100438 write_unlock(&rxrpc_connection_lock);
David Howells17926a72007-04-26 15:48:28 -0700439
440 new = "new";
441
442success:
David Howells19ffa012016-04-04 14:00:36 +0100443 _net("CONNECTION %s %d {%x}", new, conn->debug_id, conn->proto.cid);
David Howells17926a72007-04-26 15:48:28 -0700444
445 _leave(" = %p {u=%d}", conn, atomic_read(&conn->usage));
446 return conn;
447
448 /* we found the connection in the list immediately */
449found_extant_connection:
David Howells42886ff2016-06-16 13:31:07 +0100450 if (sp->hdr.securityIndex != conn->security_ix) {
David Howellsaa390bb2016-06-17 10:06:56 +0100451 read_unlock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700452 goto security_mismatch;
453 }
David Howells5627cc82016-04-04 14:00:38 +0100454 rxrpc_get_connection(conn);
David Howellsaa390bb2016-06-17 10:06:56 +0100455 read_unlock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700456 goto success;
457
458 /* we found the connection on the second time through the list */
459found_extant_second:
David Howells42886ff2016-06-16 13:31:07 +0100460 if (sp->hdr.securityIndex != conn->security_ix) {
David Howellsaa390bb2016-06-17 10:06:56 +0100461 write_unlock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700462 goto security_mismatch;
463 }
David Howells5627cc82016-04-04 14:00:38 +0100464 rxrpc_get_connection(conn);
David Howellsaa390bb2016-06-17 10:06:56 +0100465 write_unlock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700466 kfree(candidate);
467 goto success;
468
469security_mismatch:
470 kfree(candidate);
471 _leave(" = -EKEYREJECTED");
472 return ERR_PTR(-EKEYREJECTED);
473}
474
475/*
476 * find a connection based on transport and RxRPC connection ID for an incoming
477 * packet
478 */
David Howellsaa390bb2016-06-17 10:06:56 +0100479struct rxrpc_connection *rxrpc_find_connection(struct rxrpc_local *local,
480 struct rxrpc_peer *peer,
David Howells42886ff2016-06-16 13:31:07 +0100481 struct sk_buff *skb)
David Howells17926a72007-04-26 15:48:28 -0700482{
483 struct rxrpc_connection *conn;
David Howells42886ff2016-06-16 13:31:07 +0100484 struct rxrpc_skb_priv *sp = rxrpc_skb(skb);
David Howells17926a72007-04-26 15:48:28 -0700485 struct rb_node *p;
David Howells0d12f8a2016-03-04 15:53:46 +0000486 u32 epoch, cid;
David Howells17926a72007-04-26 15:48:28 -0700487
David Howells42886ff2016-06-16 13:31:07 +0100488 _enter(",{%x,%x}", sp->hdr.cid, sp->hdr.flags);
David Howells17926a72007-04-26 15:48:28 -0700489
David Howellsaa390bb2016-06-17 10:06:56 +0100490 read_lock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700491
David Howells42886ff2016-06-16 13:31:07 +0100492 cid = sp->hdr.cid & RXRPC_CIDMASK;
493 epoch = sp->hdr.epoch;
David Howells17926a72007-04-26 15:48:28 -0700494
David Howells4a3388c2016-04-04 14:00:37 +0100495 if (sp->hdr.flags & RXRPC_CLIENT_INITIATED) {
David Howellsaa390bb2016-06-17 10:06:56 +0100496 p = peer->service_conns.rb_node;
David Howells4a3388c2016-04-04 14:00:37 +0100497 while (p) {
David Howells999b69f2016-06-17 15:42:35 +0100498 conn = rb_entry(p, struct rxrpc_connection, service_node);
David Howells17926a72007-04-26 15:48:28 -0700499
David Howells4a3388c2016-04-04 14:00:37 +0100500 _debug("maybe %x", conn->proto.cid);
David Howells17926a72007-04-26 15:48:28 -0700501
David Howells4a3388c2016-04-04 14:00:37 +0100502 if (epoch < conn->proto.epoch)
503 p = p->rb_left;
504 else if (epoch > conn->proto.epoch)
505 p = p->rb_right;
506 else if (cid < conn->proto.cid)
507 p = p->rb_left;
508 else if (cid > conn->proto.cid)
509 p = p->rb_right;
510 else
511 goto found;
512 }
513 } else {
514 conn = idr_find(&rxrpc_client_conn_ids, cid >> RXRPC_CIDSHIFT);
David Howells689f4c62016-06-30 11:34:30 +0100515 if (conn &&
516 conn->proto.epoch == epoch &&
517 conn->params.peer == peer)
David Howells17926a72007-04-26 15:48:28 -0700518 goto found;
519 }
520
David Howellsaa390bb2016-06-17 10:06:56 +0100521 read_unlock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700522 _leave(" = NULL");
523 return NULL;
524
525found:
David Howells5627cc82016-04-04 14:00:38 +0100526 rxrpc_get_connection(conn);
David Howellsaa390bb2016-06-17 10:06:56 +0100527 read_unlock_bh(&peer->conn_lock);
David Howells17926a72007-04-26 15:48:28 -0700528 _leave(" = %p", conn);
529 return conn;
530}
531
532/*
David Howells999b69f2016-06-17 15:42:35 +0100533 * Disconnect a call and clear any channel it occupies when that call
534 * terminates.
535 */
536void rxrpc_disconnect_call(struct rxrpc_call *call)
537{
538 struct rxrpc_connection *conn = call->conn;
539 unsigned chan = call->channel;
540
541 _enter("%d,%d", conn->debug_id, call->channel);
542
David Howellse653cfe2016-04-04 14:00:38 +0100543 spin_lock(&conn->channel_lock);
544
David Howells999b69f2016-06-17 15:42:35 +0100545 if (conn->channels[chan] == call) {
546 rcu_assign_pointer(conn->channels[chan], NULL);
547 atomic_inc(&conn->avail_chans);
548 wake_up(&conn->channel_wq);
549 }
David Howellse653cfe2016-04-04 14:00:38 +0100550
551 spin_unlock(&conn->channel_lock);
552
553 call->conn = NULL;
554 rxrpc_put_connection(conn);
555 _leave("");
David Howells999b69f2016-06-17 15:42:35 +0100556}
557
558/*
David Howells17926a72007-04-26 15:48:28 -0700559 * release a virtual connection
560 */
561void rxrpc_put_connection(struct rxrpc_connection *conn)
562{
David Howells999b69f2016-06-17 15:42:35 +0100563 if (!conn)
564 return;
565
David Howells17926a72007-04-26 15:48:28 -0700566 _enter("%p{u=%d,d=%d}",
567 conn, atomic_read(&conn->usage), conn->debug_id);
568
569 ASSERTCMP(atomic_read(&conn->usage), >, 0);
570
Ksenija Stanojevic22a3f9a2015-09-17 18:12:53 +0200571 conn->put_time = ktime_get_seconds();
David Howells17926a72007-04-26 15:48:28 -0700572 if (atomic_dec_and_test(&conn->usage)) {
573 _debug("zombie");
David Howells651350d2007-04-26 15:50:17 -0700574 rxrpc_queue_delayed_work(&rxrpc_connection_reap, 0);
David Howells17926a72007-04-26 15:48:28 -0700575 }
576
577 _leave("");
578}
579
580/*
581 * destroy a virtual connection
582 */
583static void rxrpc_destroy_connection(struct rxrpc_connection *conn)
584{
585 _enter("%p{%d}", conn, atomic_read(&conn->usage));
586
587 ASSERTCMP(atomic_read(&conn->usage), ==, 0);
588
589 _net("DESTROY CONN %d", conn->debug_id);
590
David Howells17926a72007-04-26 15:48:28 -0700591 ASSERT(RB_EMPTY_ROOT(&conn->calls));
592 rxrpc_purge_queue(&conn->rx_queue);
593
David Howellse0e4d822016-04-07 17:23:58 +0100594 conn->security->clear(conn);
David Howells19ffa012016-04-04 14:00:36 +0100595 key_put(conn->params.key);
David Howellse0e4d822016-04-07 17:23:58 +0100596 key_put(conn->server_key);
David Howellsaa390bb2016-06-17 10:06:56 +0100597 rxrpc_put_peer(conn->params.peer);
598 rxrpc_put_local(conn->params.local);
David Howellse0e4d822016-04-07 17:23:58 +0100599
David Howells17926a72007-04-26 15:48:28 -0700600 kfree(conn);
601 _leave("");
602}
603
604/*
605 * reap dead connections
606 */
Roel Kluin5eaa65b2008-12-10 15:18:31 -0800607static void rxrpc_connection_reaper(struct work_struct *work)
David Howells17926a72007-04-26 15:48:28 -0700608{
609 struct rxrpc_connection *conn, *_p;
David Howellsaa390bb2016-06-17 10:06:56 +0100610 struct rxrpc_peer *peer;
David Howells17926a72007-04-26 15:48:28 -0700611 unsigned long now, earliest, reap_time;
612
613 LIST_HEAD(graveyard);
614
615 _enter("");
616
Ksenija Stanojevic22a3f9a2015-09-17 18:12:53 +0200617 now = ktime_get_seconds();
David Howells17926a72007-04-26 15:48:28 -0700618 earliest = ULONG_MAX;
619
David Howellsb3f57502016-06-21 16:10:03 +0100620 write_lock(&rxrpc_connection_lock);
David Howells17926a72007-04-26 15:48:28 -0700621 list_for_each_entry_safe(conn, _p, &rxrpc_connections, link) {
622 _debug("reap CONN %d { u=%d,t=%ld }",
623 conn->debug_id, atomic_read(&conn->usage),
624 (long) now - (long) conn->put_time);
625
626 if (likely(atomic_read(&conn->usage) > 0))
627 continue;
628
David Howells999b69f2016-06-17 15:42:35 +0100629 if (rxrpc_conn_is_client(conn)) {
630 struct rxrpc_local *local = conn->params.local;
631 spin_lock(&local->client_conns_lock);
632 reap_time = conn->put_time + rxrpc_connection_expiry;
David Howells17926a72007-04-26 15:48:28 -0700633
David Howells999b69f2016-06-17 15:42:35 +0100634 if (atomic_read(&conn->usage) > 0) {
635 ;
636 } else if (reap_time <= now) {
637 list_move_tail(&conn->link, &graveyard);
David Howells4a3388c2016-04-04 14:00:37 +0100638 rxrpc_put_client_connection_id(conn);
David Howells999b69f2016-06-17 15:42:35 +0100639 rb_erase(&conn->client_node,
640 &local->client_conns);
641 } else if (reap_time < earliest) {
642 earliest = reap_time;
David Howells17926a72007-04-26 15:48:28 -0700643 }
644
David Howells999b69f2016-06-17 15:42:35 +0100645 spin_unlock(&local->client_conns_lock);
646 } else {
David Howellsaa390bb2016-06-17 10:06:56 +0100647 peer = conn->params.peer;
648 write_lock_bh(&peer->conn_lock);
David Howells999b69f2016-06-17 15:42:35 +0100649 reap_time = conn->put_time + rxrpc_connection_expiry;
David Howells17926a72007-04-26 15:48:28 -0700650
David Howells999b69f2016-06-17 15:42:35 +0100651 if (atomic_read(&conn->usage) > 0) {
652 ;
653 } else if (reap_time <= now) {
654 list_move_tail(&conn->link, &graveyard);
655 rb_erase(&conn->service_node,
David Howellsaa390bb2016-06-17 10:06:56 +0100656 &peer->service_conns);
David Howells999b69f2016-06-17 15:42:35 +0100657 } else if (reap_time < earliest) {
658 earliest = reap_time;
659 }
660
David Howellsaa390bb2016-06-17 10:06:56 +0100661 write_unlock_bh(&peer->conn_lock);
David Howells999b69f2016-06-17 15:42:35 +0100662 }
David Howells17926a72007-04-26 15:48:28 -0700663 }
David Howellsb3f57502016-06-21 16:10:03 +0100664 write_unlock(&rxrpc_connection_lock);
David Howells17926a72007-04-26 15:48:28 -0700665
666 if (earliest != ULONG_MAX) {
667 _debug("reschedule reaper %ld", (long) earliest - now);
668 ASSERTCMP(earliest, >, now);
David Howells651350d2007-04-26 15:50:17 -0700669 rxrpc_queue_delayed_work(&rxrpc_connection_reap,
670 (earliest - now) * HZ);
David Howells17926a72007-04-26 15:48:28 -0700671 }
672
673 /* then destroy all those pulled out */
674 while (!list_empty(&graveyard)) {
675 conn = list_entry(graveyard.next, struct rxrpc_connection,
676 link);
677 list_del_init(&conn->link);
678
679 ASSERTCMP(atomic_read(&conn->usage), ==, 0);
680 rxrpc_destroy_connection(conn);
681 }
682
683 _leave("");
684}
685
686/*
687 * preemptively destroy all the connection records rather than waiting for them
688 * to time out
689 */
690void __exit rxrpc_destroy_all_connections(void)
691{
692 _enter("");
693
David Howells5873c082014-02-07 18:58:44 +0000694 rxrpc_connection_expiry = 0;
David Howells17926a72007-04-26 15:48:28 -0700695 cancel_delayed_work(&rxrpc_connection_reap);
David Howells651350d2007-04-26 15:50:17 -0700696 rxrpc_queue_delayed_work(&rxrpc_connection_reap, 0);
David Howells17926a72007-04-26 15:48:28 -0700697
698 _leave("");
699}