Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
freebsd
GitHub Repository: freebsd/freebsd-src
Path: blob/main/sys/rpc/clnt_bck.c
39477 views
1
/* $NetBSD: clnt_vc.c,v 1.4 2000/07/14 08:40:42 fvdl Exp $ */
2
3
/*-
4
* Copyright (c) 2009, Sun Microsystems, Inc.
5
* All rights reserved.
6
*
7
* Redistribution and use in source and binary forms, with or without
8
* modification, are permitted provided that the following conditions are met:
9
* - Redistributions of source code must retain the above copyright notice,
10
* this list of conditions and the following disclaimer.
11
* - Redistributions in binary form must reproduce the above copyright notice,
12
* this list of conditions and the following disclaimer in the documentation
13
* and/or other materials provided with the distribution.
14
* - Neither the name of Sun Microsystems, Inc. nor the names of its
15
* contributors may be used to endorse or promote products derived
16
* from this software without specific prior written permission.
17
*
18
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
22
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28
* POSSIBILITY OF SUCH DAMAGE.
29
*/
30
31
#include <sys/cdefs.h>
32
/*
33
* clnt_tcp.c, Implements a TCP/IP based, client side RPC.
34
*
35
* Copyright (C) 1984, Sun Microsystems, Inc.
36
*
37
* TCP based RPC supports 'batched calls'.
38
* A sequence of calls may be batched-up in a send buffer. The rpc call
39
* return immediately to the client even though the call was not necessarily
40
* sent. The batching occurs if the results' xdr routine is NULL (0) AND
41
* the rpc timeout value is zero (see clnt.h, rpc).
42
*
43
* Clients should NOT casually batch calls that in fact return results; that is,
44
* the server side should be aware that a call is batched and not produce any
45
* return message. Batched calls that produce many result messages can
46
* deadlock (netlock) the client and the server....
47
*
48
* Now go hang yourself.
49
*/
50
51
/*
52
* This code handles the special case of a NFSv4.n backchannel for
53
* callback RPCs. It is similar to clnt_vc.c, but uses the TCP
54
* connection provided by the client to the server.
55
*/
56
57
#include "opt_kern_tls.h"
58
59
#include <sys/param.h>
60
#include <sys/systm.h>
61
#include <sys/ktls.h>
62
#include <sys/lock.h>
63
#include <sys/malloc.h>
64
#include <sys/mbuf.h>
65
#include <sys/mutex.h>
66
#include <sys/pcpu.h>
67
#include <sys/proc.h>
68
#include <sys/protosw.h>
69
#include <sys/socket.h>
70
#include <sys/socketvar.h>
71
#include <sys/sx.h>
72
#include <sys/syslog.h>
73
#include <sys/time.h>
74
#include <sys/uio.h>
75
76
#include <net/vnet.h>
77
78
#include <netinet/tcp.h>
79
80
#include <rpc/rpc.h>
81
#include <rpc/rpc_com.h>
82
#include <rpc/krpc.h>
83
#include <rpc/rpcsec_tls.h>
84
85
struct cmessage {
86
struct cmsghdr cmsg;
87
struct cmsgcred cmcred;
88
};
89
90
static void clnt_bck_geterr(CLIENT *, struct rpc_err *);
91
static bool_t clnt_bck_freeres(CLIENT *, xdrproc_t, void *);
92
static void clnt_bck_abort(CLIENT *);
93
static bool_t clnt_bck_control(CLIENT *, u_int, void *);
94
static void clnt_bck_close(CLIENT *);
95
static void clnt_bck_destroy(CLIENT *);
96
97
static const struct clnt_ops clnt_bck_ops = {
98
.cl_abort = clnt_bck_abort,
99
.cl_geterr = clnt_bck_geterr,
100
.cl_freeres = clnt_bck_freeres,
101
.cl_close = clnt_bck_close,
102
.cl_destroy = clnt_bck_destroy,
103
.cl_control = clnt_bck_control
104
};
105
106
/*
107
* Create a client handle for a connection.
108
* Default options are set, which the user can change using clnt_control()'s.
109
* This code handles the special case of an NFSv4.1 session backchannel
110
* call, which is sent on a TCP connection created against the server
111
* by a client.
112
*/
113
void *
114
clnt_bck_create(
115
struct socket *so, /* Server transport socket. */
116
const rpcprog_t prog, /* program number */
117
const rpcvers_t vers) /* version number */
118
{
119
CLIENT *cl; /* client handle */
120
struct ct_data *ct = NULL; /* client handle */
121
struct timeval now;
122
struct rpc_msg call_msg;
123
static uint32_t disrupt;
124
XDR xdrs;
125
126
if (disrupt == 0)
127
disrupt = (uint32_t)(long)so;
128
129
cl = (CLIENT *)mem_alloc(sizeof (*cl));
130
ct = (struct ct_data *)mem_alloc(sizeof (*ct));
131
132
mtx_init(&ct->ct_lock, "ct->ct_lock", NULL, MTX_DEF);
133
ct->ct_threads = 0;
134
ct->ct_closing = FALSE;
135
ct->ct_closed = FALSE;
136
ct->ct_upcallrefs = 0;
137
ct->ct_closeit = FALSE;
138
139
/*
140
* Set up private data struct
141
*/
142
ct->ct_wait.tv_sec = -1;
143
ct->ct_wait.tv_usec = -1;
144
145
/*
146
* Initialize call message
147
*/
148
getmicrotime(&now);
149
ct->ct_xid = ((uint32_t)++disrupt) ^ __RPC_GETXID(&now);
150
call_msg.rm_xid = ct->ct_xid;
151
call_msg.rm_direction = CALL;
152
call_msg.rm_call.cb_rpcvers = RPC_MSG_VERSION;
153
call_msg.rm_call.cb_prog = (uint32_t)prog;
154
call_msg.rm_call.cb_vers = (uint32_t)vers;
155
156
/*
157
* pre-serialize the static part of the call msg and stash it away
158
*/
159
xdrmem_create(&xdrs, ct->ct_mcallc, MCALL_MSG_SIZE,
160
XDR_ENCODE);
161
if (!xdr_callhdr(&xdrs, &call_msg))
162
goto err;
163
ct->ct_mpos = XDR_GETPOS(&xdrs);
164
XDR_DESTROY(&xdrs);
165
ct->ct_waitchan = "rpcbck";
166
ct->ct_waitflag = 0;
167
cl->cl_refs = 1;
168
cl->cl_ops = &clnt_bck_ops;
169
cl->cl_private = ct;
170
cl->cl_auth = authnone_create();
171
TAILQ_INIT(&ct->ct_pending);
172
return (cl);
173
174
err:
175
mtx_destroy(&ct->ct_lock);
176
mem_free(ct, sizeof (struct ct_data));
177
mem_free(cl, sizeof (CLIENT));
178
return (NULL);
179
}
180
181
enum clnt_stat
182
clnt_bck_call(
183
CLIENT *cl, /* client handle */
184
struct rpc_callextra *ext, /* call metadata */
185
rpcproc_t proc, /* procedure number */
186
struct mbuf *args, /* pointer to args */
187
struct mbuf **resultsp, /* pointer to results */
188
struct timeval utimeout,
189
SVCXPRT *xprt)
190
{
191
struct ct_data *ct = (struct ct_data *) cl->cl_private;
192
AUTH *auth;
193
struct rpc_err *errp;
194
enum clnt_stat stat;
195
XDR xdrs;
196
struct rpc_msg reply_msg;
197
bool_t ok;
198
int nrefreshes = 2; /* number of times to refresh cred */
199
struct timeval timeout;
200
uint32_t xid;
201
struct mbuf *mreq = NULL, *results;
202
struct ct_request *cr;
203
int error, maxextsiz;
204
#ifdef KERN_TLS
205
u_int maxlen;
206
#endif
207
208
cr = malloc(sizeof(struct ct_request), M_RPC, M_WAITOK);
209
210
mtx_lock(&ct->ct_lock);
211
212
if (ct->ct_closing || ct->ct_closed) {
213
mtx_unlock(&ct->ct_lock);
214
free(cr, M_RPC);
215
return (RPC_CANTSEND);
216
}
217
ct->ct_threads++;
218
219
if (ext) {
220
auth = ext->rc_auth;
221
errp = &ext->rc_err;
222
} else {
223
auth = cl->cl_auth;
224
errp = &ct->ct_error;
225
}
226
227
cr->cr_mrep = NULL;
228
cr->cr_error = 0;
229
230
if (ct->ct_wait.tv_usec == -1)
231
timeout = utimeout; /* use supplied timeout */
232
else
233
timeout = ct->ct_wait; /* use default timeout */
234
235
call_again:
236
mtx_assert(&ct->ct_lock, MA_OWNED);
237
238
ct->ct_xid++;
239
xid = ct->ct_xid;
240
241
mtx_unlock(&ct->ct_lock);
242
243
/*
244
* Leave space to pre-pend the record mark.
245
*/
246
mreq = m_gethdr(M_WAITOK, MT_DATA);
247
mreq->m_data += sizeof(uint32_t);
248
KASSERT(ct->ct_mpos + sizeof(uint32_t) <= MHLEN,
249
("RPC header too big"));
250
bcopy(ct->ct_mcallc, mreq->m_data, ct->ct_mpos);
251
mreq->m_len = ct->ct_mpos;
252
253
/*
254
* The XID is the first thing in the request.
255
*/
256
*mtod(mreq, uint32_t *) = htonl(xid);
257
258
xdrmbuf_create(&xdrs, mreq, XDR_ENCODE);
259
260
errp->re_status = stat = RPC_SUCCESS;
261
262
if ((!XDR_PUTINT32(&xdrs, &proc)) ||
263
(!AUTH_MARSHALL(auth, xid, &xdrs,
264
m_copym(args, 0, M_COPYALL, M_WAITOK)))) {
265
errp->re_status = stat = RPC_CANTENCODEARGS;
266
mtx_lock(&ct->ct_lock);
267
goto out;
268
}
269
mreq->m_pkthdr.len = m_length(mreq, NULL);
270
271
/*
272
* Prepend a record marker containing the packet length.
273
*/
274
M_PREPEND(mreq, sizeof(uint32_t), M_WAITOK);
275
*mtod(mreq, uint32_t *) =
276
htonl(0x80000000 | (mreq->m_pkthdr.len - sizeof(uint32_t)));
277
278
cr->cr_xid = xid;
279
mtx_lock(&ct->ct_lock);
280
/*
281
* Check to see if the client end has already started to close down
282
* the connection. The svc code will have set ct_error.re_status
283
* to RPC_CANTRECV if this is the case.
284
* If the client starts to close down the connection after this
285
* point, it will be detected later when cr_error is checked,
286
* since the request is in the ct_pending queue.
287
*/
288
if (ct->ct_error.re_status == RPC_CANTRECV) {
289
if (errp != &ct->ct_error) {
290
errp->re_errno = ct->ct_error.re_errno;
291
errp->re_status = RPC_CANTRECV;
292
}
293
stat = RPC_CANTRECV;
294
goto out;
295
}
296
TAILQ_INSERT_TAIL(&ct->ct_pending, cr, cr_link);
297
mtx_unlock(&ct->ct_lock);
298
299
/* For RPC-over-TLS, copy mrep to a chain of ext_pgs. */
300
if ((xprt->xp_tls & RPCTLS_FLAGS_HANDSHAKE) != 0) {
301
/*
302
* Copy the mbuf chain to a chain of
303
* ext_pgs mbuf(s) as required by KERN_TLS.
304
*/
305
maxextsiz = TLS_MAX_MSG_SIZE_V10_2;
306
#ifdef KERN_TLS
307
if (rpctls_getinfo(&maxlen, false, false))
308
maxextsiz = min(maxextsiz, maxlen);
309
#endif
310
mreq = _rpc_copym_into_ext_pgs(mreq, maxextsiz);
311
}
312
/*
313
* sosend consumes mreq.
314
*/
315
sx_xlock(&xprt->xp_lock);
316
error = sosend(xprt->xp_socket, NULL, NULL, mreq, NULL, 0, curthread);
317
if (error != 0) printf("sosend=%d\n", error);
318
mreq = NULL;
319
if (error == EMSGSIZE) {
320
printf("emsgsize\n");
321
SOCK_SENDBUF_LOCK(xprt->xp_socket);
322
sbwait(xprt->xp_socket, SO_SND);
323
SOCK_SENDBUF_UNLOCK(xprt->xp_socket);
324
sx_xunlock(&xprt->xp_lock);
325
AUTH_VALIDATE(auth, xid, NULL, NULL);
326
mtx_lock(&ct->ct_lock);
327
TAILQ_REMOVE(&ct->ct_pending, cr, cr_link);
328
goto call_again;
329
}
330
sx_xunlock(&xprt->xp_lock);
331
332
reply_msg.acpted_rply.ar_verf.oa_flavor = AUTH_NULL;
333
reply_msg.acpted_rply.ar_verf.oa_base = cr->cr_verf;
334
reply_msg.acpted_rply.ar_verf.oa_length = 0;
335
reply_msg.acpted_rply.ar_results.where = NULL;
336
reply_msg.acpted_rply.ar_results.proc = (xdrproc_t)xdr_void;
337
338
mtx_lock(&ct->ct_lock);
339
if (error) {
340
TAILQ_REMOVE(&ct->ct_pending, cr, cr_link);
341
errp->re_errno = error;
342
errp->re_status = stat = RPC_CANTSEND;
343
goto out;
344
}
345
346
/*
347
* Check to see if we got an upcall while waiting for the
348
* lock. In both these cases, the request has been removed
349
* from ct->ct_pending.
350
*/
351
if (cr->cr_error) {
352
TAILQ_REMOVE(&ct->ct_pending, cr, cr_link);
353
errp->re_errno = cr->cr_error;
354
errp->re_status = stat = RPC_CANTRECV;
355
goto out;
356
}
357
if (cr->cr_mrep) {
358
TAILQ_REMOVE(&ct->ct_pending, cr, cr_link);
359
goto got_reply;
360
}
361
362
/*
363
* Hack to provide rpc-based message passing
364
*/
365
if (timeout.tv_sec == 0 && timeout.tv_usec == 0) {
366
TAILQ_REMOVE(&ct->ct_pending, cr, cr_link);
367
errp->re_status = stat = RPC_TIMEDOUT;
368
goto out;
369
}
370
371
error = msleep(cr, &ct->ct_lock, ct->ct_waitflag, ct->ct_waitchan,
372
tvtohz(&timeout));
373
374
TAILQ_REMOVE(&ct->ct_pending, cr, cr_link);
375
376
if (error) {
377
/*
378
* The sleep returned an error so our request is still
379
* on the list. Turn the error code into an
380
* appropriate client status.
381
*/
382
errp->re_errno = error;
383
switch (error) {
384
case EINTR:
385
stat = RPC_INTR;
386
break;
387
case EWOULDBLOCK:
388
stat = RPC_TIMEDOUT;
389
break;
390
default:
391
stat = RPC_CANTRECV;
392
}
393
errp->re_status = stat;
394
goto out;
395
} else {
396
/*
397
* We were woken up by the svc thread. If the
398
* upcall had a receive error, report that,
399
* otherwise we have a reply.
400
*/
401
if (cr->cr_error) {
402
errp->re_errno = cr->cr_error;
403
errp->re_status = stat = RPC_CANTRECV;
404
goto out;
405
}
406
}
407
408
got_reply:
409
/*
410
* Now decode and validate the response. We need to drop the
411
* lock since xdr_replymsg may end up sleeping in malloc.
412
*/
413
mtx_unlock(&ct->ct_lock);
414
415
if (ext && ext->rc_feedback)
416
ext->rc_feedback(FEEDBACK_OK, proc, ext->rc_feedback_arg);
417
418
xdrmbuf_create(&xdrs, cr->cr_mrep, XDR_DECODE);
419
ok = xdr_replymsg(&xdrs, &reply_msg);
420
cr->cr_mrep = NULL;
421
422
if (ok) {
423
if ((reply_msg.rm_reply.rp_stat == MSG_ACCEPTED) &&
424
(reply_msg.acpted_rply.ar_stat == SUCCESS))
425
errp->re_status = stat = RPC_SUCCESS;
426
else
427
stat = _seterr_reply(&reply_msg, errp);
428
429
if (stat == RPC_SUCCESS) {
430
results = xdrmbuf_getall(&xdrs);
431
if (!AUTH_VALIDATE(auth, xid,
432
&reply_msg.acpted_rply.ar_verf, &results)) {
433
errp->re_status = stat = RPC_AUTHERROR;
434
errp->re_why = AUTH_INVALIDRESP;
435
} else {
436
KASSERT(results,
437
("auth validated but no result"));
438
*resultsp = results;
439
}
440
} /* end successful completion */
441
/*
442
* If unsuccessful AND error is an authentication error
443
* then refresh credentials and try again, else break
444
*/
445
else if (stat == RPC_AUTHERROR)
446
/* maybe our credentials need to be refreshed ... */
447
if (nrefreshes > 0 && AUTH_REFRESH(auth, &reply_msg)) {
448
nrefreshes--;
449
XDR_DESTROY(&xdrs);
450
mtx_lock(&ct->ct_lock);
451
goto call_again;
452
}
453
/* end of unsuccessful completion */
454
/* end of valid reply message */
455
} else
456
errp->re_status = stat = RPC_CANTDECODERES;
457
XDR_DESTROY(&xdrs);
458
mtx_lock(&ct->ct_lock);
459
out:
460
mtx_assert(&ct->ct_lock, MA_OWNED);
461
462
KASSERT(stat != RPC_SUCCESS || *resultsp,
463
("RPC_SUCCESS without reply"));
464
465
if (mreq != NULL)
466
m_freem(mreq);
467
if (cr->cr_mrep != NULL)
468
m_freem(cr->cr_mrep);
469
470
ct->ct_threads--;
471
if (ct->ct_closing)
472
wakeup(ct);
473
474
mtx_unlock(&ct->ct_lock);
475
476
if (auth && stat != RPC_SUCCESS)
477
AUTH_VALIDATE(auth, xid, NULL, NULL);
478
479
free(cr, M_RPC);
480
481
return (stat);
482
}
483
484
static void
485
clnt_bck_geterr(CLIENT *cl, struct rpc_err *errp)
486
{
487
struct ct_data *ct = (struct ct_data *) cl->cl_private;
488
489
*errp = ct->ct_error;
490
}
491
492
static bool_t
493
clnt_bck_freeres(CLIENT *cl, xdrproc_t xdr_res, void *res_ptr)
494
{
495
XDR xdrs;
496
bool_t dummy;
497
498
xdrs.x_op = XDR_FREE;
499
dummy = (*xdr_res)(&xdrs, res_ptr);
500
501
return (dummy);
502
}
503
504
/*ARGSUSED*/
505
static void
506
clnt_bck_abort(CLIENT *cl)
507
{
508
}
509
510
static bool_t
511
clnt_bck_control(CLIENT *cl, u_int request, void *info)
512
{
513
514
return (TRUE);
515
}
516
517
static void
518
clnt_bck_close(CLIENT *cl)
519
{
520
struct ct_data *ct = (struct ct_data *) cl->cl_private;
521
522
mtx_lock(&ct->ct_lock);
523
524
if (ct->ct_closed) {
525
mtx_unlock(&ct->ct_lock);
526
return;
527
}
528
529
if (ct->ct_closing) {
530
while (ct->ct_closing)
531
msleep(ct, &ct->ct_lock, 0, "rpcclose", 0);
532
KASSERT(ct->ct_closed, ("client should be closed"));
533
mtx_unlock(&ct->ct_lock);
534
return;
535
}
536
537
ct->ct_closing = FALSE;
538
ct->ct_closed = TRUE;
539
mtx_unlock(&ct->ct_lock);
540
wakeup(ct);
541
}
542
543
static void
544
clnt_bck_destroy(CLIENT *cl)
545
{
546
struct ct_data *ct = (struct ct_data *) cl->cl_private;
547
548
clnt_bck_close(cl);
549
550
mtx_destroy(&ct->ct_lock);
551
mem_free(ct, sizeof(struct ct_data));
552
if (cl->cl_netid && cl->cl_netid[0])
553
mem_free(cl->cl_netid, strlen(cl->cl_netid) +1);
554
if (cl->cl_tp && cl->cl_tp[0])
555
mem_free(cl->cl_tp, strlen(cl->cl_tp) +1);
556
mem_free(cl, sizeof(CLIENT));
557
}
558
559
/*
560
* This call is done by the svc code when a backchannel RPC reply is
561
* received.
562
* For the server end, where callback RPCs to the client are performed,
563
* xp_p2 points to the "CLIENT" and not the associated "struct ct_data"
564
* so that svc_vc_destroy() can CLNT_RELEASE() the reference count on it.
565
*/
566
void
567
clnt_bck_svccall(void *arg, struct mbuf *mrep, uint32_t xid)
568
{
569
CLIENT *cl = (CLIENT *)arg;
570
struct ct_data *ct;
571
struct ct_request *cr;
572
int foundreq;
573
574
ct = (struct ct_data *)cl->cl_private;
575
mtx_lock(&ct->ct_lock);
576
if (ct->ct_closing || ct->ct_closed) {
577
mtx_unlock(&ct->ct_lock);
578
m_freem(mrep);
579
return;
580
}
581
582
ct->ct_upcallrefs++;
583
/*
584
* See if we can match this reply to a request.
585
*/
586
foundreq = 0;
587
TAILQ_FOREACH(cr, &ct->ct_pending, cr_link) {
588
if (cr->cr_xid == xid) {
589
/*
590
* This one matches. We leave the reply mbuf list in
591
* cr->cr_mrep. Set the XID to zero so that we will
592
* ignore any duplicated replies.
593
*/
594
cr->cr_xid = 0;
595
cr->cr_mrep = mrep;
596
cr->cr_error = 0;
597
foundreq = 1;
598
wakeup(cr);
599
break;
600
}
601
}
602
603
ct->ct_upcallrefs--;
604
if (ct->ct_upcallrefs < 0)
605
panic("rpcvc svccall refcnt");
606
if (ct->ct_upcallrefs == 0)
607
wakeup(&ct->ct_upcallrefs);
608
mtx_unlock(&ct->ct_lock);
609
if (foundreq == 0)
610
m_freem(mrep);
611
}
612
613
614