1a90cf9f2SGordon Ross /*
2a90cf9f2SGordon Ross  * This file and its contents are supplied under the terms of the
3a90cf9f2SGordon Ross  * Common Development and Distribution License ("CDDL"), version 1.0.
4a90cf9f2SGordon Ross  * You may only use this file in accordance with the terms of version
5a90cf9f2SGordon Ross  * 1.0 of the CDDL.
6a90cf9f2SGordon Ross  *
7a90cf9f2SGordon Ross  * A full copy of the text of the CDDL should have accompanied this
8a90cf9f2SGordon Ross  * source.  A copy of the CDDL is also available via the Internet at
9a90cf9f2SGordon Ross  * http://www.illumos.org/license/CDDL.
10a90cf9f2SGordon Ross  */
11a90cf9f2SGordon Ross 
12a90cf9f2SGordon Ross /*
13*525641e8SGordon Ross  * Copyright 2020 Tintri by DDN, Inc.  All rights reserved.
14a90cf9f2SGordon Ross  */
15a90cf9f2SGordon Ross 
16a90cf9f2SGordon Ross /*
17a90cf9f2SGordon Ross  * Dispatch function for SMB2_CANCEL
18a90cf9f2SGordon Ross  */
19a90cf9f2SGordon Ross 
20a90cf9f2SGordon Ross #include <smbsrv/smb2_kproto.h>
21a90cf9f2SGordon Ross 
225677e049SGordon Ross static void smb2_cancel_async(smb_request_t *);
235677e049SGordon Ross static void smb2_cancel_sync(smb_request_t *);
24a90cf9f2SGordon Ross 
25a90cf9f2SGordon Ross /*
26a90cf9f2SGordon Ross  * This handles an SMB2_CANCEL request when seen in the reader.
27a90cf9f2SGordon Ross  * (See smb2sr_newrq)  Handle this immediately, rather than
28a90cf9f2SGordon Ross  * going through the normal taskq dispatch mechanism.
29a90cf9f2SGordon Ross  * Note that Cancel does NOT get a response.
305677e049SGordon Ross  *
315677e049SGordon Ross  * Any non-zero return causes disconnect.
325677e049SGordon Ross  * SMB2 header is already decoded.
33a90cf9f2SGordon Ross  */
34a90cf9f2SGordon Ross int
smb2_newrq_cancel(smb_request_t * sr)355677e049SGordon Ross smb2_newrq_cancel(smb_request_t *sr)
36a90cf9f2SGordon Ross {
37a90cf9f2SGordon Ross 
38a90cf9f2SGordon Ross 	/*
395677e049SGordon Ross 	 * If we get SMB2 cancel as part of a compound,
405677e049SGordon Ross 	 * that's a protocol violation.  Drop 'em!
41a90cf9f2SGordon Ross 	 */
425677e049SGordon Ross 	if (sr->smb2_next_command != 0)
435677e049SGordon Ross 		return (EINVAL);
44a90cf9f2SGordon Ross 
4593bc28dbSGordon Ross 	DTRACE_SMB2_START(op__Cancel, smb_request_t *, sr);
4693bc28dbSGordon Ross 
47a90cf9f2SGordon Ross 	if (sr->smb2_hdr_flags & SMB2_FLAGS_ASYNC_COMMAND)
485677e049SGordon Ross 		smb2_cancel_async(sr);
49a90cf9f2SGordon Ross 	else
505677e049SGordon Ross 		smb2_cancel_sync(sr);
51a90cf9f2SGordon Ross 
5293bc28dbSGordon Ross 	DTRACE_SMB2_DONE(op__Cancel, smb_request_t *, sr);
5393bc28dbSGordon Ross 
54a90cf9f2SGordon Ross 	return (0);
55a90cf9f2SGordon Ross }
56a90cf9f2SGordon Ross 
575677e049SGordon Ross /*
585677e049SGordon Ross  * Dispatch handler for SMB2_CANCEL.
595677e049SGordon Ross  * Note that Cancel does NOT get a response.
605677e049SGordon Ross  */
615677e049SGordon Ross smb_sdrc_t
smb2_cancel(smb_request_t * sr)625677e049SGordon Ross smb2_cancel(smb_request_t *sr)
635677e049SGordon Ross {
645677e049SGordon Ross 
655677e049SGordon Ross 	/*
665677e049SGordon Ross 	 * If we get SMB2 cancel as part of a compound,
675677e049SGordon Ross 	 * that's a protocol violation.  Drop 'em!
685677e049SGordon Ross 	 */
695677e049SGordon Ross 	if (sr->smb2_cmd_hdr != 0 || sr->smb2_next_command != 0)
705677e049SGordon Ross 		return (SDRC_DROP_VC);
715677e049SGordon Ross 
7293bc28dbSGordon Ross 	DTRACE_SMB2_START(op__Cancel, smb_request_t *, sr);
7393bc28dbSGordon Ross 
745677e049SGordon Ross 	if (sr->smb2_hdr_flags & SMB2_FLAGS_ASYNC_COMMAND) {
755677e049SGordon Ross 		smb2_cancel_async(sr);
765677e049SGordon Ross 	} else {
775677e049SGordon Ross 		smb2_cancel_sync(sr);
785677e049SGordon Ross 	}
795677e049SGordon Ross 
8093bc28dbSGordon Ross 	DTRACE_SMB2_DONE(op__Cancel, smb_request_t *, sr);
8193bc28dbSGordon Ross 
825677e049SGordon Ross 	return (SDRC_NO_REPLY);
835677e049SGordon Ross }
845677e049SGordon Ross 
855677e049SGordon Ross /*
865677e049SGordon Ross  * SMB2 Cancel (sync) has an inherent race with the request being
875677e049SGordon Ross  * cancelled.  The request may have been received but not yet
885677e049SGordon Ross  * executed by a worker thread, in which case we'll mark the
895677e049SGordon Ross  * request state as cancelled, and when a worker thread starts
905677e049SGordon Ross  * on this request we'll cancel everything in the compound.
915677e049SGordon Ross  */
92a90cf9f2SGordon Ross static void
smb2_cancel_sync(smb_request_t * sr)935677e049SGordon Ross smb2_cancel_sync(smb_request_t *sr)
94a90cf9f2SGordon Ross {
95a90cf9f2SGordon Ross 	struct smb_request *req;
96a90cf9f2SGordon Ross 	struct smb_session *session = sr->session;
97a90cf9f2SGordon Ross 	int cnt = 0;
98a90cf9f2SGordon Ross 
99*525641e8SGordon Ross 	if (sr->smb2_messageid == 0 || sr->smb2_messageid == UINT64_MAX)
100*525641e8SGordon Ross 		return;
1015677e049SGordon Ross 
102a90cf9f2SGordon Ross 	smb_slist_enter(&session->s_req_list);
1035677e049SGordon Ross 	for (req = smb_slist_head(&session->s_req_list); req != NULL;
1045677e049SGordon Ross 	    req = smb_slist_next(&session->s_req_list, req)) {
1055677e049SGordon Ross 
1065677e049SGordon Ross 		/* never cancel self */
1075677e049SGordon Ross 		if (req == sr)
1085677e049SGordon Ross 			continue;
1095677e049SGordon Ross 
1105677e049SGordon Ross 		if (sr->smb2_messageid >= req->smb2_first_msgid &&
1115677e049SGordon Ross 		    sr->smb2_messageid < (req->smb2_first_msgid +
1125677e049SGordon Ross 		    req->smb2_total_credits)) {
113a90cf9f2SGordon Ross 			smb_request_cancel(req);
114a90cf9f2SGordon Ross 			cnt++;
115a90cf9f2SGordon Ross 		}
116a90cf9f2SGordon Ross 	}
1175677e049SGordon Ross 	smb_slist_exit(&session->s_req_list);
1185677e049SGordon Ross 
119a90cf9f2SGordon Ross 	if (cnt != 1) {
120a90cf9f2SGordon Ross 		DTRACE_PROBE2(smb2__cancel__error,
121a90cf9f2SGordon Ross 		    uint64_t, sr->smb2_messageid, int, cnt);
1225677e049SGordon Ross #ifdef	DEBUG
1235677e049SGordon Ross 		/*
1245677e049SGordon Ross 		 * It's somewhat common that we may see a cancel for a
1255677e049SGordon Ross 		 * request that has already completed, so report that
1265677e049SGordon Ross 		 * only in debug builds.
1275677e049SGordon Ross 		 */
1285677e049SGordon Ross 		cmn_err(CE_WARN, "SMB2 cancel failed, "
1295677e049SGordon Ross 		    "client=%s, MID=0x%llx",
1305677e049SGordon Ross 		    sr->session->ip_addr_str,
1315677e049SGordon Ross 		    (u_longlong_t)sr->smb2_messageid);
1325677e049SGordon Ross #endif
133a90cf9f2SGordon Ross 	}
134a90cf9f2SGordon Ross }
135a90cf9f2SGordon Ross 
1365677e049SGordon Ross /*
1375677e049SGordon Ross  * Note that cancelling an async request doesn't have a race
1385677e049SGordon Ross  * because the client doesn't learn about the async ID until we
1395677e049SGordon Ross  * send it to them in an interim reply, and by that point the
1405677e049SGordon Ross  * request has progressed to the point where smb_cancel can find
1415677e049SGordon Ross  * the request and cancel it.
1425677e049SGordon Ross  */
143a90cf9f2SGordon Ross static void
smb2_cancel_async(smb_request_t * sr)1445677e049SGordon Ross smb2_cancel_async(smb_request_t *sr)
145a90cf9f2SGordon Ross {
146a90cf9f2SGordon Ross 	struct smb_request *req;
147a90cf9f2SGordon Ross 	struct smb_session *session = sr->session;
148a90cf9f2SGordon Ross 	int cnt = 0;
149a90cf9f2SGordon Ross 
150*525641e8SGordon Ross 	if (sr->smb2_async_id == 0)
151*525641e8SGordon Ross 		return;
152*525641e8SGordon Ross 
153a90cf9f2SGordon Ross 	smb_slist_enter(&session->s_req_list);
154a90cf9f2SGordon Ross 	req = smb_slist_head(&session->s_req_list);
155a90cf9f2SGordon Ross 	while (req) {
156a90cf9f2SGordon Ross 		ASSERT(req->sr_magic == SMB_REQ_MAGIC);
157a90cf9f2SGordon Ross 		if ((req != sr) &&
158a90cf9f2SGordon Ross 		    (req->smb2_async_id == sr->smb2_async_id)) {
159a90cf9f2SGordon Ross 			smb_request_cancel(req);
160a90cf9f2SGordon Ross 			cnt++;
161a90cf9f2SGordon Ross 		}
162a90cf9f2SGordon Ross 		req = smb_slist_next(&session->s_req_list, req);
163a90cf9f2SGordon Ross 	}
164a90cf9f2SGordon Ross 	if (cnt != 1) {
165a90cf9f2SGordon Ross 		DTRACE_PROBE2(smb2__cancel__error,
166a90cf9f2SGordon Ross 		    uint64_t, sr->smb2_async_id, int, cnt);
1675677e049SGordon Ross 		/*
1685677e049SGordon Ross 		 * Not logging here, as this is normal, i.e.
1695677e049SGordon Ross 		 * when both a cancel and a handle close
1705677e049SGordon Ross 		 * terminates an SMB2_notify request.
1715677e049SGordon Ross 		 */
172a90cf9f2SGordon Ross 	}
173a90cf9f2SGordon Ross 	smb_slist_exit(&session->s_req_list);
174a90cf9f2SGordon Ross }
175