1*b819cea2SGordon Ross /*
2*b819cea2SGordon Ross  * CDDL HEADER START
3*b819cea2SGordon Ross  *
4*b819cea2SGordon Ross  * The contents of this file are subject to the terms of the
5*b819cea2SGordon Ross  * Common Development and Distribution License (the "License").
6*b819cea2SGordon Ross  * You may not use this file except in compliance with the License.
7*b819cea2SGordon Ross  *
8*b819cea2SGordon Ross  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*b819cea2SGordon Ross  * or http://www.opensolaris.org/os/licensing.
10*b819cea2SGordon Ross  * See the License for the specific language governing permissions
11*b819cea2SGordon Ross  * and limitations under the License.
12*b819cea2SGordon Ross  *
13*b819cea2SGordon Ross  * When distributing Covered Code, include this CDDL HEADER in each
14*b819cea2SGordon Ross  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*b819cea2SGordon Ross  * If applicable, add the following below this CDDL HEADER, with the
16*b819cea2SGordon Ross  * fields enclosed by brackets "[]" replaced with your own identifying
17*b819cea2SGordon Ross  * information: Portions Copyright [yyyy] [name of copyright owner]
18*b819cea2SGordon Ross  *
19*b819cea2SGordon Ross  * CDDL HEADER END
20*b819cea2SGordon Ross  */
21*b819cea2SGordon Ross /*
22*b819cea2SGordon Ross  * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
23*b819cea2SGordon Ross  * Copyright 2013 Nexenta Systems, Inc.  All rights reserved.
24*b819cea2SGordon Ross  */
25*b819cea2SGordon Ross 
26*b819cea2SGordon Ross /*
27*b819cea2SGordon Ross  * "Upcall" glue for the fake (user-mode) smbsrv module.
28*b819cea2SGordon Ross  */
29*b819cea2SGordon Ross 
30*b819cea2SGordon Ross #include <sys/types.h>
31*b819cea2SGordon Ross #include <sys/kmem.h>
32*b819cea2SGordon Ross #include <sys/ddi.h>
33*b819cea2SGordon Ross #include <sys/sunddi.h>
34*b819cea2SGordon Ross #include <sys/cmn_err.h>
35*b819cea2SGordon Ross #include <sys/door.h>
36*b819cea2SGordon Ross #include <smbsrv/smb_kproto.h>
37*b819cea2SGordon Ross #include <smbsrv/smb_door.h>
38*b819cea2SGordon Ross 
39*b819cea2SGordon Ross static int smb_kdoor_encode(smb_doorarg_t *);
40*b819cea2SGordon Ross static int smb_kdoor_decode(smb_doorarg_t *);
41*b819cea2SGordon Ross static void smb_kdoor_sethdr(smb_doorarg_t *, uint32_t);
42*b819cea2SGordon Ross static boolean_t smb_kdoor_chkhdr(smb_doorarg_t *, smb_doorhdr_t *);
43*b819cea2SGordon Ross static void smb_kdoor_free(door_arg_t *);
44*b819cea2SGordon Ross 
45*b819cea2SGordon Ross void
smb_kdoor_init(smb_server_t * sv)46*b819cea2SGordon Ross smb_kdoor_init(smb_server_t *sv)
47*b819cea2SGordon Ross {
48*b819cea2SGordon Ross 	sv->sv_kdoor_id = -1;
49*b819cea2SGordon Ross 	mutex_init(&sv->sv_kdoor_mutex, NULL, MUTEX_DEFAULT, NULL);
50*b819cea2SGordon Ross 	cv_init(&sv->sv_kdoor_cv, NULL, CV_DEFAULT, NULL);
51*b819cea2SGordon Ross }
52*b819cea2SGordon Ross 
53*b819cea2SGordon Ross void
smb_kdoor_fini(smb_server_t * sv)54*b819cea2SGordon Ross smb_kdoor_fini(smb_server_t *sv)
55*b819cea2SGordon Ross {
56*b819cea2SGordon Ross 	smb_kdoor_close(sv);
57*b819cea2SGordon Ross 	cv_destroy(&sv->sv_kdoor_cv);
58*b819cea2SGordon Ross 	mutex_destroy(&sv->sv_kdoor_mutex);
59*b819cea2SGordon Ross }
60*b819cea2SGordon Ross 
61*b819cea2SGordon Ross /*
62*b819cea2SGordon Ross  * In the "fake kernen", our "upcalls" don't use the
63*b819cea2SGordon Ross  * real door, but just call via a function pointer.
64*b819cea2SGordon Ross  * This is where we setup that pointer, which is
65*b819cea2SGordon Ross  * fksmbd_door_dispatch()
66*b819cea2SGordon Ross  */
67*b819cea2SGordon Ross void
fksmb_kdoor_open(smb_server_t * sv,void * varg)68*b819cea2SGordon Ross fksmb_kdoor_open(smb_server_t *sv, void *varg)
69*b819cea2SGordon Ross {
70*b819cea2SGordon Ross 	sv->sv_kdoor_hd = varg;
71*b819cea2SGordon Ross }
72*b819cea2SGordon Ross 
73*b819cea2SGordon Ross void
smb_kdoor_close(smb_server_t * sv)74*b819cea2SGordon Ross smb_kdoor_close(smb_server_t *sv)
75*b819cea2SGordon Ross {
76*b819cea2SGordon Ross 	sv->sv_kdoor_hd = NULL;
77*b819cea2SGordon Ross 	sv->sv_kdoor_id = -1;
78*b819cea2SGordon Ross }
79*b819cea2SGordon Ross 
80*b819cea2SGordon Ross /* ARGSUSED */
81*b819cea2SGordon Ross int
smb_kdoor_upcall(smb_server_t * sv,uint32_t cmd,void * req_data,xdrproc_t req_xdr,void * rsp_data,xdrproc_t rsp_xdr)82*b819cea2SGordon Ross smb_kdoor_upcall(smb_server_t *sv, uint32_t cmd,
83*b819cea2SGordon Ross     void *req_data, xdrproc_t req_xdr,
84*b819cea2SGordon Ross     void *rsp_data, xdrproc_t rsp_xdr)
85*b819cea2SGordon Ross {
86*b819cea2SGordon Ross 	smb_doorarg_t	da;
87*b819cea2SGordon Ross 	fksmb_kdoor_disp_func_t *func;
88*b819cea2SGordon Ross 	int rc;
89*b819cea2SGordon Ross 
90*b819cea2SGordon Ross 	bzero(&da, sizeof (smb_doorarg_t));
91*b819cea2SGordon Ross 	da.da_opcode = cmd;
92*b819cea2SGordon Ross 	da.da_opname = smb_doorhdr_opname(cmd);
93*b819cea2SGordon Ross 	da.da_req_xdr = req_xdr;
94*b819cea2SGordon Ross 	da.da_rsp_xdr = rsp_xdr;
95*b819cea2SGordon Ross 	da.da_req_data = req_data;
96*b819cea2SGordon Ross 	da.da_rsp_data = rsp_data;
97*b819cea2SGordon Ross 
98*b819cea2SGordon Ross 	if ((req_data == NULL && req_xdr != NULL) ||
99*b819cea2SGordon Ross 	    (rsp_data == NULL && rsp_xdr != NULL)) {
100*b819cea2SGordon Ross 		cmn_err(CE_WARN, "smb_kdoor_upcall[%s]: invalid param",
101*b819cea2SGordon Ross 		    da.da_opname);
102*b819cea2SGordon Ross 		return (-1);
103*b819cea2SGordon Ross 	}
104*b819cea2SGordon Ross 
105*b819cea2SGordon Ross 	/* NB: no ASYNC, nor event stuff */
106*b819cea2SGordon Ross 
107*b819cea2SGordon Ross 	func = (fksmb_kdoor_disp_func_t *)(sv->sv_kdoor_hd);
108*b819cea2SGordon Ross 	if (func == NULL)
109*b819cea2SGordon Ross 		return (EFAULT);
110*b819cea2SGordon Ross 
111*b819cea2SGordon Ross 	if ((rc = smb_kdoor_encode(&da)) != 0)
112*b819cea2SGordon Ross 		goto out;
113*b819cea2SGordon Ross 
114*b819cea2SGordon Ross 	/*
115*b819cea2SGordon Ross 	 * The "upcall" (just call via function pointer)
116*b819cea2SGordon Ross 	 * i.e. see: fksmbd_door_dispatch()
117*b819cea2SGordon Ross 	 */
118*b819cea2SGordon Ross 	if ((rc = (*func)(&da)) != 0)
119*b819cea2SGordon Ross 		goto out;
120*b819cea2SGordon Ross 
121*b819cea2SGordon Ross 	rc = smb_kdoor_decode(&da);
122*b819cea2SGordon Ross out:
123*b819cea2SGordon Ross 	smb_kdoor_free(&da.da_arg);
124*b819cea2SGordon Ross 
125*b819cea2SGordon Ross 	return (rc);
126*b819cea2SGordon Ross }
127*b819cea2SGordon Ross 
128*b819cea2SGordon Ross /* no smb_kdoor_send, smb_kdoor_receive */
129*b819cea2SGordon Ross /* no smb_kdoor_upcall_private */
130*b819cea2SGordon Ross 
131*b819cea2SGordon Ross static int
smb_kdoor_encode(smb_doorarg_t * da)132*b819cea2SGordon Ross smb_kdoor_encode(smb_doorarg_t *da)
133*b819cea2SGordon Ross {
134*b819cea2SGordon Ross 	XDR		xdrs;
135*b819cea2SGordon Ross 	char		*buf;
136*b819cea2SGordon Ross 	uint32_t	len;
137*b819cea2SGordon Ross 
138*b819cea2SGordon Ross 	len = xdr_sizeof(smb_doorhdr_xdr, &da->da_hdr);
139*b819cea2SGordon Ross 	if (da->da_req_xdr != NULL)
140*b819cea2SGordon Ross 		len += xdr_sizeof(da->da_req_xdr, da->da_req_data);
141*b819cea2SGordon Ross 
142*b819cea2SGordon Ross 	smb_kdoor_sethdr(da, len);
143*b819cea2SGordon Ross 
144*b819cea2SGordon Ross 	buf = kmem_zalloc(len, KM_SLEEP);
145*b819cea2SGordon Ross 	xdrmem_create(&xdrs, buf, len, XDR_ENCODE);
146*b819cea2SGordon Ross 
147*b819cea2SGordon Ross 	if (!smb_doorhdr_xdr(&xdrs, &da->da_hdr)) {
148*b819cea2SGordon Ross 		cmn_err(CE_WARN, "smb_kdoor_encode[%s]: header encode failed",
149*b819cea2SGordon Ross 		    da->da_opname);
150*b819cea2SGordon Ross 		kmem_free(buf, len);
151*b819cea2SGordon Ross 		xdr_destroy(&xdrs);
152*b819cea2SGordon Ross 		return (-1);
153*b819cea2SGordon Ross 	}
154*b819cea2SGordon Ross 
155*b819cea2SGordon Ross 	if (da->da_req_xdr != NULL) {
156*b819cea2SGordon Ross 		if (!da->da_req_xdr(&xdrs, da->da_req_data)) {
157*b819cea2SGordon Ross 			cmn_err(CE_WARN, "smb_kdoor_encode[%s]: encode failed",
158*b819cea2SGordon Ross 			    da->da_opname);
159*b819cea2SGordon Ross 			kmem_free(buf, len);
160*b819cea2SGordon Ross 			xdr_destroy(&xdrs);
161*b819cea2SGordon Ross 			return (-1);
162*b819cea2SGordon Ross 		}
163*b819cea2SGordon Ross 	}
164*b819cea2SGordon Ross 
165*b819cea2SGordon Ross 	da->da_arg.data_ptr = buf;
166*b819cea2SGordon Ross 	da->da_arg.data_size = len;
167*b819cea2SGordon Ross 	da->da_arg.desc_ptr = NULL;
168*b819cea2SGordon Ross 	da->da_arg.desc_num = 0;
169*b819cea2SGordon Ross 	da->da_arg.rbuf = buf;
170*b819cea2SGordon Ross 	da->da_arg.rsize = len;
171*b819cea2SGordon Ross 
172*b819cea2SGordon Ross 	xdr_destroy(&xdrs);
173*b819cea2SGordon Ross 	return (0);
174*b819cea2SGordon Ross }
175*b819cea2SGordon Ross 
176*b819cea2SGordon Ross /*
177*b819cea2SGordon Ross  * Decode the response in rbuf and rsize.
178*b819cea2SGordon Ross  */
179*b819cea2SGordon Ross static int
smb_kdoor_decode(smb_doorarg_t * da)180*b819cea2SGordon Ross smb_kdoor_decode(smb_doorarg_t *da)
181*b819cea2SGordon Ross {
182*b819cea2SGordon Ross 	XDR		xdrs;
183*b819cea2SGordon Ross 	smb_doorhdr_t	hdr;
184*b819cea2SGordon Ross 	char		*rbuf = da->da_arg.rbuf;
185*b819cea2SGordon Ross 	uint32_t	rsize = da->da_arg.rsize;
186*b819cea2SGordon Ross 
187*b819cea2SGordon Ross 	if (rbuf == NULL || rsize == 0) {
188*b819cea2SGordon Ross 		cmn_err(CE_WARN, "smb_kdoor_decode[%s]: invalid param",
189*b819cea2SGordon Ross 		    da->da_opname);
190*b819cea2SGordon Ross 		return (-1);
191*b819cea2SGordon Ross 	}
192*b819cea2SGordon Ross 
193*b819cea2SGordon Ross 	xdrmem_create(&xdrs, rbuf, rsize, XDR_DECODE);
194*b819cea2SGordon Ross 
195*b819cea2SGordon Ross 	if (!smb_doorhdr_xdr(&xdrs, &hdr)) {
196*b819cea2SGordon Ross 		cmn_err(CE_WARN, "smb_kdoor_decode[%s]: header decode failed",
197*b819cea2SGordon Ross 		    da->da_opname);
198*b819cea2SGordon Ross 		xdr_destroy(&xdrs);
199*b819cea2SGordon Ross 		return (-1);
200*b819cea2SGordon Ross 	}
201*b819cea2SGordon Ross 
202*b819cea2SGordon Ross 	if (!smb_kdoor_chkhdr(da, &hdr)) {
203*b819cea2SGordon Ross 		xdr_destroy(&xdrs);
204*b819cea2SGordon Ross 		return (-1);
205*b819cea2SGordon Ross 	}
206*b819cea2SGordon Ross 
207*b819cea2SGordon Ross 	if (hdr.dh_datalen != 0 && da->da_rsp_xdr != NULL) {
208*b819cea2SGordon Ross 		if (!da->da_rsp_xdr(&xdrs, da->da_rsp_data)) {
209*b819cea2SGordon Ross 			cmn_err(CE_WARN, "smb_kdoor_decode[%s]: decode failed",
210*b819cea2SGordon Ross 			    da->da_opname);
211*b819cea2SGordon Ross 			xdr_destroy(&xdrs);
212*b819cea2SGordon Ross 			return (-1);
213*b819cea2SGordon Ross 		}
214*b819cea2SGordon Ross 	}
215*b819cea2SGordon Ross 
216*b819cea2SGordon Ross 	xdr_destroy(&xdrs);
217*b819cea2SGordon Ross 	return (0);
218*b819cea2SGordon Ross }
219*b819cea2SGordon Ross 
220*b819cea2SGordon Ross static void
smb_kdoor_sethdr(smb_doorarg_t * da,uint32_t datalen)221*b819cea2SGordon Ross smb_kdoor_sethdr(smb_doorarg_t *da, uint32_t datalen)
222*b819cea2SGordon Ross {
223*b819cea2SGordon Ross 	smb_doorhdr_t	*hdr = &da->da_hdr;
224*b819cea2SGordon Ross 
225*b819cea2SGordon Ross 	bzero(hdr, sizeof (smb_doorhdr_t));
226*b819cea2SGordon Ross 	hdr->dh_magic = SMB_DOOR_HDR_MAGIC;
227*b819cea2SGordon Ross 	hdr->dh_flags = da->da_flags | SMB_DF_FAKE_KERNEL;
228*b819cea2SGordon Ross 	hdr->dh_op = da->da_opcode;
229*b819cea2SGordon Ross 	/* hdr->dh_txid = 0 (not used) */
230*b819cea2SGordon Ross 	hdr->dh_datalen = datalen;
231*b819cea2SGordon Ross 	hdr->dh_door_rc = SMB_DOP_NOT_CALLED;
232*b819cea2SGordon Ross }
233*b819cea2SGordon Ross 
234*b819cea2SGordon Ross static boolean_t
smb_kdoor_chkhdr(smb_doorarg_t * da,smb_doorhdr_t * hdr)235*b819cea2SGordon Ross smb_kdoor_chkhdr(smb_doorarg_t *da, smb_doorhdr_t *hdr)
236*b819cea2SGordon Ross {
237*b819cea2SGordon Ross 	if ((hdr->dh_magic != SMB_DOOR_HDR_MAGIC) ||
238*b819cea2SGordon Ross 	    (hdr->dh_op != da->da_hdr.dh_op) ||
239*b819cea2SGordon Ross 	    (hdr->dh_txid != da->da_hdr.dh_txid)) {
240*b819cea2SGordon Ross 		cmn_err(CE_WARN, "smb_kdoor_chkhdr[%s]: invalid header",
241*b819cea2SGordon Ross 		    da->da_opname);
242*b819cea2SGordon Ross 		return (B_FALSE);
243*b819cea2SGordon Ross 	}
244*b819cea2SGordon Ross 
245*b819cea2SGordon Ross 	switch (hdr->dh_door_rc) {
246*b819cea2SGordon Ross 	case SMB_DOP_SUCCESS:
247*b819cea2SGordon Ross 		break;
248*b819cea2SGordon Ross 
249*b819cea2SGordon Ross 	/* SMB_DOP_EMPTYBUF is a "normal" error (silent). */
250*b819cea2SGordon Ross 	case SMB_DOP_EMPTYBUF:
251*b819cea2SGordon Ross 		return (B_FALSE);
252*b819cea2SGordon Ross 
253*b819cea2SGordon Ross 	default:
254*b819cea2SGordon Ross 		cmn_err(CE_WARN, "smb_kdoor_chkhdr[%s]: call failed: %u",
255*b819cea2SGordon Ross 		    da->da_opname, hdr->dh_door_rc);
256*b819cea2SGordon Ross 		return (B_FALSE);
257*b819cea2SGordon Ross 	}
258*b819cea2SGordon Ross 
259*b819cea2SGordon Ross 	return (B_TRUE);
260*b819cea2SGordon Ross }
261*b819cea2SGordon Ross 
262*b819cea2SGordon Ross /*
263*b819cea2SGordon Ross  * Free both the argument and result door buffers regardless of the status
264*b819cea2SGordon Ross  * of the up-call.  The doorfs allocates a new buffer if the result buffer
265*b819cea2SGordon Ross  * passed by the client is too small.
266*b819cea2SGordon Ross  */
267*b819cea2SGordon Ross static void
smb_kdoor_free(door_arg_t * arg)268*b819cea2SGordon Ross smb_kdoor_free(door_arg_t *arg)
269*b819cea2SGordon Ross {
270*b819cea2SGordon Ross 	if (arg->rbuf != NULL && arg->rbuf != arg->data_ptr)
271*b819cea2SGordon Ross 		kmem_free(arg->rbuf, arg->rsize);
272*b819cea2SGordon Ross 
273*b819cea2SGordon Ross 	if (arg->data_ptr != NULL)
274*b819cea2SGordon Ross 		kmem_free(arg->data_ptr, arg->data_size);
275*b819cea2SGordon Ross }
276