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.

831 lines
30 KiB

  1. /*++ BUILD Version: 0009 // Increment this if a change has global effects
  2. Copyright (c) 1987-1993 Microsoft Corporation
  3. Module Name:
  4. smbcxchng.h
  5. Abstract:
  6. This is the include file that defines all constants and types for
  7. SMB exchange implementation.
  8. Author:
  9. Balan Sethu Raman (SethuR) 06-Feb-95 Created
  10. Notes:
  11. An exchange is the core abstarction on which the SMB connection engine and
  12. the mini RDR are implemented. It encapsulates the notion of sending an SMB to
  13. the server and receiving the associated response, i.e, exchanging an SMB and
  14. hence the name.
  15. The exchange of an SMB with the server involves the following steps ....
  16. 1) Submitting the formatted SMB buffer for transmission.
  17. 2) Processing a send complete indication which ensures that at the
  18. transport level the SMB has been sent to the server.
  19. 3) Processing the receive indication which contains all/part of the
  20. response sent by the server.
  21. 4) Copying additional data not indicated by the transport
  22. There are a number of variations on this theme. For example there are certain
  23. SMB's for which no response is expected, e.g., write mailslots and there are
  24. certain SMB's which are inherently multi part in nature, TRANSACT smb's.
  25. In addition the steps outlined above will not always happen in that order. The
  26. precise sequence of events is dictated by the underlying transport chosen and
  27. the network conditions. It is this dependency that makes the implementation
  28. of exchanges challenging.
  29. The two primary goals that the current implementation was designed for are (1)
  30. performance and (2) encapsulation of transport dependencies. Goal(1) is
  31. important because this constitutes an integral part of the code path for
  32. exchanging any packet with the server. Goal (2) is important to ensure
  33. customization of the Rdr for different transports. This encapsulation provides
  34. a convenient vehicle for isolating SMB protocol level decisions from transport
  35. level decisons as much as possible.
  36. In addition the following goals were used to guide the implementation process ...
  37. 1) The exchange implementation must be able to handle asynchronous
  38. operations and synchronous operations well. The trade offs were made in
  39. favour of asynchronous operations as and when required.
  40. 2) Sufficient infrastructure support must be provided so as to ease the
  41. implementation of different flavours of exchanges.
  42. The SMB_EXCHANGE consists of a dispatch vector with the following functions
  43. 1) Start -- to initiate the exchange
  44. 2) Receive -- to handle response indications from the server
  45. 3) CopyDataHandler -- to handle portions of the response not indicated
  46. 4) SendCompletionHandler -- to handle send complete indications from the transport.
  47. 5) QuiescentStateHandler -- to handle transitions to a quiescent state, i.e., no
  48. SMB connection engine operations are outstanding.
  49. Most kinds of exchange use the QuiescentStateHandler to finalize the
  50. operation and discard the exchange. However, certain kinds of exchanges
  51. which implement the notion of a macro exchange, i.e., exchange multiple
  52. SMB's use this to delineate different phases of the multiple exchange,
  53. e.g., ORDINARY_EXCHANGE which implements most file io operations.
  54. In addition to the dispatch vector the vanilla exchange consists of state
  55. information to record the current state of the exchange, sufficient context
  56. for resumption and context for handling SMB protocol related operations. The
  57. SMB protocol requires that each SMB sent to the server be stamped with a MID
  58. ( multiplex id. ) in order to distinguish between concurrent SMB exchanges.
  59. The connection engine provides this service.
  60. The exchange also encapsulates a SMBCE_EXCHANGE_CONTEXT instance which
  61. encapsulates all the information required for building a SMB_HEADER.
  62. --*/
  63. #ifndef _SMBXCHNG_H_
  64. #define _SMBXCHNG_H_
  65. typedef enum _SMBCE_STATE_ {
  66. SMBCE_START_IN_PROGRESS,
  67. SMBCE_STARTED,
  68. SMBCE_STOP_IN_PROGRESS,
  69. SMBCE_STOPPED
  70. } SMBCE_STATE, *PSMBCE_STATE;
  71. typedef struct _SMBCE_STARTSTOP_CONTEXT_ {
  72. SMBCE_STATE State;
  73. LONG ActiveExchanges;
  74. KEVENT StopEvent;
  75. PKEVENT pServerEntryTearDownEvent;
  76. LIST_ENTRY SessionSetupRequests;
  77. } SMBCE_STARTSTOP_CONTEXT, *PSMBCE_STARTSTOP_CONTEXT;
  78. extern SMBCE_STARTSTOP_CONTEXT SmbCeStartStopContext;
  79. //
  80. // SMB_PROTOCOL_EXCHANGE dispatch vector function prototypes ..
  81. //
  82. // the initiator or the start routine
  83. typedef
  84. NTSTATUS
  85. (*PSMB_EXCHANGE_START)(
  86. IN struct _SMB_EXCHANGE *pExchange);
  87. // The SMB receive handler
  88. typedef
  89. NTSTATUS
  90. (*PSMB_EXCHANGE_IND_RECEIVE)(
  91. IN struct _SMB_EXCHANGE *pExchange, // The exchange instance
  92. IN ULONG BytesIndicated,
  93. IN ULONG BytesAvailable,
  94. OUT ULONG *BytesTaken,
  95. IN PSMB_HEADER pSmbHeader,
  96. OUT PMDL *pDataBufferPointer, // buffer to copy unindicated data
  97. OUT PULONG pDataSize, // buffer size
  98. IN ULONG ReceiveFlags
  99. );
  100. // the SMB xmit callback
  101. typedef
  102. NTSTATUS
  103. (*PSMB_EXCHANGE_IND_SEND_CALLBACK)(
  104. IN struct _SMB_EXCHANGE *pExchange, // The exchange instance
  105. IN PMDL pDataBuffer,
  106. IN NTSTATUS SendCompletionStatus
  107. );
  108. // the copy data callback for fetching large data
  109. typedef
  110. NTSTATUS
  111. (*PSMB_EXCHANGE_IND_COPY_DATA_CALLBACK)(
  112. IN struct _SMB_EXCHANGE *pExchange, // the exchange instance
  113. IN PMDL pCopyDataBuffer, // the buffer
  114. IN ULONG CopyDataSize // amount of data copied
  115. );
  116. // the finalization routine
  117. // This particular routine has a signature that is NT specific the IRQL
  118. // parameter that is passed in and the notion of posting. This helps consolidate
  119. // the NT transport driver model of indications at DPC level in SmbCeFinalizeExchange.
  120. // On WIN95 the lease restrictive value of IRQL can be passed in.
  121. typedef
  122. NTSTATUS
  123. (*PSMB_EXCHANGE_FINALIZE)(
  124. IN OUT struct _SMB_EXCHANGE *pExchange,
  125. OUT BOOLEAN *pPostRequest);
  126. typedef
  127. NTSTATUS
  128. (*PSMB_EXCHANGE_IND_ASSOCIATED_EXCHANGES_COMPLETION)(
  129. IN OUT struct _SMB_EXCHANGE *pExchange,
  130. OUT BOOLEAN *pPostRequest);
  131. // The Exchange dispatch vector definition
  132. typedef struct _SMB_EXCHANGE_DISPATCH_VECTOR_ {
  133. PSMB_EXCHANGE_START Start;
  134. PSMB_EXCHANGE_IND_RECEIVE Receive;
  135. PSMB_EXCHANGE_IND_COPY_DATA_CALLBACK CopyDataHandler;
  136. PSMB_EXCHANGE_IND_SEND_CALLBACK SendCompletionHandler;
  137. PSMB_EXCHANGE_FINALIZE Finalize;
  138. PSMB_EXCHANGE_IND_ASSOCIATED_EXCHANGES_COMPLETION AssociatedExchangesCompletionHandler;
  139. } SMB_EXCHANGE_DISPATCH_VECTOR, *PSMB_EXCHANGE_DISPATCH_VECTOR;
  140. // An enumerated type listing the type of exchanges
  141. typedef enum _SMB_EXCHANGE_TYPE_ {
  142. CONSTRUCT_NETROOT_EXCHANGE,
  143. ORDINARY_EXCHANGE,
  144. TRANSACT_EXCHANGE,
  145. EXTENDED_SESSION_SETUP_EXCHANGE,
  146. ADMIN_EXCHANGE,
  147. SENTINEL_EXCHANGE
  148. } SMB_EXCHANGE_TYPE, *PSMB_EXCHANGE_TYPE;
  149. // known exchange type dispatch vectors
  150. extern SMB_EXCHANGE_DISPATCH_VECTOR ConstructNetRootExchangeDispatch;
  151. extern SMB_EXCHANGE_DISPATCH_VECTOR OrdinaryExchangeDispatch;
  152. extern SMB_EXCHANGE_DISPATCH_VECTOR TransactExchangeDispatch;
  153. // The various states of the exchange. Each exchange transitions from
  154. // the SMBCE_EXCHANGE_INITIALIZATION_START to SMBCE_EXCHANGE_INITIATED or
  155. // SMBCE_EXCHANGE_ABORTED state.
  156. typedef enum _SMBCE_EXCHANGE_STATE_ {
  157. SMBCE_EXCHANGE_INITIALIZATION_START,
  158. SMBCE_EXCHANGE_SERVER_INITIALIZED,
  159. SMBCE_EXCHANGE_SESSION_INITIALIZED,
  160. SMBCE_EXCHANGE_NETROOT_INITIALIZED,
  161. SMBCE_EXCHANGE_SECURITYBUFFER_INITIALIZED,
  162. SMBCE_EXCHANGE_INITIATED,
  163. SMBCE_EXCHANGE_ABORTED
  164. } SMBCE_EXCHANGE_STATE, *PSMBCE_EXCHANGE_STATE;
  165. // The exchange encapsulates the transport information from the clients. The
  166. // Exchange engine is sandwiched between the protocol selection engine in the
  167. // mini redirector on one side and the various transports on the other side.
  168. // The transport information encapsulates the various categories of transport
  169. // the exchange engine understands.
  170. typedef struct SMBCE_EXCHANGE_TRANSPORT_INFORMATION {
  171. union {
  172. struct {
  173. struct _SMBCE_VC *pVc;
  174. } Vcs;
  175. struct {
  176. ULONG Dummy;
  177. } Datagrams;
  178. struct {
  179. ULONG Dummy;
  180. } Hybrid;
  181. };
  182. } SMBCE_EXCHANGE_TRANSPORT_CONTEXT,
  183. *PSMBCE_EXCHANGE_TRANSPORT_CONTEXT;
  184. typedef struct _SMBCE_EXCHANGE_CONTEXT_ {
  185. PMRX_V_NET_ROOT pVNetRoot;
  186. PSMBCEDB_SERVER_ENTRY pServerEntry;
  187. PSMBCE_V_NET_ROOT_CONTEXT pVNetRootContext;
  188. SMBCE_EXCHANGE_TRANSPORT_CONTEXT TransportContext;
  189. } SMBCE_EXCHANGE_CONTEXT,*PSMBCE_EXCHANGE_CONTEXT;
  190. //
  191. // Similar to the subclassing of SMB net roots the SMB_EXCHANGE will be subclassed
  192. // further to deal with various types of SMB exchanges. SMB exchanges can be roughly
  193. // classified into the following types based on the interactions involved ...
  194. //
  195. // The SMB's that need to be exchanged need to be augmented with some admin SMB's which
  196. // are required for the maintenance of SMB's in the connection engine.
  197. #define SMBCE_EXCHANGE_MID_VALID (0x00000001)
  198. #define SMBCE_EXCHANGE_REUSE_MID (0x00000002)
  199. #define SMBCE_EXCHANGE_RETAIN_MID (SMBCE_EXCHANGE_REUSE_MID)
  200. #define SMBCE_EXCHANGE_MULTIPLE_SENDS_POSSIBLE (0x00000004)
  201. #define SMBCE_EXCHANGE_FINALIZED (0x00000008)
  202. #define SMBCE_EXCHANGE_ATTEMPT_RECONNECTS (0x00000010)
  203. #define SMBCE_EXCHANGE_INDEFINITE_DELAY_IN_RESPONSE (0x00000020)
  204. #define SMBCE_EXCHANGE_MAILSLOT_OPERATION (0x00000040)
  205. #define SMBCE_EXCHANGE_SESSION_CONSTRUCTOR (0x00000100)
  206. #define SMBCE_EXCHANGE_NETROOT_CONSTRUCTOR (0x00000200)
  207. #define SMBCE_EXCHANGE_NOT_FROM_POOL (0x00000800)
  208. #define SMBCE_EXCHANGE_TIMED_RECEIVE_OPERATION (0x00001000)
  209. #define SMBCE_EXCHANGE_TIMEDOUT (0x00002000)
  210. #define SMBCE_EXCHANGE_FULL_PROCESSID_SPECIFIED (0x00004000)
  211. #define SMBCE_EXCHANGE_SMBCE_STOPPED (0x00008000)
  212. #define SMBCE_EXCHANGE_SIGNATURE_BUFFER_ALLOCATED (0x01000000)
  213. #define SMBCE_EXCHANGE_DEBUG_SYSCACHE (0x02000000)
  214. #define SMBCE_ASSOCIATED_EXCHANGE (0x80000000)
  215. #define SMBCE_ASSOCIATED_EXCHANGES_COMPLETION_HANDLER_ACTIVATED (0x40000000)
  216. #define SMBCE_EXCHANGE_FLAGS_TO_PRESERVE (SMBCE_EXCHANGE_NOT_FROM_POOL)
  217. #define SMBCE_OPLOCK_RESPONSE_MID (0xffff)
  218. #define SMBCE_MAILSLOT_OPERATION_MID (0xffff)
  219. #define SMBCE_ECHO_PROBE_MID (0xfffe)
  220. //
  221. // The cancellation status is defined as a PVOID instead of a BOOLEAN to allow
  222. // us the use of Interlocked manipulation instructions
  223. // There are only two states SMBCE_EXCHANGE_CANCELLED, SMBCE_EXCHANGE_ACTIVE
  224. //
  225. #define SMBCE_EXCHANGE_CANCELLED (0xcccccccc)
  226. #define SMBCE_EXCHANGE_NOT_CANCELLED (0xaaaaaaaa)
  227. // The Exchange definition
  228. typedef struct _SMB_EXCHANGE {
  229. union {
  230. UCHAR Type;
  231. struct {
  232. NODE_TYPE_CODE NodeTypeCode; // node type.
  233. NODE_BYTE_SIZE NodeByteSize; // node size.
  234. LONG ReferenceCount;
  235. };
  236. };
  237. LIST_ENTRY SmbMmInUseListEntry;
  238. PRX_CONTEXT RxContext; //use of these two fields is advisory
  239. PVOID LastExecutingThread; //OE and Xact will use them
  240. union {
  241. NTSTATUS SmbStatus;
  242. PMRX_SMB_SRV_OPEN SmbSrvOpen;
  243. };
  244. NTSTATUS Status;
  245. ULONG ServerVersion;
  246. SMB_EXCHANGE_ID Id;
  247. USHORT SmbCeState;
  248. USHORT MidCookie;
  249. SMB_MPX_ID Mid;
  250. LONG CancellationStatus;
  251. ULONG SmbCeFlags;
  252. SMBCE_EXCHANGE_CONTEXT SmbCeContext;
  253. LONG SendCompletePendingOperations;
  254. LONG CopyDataPendingOperations;
  255. LONG ReceivePendingOperations;
  256. LONG LocalPendingOperations;
  257. PKEVENT pSmbCeSynchronizationEvent;
  258. LIST_ENTRY ExchangeList;
  259. LARGE_INTEGER ExpiryTime;
  260. PSMB_EXCHANGE_DISPATCH_VECTOR pDispatchVector;
  261. union {
  262. struct {
  263. struct _SMB_EXCHANGE *pMasterExchange;
  264. SINGLE_LIST_ENTRY NextAssociatedExchange;
  265. } Associated;
  266. struct {
  267. SINGLE_LIST_ENTRY AssociatedExchangesToBeFinalized;
  268. LONG PendingAssociatedExchanges;
  269. } Master;
  270. };
  271. RX_WORK_QUEUE_ITEM WorkQueueItem;
  272. PVOID BufferForServerResponse; //For Security Signature verification
  273. PMDL MdlForServerResponse; //temporarily hold the entire message for security
  274. //signature checking.
  275. ULONG SmbSecuritySignatureIndex;
  276. ULONG ExchangeTransportInitialized;
  277. NTSTATUS SessionSetupStatus;
  278. BOOLEAN IsOffLineFile;
  279. BOOLEAN IsSecuritySignatureEnabled;
  280. BOOLEAN SecuritySignatureReturned;
  281. UCHAR SmbCommand;
  282. LIST_ENTRY CancelledList;
  283. } SMB_EXCHANGE, *PSMB_EXCHANGE;
  284. INLINE PSMBCEDB_SERVER_ENTRY
  285. SmbCeGetExchangeServerEntry(PVOID pExchange)
  286. {
  287. PSMB_EXCHANGE pSmbExchange = (PSMB_EXCHANGE)pExchange;
  288. ASSERT(pSmbExchange->SmbCeContext.pServerEntry != NULL);
  289. return pSmbExchange->SmbCeContext.pServerEntry;
  290. }
  291. INLINE PSMBCE_SERVER
  292. SmbCeGetExchangeServer(PVOID pExchange)
  293. {
  294. PSMB_EXCHANGE pSmbExchange = (PSMB_EXCHANGE)pExchange;
  295. return &(pSmbExchange->SmbCeContext.pServerEntry->Server);
  296. }
  297. INLINE PSMBCEDB_SESSION_ENTRY
  298. SmbCeGetExchangeSessionEntry(PVOID pExchange)
  299. {
  300. PSMB_EXCHANGE pSmbExchange = (PSMB_EXCHANGE)pExchange;
  301. if (pSmbExchange->SmbCeContext.pVNetRootContext != NULL) {
  302. return pSmbExchange->SmbCeContext.pVNetRootContext->pSessionEntry;
  303. } else {
  304. return NULL;
  305. }
  306. }
  307. INLINE PSMBCE_SESSION
  308. SmbCeGetExchangeSession(PVOID pExchange)
  309. {
  310. PSMB_EXCHANGE pSmbExchange = (PSMB_EXCHANGE)pExchange;
  311. if (pSmbExchange->SmbCeContext.pVNetRootContext != NULL) {
  312. return &(pSmbExchange->SmbCeContext.pVNetRootContext->pSessionEntry->Session);
  313. } else {
  314. return NULL;
  315. }
  316. }
  317. INLINE PSMBCEDB_NET_ROOT_ENTRY
  318. SmbCeGetExchangeNetRootEntry(PVOID pExchange)
  319. {
  320. PSMB_EXCHANGE pSmbExchange = (PSMB_EXCHANGE)pExchange;
  321. if (pSmbExchange->SmbCeContext.pVNetRootContext != NULL) {
  322. return pSmbExchange->SmbCeContext.pVNetRootContext->pNetRootEntry;
  323. } else {
  324. return NULL;
  325. }
  326. }
  327. INLINE PSMBCE_NET_ROOT
  328. SmbCeGetExchangeNetRoot(PVOID pExchange)
  329. {
  330. PSMB_EXCHANGE pSmbExchange = (PSMB_EXCHANGE)pExchange;
  331. if (pSmbExchange->SmbCeContext.pVNetRootContext != NULL) {
  332. return &(pSmbExchange->SmbCeContext.pVNetRootContext->pNetRootEntry->NetRoot);
  333. } else {
  334. return NULL;
  335. }
  336. }
  337. INLINE PMRX_V_NET_ROOT
  338. SmbCeGetExchangeVNetRoot(PVOID pExchange)
  339. {
  340. PSMB_EXCHANGE pSmbExchange = (PSMB_EXCHANGE)pExchange;
  341. return pSmbExchange->SmbCeContext.pVNetRoot;
  342. }
  343. INLINE PSMBCE_V_NET_ROOT_CONTEXT
  344. SmbCeGetExchangeVNetRootContext(PVOID pExchange)
  345. {
  346. PSMB_EXCHANGE pSmbExchange = (PSMB_EXCHANGE)pExchange;
  347. return pSmbExchange->SmbCeContext.pVNetRootContext;
  348. }
  349. extern ULONG SmbCeTraceExchangeReferenceCount;
  350. // The following functions ( inline, macros and otherwise ) are defined
  351. // to manipulate the exchanges
  352. // The reset exchange macro provides a mechanism for forcing the exchange
  353. // instance to a well known start state. This is used by the protocol
  354. // selection engine to transceive different SMB's. A note of caution --
  355. // ensure that the conditions are O.K for initialization. There is no well
  356. // known mechanism in the exchange engine to prevent overwriting an
  357. // exchange instance while in use.
  358. #define SmbCeResetExchange(pExchange) \
  359. (pExchange)->SmbCeFlags &= ~SMBCE_EXCHANGE_FINALIZED; \
  360. (pExchange)->ReceivePendingOperations = 0; \
  361. (pExchange)->CopyDataPendingOperations = 0; \
  362. (pExchange)->SendCompletePendingOperations = 0; \
  363. (pExchange)->LocalPendingOperations = 0; \
  364. (pExchange)->Status = STATUS_SUCCESS; \
  365. (pExchange)->SmbStatus = STATUS_SUCCESS
  366. // The following macros provide a mechanism for referencing and dereferencing
  367. // the exchange. The reference count provides a mechanism for detecting
  368. // when an exchange instance can be safely discarded. The reference count
  369. // differs from the pending operations count maintained in the exchange
  370. // which are used to detect when a quiescent state is reached.
  371. #define SmbCeReferenceExchange(pExchange) \
  372. InterlockedIncrement(&(pExchange)->ReferenceCount); \
  373. if (SmbCeTraceExchangeReferenceCount) { \
  374. DbgPrint("Reference Exchange %lx Type(%ld) %s %ld %ld\n", \
  375. (pExchange), \
  376. (pExchange)->Type, \
  377. __FILE__, \
  378. __LINE__, \
  379. (pExchange)->ReferenceCount); \
  380. }
  381. #define SmbCeDereferenceExchange(pExchange) \
  382. InterlockedDecrement(&(pExchange)->ReferenceCount); \
  383. if (SmbCeTraceExchangeReferenceCount) { \
  384. DbgPrint("Dereference Exchange %lx Type(%ld) %s %ld %ld\n", \
  385. (pExchange), \
  386. (pExchange)->Type, \
  387. __FILE__, \
  388. __LINE__, \
  389. (pExchange)->ReferenceCount); \
  390. }
  391. #define SmbCeDereferenceAndDiscardExchange(pExchange) \
  392. if (InterlockedDecrement(&(pExchange)->ReferenceCount) == 0) { \
  393. SmbCeDiscardExchange(pExchange); \
  394. } \
  395. if (SmbCeTraceExchangeReferenceCount) { \
  396. DbgPrint("Dereference Exchange %lx Type(%ld) %s %ld %ld\n", \
  397. (pExchange), \
  398. (pExchange)->Type, \
  399. __FILE__, \
  400. __LINE__, \
  401. (pExchange)->ReferenceCount); \
  402. }
  403. // Macros to hide the syntactic details of dereferencing and calling a
  404. // routine in a dispatch vector. These macros are purely intended for
  405. // use in the connection engine only and is not meant for use by
  406. // other modules.
  407. #define SMB_EXCHANGE_DISPATCH(pExchange,Routine,Arguments) \
  408. (*((pExchange)->pDispatchVector->Routine))##Arguments
  409. #define SMB_EXCHANGE_POST(pExchange,Routine) \
  410. RxPostToWorkerThread(&(pExchange)->WorkItem.WorkQueueItem, \
  411. (pExchange)->pDispatchVector->Routine, \
  412. (pExchange))
  413. // The following enum type defines the result of invoking the finalization routine
  414. // on an exchange instance.
  415. typedef enum _SMBCE_EXCHANGE_STATUS_ {
  416. SmbCeExchangeAlreadyFinalized,
  417. SmbCeExchangeFinalized,
  418. SmbCeExchangeNotFinalized
  419. } SMBCE_EXCHANGE_STATUS, *PSMBCE_EXCHANGE_STATUS;
  420. // The pending operations associated with an exchange are classified into four kinds
  421. // Receive operations, Copy Data Operations, Send Complete and Local operations.
  422. // These need to be incremented under the protection of a spinlock. However they
  423. // are decremented in the absence of a spinlock ( with the respective assert ).
  424. #define SMBCE_LOCAL_OPERATION 0x1
  425. #define SMBCE_SEND_COMPLETE_OPERATION 0x2
  426. #define SMBCE_COPY_DATA_OPERATION 0x4
  427. #define SMBCE_RECEIVE_OPERATION 0x8
  428. extern NTSTATUS
  429. SmbCeIncrementPendingOperations(
  430. PSMB_EXCHANGE pExchange,
  431. ULONG PendingOperationsMask,
  432. PVOID FileName,
  433. ULONG FileLine);
  434. extern NTSTATUS
  435. SmbCeDecrementPendingOperations(
  436. PSMB_EXCHANGE pExchange,
  437. ULONG PendingOperationsMask,
  438. PVOID FileName,
  439. ULONG FileLine);
  440. extern SMBCE_EXCHANGE_STATUS
  441. SmbCeDecrementPendingOperationsAndFinalize(
  442. PSMB_EXCHANGE pExchange,
  443. ULONG PendingOperationsMask,
  444. PVOID FileName,
  445. ULONG FileLine);
  446. // the pending operations increment routines
  447. #define SmbCeIncrementPendingReceiveOperations(pExchange) \
  448. SmbCeIncrementPendingOperations(pExchange,(SMBCE_RECEIVE_OPERATION),__FILE__,__LINE__)
  449. #define SmbCeIncrementPendingSendCompleteOperations(pExchange) \
  450. SmbCeIncrementPendingOperations(pExchange,(SMBCE_SEND_COMPLETE_OPERATION),__FILE__,__LINE__)
  451. #define SmbCeIncrementPendingCopyDataOperations(pExchange) \
  452. SmbCeIncrementPendingOperations(pExchange,(SMBCE_COPY_DATA_OPERATION),__FILE__,__LINE__)
  453. #define SmbCeIncrementPendingLocalOperations(pExchange) \
  454. SmbCeIncrementPendingOperations(pExchange,(SMBCE_LOCAL_OPERATION),__FILE__,__LINE__)
  455. // The pending operations decrement routines
  456. // Note the special casing of ReceivePendingOperations since it is the only one
  457. // that can be forced by a disconnect indication. There are two variations in
  458. // the decrement macros. The first flavour is to be used when it can be
  459. // guaranteed that the decrement operation will not lead to the finalization
  460. // of the exchange and the second is to be used when we cannot ensure the criterion
  461. // for the first. The difference between the two is that it eliminates
  462. // acquisition/release of a spinlock.
  463. #define SmbCeDecrementPendingReceiveOperations(pExchange) \
  464. SmbCeDecrementPendingOperations(pExchange,(SMBCE_RECEIVE_OPERATION),__FILE__,__LINE__)
  465. #define SmbCeDecrementPendingSendCompleteOperations(pExchange) \
  466. SmbCeDecrementPendingOperations(pExchange,(SMBCE_SEND_COMPLETE_OPERATION),__FILE__,__LINE__)
  467. #define SmbCeDecrementPendingCopyDataOperations(pExchange) \
  468. SmbCeDecrementPendingOperations(pExchange,(SMBCE_COPY_DATA_OPERATION),__FILE__,__LINE__)
  469. #define SmbCeDecrementPendingLocalOperations(pExchange) \
  470. SmbCeDecrementPendingOperations(pExchange,(SMBCE_LOCAL_OPERATION),__FILE__,__LINE__)
  471. // The pending operations decrement routines
  472. #define SmbCeDecrementPendingReceiveOperationsAndFinalize(pExchange) \
  473. SmbCeDecrementPendingOperationsAndFinalize(pExchange,(SMBCE_RECEIVE_OPERATION),__FILE__,__LINE__)
  474. #define SmbCeDecrementPendingSendCompleteOperationsAndFinalize(pExchange) \
  475. SmbCeDecrementPendingOperationsAndFinalize(pExchange,(SMBCE_SEND_COMPLETE_OPERATION),__FILE__,__LINE__)
  476. #define SmbCeDecrementPendingCopyDataOperationsAndFinalize(pExchange) \
  477. SmbCeDecrementPendingOperationsAndFinalize(pExchange,(SMBCE_COPY_DATA_OPERATION),__FILE__,__LINE__)
  478. #define SmbCeDecrementPendingLocalOperationsAndFinalize(pExchange) \
  479. SmbCeDecrementPendingOperationsAndFinalize(pExchange,(SMBCE_LOCAL_OPERATION),__FILE__,__LINE__)
  480. //
  481. // This is the pid that will be used by the rdr; rdr1 used 0xcafe.
  482. // only this pid is ever sent except for nt<-->nt creates. in these cases,
  483. // we have to send the full 32bit process id for RPC. actually, we only have to do
  484. // for pipes but we do it all the time instead.
  485. //
  486. #define MRXSMB_PROCESS_ID (0xfeff)
  487. INLINE VOID
  488. SmbCeSetFullProcessIdInHeader(
  489. PSMB_EXCHANGE pExchange,
  490. ULONG ProcessId,
  491. PNT_SMB_HEADER pNtSmbHeader)
  492. {
  493. pExchange->SmbCeFlags |= SMBCE_EXCHANGE_FULL_PROCESSID_SPECIFIED;
  494. SmbPutUshort(&pNtSmbHeader->Pid, (USHORT)((ProcessId) & 0xFFFF));
  495. SmbPutUshort(&pNtSmbHeader->PidHigh, (USHORT)((ProcessId) >> 16));
  496. }
  497. // The exchange engine API, for creation and manipulation of exchange instances
  498. // Initialization/Creation of an exchange instance
  499. extern NTSTATUS
  500. SmbCepInitializeExchange(
  501. PSMB_EXCHANGE *pExchangePointer,
  502. PRX_CONTEXT pRxContext,
  503. PSMBCEDB_SERVER_ENTRY pServerEntry,
  504. PMRX_V_NET_ROOT pVNetRoot,
  505. SMB_EXCHANGE_TYPE ExchangeType,
  506. PSMB_EXCHANGE_DISPATCH_VECTOR pDispatchVector);
  507. INLINE NTSTATUS
  508. SmbCeInitializeExchange(
  509. PSMB_EXCHANGE *pExchangePointer,
  510. PRX_CONTEXT pRxContext,
  511. PMRX_V_NET_ROOT pVNetRoot,
  512. SMB_EXCHANGE_TYPE ExchangeType,
  513. PSMB_EXCHANGE_DISPATCH_VECTOR pDispatchVector)
  514. {
  515. return SmbCepInitializeExchange(
  516. pExchangePointer,
  517. pRxContext,
  518. NULL,
  519. pVNetRoot,
  520. ExchangeType,
  521. pDispatchVector);
  522. }
  523. INLINE NTSTATUS
  524. SmbCeInitializeExchange2(
  525. PSMB_EXCHANGE *pExchangePointer,
  526. PRX_CONTEXT pRxContext,
  527. PSMBCEDB_SERVER_ENTRY pServerEntry,
  528. SMB_EXCHANGE_TYPE ExchangeType,
  529. PSMB_EXCHANGE_DISPATCH_VECTOR pDispatchVector)
  530. {
  531. return SmbCepInitializeExchange(
  532. pExchangePointer,
  533. pRxContext,
  534. pServerEntry,
  535. NULL,
  536. ExchangeType,
  537. pDispatchVector);
  538. }
  539. extern NTSTATUS
  540. SmbCeInitializeAssociatedExchange(
  541. PSMB_EXCHANGE *pAssociatedExchangePointer,
  542. PSMB_EXCHANGE pMasterExchange,
  543. SMB_EXCHANGE_TYPE Type,
  544. PSMB_EXCHANGE_DISPATCH_VECTOR pDispatchVector);
  545. // converting one type of exchange to another
  546. extern NTSTATUS
  547. SmbCeTransformExchange(
  548. PSMB_EXCHANGE pExchange,
  549. SMB_EXCHANGE_TYPE NewType,
  550. PSMB_EXCHANGE_DISPATCH_VECTOR pDispatchVector);
  551. // Initiating an exchange
  552. extern NTSTATUS
  553. SmbCeInitiateExchange(PSMB_EXCHANGE pExchange);
  554. extern NTSTATUS
  555. SmbCeInitiateAssociatedExchange(
  556. PSMB_EXCHANGE pAssociatedExchange,
  557. BOOLEAN EnableCompletionHandlerInMasterExchange);
  558. // Resuming an exchange
  559. extern NTSTATUS
  560. SmbCeResumeExchange(PSMB_EXCHANGE pExchange);
  561. // aborting an initiated exchange
  562. extern NTSTATUS
  563. SmbCeAbortExchange(PSMB_EXCHANGE pExchange);
  564. // discarding an exchnge instance
  565. extern VOID
  566. SmbCeDiscardExchange(PVOID pExchange);
  567. // In addition to providing a flexible mechanism for exchanging packets with
  568. // the server the exchange engine also provides a mechanism for building and
  569. // parsing SMB_HEADER's. This functionality is built into the connection
  570. // engine because the meta data in the headers is used to update the connection
  571. // engine database.
  572. // building SMB headers
  573. extern NTSTATUS
  574. SmbCeBuildSmbHeader(
  575. IN OUT PSMB_EXCHANGE pExchange,
  576. IN OUT PVOID pBuffer,
  577. IN ULONG BufferLength,
  578. OUT PULONG pRemainingBuffer,
  579. OUT PUCHAR pLastCommandInHeader,
  580. OUT PUCHAR *pNextCommand);
  581. // parsing SMB headers.
  582. extern NTSTATUS
  583. SmbCeParseSmbHeader(
  584. PSMB_EXCHANGE pExchange,
  585. PSMB_HEADER pSmbHeader,
  586. PGENERIC_ANDX pCommandToProcess,
  587. NTSTATUS *pSmbResponseStatus,
  588. ULONG BytesAvailable,
  589. ULONG BytesIndicated,
  590. PULONG pBytesConsumed);
  591. // The following routines are intended for use in the connection engine only.
  592. extern NTSTATUS
  593. MRxSmbInitializeSmbCe();
  594. extern NTSTATUS
  595. MRxSmbTearDownSmbCe();
  596. extern NTSTATUS
  597. SmbCePrepareExchangeForReuse(PSMB_EXCHANGE pExchange);
  598. extern PVOID
  599. SmbCeMapSendBufferToCompletionContext(
  600. PSMB_EXCHANGE pExchange,
  601. PVOID pBuffer);
  602. extern PVOID
  603. SmbCeMapSendCompletionContextToBuffer(
  604. PSMB_EXCHANGE pExchange,
  605. PVOID pContext);
  606. extern SMBCE_EXCHANGE_STATUS
  607. SmbCeFinalizeExchange(PSMB_EXCHANGE pExchange);
  608. extern VOID
  609. SmbCeFinalizeExchangeOnDisconnect(
  610. PSMB_EXCHANGE pExchange);
  611. extern NTSTATUS
  612. SmbCeReferenceServer(
  613. PSMB_EXCHANGE pExchange);
  614. extern NTSTATUS
  615. SmbCeIncrementActiveExchangeCount();
  616. extern VOID
  617. SmbCeDecrementActiveExchangeCount();
  618. extern VOID
  619. SmbCeSetExpiryTime(
  620. PSMB_EXCHANGE pExchange);
  621. extern BOOLEAN
  622. SmbCeDetectExpiredExchanges(
  623. PSMBCEDB_SERVER_ENTRY pServerEntry);
  624. extern VOID
  625. SmbCepFinalizeAssociatedExchange(
  626. PSMB_EXCHANGE pExchange);
  627. extern NTSTATUS
  628. SmbCeCancelExchange(
  629. PRX_CONTEXT pRxContext);
  630. typedef struct _SMB_CONSTRUCT_NETROOT_EXCHANGE_ {
  631. union {
  632. SMB_EXCHANGE;
  633. SMB_EXCHANGE Exchange;
  634. };
  635. SMB_TREE_ID TreeId;
  636. SMB_USER_ID UserId;
  637. BOOLEAN fUpdateDefaultSessionEntry;
  638. BOOLEAN fInitializeNetRoot;
  639. PMRX_NETROOT_CALLBACK NetRootCallback;
  640. PMDL pSmbRequestMdl;
  641. PMDL pSmbResponseMdl;
  642. PVOID pSmbActualBuffer; // Originally allocated buffer
  643. PVOID pSmbBuffer; // Start of header
  644. PMRX_CREATENETROOT_CONTEXT pCreateNetRootContext;
  645. CSC_SHARE_HANDLE hShare;
  646. } SMB_CONSTRUCT_NETROOT_EXCHANGE, *PSMB_CONSTRUCT_NETROOT_EXCHANGE;
  647. extern
  648. NTSTATUS
  649. GetSmbResponseNtStatus(
  650. IN PSMB_HEADER pSmbHeader,
  651. IN PSMB_EXCHANGE pExchange
  652. );
  653. extern CHAR InitialSecuritySignature[];
  654. #endif // _SMBXCHNG_H_