1 /* 2 * This file and its contents are supplied under the terms of the 3 * Common Development and Distribution License ("CDDL"), version 1.0. 4 * You may only use this file in accordance with the terms of version 5 * 1.0 of the CDDL. 6 * 7 * A full copy of the text of the CDDL should have accompanied this 8 * source. A copy of the CDDL is also available via the Internet at 9 * http://www.illumos.org/license/CDDL. 10 */ 11 12 /* 13 * Copyright 2015 Nexenta Systems, Inc. All rights reserved. 14 */ 15 16 17 #include <smbsrv/smb2_kproto.h> 18 #include <smbsrv/smb_kstat.h> 19 #include <smbsrv/smb2.h> 20 21 /* 22 * Saved state for a command that "goes async". When a compound request 23 * contains a command that may block indefinitely, the compound reply is 24 * composed with an "interim response" for that command, and information 25 * needed to actually dispatch that command is saved on a list of "async" 26 * commands for this compound request. After the compound reply is sent, 27 * the list of async commands is processed, and those may block as long 28 * as they need to without affecting the initial compound request. 29 * 30 * Now interestingly, this "async" mechanism is not used with the full 31 * range of asynchrony that one might imagine. The design of async 32 * request processing can be drastically simplified if we can assume 33 * that there's no need to run more than one async command at a time. 34 * With that simplifying assumption, we can continue using the current 35 * "one worker thread per request message" model, which has very simple 36 * locking rules etc. The same worker thread that handles the initial 37 * compound request can handle the list of async requests. 38 * 39 * As it turns out, SMB2 clients do not try to use more than one "async" 40 * command in a compound. If they were to do so, the [MS-SMB2] spec. 41 * allows us to decline additional async requests with an error. 42 * 43 * smb_async_req_t is the struct used to save an "async" request on 44 * the list of requests that had an interim reply in the initial 45 * compound reply. This includes everything needed to restart 46 * processing at the async command. 47 */ 48 49 typedef struct smb2_async_req { 50 51 smb_sdrc_t (*ar_func)(smb_request_t *); 52 53 int ar_cmd_hdr; /* smb2_cmd_hdr offset */ 54 int ar_cmd_len; /* length from hdr */ 55 56 /* 57 * SMB2 header fields. 58 */ 59 uint16_t ar_cmd_code; 60 uint16_t ar_uid; 61 uint16_t ar_tid; 62 uint32_t ar_pid; 63 uint32_t ar_hdr_flags; 64 uint64_t ar_messageid; 65 } smb2_async_req_t; 66 67 void smb2sr_do_async(smb_request_t *); 68 smb_sdrc_t smb2_invalid_cmd(smb_request_t *); 69 static void smb2_tq_work(void *); 70 71 static const smb_disp_entry_t 72 smb2_disp_table[SMB2__NCMDS] = { 73 74 /* text-name, pre, func, post, cmd-code, dialect, flags */ 75 76 { "smb2_negotiate", NULL, 77 smb2_negotiate, NULL, 0, 0, 78 SDDF_SUPPRESS_TID | SDDF_SUPPRESS_UID }, 79 80 { "smb2_session_setup", NULL, 81 smb2_session_setup, NULL, 0, 0, 82 SDDF_SUPPRESS_TID | SDDF_SUPPRESS_UID }, 83 84 { "smb2_logoff", NULL, 85 smb2_logoff, NULL, 0, 0, 86 SDDF_SUPPRESS_TID }, 87 88 { "smb2_tree_connect", NULL, 89 smb2_tree_connect, NULL, 0, 0, 90 SDDF_SUPPRESS_TID }, 91 92 { "smb2_tree_disconn", NULL, 93 smb2_tree_disconn, NULL, 0, 0 }, 94 95 { "smb2_create", NULL, 96 smb2_create, NULL, 0, 0 }, 97 98 { "smb2_close", NULL, 99 smb2_close, NULL, 0, 0 }, 100 101 { "smb2_flush", NULL, 102 smb2_flush, NULL, 0, 0 }, 103 104 { "smb2_read", NULL, 105 smb2_read, NULL, 0, 0 }, 106 107 { "smb2_write", NULL, 108 smb2_write, NULL, 0, 0 }, 109 110 { "smb2_lock", NULL, 111 smb2_lock, NULL, 0, 0 }, 112 113 { "smb2_ioctl", NULL, 114 smb2_ioctl, NULL, 0, 0 }, 115 116 /* 117 * Note: Cancel gets the "invalid command" handler because 118 * that's always handled directly in the reader. We should 119 * never get to the function using this table, but note: 120 * We CAN get here if a nasty client adds cancel to some 121 * compound message, which is a protocol violation. 122 */ 123 { "smb2_cancel", NULL, 124 smb2_invalid_cmd, NULL, 0, 0 }, 125 126 { "smb2_echo", NULL, 127 smb2_echo, NULL, 0, 0, 128 SDDF_SUPPRESS_UID | SDDF_SUPPRESS_TID }, 129 130 { "smb2_query_dir", NULL, 131 smb2_query_dir, NULL, 0, 0 }, 132 133 { "smb2_change_notify", NULL, 134 smb2_change_notify, NULL, 0, 0 }, 135 136 { "smb2_query_info", NULL, 137 smb2_query_info, NULL, 0, 0 }, 138 139 { "smb2_set_info", NULL, 140 smb2_set_info, NULL, 0, 0 }, 141 142 { "smb2_oplock_break_ack", NULL, 143 smb2_oplock_break_ack, NULL, 0, 0 }, 144 145 { "smb2_invalid_cmd", NULL, 146 smb2_invalid_cmd, NULL, 0, 0, 147 SDDF_SUPPRESS_UID | SDDF_SUPPRESS_TID }, 148 }; 149 150 smb_sdrc_t 151 smb2_invalid_cmd(smb_request_t *sr) 152 { 153 #ifdef DEBUG 154 cmn_err(CE_NOTE, "clnt %s bad SMB2 cmd code", 155 sr->session->ip_addr_str); 156 #endif 157 sr->smb2_status = NT_STATUS_INVALID_PARAMETER; 158 return (SDRC_DROP_VC); 159 } 160 161 /* 162 * This is the SMB2 handler for new smb requests, called from 163 * smb_session_reader after SMB negotiate is done. For most SMB2 164 * requests, we just enqueue them for the smb_session_worker to 165 * execute via the task queue, so they can block for resources 166 * without stopping the reader thread. A few protocol messages 167 * are special cases and are handled directly here in the reader 168 * thread so they don't wait for taskq scheduling. 169 * 170 * This function must either enqueue the new request for 171 * execution via the task queue, or execute it directly 172 * and then free it. If this returns non-zero, the caller 173 * will drop the session. 174 */ 175 int 176 smb2sr_newrq(smb_request_t *sr) 177 { 178 uint32_t magic; 179 uint16_t command; 180 int rc; 181 182 magic = LE_IN32(sr->sr_request_buf); 183 if (magic != SMB2_PROTOCOL_MAGIC) { 184 smb_request_free(sr); 185 /* will drop the connection */ 186 return (EPROTO); 187 } 188 189 /* 190 * Execute Cancel requests immediately, (here in the 191 * reader thread) so they won't wait for any other 192 * commands we might already have in the task queue. 193 * Cancel also skips signature verification and 194 * does not consume a sequence number. 195 * [MS-SMB2] 3.2.4.24 Cancellation... 196 */ 197 command = LE_IN16((uint8_t *)sr->sr_request_buf + 12); 198 if (command == SMB2_CANCEL) { 199 rc = smb2sr_newrq_cancel(sr); 200 smb_request_free(sr); 201 return (rc); 202 } 203 204 /* 205 * Submit the request to the task queue, which calls 206 * smb2_tq_work when the workload permits. 207 */ 208 sr->sr_time_submitted = gethrtime(); 209 sr->sr_state = SMB_REQ_STATE_SUBMITTED; 210 smb_srqueue_waitq_enter(sr->session->s_srqueue); 211 (void) taskq_dispatch(sr->sr_server->sv_worker_pool, 212 smb2_tq_work, sr, TQ_SLEEP); 213 214 return (0); 215 } 216 217 static void 218 smb2_tq_work(void *arg) 219 { 220 smb_request_t *sr; 221 smb_srqueue_t *srq; 222 223 sr = (smb_request_t *)arg; 224 SMB_REQ_VALID(sr); 225 226 srq = sr->session->s_srqueue; 227 smb_srqueue_waitq_to_runq(srq); 228 sr->sr_worker = curthread; 229 sr->sr_time_active = gethrtime(); 230 231 /* 232 * In contrast with SMB1, SMB2 must _always_ dispatch to 233 * the handler function, because cancelled requests need 234 * an error reply (NT_STATUS_CANCELLED). 235 */ 236 smb2sr_work(sr); 237 238 smb_srqueue_runq_exit(srq); 239 } 240 241 /* 242 * smb2sr_work 243 * 244 * This function processes each SMB command in the current request 245 * (which may be a compound request) building a reply containing 246 * SMB reply messages, one-to-one with the SMB commands. Some SMB 247 * commands (change notify, blocking locks) may require both an 248 * "interim response" and a later "async response" at completion. 249 * In such cases, we'll encode the interim response in the reply 250 * compound we're building, and put the (now async) command on a 251 * list of commands that need further processing. After we've 252 * finished processing the commands in this compound and building 253 * the compound reply, we'll send the compound reply, and finally 254 * process the list of async commands. 255 * 256 * As we work our way through the compound request and reply, 257 * we need to keep track of the bounds of the current request 258 * and reply. For the request, this uses an MBC_SHADOW_CHAIN 259 * that begins at smb2_cmd_hdr. The reply is appended to the 260 * sr->reply chain starting at smb2_reply_hdr. 261 * 262 * This function must always free the smb request. 263 */ 264 void 265 smb2sr_work(struct smb_request *sr) 266 { 267 const smb_disp_entry_t *sdd; 268 smb_disp_stats_t *sds; 269 smb_session_t *session; 270 uint32_t msg_len; 271 uint16_t cmd_idx; 272 int rc = 0; 273 boolean_t disconnect = B_FALSE; 274 boolean_t related; 275 276 session = sr->session; 277 278 ASSERT(sr->tid_tree == 0); 279 ASSERT(sr->uid_user == 0); 280 ASSERT(sr->fid_ofile == 0); 281 sr->smb_fid = (uint16_t)-1; 282 sr->smb2_status = 0; 283 284 /* temporary until we identify a user */ 285 sr->user_cr = zone_kcred(); 286 287 mutex_enter(&sr->sr_mutex); 288 switch (sr->sr_state) { 289 case SMB_REQ_STATE_SUBMITTED: 290 case SMB_REQ_STATE_CLEANED_UP: 291 sr->sr_state = SMB_REQ_STATE_ACTIVE; 292 break; 293 default: 294 ASSERT(0); 295 /* FALLTHROUGH */ 296 case SMB_REQ_STATE_CANCELED: 297 sr->smb2_status = NT_STATUS_CANCELLED; 298 break; 299 } 300 mutex_exit(&sr->sr_mutex); 301 302 cmd_start: 303 /* 304 * Decode the request header 305 * 306 * Most problems with decoding will result in the error 307 * STATUS_INVALID_PARAMETER. If the decoding problem 308 * prevents continuing, we'll close the connection. 309 * [MS-SMB2] 3.3.5.2.6 Handling Incorrectly Formatted... 310 * 311 * We treat some status codes as if "sticky", meaning 312 * once they're set after some command handler returns, 313 * all remaining commands get this status without even 314 * calling the command-specific handler. The cancelled 315 * status is used above, and insufficient_resources is 316 * used when smb2sr_go_async declines to "go async". 317 * Otherwise initialize to zero (success). 318 */ 319 if (sr->smb2_status != NT_STATUS_CANCELLED && 320 sr->smb2_status != NT_STATUS_INSUFFICIENT_RESOURCES) 321 sr->smb2_status = 0; 322 323 sr->smb2_cmd_hdr = sr->command.chain_offset; 324 if ((rc = smb2_decode_header(sr)) != 0) { 325 cmn_err(CE_WARN, "clnt %s bad SMB2 header", 326 session->ip_addr_str); 327 disconnect = B_TRUE; 328 goto cleanup; 329 } 330 331 /* 332 * The SMB2_FLAGS_SERVER_TO_REDIR should only appear 333 * in messages from the server back to the client. 334 */ 335 if ((sr->smb2_hdr_flags & SMB2_FLAGS_SERVER_TO_REDIR) != 0) { 336 cmn_err(CE_WARN, "clnt %s bad SMB2 flags", 337 session->ip_addr_str); 338 disconnect = B_TRUE; 339 goto cleanup; 340 } 341 related = (sr->smb2_hdr_flags & SMB2_FLAGS_RELATED_OPERATIONS); 342 343 /* 344 * In case we bail out with an error before we get to the 345 * section that computes the credit grant, initialize the 346 * response header fields so that credits won't change. 347 * Note: SMB 2.02 clients may send credit charge zero. 348 */ 349 if (sr->smb2_credit_charge == 0) 350 sr->smb2_credit_charge = 1; 351 sr->smb2_credit_response = sr->smb2_credit_charge; 352 353 /* 354 * Reserve space for the reply header, and save the offset. 355 * The reply header will be overwritten later. If we have 356 * already exhausted the output space, then this client is 357 * trying something funny. Log it and kill 'em. 358 */ 359 sr->smb2_reply_hdr = sr->reply.chain_offset; 360 if ((rc = smb2_encode_header(sr, B_FALSE)) != 0) { 361 cmn_err(CE_WARN, "clnt %s excessive reply", 362 session->ip_addr_str); 363 disconnect = B_TRUE; 364 goto cleanup; 365 } 366 367 /* 368 * Figure out the length of data following the SMB2 header. 369 * It ends at either the next SMB2 header if there is one 370 * (smb2_next_command != 0) or at the end of the message. 371 */ 372 if (sr->smb2_next_command != 0) { 373 /* [MS-SMB2] says this is 8-byte aligned */ 374 msg_len = sr->smb2_next_command; 375 if ((msg_len & 7) != 0 || (msg_len < SMB2_HDR_SIZE) || 376 ((sr->smb2_cmd_hdr + msg_len) > sr->command.max_bytes)) { 377 cmn_err(CE_WARN, "clnt %s bad SMB2 next cmd", 378 session->ip_addr_str); 379 disconnect = B_TRUE; 380 goto cleanup; 381 } 382 } else { 383 msg_len = sr->command.max_bytes - sr->smb2_cmd_hdr; 384 } 385 386 /* 387 * Setup a shadow chain for this SMB2 command, starting 388 * with the header and ending at either the next command 389 * or the end of the message. The signing check below 390 * needs the entire SMB2 command. After that's done, we 391 * advance chain_offset to the end of the header where 392 * the command specific handlers continue decoding. 393 */ 394 (void) MBC_SHADOW_CHAIN(&sr->smb_data, &sr->command, 395 sr->smb2_cmd_hdr, msg_len); 396 397 /* 398 * Validate the commmand code, get dispatch table entries. 399 * [MS-SMB2] 3.3.5.2.6 Handling Incorrectly Formatted... 400 * 401 * The last slot in the dispatch table is used to handle 402 * invalid commands. Same for statistics. 403 */ 404 if (sr->smb2_cmd_code < SMB2_INVALID_CMD) 405 cmd_idx = sr->smb2_cmd_code; 406 else 407 cmd_idx = SMB2_INVALID_CMD; 408 sdd = &smb2_disp_table[cmd_idx]; 409 sds = &session->s_server->sv_disp_stats2[cmd_idx]; 410 411 /* 412 * If this command is NOT "related" to the previous, 413 * clear out the UID, TID, FID state that might be 414 * left over from the previous command. 415 * 416 * If the command IS related, any new IDs are ignored, 417 * and we simply continue with the previous user, tree, 418 * and open file. 419 */ 420 if (!related) { 421 /* 422 * Drop user, tree, file; carefully ordered to 423 * avoid dangling references: file, tree, user 424 */ 425 if (sr->fid_ofile != NULL) { 426 smb_ofile_request_complete(sr->fid_ofile); 427 smb_ofile_release(sr->fid_ofile); 428 sr->fid_ofile = NULL; 429 } 430 if (sr->tid_tree != NULL) { 431 smb_tree_release(sr->tid_tree); 432 sr->tid_tree = NULL; 433 } 434 if (sr->uid_user != NULL) { 435 smb_user_release(sr->uid_user); 436 sr->uid_user = NULL; 437 sr->user_cr = zone_kcred(); 438 } 439 } 440 441 /* 442 * Make sure we have a user and tree as needed 443 * according to the flags for the this command. 444 * Note that we may have inherited these. 445 */ 446 if ((sdd->sdt_flags & SDDF_SUPPRESS_UID) == 0) { 447 /* 448 * This command requires a user session. 449 */ 450 if (related) { 451 /* 452 * Previous command should have given us a user. 453 * [MS-SMB2] 3.3.5.2 Handling Related Requests 454 */ 455 if (sr->uid_user == NULL) { 456 smb2sr_put_error(sr, 457 NT_STATUS_INVALID_PARAMETER); 458 goto cmd_done; 459 } 460 sr->smb_uid = sr->uid_user->u_uid; 461 } else { 462 /* 463 * Lookup the UID 464 * [MS-SMB2] 3.3.5.2 Verifying the Session 465 */ 466 ASSERT(sr->uid_user == NULL); 467 sr->uid_user = smb_session_lookup_uid(session, 468 sr->smb_uid); 469 if (sr->uid_user == NULL) { 470 smb2sr_put_error(sr, 471 NT_STATUS_USER_SESSION_DELETED); 472 goto cmd_done; 473 } 474 sr->user_cr = smb_user_getcred(sr->uid_user); 475 } 476 ASSERT(sr->uid_user != NULL); 477 } 478 479 if ((sdd->sdt_flags & SDDF_SUPPRESS_TID) == 0) { 480 /* 481 * This command requires a tree connection. 482 */ 483 if (related) { 484 /* 485 * Previous command should have given us a tree. 486 * [MS-SMB2] 3.3.5.2 Handling Related Requests 487 */ 488 if (sr->tid_tree == NULL) { 489 smb2sr_put_error(sr, 490 NT_STATUS_INVALID_PARAMETER); 491 goto cmd_done; 492 } 493 sr->smb_tid = sr->tid_tree->t_tid; 494 } else { 495 /* 496 * Lookup the TID 497 * [MS-SMB2] 3.3.5.2 Verifying the Tree Connect 498 */ 499 ASSERT(sr->tid_tree == NULL); 500 sr->tid_tree = smb_session_lookup_tree(session, 501 sr->smb_tid); 502 if (sr->tid_tree == NULL) { 503 smb2sr_put_error(sr, 504 NT_STATUS_NETWORK_NAME_DELETED); 505 goto cmd_done; 506 } 507 } 508 ASSERT(sr->tid_tree != NULL); 509 } 510 511 /* 512 * SMB2 signature verification, two parts: 513 * (a) Require SMB2_FLAGS_SIGNED (for most request types) 514 * (b) If SMB2_FLAGS_SIGNED is set, check the signature. 515 * [MS-SMB2] 3.3.5.2.4 Verifying the Signature 516 */ 517 518 /* 519 * No user session means no signature check. That's OK, 520 * i.e. for commands marked SDDF_SUPPRESS_UID above. 521 * Note, this also means we won't sign the reply. 522 */ 523 if (sr->uid_user == NULL) 524 sr->smb2_hdr_flags &= ~SMB2_FLAGS_SIGNED; 525 526 /* 527 * The SDDF_SUPPRESS_UID dispatch is set for requests that 528 * don't need a UID (user). These also don't require a 529 * signature check here. 530 */ 531 if ((sdd->sdt_flags & SDDF_SUPPRESS_UID) == 0 && 532 sr->uid_user != NULL && 533 (sr->uid_user->u_sign_flags & SMB_SIGNING_CHECK) != 0) { 534 /* 535 * This request type should be signed, and 536 * we're configured to require signatures. 537 */ 538 if ((sr->smb2_hdr_flags & SMB2_FLAGS_SIGNED) == 0) { 539 smb2sr_put_error(sr, NT_STATUS_ACCESS_DENIED); 540 goto cmd_done; 541 } 542 rc = smb2_sign_check_request(sr); 543 if (rc != 0) { 544 DTRACE_PROBE1(smb2__sign__check, smb_request_t, sr); 545 smb2sr_put_error(sr, NT_STATUS_ACCESS_DENIED); 546 goto cmd_done; 547 } 548 } 549 550 /* 551 * Now that the signing check is done with smb_data, 552 * advance past the SMB2 header we decoded earlier. 553 * This leaves sr->smb_data correctly positioned 554 * for command-specific decoding in the dispatch 555 * function called next. 556 */ 557 sr->smb_data.chain_offset = sr->smb2_cmd_hdr + SMB2_HDR_SIZE; 558 559 /* 560 * SMB2 credits determine how many simultaneous commands the 561 * client may issue, and bounds the range of message IDs those 562 * commands may use. With multi-credit support, commands may 563 * use ranges of message IDs, where the credits used by each 564 * command are proportional to their data transfer size. 565 * 566 * Every command may request an increase or decrease of 567 * the currently granted credits, based on the difference 568 * between the credit request and the credit charge. 569 * [MS-SMB2] 3.3.1.2 Algorithm for the Granting of Credits 570 * 571 * Most commands have credit_request=1, credit_charge=1, 572 * which keeps the credit grant unchanged. 573 * 574 * All we're really doing here (for now) is reducing the 575 * credit_response if the client requests a credit increase 576 * that would take their credit over the maximum, and 577 * limiting the decrease so they don't run out of credits. 578 * 579 * Later, this could do something dynamic based on load. 580 * 581 * One other non-obvious bit about credits: We keep the 582 * session s_max_credits low until the 1st authentication, 583 * at which point we'll set the normal maximum_credits. 584 * Some clients ask for more credits with session setup, 585 * and we need to handle that requested increase _after_ 586 * the command-specific handler returns so it won't be 587 * restricted to the lower (pre-auth) limit. 588 */ 589 sr->smb2_credit_response = sr->smb2_credit_request; 590 if (sr->smb2_credit_request < sr->smb2_credit_charge) { 591 uint16_t cur, d; 592 593 mutex_enter(&session->s_credits_mutex); 594 cur = session->s_cur_credits; 595 596 /* Handle credit decrease. */ 597 d = sr->smb2_credit_charge - sr->smb2_credit_request; 598 cur -= d; 599 if (cur & 0x8000) { 600 /* 601 * underflow (bad credit charge or request) 602 * leave credits unchanged (response=charge) 603 */ 604 cur = session->s_cur_credits; 605 sr->smb2_credit_response = sr->smb2_credit_charge; 606 DTRACE_PROBE1(smb2__credit__neg, smb_request_t, sr); 607 } 608 609 /* 610 * The server MUST ensure that the number of credits 611 * held by the client is never reduced to zero. 612 * [MS-SMB2] 3.3.1.2 613 */ 614 if (cur == 0) { 615 cur = 1; 616 sr->smb2_credit_response += 1; 617 DTRACE_PROBE1(smb2__credit__min, smb_request_t, sr); 618 } 619 620 DTRACE_PROBE3(smb2__credit__decrease, 621 smb_request_t, sr, int, (int)cur, 622 int, (int)session->s_cur_credits); 623 624 session->s_cur_credits = cur; 625 mutex_exit(&session->s_credits_mutex); 626 } 627 628 /* 629 * The real work: call the SMB2 command handler 630 * (except for "sticky" smb2_status - see above) 631 */ 632 sr->sr_time_start = gethrtime(); 633 rc = SDRC_SUCCESS; 634 if (sr->smb2_status == 0) { 635 /* NB: not using pre_op */ 636 rc = (*sdd->sdt_function)(sr); 637 /* NB: not using post_op */ 638 } 639 640 MBC_FLUSH(&sr->raw_data); 641 642 /* 643 * Second half of SMB2 credit handling (increases) 644 */ 645 if (sr->smb2_credit_request > sr->smb2_credit_charge) { 646 uint16_t cur, d; 647 648 mutex_enter(&session->s_credits_mutex); 649 cur = session->s_cur_credits; 650 651 /* Handle credit increase. */ 652 d = sr->smb2_credit_request - sr->smb2_credit_charge; 653 cur += d; 654 655 /* 656 * If new credits would be above max, 657 * reduce the credit grant. 658 */ 659 if (cur > session->s_max_credits) { 660 d = cur - session->s_max_credits; 661 cur = session->s_max_credits; 662 sr->smb2_credit_response -= d; 663 DTRACE_PROBE1(smb2__credit__max, smb_request_t, sr); 664 } 665 666 DTRACE_PROBE3(smb2__credit__increase, 667 smb_request_t, sr, int, (int)cur, 668 int, (int)session->s_cur_credits); 669 670 session->s_cur_credits = cur; 671 mutex_exit(&session->s_credits_mutex); 672 } 673 674 cmd_done: 675 /* 676 * Pad the reply to align(8) if necessary. 677 */ 678 if (sr->reply.chain_offset & 7) { 679 int padsz = 8 - (sr->reply.chain_offset & 7); 680 (void) smb_mbc_encodef(&sr->reply, "#.", padsz); 681 } 682 ASSERT((sr->reply.chain_offset & 7) == 0); 683 684 /* 685 * Record some statistics: latency, rx bytes, tx bytes. 686 */ 687 smb_latency_add_sample(&sds->sdt_lat, 688 gethrtime() - sr->sr_time_start); 689 atomic_add_64(&sds->sdt_rxb, 690 (int64_t)(sr->command.chain_offset - sr->smb2_cmd_hdr)); 691 atomic_add_64(&sds->sdt_txb, 692 (int64_t)(sr->reply.chain_offset - sr->smb2_reply_hdr)); 693 694 switch (rc) { 695 case SDRC_SUCCESS: 696 break; 697 default: 698 /* 699 * SMB2 does not use the other dispatch return codes. 700 * If we see something else, log an event so we'll 701 * know something is returning bogus status codes. 702 * If you see these in the log, use dtrace to find 703 * the code returning something else. 704 */ 705 #ifdef DEBUG 706 cmn_err(CE_NOTE, "handler for %u returned 0x%x", 707 sr->smb2_cmd_code, rc); 708 #endif 709 /* FALLTHROUGH */ 710 case SDRC_ERROR: 711 if (sr->smb2_status == 0) 712 sr->smb2_status = NT_STATUS_INTERNAL_ERROR; 713 break; 714 case SDRC_DROP_VC: 715 disconnect = B_TRUE; 716 goto cleanup; 717 } 718 719 /* 720 * If there's a next command, figure out where it starts, 721 * and fill in the next command offset for the reply. 722 * Note: We sanity checked smb2_next_command above 723 * (the offset to the next command). Similarly set 724 * smb2_next_reply as the offset to the next reply. 725 */ 726 if (sr->smb2_next_command != 0) { 727 sr->command.chain_offset = 728 sr->smb2_cmd_hdr + sr->smb2_next_command; 729 sr->smb2_next_reply = 730 sr->reply.chain_offset - sr->smb2_reply_hdr; 731 } else { 732 sr->smb2_next_reply = 0; 733 } 734 735 /* 736 * Overwrite the SMB2 header for the response of 737 * this command (possibly part of a compound). 738 * encode_header adds: SMB2_FLAGS_SERVER_TO_REDIR 739 */ 740 (void) smb2_encode_header(sr, B_TRUE); 741 742 if (sr->smb2_hdr_flags & SMB2_FLAGS_SIGNED) 743 smb2_sign_reply(sr); 744 745 if (sr->smb2_next_command != 0) 746 goto cmd_start; 747 748 /* 749 * We've done all the commands in this compound. 750 * Send it out. 751 */ 752 smb2_send_reply(sr); 753 754 /* 755 * If any of the requests "went async", process those now. 756 * The async. function "keeps" this sr, changing its state 757 * to completed and calling smb_request_free(). 758 */ 759 if (sr->sr_async_req != NULL) { 760 smb2sr_do_async(sr); 761 return; 762 } 763 764 cleanup: 765 if (disconnect) { 766 smb_rwx_rwenter(&session->s_lock, RW_WRITER); 767 switch (session->s_state) { 768 case SMB_SESSION_STATE_DISCONNECTED: 769 case SMB_SESSION_STATE_TERMINATED: 770 break; 771 default: 772 smb_soshutdown(session->sock); 773 session->s_state = SMB_SESSION_STATE_DISCONNECTED; 774 break; 775 } 776 smb_rwx_rwexit(&session->s_lock); 777 } 778 779 mutex_enter(&sr->sr_mutex); 780 sr->sr_state = SMB_REQ_STATE_COMPLETED; 781 mutex_exit(&sr->sr_mutex); 782 783 smb_request_free(sr); 784 } 785 786 /* 787 * Dispatch an async request using saved information. 788 * See smb2sr_save_async and [MS-SMB2] 3.3.4.2 789 * 790 * This is sort of a "lite" version of smb2sr_work. Initialize the 791 * command and reply areas as they were when the command-speicific 792 * handler started (in case it needs to decode anything again). 793 * Call the async function, which builds the command-specific part 794 * of the response. Finally, send the response and free the sr. 795 */ 796 void 797 smb2sr_do_async(smb_request_t *sr) 798 { 799 const smb_disp_entry_t *sdd; 800 smb_disp_stats_t *sds; 801 smb2_async_req_t *ar; 802 int rc = 0; 803 804 /* 805 * Restore what smb2_decode_header found. 806 * (In lieu of decoding it again.) 807 */ 808 ar = sr->sr_async_req; 809 sr->smb2_cmd_hdr = ar->ar_cmd_hdr; 810 sr->smb2_cmd_code = ar->ar_cmd_code; 811 sr->smb2_hdr_flags = ar->ar_hdr_flags; 812 sr->smb2_async_id = (uintptr_t)ar; 813 sr->smb2_messageid = ar->ar_messageid; 814 sr->smb_pid = ar->ar_pid; 815 sr->smb_tid = ar->ar_tid; 816 sr->smb_uid = ar->ar_uid; 817 sr->smb2_status = 0; 818 819 /* 820 * Async requests don't grant credits, because any credits 821 * should have gone out with the interim reply. 822 * An async reply goes alone (no next reply). 823 */ 824 sr->smb2_credit_response = 0; 825 sr->smb2_next_reply = 0; 826 827 /* 828 * Setup input mbuf_chain 829 */ 830 ASSERT(ar->ar_cmd_len >= SMB2_HDR_SIZE); 831 (void) MBC_SHADOW_CHAIN(&sr->smb_data, &sr->command, 832 sr->smb2_cmd_hdr + SMB2_HDR_SIZE, 833 ar->ar_cmd_len - SMB2_HDR_SIZE); 834 835 /* 836 * Setup output mbuf_chain 837 */ 838 MBC_FLUSH(&sr->reply); 839 sr->smb2_reply_hdr = sr->reply.chain_offset; 840 (void) smb2_encode_header(sr, B_FALSE); 841 842 VERIFY3U(sr->smb2_cmd_code, <, SMB2_INVALID_CMD); 843 sdd = &smb2_disp_table[sr->smb2_cmd_code]; 844 sds = sr->session->s_server->sv_disp_stats2; 845 sds = &sds[sr->smb2_cmd_code]; 846 847 /* 848 * Keep the UID, TID, ofile we have. 849 */ 850 if ((sdd->sdt_flags & SDDF_SUPPRESS_UID) == 0 && 851 sr->uid_user == NULL) { 852 smb2sr_put_error(sr, NT_STATUS_USER_SESSION_DELETED); 853 goto cmd_done; 854 } 855 if ((sdd->sdt_flags & SDDF_SUPPRESS_TID) == 0 && 856 sr->tid_tree == NULL) { 857 smb2sr_put_error(sr, NT_STATUS_NETWORK_NAME_DELETED); 858 goto cmd_done; 859 } 860 861 /* 862 * Signature already verified 863 * Credits handled... 864 * 865 * Just call the async handler function. 866 */ 867 rc = ar->ar_func(sr); 868 if (rc != 0 && sr->smb2_status == 0) 869 sr->smb2_status = NT_STATUS_INTERNAL_ERROR; 870 871 cmd_done: 872 /* 873 * Pad the reply to align(8) if necessary. 874 */ 875 if (sr->reply.chain_offset & 7) { 876 int padsz = 8 - (sr->reply.chain_offset & 7); 877 (void) smb_mbc_encodef(&sr->reply, "#.", padsz); 878 } 879 ASSERT((sr->reply.chain_offset & 7) == 0); 880 881 /* 882 * Record some statistics: (just tx bytes here) 883 */ 884 atomic_add_64(&sds->sdt_txb, 885 (int64_t)(sr->reply.chain_offset - sr->smb2_reply_hdr)); 886 887 /* 888 * Overwrite the SMB2 header for the response of 889 * this command (possibly part of a compound). 890 * The call adds: SMB2_FLAGS_SERVER_TO_REDIR 891 */ 892 (void) smb2_encode_header(sr, B_TRUE); 893 894 if (sr->smb2_hdr_flags & SMB2_FLAGS_SIGNED) 895 smb2_sign_reply(sr); 896 897 smb2_send_reply(sr); 898 899 /* 900 * Done. Unlink and free. 901 */ 902 sr->sr_async_req = NULL; 903 kmem_free(ar, sizeof (*ar)); 904 905 mutex_enter(&sr->sr_mutex); 906 sr->sr_state = SMB_REQ_STATE_COMPLETED; 907 mutex_exit(&sr->sr_mutex); 908 909 smb_request_free(sr); 910 } 911 912 /* 913 * In preparation for sending an "interim response", save 914 * all the state we'll need to run an async command later, 915 * and assign an "async id" for this (now async) command. 916 * See [MS-SMB2] 3.3.4.2 917 * 918 * If more than one request in a compound request tries to 919 * "go async", we can "say no". See [MS-SMB2] 3.3.4.2 920 * If an operation would require asynchronous processing 921 * but resources are constrained, the server MAY choose to 922 * fail that operation with STATUS_INSUFFICIENT_RESOURCES. 923 * 924 * For simplicity, we further restrict the cases where we're 925 * willing to "go async", and only allow the last command in a 926 * compound to "go async". It happens that this is the only 927 * case where we're actually asked to go async anyway. This 928 * simplification also means there can be at most one command 929 * in a compound that "goes async" (the last one). 930 * 931 * If we agree to "go async", this should return STATUS_PENDING. 932 * Otherwise return STATUS_INSUFFICIENT_RESOURCES for this and 933 * all requests following this request. (See the comments re. 934 * "sticky" smb2_status values in smb2sr_work). 935 * 936 * Note: the Async ID we assign here is arbitrary, and need only 937 * be unique among pending async responses on this connection, so 938 * this just uses an object address as the Async ID. 939 * 940 * Also, the assigned worker is the ONLY thread using this 941 * async request object (sr_async_req) so no locking. 942 */ 943 uint32_t 944 smb2sr_go_async(smb_request_t *sr, 945 smb_sdrc_t (*async_func)(smb_request_t *)) 946 { 947 smb2_async_req_t *ar; 948 949 if (sr->smb2_next_command != 0) 950 return (NT_STATUS_INSUFFICIENT_RESOURCES); 951 952 ASSERT(sr->sr_async_req == NULL); 953 ar = kmem_zalloc(sizeof (*ar), KM_SLEEP); 954 955 /* 956 * Place an interim response in the compound reply. 957 * 958 * Turn on the "async" flag for both the (synchronous) 959 * interim response and the (later) async response, 960 * by storing that in flags before coping into ar. 961 * 962 * The "related" flag should always be off for the 963 * async part because we're no longer operating on a 964 * sequence of commands when we execute that. 965 */ 966 sr->smb2_hdr_flags |= SMB2_FLAGS_ASYNC_COMMAND; 967 sr->smb2_async_id = (uintptr_t)ar; 968 969 ar->ar_func = async_func; 970 ar->ar_cmd_hdr = sr->smb2_cmd_hdr; 971 ar->ar_cmd_len = sr->smb_data.max_bytes - sr->smb2_cmd_hdr; 972 973 ar->ar_cmd_code = sr->smb2_cmd_code; 974 ar->ar_hdr_flags = sr->smb2_hdr_flags & 975 ~SMB2_FLAGS_RELATED_OPERATIONS; 976 ar->ar_messageid = sr->smb2_messageid; 977 ar->ar_pid = sr->smb_pid; 978 ar->ar_tid = sr->smb_tid; 979 ar->ar_uid = sr->smb_uid; 980 981 sr->sr_async_req = ar; 982 983 /* Interim responses are NOT signed. */ 984 sr->smb2_hdr_flags &= ~SMB2_FLAGS_SIGNED; 985 986 return (NT_STATUS_PENDING); 987 } 988 989 int 990 smb2_decode_header(smb_request_t *sr) 991 { 992 uint64_t ssnid; 993 uint32_t pid, tid; 994 uint16_t hdr_len; 995 int rc; 996 997 rc = smb_mbc_decodef( 998 &sr->command, "Nwww..wwllqllq16c", 999 &hdr_len, /* w */ 1000 &sr->smb2_credit_charge, /* w */ 1001 &sr->smb2_chan_seq, /* w */ 1002 /* reserved .. */ 1003 &sr->smb2_cmd_code, /* w */ 1004 &sr->smb2_credit_request, /* w */ 1005 &sr->smb2_hdr_flags, /* l */ 1006 &sr->smb2_next_command, /* l */ 1007 &sr->smb2_messageid, /* q */ 1008 &pid, /* l */ 1009 &tid, /* l */ 1010 &ssnid, /* q */ 1011 sr->smb2_sig); /* 16c */ 1012 if (rc) 1013 return (rc); 1014 1015 if (hdr_len != SMB2_HDR_SIZE) 1016 return (-1); 1017 1018 sr->smb_uid = (uint16_t)ssnid; /* XXX wide UIDs */ 1019 1020 if (sr->smb2_hdr_flags & SMB2_FLAGS_ASYNC_COMMAND) { 1021 sr->smb2_async_id = pid | 1022 ((uint64_t)tid) << 32; 1023 } else { 1024 sr->smb_pid = pid; 1025 sr->smb_tid = (uint16_t)tid; /* XXX wide TIDs */ 1026 } 1027 1028 return (rc); 1029 } 1030 1031 int 1032 smb2_encode_header(smb_request_t *sr, boolean_t overwrite) 1033 { 1034 uint64_t ssnid = sr->smb_uid; 1035 uint64_t pid_tid_aid; /* pid+tid, or async id */ 1036 uint32_t reply_hdr_flags; 1037 int rc; 1038 1039 if (sr->smb2_hdr_flags & SMB2_FLAGS_ASYNC_COMMAND) { 1040 pid_tid_aid = sr->smb2_async_id; 1041 } else { 1042 pid_tid_aid = sr->smb_pid | 1043 ((uint64_t)sr->smb_tid) << 32; 1044 } 1045 reply_hdr_flags = sr->smb2_hdr_flags | SMB2_FLAGS_SERVER_TO_REDIR; 1046 1047 if (overwrite) { 1048 rc = smb_mbc_poke(&sr->reply, 1049 sr->smb2_reply_hdr, 1050 "Nwwlwwllqqq16c", 1051 SMB2_HDR_SIZE, /* w */ 1052 sr->smb2_credit_charge, /* w */ 1053 sr->smb2_status, /* l */ 1054 sr->smb2_cmd_code, /* w */ 1055 sr->smb2_credit_response, /* w */ 1056 reply_hdr_flags, /* l */ 1057 sr->smb2_next_reply, /* l */ 1058 sr->smb2_messageid, /* q */ 1059 pid_tid_aid, /* q */ 1060 ssnid, /* q */ 1061 sr->smb2_sig); /* 16c */ 1062 } else { 1063 rc = smb_mbc_encodef(&sr->reply, 1064 "Nwwlwwllqqq16c", 1065 SMB2_HDR_SIZE, /* w */ 1066 sr->smb2_credit_charge, /* w */ 1067 sr->smb2_status, /* l */ 1068 sr->smb2_cmd_code, /* w */ 1069 sr->smb2_credit_response, /* w */ 1070 reply_hdr_flags, /* l */ 1071 sr->smb2_next_reply, /* l */ 1072 sr->smb2_messageid, /* q */ 1073 pid_tid_aid, /* q */ 1074 ssnid, /* q */ 1075 sr->smb2_sig); /* 16c */ 1076 } 1077 1078 return (rc); 1079 } 1080 1081 void 1082 smb2_send_reply(smb_request_t *sr) 1083 { 1084 1085 if (smb_session_send(sr->session, 0, &sr->reply) == 0) 1086 sr->reply.chain = 0; 1087 } 1088 1089 /* 1090 * This wrapper function exists to help catch calls to smbsr_status() 1091 * (which is SMB1-specific) in common code. See smbsr_status(). 1092 * If the log message below is seen, put a dtrace probe on this 1093 * function with a stack() action to see who is calling the SMB1 1094 * "put error" from common code, and fix it. 1095 */ 1096 void 1097 smbsr_status_smb2(smb_request_t *sr, DWORD status) 1098 { 1099 const char *name; 1100 1101 if (sr->smb2_cmd_code < SMB2__NCMDS) 1102 name = smb2_disp_table[sr->smb2_cmd_code].sdt_name; 1103 else 1104 name = "<unknown>"; 1105 #ifdef DEBUG 1106 cmn_err(CE_NOTE, "smbsr_status called for %s", name); 1107 #endif 1108 1109 smb2sr_put_error_data(sr, status, NULL); 1110 } 1111 1112 void 1113 smb2sr_put_errno(struct smb_request *sr, int errnum) 1114 { 1115 uint32_t status = smb_errno2status(errnum); 1116 smb2sr_put_error_data(sr, status, NULL); 1117 } 1118 1119 void 1120 smb2sr_put_error(smb_request_t *sr, uint32_t status) 1121 { 1122 smb2sr_put_error_data(sr, status, NULL); 1123 } 1124 1125 /* 1126 * Build an SMB2 error response. [MS-SMB2] 2.2.2 1127 */ 1128 void 1129 smb2sr_put_error_data(smb_request_t *sr, uint32_t status, mbuf_chain_t *mbc) 1130 { 1131 DWORD len; 1132 1133 /* 1134 * The common dispatch code writes this when it 1135 * updates the SMB2 header before sending. 1136 */ 1137 sr->smb2_status = status; 1138 1139 /* Rewind to the end of the SMB header. */ 1140 sr->reply.chain_offset = sr->smb2_reply_hdr + SMB2_HDR_SIZE; 1141 1142 /* 1143 * NB: Must provide at least one byte of error data, 1144 * per [MS-SMB2] 2.2.2 1145 */ 1146 if (mbc != NULL && (len = MBC_LENGTH(mbc)) != 0) { 1147 (void) smb_mbc_encodef( 1148 &sr->reply, 1149 "wwlC", 1150 9, /* StructSize */ /* w */ 1151 0, /* reserved */ /* w */ 1152 len, /* l */ 1153 mbc); /* C */ 1154 } else { 1155 (void) smb_mbc_encodef( 1156 &sr->reply, 1157 "wwl.", 1158 9, /* StructSize */ /* w */ 1159 0, /* reserved */ /* w */ 1160 0); /* l. */ 1161 } 1162 } 1163 1164 /* 1165 * smb2sr_lookup_fid 1166 * 1167 * Setup sr->fid_ofile, either inherited from a related command, 1168 * or obtained via FID lookup. Similar inheritance logic as in 1169 * smb2sr_work. 1170 */ 1171 uint32_t 1172 smb2sr_lookup_fid(smb_request_t *sr, smb2fid_t *fid) 1173 { 1174 boolean_t related = sr->smb2_hdr_flags & 1175 SMB2_FLAGS_RELATED_OPERATIONS; 1176 1177 if (related) { 1178 if (sr->fid_ofile == NULL) 1179 return (NT_STATUS_INVALID_PARAMETER); 1180 sr->smb_fid = sr->fid_ofile->f_fid; 1181 return (0); 1182 } 1183 1184 /* 1185 * If we could be sure this is called only once per cmd, 1186 * we could simply ASSERT(sr->fid_ofile == NULL) here. 1187 * However, there are cases where it can be called again 1188 * handling the same command, so let's tolerate that. 1189 */ 1190 if (sr->fid_ofile == NULL) { 1191 sr->smb_fid = (uint16_t)fid->temporal; 1192 sr->fid_ofile = smb_ofile_lookup_by_fid(sr, sr->smb_fid); 1193 } 1194 if (sr->fid_ofile == NULL) 1195 return (NT_STATUS_FILE_CLOSED); 1196 1197 return (0); 1198 } 1199 1200 /* 1201 * smb2_dispatch_stats_init 1202 * 1203 * Initializes dispatch statistics for SMB2. 1204 * See also smb_dispatch_stats_init(), which fills in 1205 * the lower part of the statistics array, from zero 1206 * through SMB_COM_NUM; 1207 */ 1208 void 1209 smb2_dispatch_stats_init(smb_server_t *sv) 1210 { 1211 smb_disp_stats_t *sds = sv->sv_disp_stats2; 1212 smb_kstat_req_t *ksr; 1213 int i; 1214 1215 ksr = ((smbsrv_kstats_t *)sv->sv_ksp->ks_data)->ks_reqs2; 1216 1217 for (i = 0; i < SMB2__NCMDS; i++, ksr++) { 1218 smb_latency_init(&sds[i].sdt_lat); 1219 (void) strlcpy(ksr->kr_name, smb2_disp_table[i].sdt_name, 1220 sizeof (ksr->kr_name)); 1221 } 1222 } 1223 1224 /* 1225 * smb2_dispatch_stats_fini 1226 * 1227 * Frees and destroyes the resources used for statistics. 1228 */ 1229 void 1230 smb2_dispatch_stats_fini(smb_server_t *sv) 1231 { 1232 smb_disp_stats_t *sds = sv->sv_disp_stats2; 1233 int i; 1234 1235 for (i = 0; i < SMB2__NCMDS; i++) 1236 smb_latency_destroy(&sds[i].sdt_lat); 1237 } 1238 1239 void 1240 smb2_dispatch_stats_update(smb_server_t *sv, 1241 smb_kstat_req_t *ksr, int first, int nreq) 1242 { 1243 smb_disp_stats_t *sds = sv->sv_disp_stats2; 1244 int i; 1245 int last; 1246 1247 last = first + nreq - 1; 1248 1249 if ((first < SMB2__NCMDS) && (last < SMB2__NCMDS)) { 1250 for (i = first; i <= last; i++, ksr++) { 1251 ksr->kr_rxb = sds[i].sdt_rxb; 1252 ksr->kr_txb = sds[i].sdt_txb; 1253 mutex_enter(&sds[i].sdt_lat.ly_mutex); 1254 ksr->kr_nreq = sds[i].sdt_lat.ly_a_nreq; 1255 ksr->kr_sum = sds[i].sdt_lat.ly_a_sum; 1256 ksr->kr_a_mean = sds[i].sdt_lat.ly_a_mean; 1257 ksr->kr_a_stddev = 1258 sds[i].sdt_lat.ly_a_stddev; 1259 ksr->kr_d_mean = sds[i].sdt_lat.ly_d_mean; 1260 ksr->kr_d_stddev = 1261 sds[i].sdt_lat.ly_d_stddev; 1262 sds[i].sdt_lat.ly_d_mean = 0; 1263 sds[i].sdt_lat.ly_d_nreq = 0; 1264 sds[i].sdt_lat.ly_d_stddev = 0; 1265 sds[i].sdt_lat.ly_d_sum = 0; 1266 mutex_exit(&sds[i].sdt_lat.ly_mutex); 1267 } 1268 } 1269 } 1270