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.

954 lines
38 KiB

  1. /*++
  2. Copyright (c) 1989 Microsoft Corporation
  3. Module Name:
  4. Stuffer.c
  5. Abstract:
  6. This module implements the SMBstuffer formating primitives. the following controlstring
  7. characters are defined for the stuffer: (** means nyi...**d means downlevel part not implemented)
  8. 0 placeholder for the wct
  9. 1 pad to word boundary
  10. X placeholderfor&X
  11. W,w format a word from the next parameter
  12. D,d format the next parameter as a Dword
  13. Y,y format the next parameter as a byte
  14. L,l the next parameter is a PLARGE_INTEGER; format it in
  15. M,m format a zero byte
  16. ** 2 the next parameter points to a tagged dialect ASCIZI string to be copied in
  17. ** 3 the next parameter points to a tagged devicename ASCIIZ string
  18. 4 the next parameter is either 04-tagged ASCIIZ or UNICODEZ as determined by flags2
  19. > the next parameters is ASCIIZ or UNICODEZ as determined by flags2; it is to be appended
  20. to the previous 04-tagged item by backing up over the previous null.
  21. A,a the next parameter is an ASCIIZ string
  22. U,u the next parameter is a UNICODEZ string
  23. V,v the next parameter is a UNICODEnoZ string
  24. z the next parameter is a PUNICODE_STRING to be stringed as ASCIZI
  25. or UNICODEZ as determined by flags2
  26. N,n the next parameter is a PNET_ROOT whose name is to be stringed as ASCIIZ
  27. or UNICODEZ as determined by flags2
  28. R,r the next 2 parameters are a PBYTE* and a size; reserve the region and store the pointer
  29. Q,q the current position is the data offset WORD...remember it
  30. 5 the current position is the start of the data; fill in the data pointer
  31. P,p the current position is the parameter offset WORD...remember it
  32. 6 the current position is the start of the parameters; fill in the param pointer
  33. B,b the current position is the Bcc WORD...remember it; also, fill in wct
  34. s the next parameter has the alignment information....pad accordingly
  35. S pad to DWORD
  36. c the next 2 parameters are count/addr...copy in the data.
  37. ! End of this protocol; fill in the bcc field
  38. ? next parameter is BOOLEAN_ULONG; 0=>immediate return
  39. . NOOP
  40. For controls with a upper/lowercase pair, the uppercase version indicates that a position tag
  41. is supplied in the checked version.
  42. Author:
  43. Joe Linn 3-3-95
  44. Revision History:
  45. --*/
  46. #include "precomp.h"
  47. #pragma hdrstop
  48. #include <stdio.h>
  49. #include <stdarg.h>
  50. #ifdef ALLOC_PRAGMA
  51. #pragma alloc_text(PAGE, SmbStuffWrapRtlAssert)
  52. #pragma alloc_text(PAGE, SmbMrxInitializeStufferFacilities)
  53. #pragma alloc_text(PAGE, SmbMrxFinalizeStufferFacilities)
  54. #pragma alloc_text(PAGE, MRxSmbSetInitialSMB)
  55. #pragma alloc_text(PAGE, MRxSmbStartSMBCommand)
  56. #pragma alloc_text(PAGE, MrxSMBWillThisFit)
  57. #pragma alloc_text(PAGE, MRxSmbStuffSMB)
  58. #pragma alloc_text(PAGE, MRxSmbStuffAppendRawData)
  59. #pragma alloc_text(PAGE, MRxSmbStuffAppendSmbData)
  60. #pragma alloc_text(PAGE, MRxSmbStuffSetByteCount)
  61. #endif
  62. //
  63. // The local debug trace level
  64. //
  65. #define Dbg (DEBUG_TRACE_ALWAYS)
  66. #define MRXSMB_INITIAL_WCT (0xcc)
  67. #define MRXSMB_INITIAL_BCC (0xface)
  68. #define MRXSMB_INITIAL_DATAOFFSET (0xd0ff)
  69. #define MRXSMB_INITIAL_PARAMOFFSET (0xb0ff)
  70. #define MRXSMB_INITIAL_ANDX (0xdede00ff)
  71. #if 0
  72. //this is old...........
  73. #if DBG
  74. // a little presto-changeo to get assert messages in user mode
  75. // the key is that MRxSmbRxImports->pRxNetNameTable will not be NULL...
  76. // it will point to the netnametable. this
  77. // seems like a small enough price to pay on the way to an rtl assert!
  78. VOID
  79. SmbStuffWrapRtlAssert(
  80. IN PVOID FailedAssertion,
  81. IN PVOID FileName,
  82. IN ULONG LineNumber,
  83. IN PCHAR Message
  84. )
  85. {
  86. char STARS[] = "**************************************";
  87. PAGED_CODE();
  88. if (MRxSmbRxImports->pRxNetNameTable == NULL){
  89. // do our own thing
  90. RxDbgTrace(0,Dbg,("%s\n%s\n",STARS,STARS));
  91. RxDbgTrace (0,Dbg,("Failed Assertion %s\n",FailedAssertion));
  92. RxDbgTrace(0,Dbg,("%s at line %lu\n",FileName,LineNumber));
  93. if (Message) {
  94. RxDbgTrace (0,Dbg,("%s\n",Message));
  95. }
  96. RxDbgTrace(0,Dbg,("%s\n%s\n",STARS,STARS));
  97. } else RtlAssert(FailedAssertion,FileName,LineNumber,Message);
  98. }
  99. #ifdef RtlAssert
  100. #undef RtlAssert
  101. #endif //ifdef RtlAssert
  102. #define RtlAssert SmbStuffWrapRtlAssert
  103. #endif
  104. #endif //if 0
  105. NTSTATUS
  106. SmbMrxInitializeStufferFacilities(
  107. void
  108. )
  109. /*++
  110. Routine Description:
  111. This routine initializes things for the SMB minirdr. we will allocate enough stuff
  112. to get us going. right now....we do nothing.
  113. Arguments:
  114. Return Value:
  115. RXSTATUS - The return status for the operation
  116. --*/
  117. {
  118. PAGED_CODE();
  119. return(RX_MAP_STATUS(SUCCESS));
  120. }
  121. NTSTATUS
  122. SmbMrxFinalizeStufferFacilities(
  123. void
  124. )
  125. /*++
  126. Routine Description:
  127. This routine finalizes things for the SMB minirdr. we give back everything that
  128. we have allocated. right now....we do nothing.
  129. Arguments:
  130. Return Value:
  131. RXSTATUS - The return status for the operation
  132. --*/
  133. {
  134. PAGED_CODE();
  135. return(RX_MAP_STATUS(SUCCESS));
  136. }
  137. #if DBG
  138. #define BUILD_HEADER_ROUTINE BuildHeaderRoutine
  139. typedef
  140. NTSTATUS
  141. (NTAPI *PMRXSMB_BUILD_HEADER_ROUTINE) (
  142. PSMB_EXCHANGE pExchange,
  143. PVOID pBuffer,
  144. ULONG BufferLength,
  145. PULONG pBufferConsumed,
  146. PUCHAR pLastCommandInHeader,
  147. PUCHAR *pNextCommandPtr
  148. );
  149. #else
  150. #define BUILD_HEADER_ROUTINE SmbCeBuildSmbHeader
  151. #endif
  152. NTSTATUS
  153. MRxSmbSetInitialSMB (
  154. IN OUT PSMBSTUFFER_BUFFER_STATE StufferState
  155. STUFFERTRACE_CONTROLPOINT_ARGS
  156. )
  157. {
  158. NTSTATUS Status;
  159. PNT_SMB_HEADER NtSmbHeader;
  160. ULONG BufferConsumed;
  161. PBYTE ScanPosition;
  162. PUCHAR pCommand;
  163. #if DBG
  164. PMRXSMB_BUILD_HEADER_ROUTINE BUILD_HEADER_ROUTINE = SmbCeBuildSmbHeader;
  165. #endif //if DBG
  166. PAGED_CODE();
  167. ASSERT ( StufferState != NULL ); //CODE.IMPROVEMENT shouldn't we have a nodetype??
  168. ASSERT ( sizeof(NT_SMB_HEADER) == sizeof(SMB_HEADER) );
  169. //RxDbgTrace(0, Dbg, ("MrxSMBSetInitialSMB base=%08lx,limit=%08lx\n",
  170. // StufferState->BufferBase,StufferState->BufferLimit));
  171. ASSERT ( (StufferState->BufferLimit - StufferState->BufferBase) > sizeof(SMB_HEADER));
  172. NtSmbHeader = (PNT_SMB_HEADER)(StufferState->BufferBase);
  173. RtlZeroMemory(NtSmbHeader,sizeof(NT_SMB_HEADER));
  174. //this stuff is reinitialized
  175. StufferState->DataMdl = NULL; //note that this is not finalized or anything
  176. StufferState->DataSize = 0;
  177. StufferState->CurrentWct = NULL;
  178. StufferState->PreviousCommand = SMB_COM_NO_ANDX_COMMAND;
  179. StufferState->CurrentCommand = SMB_COM_NO_ANDX_COMMAND;
  180. StufferState->FlagsCopy = 0;
  181. StufferState->Flags2Copy = 0;
  182. StufferState->CurrentPosition = ((PBYTE)NtSmbHeader);
  183. RxDbgTraceDoit(
  184. StufferState->ControlPoint = ControlPoint;
  185. StufferState->PrintCLoop = FALSE;
  186. StufferState->PrintFLoop = FALSE;
  187. while (EnablePrints) {
  188. ULONG c = EnablePrints & 0xff;
  189. EnablePrints >>= 8;
  190. if (c=='C') StufferState->PrintCLoop = TRUE;
  191. if (c=='F') StufferState->PrintFLoop = TRUE;
  192. if (c=='X') BUILD_HEADER_ROUTINE = MRxSmbBuildSmbHeaderTestSurrogate;
  193. }
  194. )
  195. Status = BUILD_HEADER_ROUTINE(
  196. StufferState->Exchange,
  197. NtSmbHeader,
  198. (ULONG)(StufferState->BufferLimit - StufferState->BufferBase),
  199. &BufferConsumed,
  200. &StufferState->PreviousCommand,
  201. &pCommand);
  202. if (Status!=RX_MAP_STATUS(SUCCESS)) {
  203. RxDbgTrace(0, Dbg, ("MrxSMBSetInitialSMB buildhdr failure st=%08lx\n",Status));
  204. RxLog(("BuildHdr failed %lx %lx",StufferState->Exchange,Status));
  205. SmbLog(LOG,
  206. MRxSmbSetInitialSMB,
  207. LOGPTR(StufferState->Exchange)
  208. LOGULONG(Status));
  209. return Status;
  210. }
  211. //copy the flags
  212. StufferState->FlagsCopy = NtSmbHeader->Flags;
  213. StufferState->Flags2Copy = SmbGetAlignedUshort(&NtSmbHeader->Flags2);
  214. if (StufferState->Exchange->Type == ORDINARY_EXCHANGE) {
  215. PSMB_PSE_ORDINARY_EXCHANGE OrdinaryExchange = (PSMB_PSE_ORDINARY_EXCHANGE)StufferState->Exchange;
  216. if (BooleanFlagOn(OrdinaryExchange->Flags,SMBPSE_OE_FLAG_TURNON_DFS_FLAG)) {
  217. StufferState->Flags2Copy |= SMB_FLAGS2_DFS;
  218. SmbPutUshort(&NtSmbHeader->Flags2,(USHORT)StufferState->Flags2Copy);
  219. }
  220. }
  221. StufferState->CurrentPosition += BufferConsumed;
  222. if (BufferConsumed > sizeof(SMB_HEADER)) {
  223. if (pCommand != NULL) {
  224. *pCommand = SMB_COM_NO_ANDX_COMMAND;
  225. }
  226. StufferState->CurrentWct = StufferState->CurrentPosition;
  227. }
  228. return Status;
  229. }
  230. #define RETURN_A_START_PROBLEM(xxyy) {\
  231. RxDbgTrace(0,Dbg,("MRxSmbStartSMBCommand gotta problem= %lu\n",xxyy)); \
  232. StufferState->SpecificProblem = xxyy; \
  233. return(RX_MAP_STATUS(INVALID_PARAMETER)); \
  234. }
  235. NTSTATUS
  236. MRxSmbStartSMBCommand (
  237. IN OUT PSMBSTUFFER_BUFFER_STATE StufferState,
  238. IN INITIAL_SMBBUG_DISPOSITION InitialSMBDisposition,
  239. IN UCHAR Command, //joejoe this next four params could come from a table...2offset and you're smaller
  240. IN ULONG MaximumBufferUsed,
  241. IN ULONG MaximumSize,
  242. IN ULONG InitialAlignment,
  243. IN ULONG MaximumResponseHeader,
  244. IN UCHAR Flags,
  245. IN UCHAR FlagsMask,
  246. IN USHORT Flags2,
  247. IN USHORT Flags2Mask
  248. STUFFERTRACE_CONTROLPOINT_ARGS
  249. )
  250. /*++
  251. Routine Description:
  252. The routine checks to see if the condition is stable. If not, it
  253. goes into a wait loop alternately getting the resource and then
  254. waiting on the event.
  255. Arguments:
  256. joejoe review this
  257. StufferState - the header buffer being used
  258. InitialSMBDisposition tells when/if to reinit the stuffer state
  259. Command - the smb command being set up
  260. MaximumBufferUsed - the amount of the header buffer that will be used (as opposed to the data)
  261. this has to be conjured up in advance. if you're not willing to do this, then
  262. just push out the current smb. this value should include any data pads!
  263. MaximumSize - the size of the data. this is to keep from overrunning the srv's smbbuf
  264. InitialAlignment - a compound argument (i.e. you get it from a constant) the top half
  265. tells the alignment unit and the bottom gives the spacing within
  266. MaximumResponseHeader - how much of the srv's response buffer this will use up
  267. Flags - the required flags settings
  268. FlagsMask - which bits of the flags are important
  269. Flags2 - the required flags2 settings
  270. Flags2Mask - which flags2 bits are important
  271. Return Value:
  272. none.
  273. --*/
  274. {
  275. UCHAR NewFlags;
  276. USHORT NewFlags2;
  277. PBYTE *CurrentPosition = &(StufferState->CurrentPosition);
  278. PNT_SMB_HEADER NtSmbHeader = (PNT_SMB_HEADER)(StufferState->BufferBase);
  279. ULONG AlignmentUnit = InitialAlignment >> 16;
  280. ULONG StufferStateRequirement = MaximumBufferUsed + AlignmentUnit;
  281. #if DBG
  282. PBYTE OriginalPosition = *CurrentPosition;
  283. #endif
  284. PAGED_CODE();
  285. if (StufferState->DataSize) {
  286. StufferState->SpecificProblem = xSMBbufSTATUS_CANT_COMPOUND;
  287. return(RX_MAP_STATUS(INVALID_PARAMETER));
  288. }
  289. if ((InitialSMBDisposition==SetInitialSMB_yyUnconditionally)
  290. || ((InitialSMBDisposition==SetInitialSMB_ForReuse)&&(StufferState->Started))) {
  291. MRxSmbSetInitialSMB( StufferState STUFFERTRACE_NOPREFIX(ControlPoint,EnablePrints) );
  292. }
  293. StufferState->Started = TRUE;
  294. //joejoe temporary hack
  295. switch (StufferState->CurrentCommand) {
  296. case SMB_COM_LOCKING_ANDX:
  297. case SMB_COM_OPEN_ANDX:
  298. case SMB_COM_READ_ANDX:
  299. case SMB_COM_WRITE_ANDX:
  300. case SMB_COM_SESSION_SETUP_ANDX:
  301. //case SMB_COM_LOGOFF_ANDX:
  302. case SMB_COM_TREE_CONNECT_ANDX:
  303. case SMB_COM_NT_CREATE_ANDX:
  304. case SMB_COM_NO_ANDX_COMMAND:
  305. break;
  306. default:
  307. StufferState->SpecificProblem = xSMBbufSTATUS_CANT_COMPOUND;
  308. return(RX_MAP_STATUS(INVALID_PARAMETER));
  309. }
  310. if (*CurrentPosition+StufferStateRequirement >= StufferState->BufferLimit ) {
  311. StufferState->SpecificProblem = xSMBbufSTATUS_CANT_COMPOUND;
  312. return(RX_MAP_STATUS(INVALID_PARAMETER));
  313. }
  314. if (StufferState->RxContext) {
  315. PRX_CONTEXT RxContext = StufferState->RxContext;
  316. PMRX_SRV_CALL SrvCall;
  317. ULONG CurrentOffset;
  318. if (RxContext->MajorFunction != IRP_MJ_CREATE) {
  319. SrvCall = RxContext->pFcb->pNetRoot->pSrvCall; //joejoe cache it?
  320. } else {
  321. SrvCall = RxContext->Create.pSrvCall;
  322. }
  323. ASSERT(SrvCall);
  324. CurrentOffset = (ULONG)(*CurrentPosition - StufferState->BufferBase);
  325. if (CurrentOffset+StufferStateRequirement+MaximumSize
  326. > GetServerMaximumBufferSize(SrvCall) ) {
  327. StufferState->SpecificProblem = xSMBbufSTATUS_SERVER_OVERRUN;
  328. return(RX_MAP_STATUS(INVALID_PARAMETER));
  329. }
  330. }
  331. NewFlags = Flags | (UCHAR)(StufferState->FlagsCopy);
  332. NewFlags2 = Flags2 | (USHORT)(StufferState->Flags2Copy);
  333. if ( ((NewFlags&FlagsMask)!=Flags) ||
  334. ((NewFlags2&Flags2Mask)!=Flags2) ) {
  335. StufferState->SpecificProblem = xSMBbufSTATUS_FLAGS_CONFLICT;
  336. return(RX_MAP_STATUS(INVALID_PARAMETER));
  337. }
  338. StufferState->FlagsCopy = NtSmbHeader->Flags = NewFlags;
  339. StufferState->Flags2Copy = NewFlags2;
  340. SmbPutAlignedUshort(&NtSmbHeader->Flags2, NewFlags2);
  341. if (!StufferState->CurrentWct) {
  342. NtSmbHeader->Command = Command;
  343. } else {
  344. PGENERIC_ANDX GenericAndX = (PGENERIC_ANDX)StufferState->CurrentWct;
  345. if (AlignmentUnit) {
  346. ULONG AlignmentMask = (AlignmentUnit-1);
  347. ULONG AlignmentResidue = InitialAlignment&AlignmentMask;
  348. RxDbgTrace(0, Dbg, ("Aligning start of smb cp&m,m,r=%08lx %08lx %08lx\n",
  349. ((ULONG)(ULONG_PTR)(*CurrentPosition))&AlignmentMask,
  350. AlignmentMask, AlignmentResidue)
  351. );
  352. for (;(((ULONG_PTR)(*CurrentPosition))&AlignmentMask)!=AlignmentResidue;) {
  353. **CurrentPosition = ',';
  354. *CurrentPosition += 1;
  355. }
  356. }
  357. GenericAndX->AndXCommand = Command;
  358. GenericAndX->AndXReserved = 0;
  359. SmbPutUshort (&GenericAndX->AndXOffset,
  360. (USHORT)(*CurrentPosition - StufferState->BufferBase));
  361. }
  362. StufferState->CurrentWct = *CurrentPosition;
  363. StufferState->CurrentCommand = Command;
  364. StufferState->CurrentDataOffset = 0;
  365. return RX_MAP_STATUS(SUCCESS);
  366. }
  367. BOOLEAN
  368. MrxSMBWillThisFit(
  369. IN PSMBSTUFFER_BUFFER_STATE StufferState,
  370. IN ULONG AlignmentUnit,
  371. IN ULONG DataSize
  372. )
  373. {
  374. //joejoe actually some stuff will fit that this says no...........
  375. return(StufferState->CurrentPosition+AlignmentUnit+DataSize<StufferState->BufferLimit);
  376. }
  377. #if RDBSSTRACE
  378. #define StufferFLoopTrace(Z) { if (StufferState->PrintFLoop) {RxDbgTraceLV__norx(0,StufferState->ControlPoint,900,Z);}}
  379. #define StufferCLoopTrace(Z) { if (StufferState->PrintCLoop) {RxDbgTraceLV__norx(0,StufferState->ControlPoint,800,Z);}}
  380. #else // DBG
  381. #define StufferFLoopTrace(Z)
  382. #define StufferCLoopTrace(Z)
  383. #endif // DBG
  384. NTSTATUS
  385. MRxSmbStuffSMB (
  386. IN OUT PSMBSTUFFER_BUFFER_STATE StufferState,
  387. ...
  388. )
  389. {
  390. va_list AP;
  391. PBYTE BufferBase = (StufferState->BufferBase);
  392. PBYTE *CurrentPosition = &(StufferState->CurrentPosition);
  393. PBYTE *CurrentWct = &(StufferState->CurrentWct);
  394. PBYTE *CurrentBcc = &(StufferState->CurrentBcc);
  395. PBYTE *CurrentDataOffset = &(StufferState->CurrentDataOffset);
  396. PBYTE *CurrentParamOffset = &(StufferState->CurrentParamOffset);
  397. SMB_STUFFER_CONTROLS CurrentStufferControl = STUFFER_CTL_NORMAL;
  398. PSMB_HEADER SmbHeader = (PSMB_HEADER)BufferBase;
  399. PSZ CurrentFormatString = NULL;
  400. ULONG arg;
  401. UCHAR WordCount;
  402. USHORT ByteCount;
  403. //joejoe change this to zero later.....apparently some servers croak on nonzero pad
  404. #define PADBYTE ((UCHAR)0xee)
  405. PBYTE CopyPtr; ULONG CopyCount,EarlyReturn;
  406. PBYTE *RegionPtr;
  407. PUNICODE_STRING Zstring;
  408. PSZ Astring;
  409. PNET_ROOT NetRoot;
  410. PLARGE_INTEGER LargeInteger;
  411. PBYTE PreviousPosition;
  412. #if DBG
  413. ULONG offset, required_WCT;
  414. ULONG CurrentOffset_tmp;
  415. #endif
  416. PAGED_CODE();
  417. va_start(AP,StufferState);
  418. for (;;) {
  419. switch (CurrentStufferControl) {
  420. case STUFFER_CTL_SKIP:
  421. case STUFFER_CTL_NORMAL:
  422. CurrentFormatString = va_arg(AP,PSZ);
  423. StufferCLoopTrace(("StufferAC = %s\n",CurrentFormatString));
  424. ASSERT (CurrentFormatString);
  425. for (;*CurrentFormatString;CurrentFormatString++) {
  426. char CurrentFormatChar = *CurrentFormatString;
  427. #if DBG
  428. { char msgbuf[80];
  429. switch (CurrentFormatChar) {
  430. case 'W': case 'w':
  431. case 'D': case 'd':
  432. case 'Y': case 'y':
  433. case 'M': case 'm':
  434. case 'L': case 'l':
  435. case 'c': case '4': case '>':
  436. case '!':
  437. //this guys are skipable
  438. break;
  439. default:
  440. if (CurrentStufferControl != STUFFER_CTL_SKIP) break;
  441. DbgPrint("Bad skip char '%c'\n",*CurrentFormatString);
  442. //DbgBreakPoint();
  443. }}
  444. //these are the ones that we do the offset check for
  445. { char msgbuf[80];
  446. #ifndef WIN9X
  447. RxSprintf(msgbuf,"control char '%c'\n",*CurrentFormatString);
  448. #endif
  449. switch (CurrentFormatChar) {
  450. case 'W': case 'D': case 'Y': case 'M': case 'B':
  451. case 'Q': case 'A': case 'U': case 'V':
  452. case 'N':
  453. case 'L':
  454. case 'R':
  455. case 'P':
  456. offset = va_arg(AP,ULONG);
  457. required_WCT = offset>>16;
  458. offset = offset & 0xffff;
  459. CurrentOffset_tmp = (ULONG)(*CurrentPosition-*CurrentWct);
  460. if (offset && (offset != CurrentOffset_tmp)){
  461. DbgPrint("Bad offset %d; should be %d\n",offset,CurrentOffset_tmp);
  462. //DbgBreakPoint();
  463. }
  464. break;
  465. default:
  466. break;
  467. }}
  468. #endif
  469. switch (CurrentFormatChar) {
  470. case '0':
  471. StufferFLoopTrace((" StufferFloop '0'\n",0));
  472. //just do the wct field...
  473. **CurrentPosition = (UCHAR)MRXSMB_INITIAL_WCT;
  474. *CurrentPosition+=1;
  475. break;
  476. case 'X':
  477. StufferFLoopTrace((" StufferFloop 'X'\n",0));
  478. //do the wct field and the &x
  479. **CurrentPosition = (UCHAR)MRXSMB_INITIAL_WCT;
  480. *CurrentPosition+=1;
  481. SmbPutUlong (*CurrentPosition, (ULONG)MRXSMB_INITIAL_ANDX);
  482. *CurrentPosition+=sizeof(ULONG);
  483. break;
  484. case 'W':
  485. case 'w':
  486. arg = va_arg(AP,ULONG);
  487. if (CurrentStufferControl == STUFFER_CTL_SKIP) break;
  488. StufferFLoopTrace((" StufferFloop 'w' arg=%lu\n",arg));
  489. SmbPutUshort (*CurrentPosition, (USHORT)arg);
  490. *CurrentPosition+=sizeof(USHORT);
  491. break;
  492. case 'Y':
  493. case 'y':
  494. arg = va_arg(AP,UCHAR);
  495. if (CurrentStufferControl == STUFFER_CTL_SKIP) break;
  496. StufferFLoopTrace((" StufferFloop 'y' arg=%lu\n",arg));
  497. **CurrentPosition = (UCHAR)arg;
  498. *CurrentPosition+=sizeof(UCHAR);
  499. break;
  500. case 'M':
  501. case 'm':
  502. if (CurrentStufferControl == STUFFER_CTL_SKIP) break;
  503. StufferFLoopTrace((" StufferFloop 'm'\n",0));
  504. **CurrentPosition = 0;
  505. *CurrentPosition+=sizeof(UCHAR);
  506. break;
  507. case 'D':
  508. case 'd':
  509. arg = va_arg(AP,ULONG);
  510. if (CurrentStufferControl == STUFFER_CTL_SKIP) break;
  511. StufferFLoopTrace((" StufferFloop 'd' arg=%lu\n",arg));
  512. SmbPutUlong (*CurrentPosition, arg);
  513. *CurrentPosition+=sizeof(ULONG);
  514. break;
  515. case 'L':
  516. case 'l':
  517. LargeInteger = va_arg(AP,PLARGE_INTEGER);
  518. if (CurrentStufferControl == STUFFER_CTL_SKIP) break;
  519. StufferFLoopTrace((" StufferFloop 'l' arg=%0lx %0lx\n",
  520. LargeInteger->HighPart, LargeInteger->LowPart));
  521. SmbPutUlong (*CurrentPosition, LargeInteger->LowPart);
  522. SmbPutUlong (*CurrentPosition, LargeInteger->HighPart);
  523. *CurrentPosition+=2*sizeof(ULONG);
  524. break;
  525. case 'B':
  526. case 'b':
  527. ASSERT (**CurrentWct == MRXSMB_INITIAL_WCT);
  528. WordCount = (UCHAR)((*CurrentPosition-*CurrentWct)>>1); //the one gets shifted off
  529. StufferFLoopTrace((" StufferFloop 'b' Wct=%lu\n",WordCount));
  530. DbgDoit( ASSERT(!required_WCT || (WordCount == (required_WCT&0x7fff))); )
  531. **CurrentWct = (UCHAR)WordCount;
  532. SmbPutUshort (*CurrentPosition, (USHORT)MRXSMB_INITIAL_BCC);
  533. *CurrentBcc = *CurrentPosition;
  534. *CurrentPosition+=sizeof(USHORT);
  535. break;
  536. case 'Q':
  537. case 'q':
  538. StufferFLoopTrace((" StufferFloop 'q' \n",0));
  539. SmbPutUshort (*CurrentPosition, (USHORT)MRXSMB_INITIAL_DATAOFFSET);
  540. *CurrentDataOffset = *CurrentPosition;
  541. *CurrentPosition+=sizeof(USHORT);
  542. break;
  543. case '5':
  544. //fill in the data offset
  545. ASSERT (SmbGetUshort (*CurrentDataOffset) == MRXSMB_INITIAL_DATAOFFSET);
  546. ByteCount = (USHORT)(*CurrentPosition-BufferBase);
  547. StufferFLoopTrace((" StufferFloop '5' offset=%lu\n",ByteCount));
  548. SmbPutUshort (*CurrentDataOffset, (USHORT)ByteCount);
  549. break;
  550. case 'P':
  551. case 'p':
  552. StufferFLoopTrace((" StufferFloop 'p' \n",0));
  553. SmbPutUshort (*CurrentPosition, (USHORT)MRXSMB_INITIAL_PARAMOFFSET);
  554. *CurrentParamOffset = *CurrentPosition;
  555. *CurrentPosition+=sizeof(USHORT);
  556. break;
  557. case '6':
  558. //fill in the data offset
  559. ASSERT (SmbGetUshort (*CurrentParamOffset) == MRXSMB_INITIAL_PARAMOFFSET);
  560. ByteCount = (USHORT)(*CurrentPosition-BufferBase);
  561. StufferFLoopTrace((" StufferFloop '6' offset=%lu\n",ByteCount));
  562. SmbPutUshort (*CurrentParamOffset, (USHORT)ByteCount);
  563. break;
  564. case 'S':
  565. // pad to ULONG; we loop behind instead of adding so we can clear
  566. // out behind ourselves; apparently, some server croak on nonzero padding
  567. StufferFLoopTrace((" StufferFloop 'S' \n",0));
  568. PreviousPosition = *CurrentPosition;
  569. *CurrentPosition = (PBYTE)QuadAlignPtr(*CurrentPosition);
  570. for (;PreviousPosition!=*CurrentPosition;) {
  571. //StufferFLoopTrace((" StufferFloop 'S' prev,curr=%08lx %08lx\n",PreviousPosition,*CurrentPosition));
  572. *PreviousPosition++ = PADBYTE;
  573. }
  574. break;
  575. case 's':
  576. // pad to arg; we loop behind instead of adding so we can clear
  577. // out behind ourselves; apparently, some server croak on nonzero padding
  578. arg = va_arg(AP,ULONG);
  579. StufferFLoopTrace((" StufferFloop 's' arg=\n",arg));
  580. PreviousPosition = *CurrentPosition;
  581. *CurrentPosition += arg-1;
  582. *CurrentPosition = (PBYTE)( ((ULONG_PTR)(*CurrentPosition)) & ~((LONG)(arg-1)) );
  583. for (;PreviousPosition!=*CurrentPosition;) {
  584. //StufferFLoopTrace((" StufferFloop 'S' prev,curr=%08lx %08lx\n",PreviousPosition,*CurrentPosition));
  585. *PreviousPosition++ = PADBYTE;
  586. }
  587. break;
  588. case '1':
  589. // pad to USHORT; we loop behind instead of adding so we can clear
  590. // out behind ourselves; apparently, some server croak on nonzero padding
  591. StufferFLoopTrace((" StufferFloop '1' Curr=%08lx \n",*CurrentPosition));
  592. PreviousPosition = *CurrentPosition;
  593. *CurrentPosition += sizeof(USHORT)-1;
  594. StufferFLoopTrace((" Curr=%08lx \n",*CurrentPosition));
  595. *CurrentPosition = (PBYTE)( ((ULONG_PTR)(*CurrentPosition)) & ~((LONG)(sizeof(USHORT)-1)) );
  596. StufferFLoopTrace((" Curr=%08lx \n",*CurrentPosition));
  597. for (;PreviousPosition!=*CurrentPosition;) {
  598. StufferFLoopTrace((" StufferFloop '1' prev,curr=%08lx %08lx\n",PreviousPosition,*CurrentPosition));
  599. *PreviousPosition++ = PADBYTE;
  600. }
  601. break;
  602. case 'c':
  603. // copy in the bytes....used a lot in transact
  604. CopyCount = va_arg(AP,ULONG);
  605. CopyPtr = va_arg(AP,PBYTE);
  606. if (CurrentStufferControl == STUFFER_CTL_SKIP) break;
  607. StufferFLoopTrace((" StufferFloop 'c' copycount = %lu\n", CopyCount));
  608. PreviousPosition = *CurrentPosition;
  609. *CurrentPosition += CopyCount;
  610. for (;PreviousPosition!=*CurrentPosition;) {
  611. //StufferFLoopTrace((" StufferFloop 'S' prev,curr=%08lx %08lx\n",PreviousPosition,*CurrentPosition));
  612. *PreviousPosition++ = *CopyPtr++;
  613. }
  614. break;
  615. case 'R':
  616. case 'r':
  617. // copy in the bytes....used a lot in transact
  618. RegionPtr = va_arg(AP,PBYTE*);
  619. CopyCount = va_arg(AP,ULONG);
  620. StufferFLoopTrace((" StufferFloop 'r' regionsize = %lu\n", CopyCount));
  621. *RegionPtr = *CurrentPosition;
  622. *CurrentPosition += CopyCount;
  623. IF_DEBUG {
  624. PreviousPosition = *RegionPtr;
  625. for (;PreviousPosition!=*CurrentPosition;) {
  626. //StufferFLoopTrace((" StufferFloop 'S' prev,curr=%08lx %08lx\n",PreviousPosition,*CurrentPosition));
  627. *PreviousPosition++ = '-';
  628. }
  629. }
  630. break;
  631. case 'A':
  632. case 'a':
  633. //copy byte from an asciiz including the trailing NULL
  634. Astring = va_arg(AP,PSZ);
  635. StufferFLoopTrace((" StufferFloop 'a' stringing = %s\n", Astring));
  636. CopyCount = strlen(Astring)+1;
  637. //if (((ULONG)(*CurrentPosition))&1) {
  638. // StufferFLoopTrace((" StufferFloop 'a' aligning\n", 0));
  639. // *CurrentPosition+=1;
  640. //}
  641. PreviousPosition = *CurrentPosition;
  642. *CurrentPosition += CopyCount;
  643. if (*CurrentPosition >= StufferState->BufferLimit) {
  644. StufferFLoopTrace((" StufferFloop 'a' bufferoverrun\n", 0));
  645. ASSERT(!"BufferOverrun");
  646. return(RX_MAP_STATUS(BUFFER_OVERFLOW));
  647. }
  648. RtlCopyMemory(PreviousPosition,Astring,CopyCount);
  649. break;
  650. case 'z':
  651. case '4':
  652. case '>':
  653. Zstring = va_arg(AP,PUNICODE_STRING);
  654. StufferFLoopTrace((" StufferFloop '4/z/>' stringing = %wZ, cp=\n", Zstring,*CurrentPosition ));
  655. if (CurrentStufferControl == STUFFER_CTL_SKIP) break;
  656. if (CurrentFormatChar=='4') {
  657. //first lay down a x'04' and then copy either a asciiz or a unicodez depending on the flags setting
  658. **CurrentPosition = (UCHAR)4; //ascii marker
  659. *CurrentPosition+=1;
  660. } else if (CurrentFormatChar=='>'){
  661. //back up over the previous NULL
  662. //
  663. *CurrentPosition-=(FlagOn(SmbHeader->Flags2,SMB_FLAGS2_UNICODE)?sizeof(WCHAR):sizeof(char));
  664. StufferFLoopTrace((" StufferFloop '4/z/>' afterroolback, cp=\n", *CurrentPosition ));
  665. }
  666. if (FlagOn(SmbHeader->Flags2,SMB_FLAGS2_UNICODE)){
  667. if (((ULONG_PTR)(*CurrentPosition))&1) {
  668. StufferFLoopTrace((" StufferFloop '4/z/>' aligning\n", 0));
  669. *CurrentPosition+=1;
  670. }
  671. PreviousPosition = *CurrentPosition;
  672. *CurrentPosition += (Zstring->Length + sizeof(WCHAR));
  673. if (*CurrentPosition >= StufferState->BufferLimit) {
  674. StufferFLoopTrace((" StufferFloop '4/z/>' bufferoverrun\n", 0));
  675. ASSERT(!"BufferOverrun");
  676. return(RX_MAP_STATUS(BUFFER_OVERFLOW));
  677. }
  678. RtlCopyMemory(PreviousPosition,Zstring->Buffer,Zstring->Length);
  679. *(((PWCHAR)(*CurrentPosition))-1) = 0;
  680. } else {
  681. NTSTATUS Status;
  682. OEM_STRING OemString;
  683. OemString.Length =
  684. OemString.MaximumLength =
  685. (USHORT)( StufferState->BufferLimit - *CurrentPosition - sizeof(CHAR));
  686. OemString.Buffer = *CurrentPosition;
  687. if (FlagOn(SmbHeader->Flags,SMB_FLAGS_CASE_INSENSITIVE) &&
  688. !FlagOn(SmbHeader->Flags2,SMB_FLAGS2_KNOWS_LONG_NAMES)) {
  689. Status = RtlUpcaseUnicodeStringToOemString(
  690. &OemString,
  691. Zstring,
  692. FALSE);
  693. } else {
  694. Status = RtlUnicodeStringToOemString(
  695. &OemString,
  696. Zstring,
  697. FALSE);
  698. }
  699. if (!NT_SUCCESS(Status)) {
  700. StufferFLoopTrace((" StufferFloop '4/z/>' bufferoverrun(ascii)\n", 0));
  701. ASSERT(!"BufferOverrun");
  702. return(RX_MAP_STATUS(BUFFER_OVERFLOW));
  703. }
  704. *CurrentPosition += OemString.Length + 1;
  705. *(*CurrentPosition-1) = 0;
  706. }
  707. break;
  708. case 'U':
  709. case 'u':
  710. //copy bytes from an UNICODE string including a trailing NULL
  711. Zstring = va_arg(AP,PUNICODE_STRING);
  712. StufferFLoopTrace((" StufferFloop 'u' stringing = %wZ\n", Zstring));
  713. if (((ULONG_PTR)(*CurrentPosition))&1) {
  714. StufferFLoopTrace((" StufferFloop 'u' aligning\n", 0));
  715. *CurrentPosition+=1;
  716. }
  717. PreviousPosition = *CurrentPosition;
  718. *CurrentPosition += (Zstring->Length + sizeof(WCHAR));
  719. if (*CurrentPosition >= StufferState->BufferLimit) {
  720. StufferFLoopTrace((" StufferFloop 'u' bufferoverrun\n", 0));
  721. return(RX_MAP_STATUS(BUFFER_OVERFLOW));
  722. }
  723. RtlCopyMemory(PreviousPosition,Zstring->Buffer,Zstring->Length);
  724. *(((PWCHAR)(*CurrentPosition))-1) = 0;
  725. break;
  726. case 'V':
  727. case 'v':
  728. //copy bytes from an UNICODE string no trailing NUL
  729. Zstring = va_arg(AP,PUNICODE_STRING);
  730. StufferFLoopTrace((" StufferFloop 'v' stringing = %wZ\n", Zstring));
  731. if (((ULONG_PTR)(*CurrentPosition))&1) {
  732. StufferFLoopTrace((" StufferFloop 'v' aligning\n", 0));
  733. *CurrentPosition+=1;
  734. }
  735. PreviousPosition = *CurrentPosition;
  736. *CurrentPosition += Zstring->Length;
  737. if (*CurrentPosition >= StufferState->BufferLimit) {
  738. StufferFLoopTrace((" StufferFloop 'v' bufferoverrun\n", 0));
  739. ASSERT(!"BufferOverrun");
  740. return(RX_MAP_STATUS(BUFFER_OVERFLOW));
  741. }
  742. RtlCopyMemory(PreviousPosition,Zstring->Buffer,Zstring->Length);
  743. break;
  744. case 'N':
  745. case 'n':
  746. //copy bytes from a NetRoot name....w null
  747. //joejoe we need to do the # thing here
  748. NetRoot = va_arg(AP,PNET_ROOT);
  749. ASSERT(NodeType(NetRoot)==RDBSS_NTC_NETROOT);
  750. Zstring = &NetRoot->PrefixEntry.Prefix;
  751. StufferFLoopTrace((" StufferFloop 'n' stringing = %wZ\n", Zstring));
  752. if (StufferState->Flags2Copy&SMB_FLAGS2_UNICODE) {
  753. if (((ULONG_PTR)(*CurrentPosition))&1) {
  754. StufferFLoopTrace((" StufferFloop 'n' aligning\n", 0));
  755. *CurrentPosition+=1;
  756. }
  757. PreviousPosition = *CurrentPosition;
  758. *CurrentPosition += (Zstring->Length + 2 * sizeof(WCHAR)); //extra \ plus a nul
  759. if (*CurrentPosition >= StufferState->BufferLimit) {
  760. StufferFLoopTrace((" StufferFloop 'n' bufferoverrun\n", 0));
  761. ASSERT(!"BufferOverrun");
  762. return(RX_MAP_STATUS(BUFFER_OVERFLOW));
  763. }
  764. *((PWCHAR)PreviousPosition) = '\\';
  765. RtlCopyMemory(PreviousPosition+sizeof(WCHAR),Zstring->Buffer,Zstring->Length);
  766. *(((PWCHAR)(*CurrentPosition))-1) = 0;
  767. }
  768. break;
  769. case '?':
  770. //early out....used in transact to do the setup
  771. EarlyReturn = va_arg(AP,ULONG);
  772. StufferFLoopTrace((" StufferFloop '?' out if 0==%08lx\n",EarlyReturn));
  773. if (EarlyReturn==0) return RX_MAP_STATUS(SUCCESS);
  774. break;
  775. case '.':
  776. //noop...used to reenter without a real formatting string
  777. StufferFLoopTrace((" StufferFloop '.'\n",0));
  778. break;
  779. case '!':
  780. if (CurrentStufferControl == STUFFER_CTL_SKIP) break;
  781. ASSERT (SmbGetUshort (*CurrentBcc) == MRXSMB_INITIAL_BCC);
  782. ByteCount = (USHORT)(*CurrentPosition-*CurrentBcc-sizeof(USHORT));
  783. StufferFLoopTrace((" StufferFloop '!' arg=%lu\n",ByteCount));
  784. SmbPutUshort (*CurrentBcc, (USHORT)ByteCount);
  785. return RX_MAP_STATUS(SUCCESS);
  786. default:
  787. StufferFLoopTrace((" StufferFloop '%c' BADBADBAD\n",*CurrentFormatString));
  788. ASSERT(!"Illegal Controlstring character\n");
  789. } //switch
  790. }//for
  791. break;
  792. case 0:
  793. return RX_MAP_STATUS(SUCCESS);
  794. default:
  795. StufferCLoopTrace((" StufferCloop %u BADBADBAD\n",CurrentStufferControl));
  796. ASSERT(!"IllegalStufferControl\n");
  797. }//switch
  798. CurrentStufferControl = va_arg(AP,SMB_STUFFER_CONTROLS);
  799. StufferCLoopTrace((" StufferCloop NewStufferControl=%u \n",CurrentStufferControl));
  800. } //for
  801. return RX_MAP_STATUS(SUCCESS);
  802. }
  803. VOID
  804. MRxSmbStuffAppendRawData(
  805. IN OUT PSMBSTUFFER_BUFFER_STATE StufferState,
  806. IN PMDL Mdl
  807. )
  808. {
  809. PMDL pMdl;
  810. PAGED_CODE();
  811. ASSERT(!StufferState->DataMdl);
  812. pMdl = StufferState->DataMdl = Mdl;
  813. StufferState->DataSize = 0;
  814. while (pMdl != NULL) {
  815. StufferState->DataSize += pMdl->ByteCount;
  816. pMdl = pMdl->Next;
  817. }
  818. return;
  819. }
  820. VOID
  821. MRxSmbStuffAppendSmbData(
  822. IN OUT PSMBSTUFFER_BUFFER_STATE StufferState,
  823. IN PMDL Mdl
  824. )
  825. {
  826. ULONG Offset;
  827. PAGED_CODE();
  828. ASSERT(!StufferState->DataMdl);
  829. StufferState->DataMdl = Mdl;
  830. StufferState->DataSize = Mdl->ByteCount;
  831. //now reach back into the buffer and set the SMB data offset; if it is already set...just get out
  832. if (SmbGetUshort (StufferState->CurrentDataOffset) == MRXSMB_INITIAL_DATAOFFSET){
  833. Offset = (ULONG)(StufferState->CurrentPosition - StufferState->BufferBase);
  834. RxDbgTrace(0, Dbg,("MRxSmbStuffAppendSmbData offset=%lu\n",Offset));
  835. SmbPutUshort (StufferState->CurrentDataOffset, (USHORT)Offset);
  836. }
  837. return;
  838. }
  839. VOID
  840. MRxSmbStuffSetByteCount(
  841. IN OUT PSMBSTUFFER_BUFFER_STATE StufferState
  842. )
  843. {
  844. ULONG ByteCount;
  845. PAGED_CODE();
  846. ASSERT (SmbGetUshort (StufferState->CurrentBcc) == MRXSMB_INITIAL_BCC);
  847. ByteCount = (ULONG)(StufferState->CurrentPosition
  848. - StufferState->CurrentBcc
  849. - sizeof(USHORT)
  850. + StufferState->DataSize);
  851. RxDbgTrace(0, Dbg,("MRxSmbStuffSetByteCount ByteCount=%lu\n",ByteCount));
  852. SmbPutUshort (StufferState->CurrentBcc, (USHORT)ByteCount);
  853. return;
  854. }