Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

849 lines
28 KiB

  1. /*++
  2. Copyright (c) 1989 Microsoft Corporation
  3. Module Name:
  4. exsessup.c
  5. Abstract:
  6. This module implements the routines for setting up a session using the
  7. the securitt negotiation mechanism ( post NT40 ).
  8. Author:
  9. Balan Sethu Raman [SethuR] 7-March-1995
  10. Revision History:
  11. Notes:
  12. The extended session setup is used in the new security negotiation scheme
  13. which involves multiple round trips to the server before the user can be
  14. successfully authenticated and a session established.
  15. In the modified scheme the negotiate returns a BLOB which is passed to the
  16. client side security package to initiate the session setup procedure. The
  17. BLOB returned by the server contains an encoding of the security packages
  18. supported by the server.
  19. The client side security package when presented with this BLOB chooses a
  20. security package and encodes the client credentials in the form of a
  21. BLOB which is shipped to the server using the EXTENDED_SESSION_SETUP_ANDX
  22. SMB.
  23. The server has one of three responses to an EXTENDED_SESSION_SETUP_ANDX
  24. SMB presented by the client.
  25. 1) The server has enough information to establish the session.
  26. 2) The server cannot proceed with the session setup because of an
  27. error in the information presented by the client or otherwise.
  28. 3) The security package on the server needs an additional round trip
  29. before the session setup can be established. This is especially
  30. true of new security packages which support mutual authentication
  31. between the client and server.
  32. In the first two cases no further round trips are required. The action taken
  33. on the client side depends upon whether the server returned a BLOB. If the
  34. server returned a BLOB it must be presented to the client side security
  35. package to complete the session setup procedure.
  36. In the case of (3) the BLOB returned by the server must be presented to the
  37. client and the BLOB generated by the security package must be shipped back
  38. to the server.
  39. In the SMBCE_EXTENDED_SESSION_SETUP_EXCHANGE the following parameters support
  40. the protocol outlined above. A buffer with maximum server buffer size is allocated,
  41. locked and a MDL created as part of the exchange initialization. This buffer is
  42. used to hold the server response BLOB. Notice that this avoids redundant copying
  43. and handles all the known cases.
  44. --*/
  45. #include "precomp.h"
  46. #pragma hdrstop
  47. #include "exsessup.h"
  48. #ifdef ALLOC_PRAGMA
  49. #pragma alloc_text(PAGE, SmbCeInitializeExtendedSessionSetupExchange)
  50. #pragma alloc_text(PAGE, SmbCeDiscardExtendedSessionSetupExchange)
  51. #pragma alloc_text(PAGE, SmbExtSecuritySessionSetupExchangeStart)
  52. #pragma alloc_text(PAGE, SmbExtSecuritySessionSetupExchangeSendCompletionHandler)
  53. #endif
  54. // this string is used to test whether the server really supports security signature.
  55. // if the server returns back the deferent string on SMB header security signature of the
  56. // extended session setup response from the client sends out on the request, the server
  57. // does support security signature.
  58. CHAR InitialSecuritySignature[] = {"BSRSPYL "};
  59. extern BOOLEAN MRxSmbSecuritySignaturesEnabled;
  60. //
  61. // The Bug check file id for this module
  62. //
  63. #define BugCheckFileId (RDBSS_BUG_CHECK_SMB_NETROOT)
  64. //
  65. // The local debug trace level
  66. //
  67. #define Dbg (DEBUG_TRACE_DISPATCH)
  68. //
  69. // Forward declarations ...
  70. //
  71. NTSTATUS
  72. SmbCeInitializeExtendedSessionSetupExchange(
  73. PSMB_EXCHANGE* pExchangePtr,
  74. PMRX_V_NET_ROOT pVNetRoot)
  75. /*++
  76. Routine Description:
  77. This routine initializes an instance of a session setup exchange.
  78. Arguments:
  79. pExchange - the exchange instance
  80. pVNetRoot - the MRX_V_NET_ROOT instance associated with the exchange.
  81. Return Value:
  82. NTSTATUS - The return status for the operation
  83. --*/
  84. {
  85. NTSTATUS Status;
  86. PSMB_EXTENDED_SESSION_SETUP_EXCHANGE pExtSessionSetupExchange;
  87. PAGED_CODE();
  88. ASSERT((pExchangePtr == NULL) ||
  89. ((*pExchangePtr)->Type == EXTENDED_SESSION_SETUP_EXCHANGE));
  90. Status = SmbCeInitializeExchange(
  91. pExchangePtr,
  92. NULL,
  93. pVNetRoot,
  94. EXTENDED_SESSION_SETUP_EXCHANGE,
  95. &ExtendedSessionSetupExchangeDispatch);
  96. if (Status == STATUS_SUCCESS) {
  97. PSMBCEDB_SERVER_ENTRY pServerEntry;
  98. pServerEntry = SmbCeGetExchangeServerEntry(*pExchangePtr);
  99. pExtSessionSetupExchange = (PSMB_EXTENDED_SESSION_SETUP_EXCHANGE)
  100. (*pExchangePtr);
  101. pExtSessionSetupExchange->SmbCeFlags |= SMBCE_EXCHANGE_TIMED_RECEIVE_OPERATION;
  102. // Allocate the buffer to hold the server response.
  103. pExtSessionSetupExchange->BufferLength =
  104. pServerEntry->Server.MaximumBufferSize;
  105. pExtSessionSetupExchange->pActualBuffer = RxAllocatePoolWithTag(
  106. PagedPool,
  107. (pExtSessionSetupExchange->BufferLength + TRANSPORT_HEADER_SIZE),
  108. MRXSMB_KERBEROS_POOLTAG);
  109. pExtSessionSetupExchange->pServerResponseBlob = NULL;
  110. pExtSessionSetupExchange->ServerResponseBlobLength = 0;
  111. pExtSessionSetupExchange->Reparse = TRUE;
  112. if (pExtSessionSetupExchange->pActualBuffer != NULL) {
  113. (PCHAR) pExtSessionSetupExchange->pBuffer =
  114. (PCHAR) pExtSessionSetupExchange->pActualBuffer + TRANSPORT_HEADER_SIZE;
  115. RxAllocateHeaderMdl(
  116. pExtSessionSetupExchange->pBuffer,
  117. pExtSessionSetupExchange->BufferLength,
  118. pExtSessionSetupExchange->pBufferAsMdl
  119. );
  120. if (pExtSessionSetupExchange->pBufferAsMdl != NULL) {
  121. RxProbeAndLockHeaderPages(
  122. pExtSessionSetupExchange->pBufferAsMdl,
  123. KernelMode,
  124. IoModifyAccess,
  125. Status);
  126. } else {
  127. Status = STATUS_INSUFFICIENT_RESOURCES;
  128. }
  129. } else {
  130. Status = STATUS_INSUFFICIENT_RESOURCES;
  131. }
  132. if (Status != STATUS_SUCCESS) {
  133. if (pExtSessionSetupExchange->pBufferAsMdl != NULL) {
  134. IoFreeMdl(pExtSessionSetupExchange->pBufferAsMdl);
  135. }
  136. if (pExtSessionSetupExchange->pActualBuffer != NULL) {
  137. RxFreePool(pExtSessionSetupExchange->pActualBuffer);
  138. }
  139. SmbCePrepareExchangeForReuse(*pExchangePtr);
  140. }
  141. }
  142. return Status;
  143. }
  144. VOID
  145. SmbCeDiscardExtendedSessionSetupExchange(
  146. PSMB_EXTENDED_SESSION_SETUP_EXCHANGE pExtSessionSetupExchange)
  147. /*++
  148. Routine Description:
  149. This routine discards an instance of a session setup exchange.
  150. Arguments:
  151. pExchange - the exchange instance
  152. --*/
  153. {
  154. PAGED_CODE();
  155. if (pExtSessionSetupExchange->pBufferAsMdl != NULL) {
  156. RxUnlockHeaderPages(pExtSessionSetupExchange->pBufferAsMdl);
  157. IoFreeMdl(pExtSessionSetupExchange->pBufferAsMdl);
  158. }
  159. if (pExtSessionSetupExchange->pActualBuffer != NULL) {
  160. RxFreePool(pExtSessionSetupExchange->pActualBuffer);
  161. }
  162. if (pExtSessionSetupExchange->pServerResponseBlob != NULL) {
  163. RxFreePool(pExtSessionSetupExchange->pServerResponseBlob);
  164. }
  165. // Normally discrading the exchange results in the session state being
  166. // updated. In order to avoid race conditions between those exchanges
  167. // which are awaiting this construction and the updating of the session
  168. // state it is done locally. COnsequently the exchange state needs to be
  169. // updated so the the discard routine does not attempt it again.
  170. pExtSessionSetupExchange->SmbCeFlags &= ~SMBCE_EXCHANGE_SESSION_CONSTRUCTOR;
  171. SmbCeDiscardExchange(
  172. (PSMB_EXCHANGE)pExtSessionSetupExchange);
  173. }
  174. NTSTATUS
  175. SmbExtSecuritySessionSetupExchangeStart(
  176. PSMB_EXCHANGE pExchange)
  177. /*++
  178. Routine Description:
  179. This is the start routine for net root construction exchanges. This initiates the
  180. construction of the appropriate SMB's if required.
  181. Arguments:
  182. pExchange - the exchange instance
  183. Return Value:
  184. RXSTATUS - The return status for the operation
  185. --*/
  186. {
  187. NTSTATUS Status;
  188. PSMB_EXTENDED_SESSION_SETUP_EXCHANGE pExtSessionSetupExchange;
  189. PSMB_HEADER pSmbHeader;
  190. PREQ_NT_EXTENDED_SESSION_SETUP_ANDX pSessionSetupRequest;
  191. PGENERIC_ANDX pGenericAndX;
  192. PSMBCEDB_SERVER_ENTRY pServerEntry;
  193. PSMBCEDB_SESSION_ENTRY pSessionEntry;
  194. ULONG SmbBufferUnconsumed;
  195. USHORT Flags2 = 0;
  196. PAGED_CODE();
  197. pServerEntry = SmbCeGetExchangeServerEntry(pExchange);
  198. pSessionEntry = SmbCeGetExchangeSessionEntry(pExchange);
  199. pExtSessionSetupExchange = (PSMB_EXTENDED_SESSION_SETUP_EXCHANGE)pExchange;
  200. if (!pExtSessionSetupExchange->FirstSessionSetup) {
  201. if (pSessionEntry->Header.State == SMBCEDB_ACTIVE) {
  202. return STATUS_SUCCESS;
  203. } else {
  204. return STATUS_USER_SESSION_DELETED;
  205. }
  206. }
  207. ASSERT((pExtSessionSetupExchange->Type == EXTENDED_SESSION_SETUP_EXCHANGE) &&
  208. (pExtSessionSetupExchange->pBuffer != NULL) &&
  209. (pExtSessionSetupExchange->pBufferAsMdl != NULL));
  210. SmbCeLog(("ExtSecSessSetup - %lx %lx\n",
  211. pExtSessionSetupExchange->pServerResponseBlob, pSessionEntry));
  212. SmbLog(LOG,
  213. SmbExtSecuritySessionSetupExchangeStart,
  214. LOGPTR(pExtSessionSetupExchange->pServerResponseBlob)
  215. LOGPTR(pSessionEntry));
  216. pSmbHeader = (PSMB_HEADER)(pExtSessionSetupExchange->pBuffer);
  217. // Fill in the buffer header
  218. pSessionSetupRequest = (PREQ_NT_EXTENDED_SESSION_SETUP_ANDX)(pSmbHeader + 1);
  219. pGenericAndX = (PGENERIC_ANDX)pSessionSetupRequest;
  220. SmbBufferUnconsumed = pExtSessionSetupExchange->BufferLength - sizeof(SMB_HEADER);
  221. Flags2 |= (SMB_FLAGS2_UNICODE |
  222. SMB_FLAGS2_KNOWS_EAS |
  223. SMB_FLAGS2_KNOWS_LONG_NAMES |
  224. SMB_FLAGS2_NT_STATUS |
  225. SMB_FLAGS2_EXTENDED_SECURITY);
  226. *((PULONG)&pSmbHeader->Protocol) = SMB_HEADER_PROTOCOL;
  227. pSmbHeader->Flags = (SMB_FLAGS_CASE_INSENSITIVE | SMB_FLAGS_CANONICALIZED_PATHS);
  228. pSmbHeader->Flags2 = Flags2;
  229. pSmbHeader->Pid = MRXSMB_PROCESS_ID;
  230. pSmbHeader->Uid = pSessionEntry->Session.UserId;
  231. pSmbHeader->Tid = 0;
  232. pSmbHeader->ErrorClass = 0;
  233. pSmbHeader->Reserved = 0;
  234. pSmbHeader->Command = SMB_COM_SESSION_SETUP_ANDX;
  235. SmbPutUshort(&pSmbHeader->Error,0);
  236. if (MRxSmbSecuritySignaturesEnabled) {
  237. pSmbHeader->Flags2 |= SMB_FLAGS2_SMB_SECURITY_SIGNATURE;
  238. }
  239. // Build the session setup and x.
  240. Status = SMBCE_SERVER_DIALECT_DISPATCH(
  241. &pServerEntry->Server,
  242. BuildSessionSetup,
  243. (pExchange,
  244. pGenericAndX,
  245. &SmbBufferUnconsumed));
  246. if (Status == STATUS_SUCCESS) {
  247. // Update the buffer for the construction of the following SMB.
  248. SmbPutUshort(
  249. &pSessionSetupRequest->AndXOffset,
  250. (USHORT)(pExtSessionSetupExchange->BufferLength - SmbBufferUnconsumed));
  251. pSessionSetupRequest->AndXCommand = SMB_COM_NO_ANDX_COMMAND;
  252. pSessionSetupRequest->AndXReserved = 0;
  253. if (pServerEntry->SecuritySignaturesEnabled &&
  254. !pServerEntry->SecuritySignaturesActive) {
  255. RtlCopyMemory(pSmbHeader->SecuritySignature,InitialSecuritySignature,SMB_SECURITY_SIGNATURE_LENGTH);
  256. }
  257. }
  258. if (Status == STATUS_SUCCESS) {
  259. Status = SmbCeTranceive(
  260. pExchange,
  261. (RXCE_SEND_PARTIAL | RXCE_SEND_SYNCHRONOUS),
  262. pExtSessionSetupExchange->pBufferAsMdl,
  263. (pExtSessionSetupExchange->BufferLength -
  264. SmbBufferUnconsumed));
  265. RxDbgTrace( 0, Dbg, ("Net Root SmbCeTranceive returned %lx\n",Status));
  266. }
  267. return Status;
  268. }
  269. NTSTATUS
  270. ParseExtSecuritySessionSetupResponse(
  271. IN PSMB_EXTENDED_SESSION_SETUP_EXCHANGE pExtSessionSetupExchange,
  272. IN ULONG BytesIndicated,
  273. IN ULONG BytesAvailable,
  274. IN PSMB_HEADER pSmbHeader)
  275. /*++
  276. Routine Description:
  277. This is the routine used to parse the extended session setup response from
  278. the server.
  279. Arguments:
  280. pExtSessionSetupExchange -- the exchange instance
  281. BytesIndicated -- the number of bytes indicated
  282. BytesAvailable -- the total number of bytes sent by the server
  283. pSmbHeader -- the SMB header ( beginning of the response)
  284. Return Value:
  285. NTSTATUS - The return status for the operation
  286. --*/
  287. {
  288. NTSTATUS Status;
  289. ULONG ResponseLength;
  290. PRESP_NT_EXTENDED_SESSION_SETUP_ANDX pSessionSetupResponse;
  291. PSMBCEDB_SESSION_ENTRY pSessionEntry;
  292. if (BytesIndicated < sizeof(SMB_HEADER) + 1) {
  293. // Abort the exchange. No further processing can be done.
  294. return STATUS_INVALID_NETWORK_RESPONSE;
  295. }
  296. pSessionEntry = SmbCeGetExchangeSessionEntry(pExtSessionSetupExchange);
  297. pSessionSetupResponse = (PRESP_NT_EXTENDED_SESSION_SETUP_ANDX)(pSmbHeader + 1);
  298. if ((pSessionSetupResponse->WordCount != 4) &&
  299. (pSessionSetupResponse->WordCount != 0)) {
  300. return STATUS_INVALID_NETWORK_RESPONSE;
  301. }
  302. // Parse the header and extract the status. The status has special significance
  303. // for further processing. If the server returns a BLOB and
  304. // STATUS_MORE_PROCESSING_REQUIRED additional round trips are required.
  305. pExtSessionSetupExchange->Status = GetSmbResponseNtStatus(pSmbHeader,(PSMB_EXCHANGE)pExtSessionSetupExchange);
  306. // Mask the errors returned by the security packagte on the server
  307. if (pExtSessionSetupExchange->Status == STATUS_INVALID_HANDLE) {
  308. pExtSessionSetupExchange->Status = STATUS_INVALID_NETWORK_RESPONSE;
  309. }
  310. //if no blob came back.....just get out
  311. if (pSessionSetupResponse->WordCount == 0) {
  312. pExtSessionSetupExchange->ServerResponseBlobLength = 0;
  313. return STATUS_SUCCESS;
  314. }
  315. // Squirrel away the UID on the first response. This UID needs to be used in
  316. // subsequent trips to complete the session establishment as it identifies
  317. // the session to the server.
  318. pSessionEntry->Session.UserId = pSmbHeader->Uid;
  319. if (FlagOn(SmbGetUshort(&pSessionSetupResponse->Action), SMB_SETUP_GUEST)) {
  320. pSessionEntry->Session.Flags |= SMBCE_SESSION_FLAGS_GUEST_SESSION;
  321. }
  322. RxDbgTrace( 0, (DEBUG_TRACE_ALWAYS), ("ParseExtSecuritySessionSetupResponse BytesIndicated %ld\n",BytesIndicated));
  323. RxDbgTrace( 0, (DEBUG_TRACE_ALWAYS), ("ParseExtSecuritySessionSetupResponse BytesAvailable %ld\n",BytesAvailable));
  324. // The bytes indicated should be atleast cover the SMB_HEADER and the
  325. // session setup response ( fixed portion )
  326. ResponseLength = sizeof(SMB_HEADER) +
  327. FIELD_OFFSET(
  328. RESP_NT_EXTENDED_SESSION_SETUP_ANDX,
  329. Buffer);
  330. if (BytesIndicated > ResponseLength) {
  331. // Compute the extended session setup response length.
  332. pExtSessionSetupExchange->ResponseLength =
  333. ResponseLength +
  334. SmbGetUshort(
  335. &pSessionSetupResponse->ByteCount);
  336. RxDbgTrace(0,Dbg,("Kerberos session setup response length %ld\n",pExtSessionSetupExchange->ResponseLength));
  337. if (BytesIndicated < pExtSessionSetupExchange->ResponseLength) {
  338. // Set up the response for copying the data.
  339. if (pExtSessionSetupExchange->ResponseLength > pExtSessionSetupExchange->BufferLength) {
  340. Status = STATUS_BUFFER_OVERFLOW;
  341. } else {
  342. Status = pExtSessionSetupExchange->Reparse
  343. ? STATUS_MORE_PROCESSING_REQUIRED
  344. : STATUS_INVALID_NETWORK_RESPONSE;
  345. }
  346. pExtSessionSetupExchange->Status = Status;
  347. } else {
  348. // set up the offsets in the response.
  349. pExtSessionSetupExchange->ServerResponseBlobOffset =
  350. sizeof(SMB_HEADER) +
  351. FIELD_OFFSET(
  352. RESP_NT_EXTENDED_SESSION_SETUP_ANDX,
  353. Buffer);
  354. pExtSessionSetupExchange->ServerResponseBlobLength =
  355. pSessionSetupResponse->SecurityBlobLength;
  356. // Copy the response onto the buffer associated with the exchange.
  357. RtlCopyMemory(
  358. pExtSessionSetupExchange->pBuffer,
  359. pSmbHeader,
  360. pExtSessionSetupExchange->ResponseLength);
  361. Status = STATUS_SUCCESS;
  362. }
  363. } else {
  364. // Abort the exchange. No further processing can be done.
  365. Status = STATUS_INVALID_NETWORK_RESPONSE;
  366. pExtSessionSetupExchange->Status = Status;
  367. }
  368. return Status;
  369. }
  370. NTSTATUS
  371. SmbExtSecuritySessionSetupExchangeReceive(
  372. IN struct _SMB_EXCHANGE *pExchange, // The exchange instance
  373. IN ULONG BytesIndicated,
  374. IN ULONG BytesAvailable,
  375. OUT ULONG *pBytesTaken,
  376. IN PSMB_HEADER pSmbHeader,
  377. OUT PMDL *pDataBufferPointer,
  378. OUT PULONG pDataSize,
  379. IN ULONG ReceiveFlags)
  380. /*++
  381. Routine Description:
  382. This is the recieve indication handling routine for net root construction exchanges
  383. Arguments:
  384. pExchange - the exchange instance
  385. BytesIndicated - the number of bytes indicated
  386. Bytes Available - the number of bytes available
  387. pBytesTaken - the number of bytes consumed
  388. pSmbHeader - the byte buffer
  389. pDataBufferPointer - the buffer into which the remaining data is to be copied.
  390. pDataSize - the buffer size.
  391. Return Value:
  392. RXSTATUS - The return status for the operation
  393. Notes:
  394. This routine is called at DPC level.
  395. --*/
  396. {
  397. NTSTATUS Status;
  398. PSMB_EXTENDED_SESSION_SETUP_EXCHANGE pExtSessionSetupExchange;
  399. PSMBCEDB_SERVER_ENTRY pServerEntry = SmbCeGetExchangeServerEntry(pExchange);
  400. ULONG SessionSetupResponseLength = 0;
  401. pExtSessionSetupExchange = (PSMB_EXTENDED_SESSION_SETUP_EXCHANGE)pExchange;
  402. if (pServerEntry->SecuritySignaturesEnabled &&
  403. !pServerEntry->SecuritySignaturesActive &&
  404. RtlCompareMemory(pSmbHeader->SecuritySignature,
  405. InitialSecuritySignature,
  406. SMB_SECURITY_SIGNATURE_LENGTH) != SMB_SECURITY_SIGNATURE_LENGTH) {
  407. pExtSessionSetupExchange->pResumptionContext->SecuritySignatureReturned = TRUE;
  408. }
  409. // Parse the response.
  410. Status = ParseExtSecuritySessionSetupResponse(
  411. pExtSessionSetupExchange,
  412. BytesIndicated,
  413. BytesAvailable,
  414. pSmbHeader);
  415. if (Status != STATUS_MORE_PROCESSING_REQUIRED) {
  416. *pBytesTaken = BytesAvailable;
  417. Status = STATUS_SUCCESS;
  418. } else {
  419. *pBytesTaken = 0;
  420. *pDataBufferPointer = pExtSessionSetupExchange->pBufferAsMdl;
  421. *pDataSize = pExtSessionSetupExchange->ResponseLength;
  422. }
  423. return Status;
  424. }
  425. NTSTATUS
  426. SmbExtSecuritySessionSetupExchangeSendCompletionHandler(
  427. IN PSMB_EXCHANGE pExchange, // The exchange instance
  428. IN PMDL pXmitBuffer,
  429. IN NTSTATUS SendCompletionStatus)
  430. /*++
  431. Routine Description:
  432. This is the send call back indication handling routine for net root construction exchanges
  433. Arguments:
  434. pExchange - the exchange instance
  435. Return Value:
  436. RXSTATUS - The return status for the operation
  437. --*/
  438. {
  439. return STATUS_SUCCESS;
  440. }
  441. NTSTATUS
  442. SmbExtSecuritySessionSetupExchangeCopyDataHandler(
  443. IN PSMB_EXCHANGE pExchange, // The exchange instance
  444. IN PMDL pCopyDataBuffer,
  445. IN ULONG DataSize)
  446. /*++
  447. Routine Description:
  448. This is the copy data handling routine for net root construction exchanges
  449. Arguments:
  450. pExchange - the exchange instance
  451. Return Value:
  452. RXSTATUS - The return status for the operation
  453. --*/
  454. {
  455. NTSTATUS Status;
  456. PSMB_EXTENDED_SESSION_SETUP_EXCHANGE pExtSessionSetupExchange;
  457. PSMB_HEADER pSmbHeader;
  458. pExtSessionSetupExchange = (PSMB_EXTENDED_SESSION_SETUP_EXCHANGE)pExchange;
  459. pSmbHeader = (PSMB_HEADER)(pExtSessionSetupExchange->pBuffer);
  460. Status = ParseExtSecuritySessionSetupResponse(
  461. pExtSessionSetupExchange,
  462. DataSize,
  463. DataSize,
  464. pSmbHeader);
  465. // At this time the parse routine cannot return STATUS_MORE_PROCESSING_REQUIRED
  466. // as the entire response has been consumed.
  467. if (Status == STATUS_MORE_PROCESSING_REQUIRED) {
  468. DbgPrint("Mapping Incomplete Server Response to Invalid Response\n");
  469. SmbLogError(Status,
  470. LOG,
  471. SmbExtSecuritySessionSetupExchangeCopyDataHandler,
  472. LOGPTR(pExtSessionSetupExchange));
  473. pExtSessionSetupExchange->Status = STATUS_INVALID_NETWORK_RESPONSE;
  474. }
  475. return STATUS_SUCCESS;
  476. }
  477. NTSTATUS
  478. SmbExtSecuritySessionSetupExchangeFinalize(
  479. PSMB_EXCHANGE pExchange,
  480. BOOLEAN *pPostFinalize)
  481. /*++
  482. Routine Description:
  483. This routine finalkzes the construct net root exchange. It resumes the RDBSS by invoking
  484. the call back and discards the exchange
  485. Arguments:
  486. pExchange - the exchange instance
  487. CurrentIrql - the current interrupt request level
  488. pPostFinalize - a pointer to a BOOLEAN if the request should be posted
  489. Return Value:
  490. RXSTATUS - The return status for the operation
  491. --*/
  492. {
  493. NTSTATUS Status;
  494. PSMB_EXTENDED_SESSION_SETUP_EXCHANGE pExtSessionSetupExchange;
  495. PSMBCE_RESUMPTION_CONTEXT pResumptionContext;
  496. pExtSessionSetupExchange = (PSMB_EXTENDED_SESSION_SETUP_EXCHANGE)pExchange;
  497. if (!pExtSessionSetupExchange->RequestPosted) {
  498. pExtSessionSetupExchange->RequestPosted = TRUE;
  499. *pPostFinalize = TRUE;
  500. return STATUS_SUCCESS;
  501. } else {
  502. // reset the flag since the exchange will be reused
  503. pExtSessionSetupExchange->RequestPosted = FALSE;
  504. *pPostFinalize = FALSE;
  505. }
  506. // Determine if further processing is required. If not finalize the
  507. // session entry.
  508. RxDbgTrace(0,Dbg,
  509. ("SmbExtSecuritySessionSetupExchangeFinalize: pESSExchange->Status = %lx\n",pExtSessionSetupExchange->Status));
  510. // If the server returned STATUS_MORE_PROCESSING_REQUIRED and a BLOB was present
  511. // another trip to the server is required and we start all over again.
  512. if ((Status = pExtSessionSetupExchange->Status) != STATUS_MORE_PROCESSING_REQUIRED) {
  513. // The server returned an error other than STATUS_MORE_PROCESSING_REQUIRED
  514. // The session establishment can be completed based on whether the server
  515. // returned a BLOB. If a BLOB was returned it needs to be passed to the
  516. // local security package. This will enable the local security package to
  517. // either complete the session establishment successfully or to extract
  518. // extended error information from the BLOB. This can in turn be used to
  519. // propogate more meaningful errors to the client.
  520. if (Status == STATUS_SUCCESS &&
  521. pExtSessionSetupExchange->ServerResponseBlobLength != 0) {
  522. PVOID pServerResponseBlob;
  523. // If we are not going back to the server an additional copy of the
  524. // server response BLOB is not required.
  525. pServerResponseBlob =
  526. ((PBYTE)pExtSessionSetupExchange->pBuffer +
  527. pExtSessionSetupExchange->ServerResponseBlobOffset);
  528. Status = ValidateServerExtendedSessionSetupResponse(
  529. pExtSessionSetupExchange,
  530. pServerResponseBlob,
  531. pExtSessionSetupExchange->ServerResponseBlobLength);
  532. }
  533. } else {
  534. // Make a copy of the server response blob so that the new session setup SMB
  535. // can be constructed
  536. if (pExtSessionSetupExchange->ServerResponseBlobLength != 0) {
  537. if (pExtSessionSetupExchange->pServerResponseBlob != NULL) {
  538. RxFreePool(pExtSessionSetupExchange->pServerResponseBlob);
  539. }
  540. pExtSessionSetupExchange->pServerResponseBlob =
  541. RxAllocatePoolWithTag(
  542. PagedPool,
  543. pExtSessionSetupExchange->ServerResponseBlobLength,
  544. MRXSMB_KERBEROS_POOLTAG);
  545. if (pExtSessionSetupExchange->pServerResponseBlob != NULL) {
  546. RtlCopyMemory(
  547. pExtSessionSetupExchange->pServerResponseBlob,
  548. ((PBYTE)pExtSessionSetupExchange->pBuffer +
  549. pExtSessionSetupExchange->ServerResponseBlobOffset),
  550. pExtSessionSetupExchange->ServerResponseBlobLength);
  551. } else {
  552. Status = STATUS_INSUFFICIENT_RESOURCES;
  553. }
  554. }
  555. }
  556. if (Status == STATUS_MORE_PROCESSING_REQUIRED) {
  557. PMRX_V_NET_ROOT pVNetRoot;
  558. USHORT SmbCeFlags;
  559. PRX_CONTEXT RxContext;
  560. pVNetRoot = pExtSessionSetupExchange->SmbCeContext.pVNetRoot;
  561. RxContext = pExtSessionSetupExchange->RxContext;
  562. // This is required so that the session state is not effected as the exchange
  563. // is prepared for reuse. This will enable us to avoid redundant initialization
  564. // as well as to carry over the state from one trip to another easily.
  565. ClearFlag(
  566. pExtSessionSetupExchange->SmbCeFlags,
  567. SMBCE_EXCHANGE_SESSION_CONSTRUCTOR);
  568. SmbCePrepareExchangeForReuse((PSMB_EXCHANGE)pExtSessionSetupExchange);
  569. // Note: By invoking SmbCeInitializeExchange as opposed to
  570. // SmbCeInitializeExtendedSessionSetupExchange only the connection engine
  571. // portion of the exchange is initialized. This will enable us to carry
  572. // over the state ( the server response BLOB ) from one trip to another
  573. // easily.
  574. Status = SmbCeInitializeExchange(
  575. (PSMB_EXCHANGE *)&pExtSessionSetupExchange,
  576. NULL,
  577. pVNetRoot,
  578. EXTENDED_SESSION_SETUP_EXCHANGE,
  579. &ExtendedSessionSetupExchangeDispatch);
  580. if (Status == STATUS_SUCCESS) {
  581. pExtSessionSetupExchange->SmbCeFlags |= SMBCE_EXCHANGE_TIMED_RECEIVE_OPERATION;
  582. pExtSessionSetupExchange->RxContext = RxContext;
  583. // Avoid duplicate counting during Exchange Initialization.
  584. SmbCeDecrementActiveExchangeCount();
  585. // Set the session echange state to SMBCE_EXCHANGE_SESSION_INITIALIZED
  586. // so that the connection engine does not queue up the request the normal
  587. // way. This will force the connection engine to initiate immediately.
  588. pExtSessionSetupExchange->SmbCeState = SMBCE_EXCHANGE_SESSION_INITIALIZED;
  589. Status = SmbCeInitiateExchange((PSMB_EXCHANGE)pExtSessionSetupExchange);
  590. }
  591. }
  592. if (Status != STATUS_PENDING) {
  593. PSMBCEDB_SERVER_ENTRY pServerEntry;
  594. PSMBCEDB_SESSION_ENTRY pSessionEntry;
  595. SMBCEDB_OBJECT_STATE SessionState;
  596. RxDbgTrace(0,Dbg,("Kerberos Exchange Session Final Status(%lx)\n",Status));
  597. pServerEntry = SmbCeGetExchangeServerEntry(pExtSessionSetupExchange);
  598. pSessionEntry = SmbCeGetExchangeSessionEntry(pExtSessionSetupExchange);
  599. pResumptionContext = pExtSessionSetupExchange->pResumptionContext;
  600. // Tear down the exchange instance ...
  601. SmbCeDiscardExtendedSessionSetupExchange(pExtSessionSetupExchange);
  602. if (pResumptionContext != NULL) {
  603. pResumptionContext->Status = Status;
  604. SmbCeResume(pResumptionContext);
  605. }
  606. }
  607. return STATUS_SUCCESS;
  608. }
  609. SMB_EXCHANGE_DISPATCH_VECTOR
  610. ExtendedSessionSetupExchangeDispatch =
  611. {
  612. SmbExtSecuritySessionSetupExchangeStart,
  613. SmbExtSecuritySessionSetupExchangeReceive,
  614. SmbExtSecuritySessionSetupExchangeCopyDataHandler,
  615. NULL,
  616. SmbExtSecuritySessionSetupExchangeFinalize,
  617. NULL
  618. };
  619.