SMB1Command.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. /* Copyright (C) 2014-2017 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
  2. *
  3. * You can redistribute this program and/or modify it under the terms of
  4. * the GNU Lesser Public License as published by the Free Software Foundation,
  5. * either version 3 of the License, or (at your option) any later version.
  6. */
  7. using System;
  8. using System.Collections.Generic;
  9. using System.Text;
  10. using Utilities;
  11. namespace SMBLibrary.SMB1
  12. {
  13. public abstract class SMB1Command
  14. {
  15. protected byte[] SMBParameters; // SMB_Parameters
  16. protected byte[] SMBData; // SMB_Data
  17. public SMB1Command()
  18. {
  19. SMBParameters = new byte[0];
  20. SMBData = new byte[0];
  21. }
  22. public SMB1Command(byte[] buffer, int offset, bool isUnicode)
  23. {
  24. byte wordCount = ByteReader.ReadByte(buffer, ref offset);
  25. SMBParameters = ByteReader.ReadBytes(buffer, ref offset, wordCount * 2);
  26. ushort byteCount = LittleEndianReader.ReadUInt16(buffer, ref offset);
  27. SMBData = ByteReader.ReadBytes(buffer, ref offset, byteCount);
  28. }
  29. public abstract CommandName CommandName
  30. {
  31. get;
  32. }
  33. public virtual byte[] GetBytes(bool isUnicode)
  34. {
  35. if (SMBParameters.Length % 2 > 0)
  36. {
  37. throw new Exception("SMB_Parameters Length must be a multiple of 2");
  38. }
  39. int length = 1 + SMBParameters.Length + 2 + SMBData.Length;
  40. byte[] buffer = new byte[length];
  41. byte wordCount = (byte)(SMBParameters.Length / 2);
  42. if (this is NTCreateAndXResponseExtended)
  43. {
  44. // [MS-SMB] Section 2.2.4.9.2 and Note <51>:
  45. // Windows-based SMB servers send 50 (0x32) words in the extended response
  46. // although they set the WordCount field to 0x2A
  47. // wordCount SHOULD be 0x2A
  48. wordCount = 0x2A;
  49. }
  50. ushort byteCount = (ushort)SMBData.Length;
  51. int offset = 0;
  52. ByteWriter.WriteByte(buffer, ref offset, wordCount);
  53. ByteWriter.WriteBytes(buffer, ref offset, SMBParameters);
  54. LittleEndianWriter.WriteUInt16(buffer, ref offset, byteCount);
  55. ByteWriter.WriteBytes(buffer, ref offset, SMBData);
  56. return buffer;
  57. }
  58. public static SMB1Command ReadCommand(byte[] buffer, int offset, CommandName commandName, SMB1Header header)
  59. {
  60. if ((header.Flags & HeaderFlags.Reply) > 0)
  61. {
  62. return ReadCommandResponse(buffer, offset, commandName, header.UnicodeFlag);
  63. }
  64. else
  65. {
  66. return ReadCommandRequest(buffer, offset, commandName, header.UnicodeFlag);
  67. }
  68. }
  69. public static SMB1Command ReadCommandRequest(byte[] buffer, int offset, CommandName commandName, bool isUnicode)
  70. {
  71. switch (commandName)
  72. {
  73. case CommandName.SMB_COM_CREATE_DIRECTORY:
  74. return new CreateDirectoryRequest(buffer, offset, isUnicode);
  75. case CommandName.SMB_COM_DELETE_DIRECTORY:
  76. return new DeleteDirectoryRequest(buffer, offset, isUnicode);
  77. case CommandName.SMB_COM_CLOSE:
  78. return new CloseRequest(buffer, offset);
  79. case CommandName.SMB_COM_FLUSH:
  80. return new FlushRequest(buffer, offset);
  81. case CommandName.SMB_COM_DELETE:
  82. return new DeleteRequest(buffer, offset, isUnicode);
  83. case CommandName.SMB_COM_RENAME:
  84. return new RenameRequest(buffer, offset, isUnicode);
  85. case CommandName.SMB_COM_QUERY_INFORMATION:
  86. return new QueryInformationRequest(buffer, offset, isUnicode);
  87. case CommandName.SMB_COM_SET_INFORMATION:
  88. return new SetInformationRequest(buffer, offset, isUnicode);
  89. case CommandName.SMB_COM_READ:
  90. return new ReadRequest(buffer, offset);
  91. case CommandName.SMB_COM_WRITE:
  92. return new WriteRequest(buffer, offset);
  93. case CommandName.SMB_COM_CHECK_DIRECTORY:
  94. return new CheckDirectoryRequest(buffer, offset, isUnicode);
  95. case CommandName.SMB_COM_WRITE_RAW:
  96. return new WriteRawRequest(buffer, offset);
  97. case CommandName.SMB_COM_SET_INFORMATION2:
  98. return new SetInformation2Request(buffer, offset);
  99. case CommandName.SMB_COM_LOCKING_ANDX:
  100. return new LockingAndXRequest(buffer, offset);
  101. case CommandName.SMB_COM_TRANSACTION:
  102. return new TransactionRequest(buffer, offset, isUnicode);
  103. case CommandName.SMB_COM_TRANSACTION_SECONDARY:
  104. return new TransactionSecondaryRequest(buffer, offset);
  105. case CommandName.SMB_COM_ECHO:
  106. return new EchoRequest(buffer, offset);
  107. case CommandName.SMB_COM_OPEN_ANDX:
  108. return new OpenAndXRequest(buffer, offset, isUnicode);
  109. case CommandName.SMB_COM_READ_ANDX:
  110. return new ReadAndXRequest(buffer, offset);
  111. case CommandName.SMB_COM_WRITE_ANDX:
  112. return new WriteAndXRequest(buffer, offset, isUnicode);
  113. case CommandName.SMB_COM_TRANSACTION2:
  114. return new Transaction2Request(buffer, offset, isUnicode);
  115. case CommandName.SMB_COM_TRANSACTION2_SECONDARY:
  116. return new Transaction2SecondaryRequest(buffer, offset);
  117. case CommandName.SMB_COM_FIND_CLOSE2:
  118. return new FindClose2Request(buffer, offset);
  119. case CommandName.SMB_COM_TREE_DISCONNECT:
  120. return new TreeDisconnectRequest(buffer, offset);
  121. case CommandName.SMB_COM_NEGOTIATE:
  122. return new NegotiateRequest(buffer, offset);
  123. case CommandName.SMB_COM_SESSION_SETUP_ANDX:
  124. {
  125. byte wordCount = ByteReader.ReadByte(buffer, offset);
  126. if (wordCount * 2 == SessionSetupAndXRequest.ParametersLength)
  127. {
  128. return new SessionSetupAndXRequest(buffer, offset, isUnicode);
  129. }
  130. else if (wordCount * 2 == SessionSetupAndXRequestExtended.ParametersLength)
  131. {
  132. return new SessionSetupAndXRequestExtended(buffer, offset, isUnicode);
  133. }
  134. else
  135. {
  136. throw new InvalidRequestException();
  137. }
  138. }
  139. case CommandName.SMB_COM_LOGOFF_ANDX:
  140. return new LogoffAndXRequest(buffer, offset);
  141. case CommandName.SMB_COM_TREE_CONNECT_ANDX:
  142. return new TreeConnectAndXRequest(buffer, offset, isUnicode);
  143. case CommandName.SMB_COM_NT_TRANSACT:
  144. return new NTTransactRequest(buffer, offset);
  145. case CommandName.SMB_COM_NT_TRANSACT_SECONDARY:
  146. return new NTTransactSecondaryRequest(buffer, offset);
  147. case CommandName.SMB_COM_NT_CREATE_ANDX:
  148. return new NTCreateAndXRequest(buffer, offset, isUnicode);
  149. case CommandName.SMB_COM_NT_CANCEL:
  150. return new NTCancelRequest(buffer, offset);
  151. default:
  152. throw new NotImplementedException("SMB Command 0x" + commandName.ToString("X"));
  153. }
  154. }
  155. public static SMB1Command ReadCommandResponse(byte[] buffer, int offset, CommandName commandName, bool isUnicode)
  156. {
  157. byte wordCount = ByteReader.ReadByte(buffer, offset);
  158. switch (commandName)
  159. {
  160. case CommandName.SMB_COM_CREATE_DIRECTORY:
  161. return new CreateDirectoryResponse(buffer, offset);
  162. case CommandName.SMB_COM_DELETE_DIRECTORY:
  163. return new DeleteDirectoryResponse(buffer, offset);
  164. case CommandName.SMB_COM_CLOSE:
  165. return new CloseResponse(buffer, offset);
  166. case CommandName.SMB_COM_FLUSH:
  167. return new FlushResponse(buffer, offset);
  168. case CommandName.SMB_COM_DELETE:
  169. return new DeleteResponse(buffer, offset);
  170. case CommandName.SMB_COM_RENAME:
  171. return new RenameResponse(buffer, offset);
  172. case CommandName.SMB_COM_QUERY_INFORMATION:
  173. return new QueryInformationResponse(buffer, offset);
  174. case CommandName.SMB_COM_SET_INFORMATION:
  175. return new SetInformationResponse(buffer, offset);
  176. case CommandName.SMB_COM_READ:
  177. return new ReadResponse(buffer, offset);
  178. case CommandName.SMB_COM_WRITE:
  179. return new WriteResponse(buffer, offset);
  180. case CommandName.SMB_COM_CHECK_DIRECTORY:
  181. return new CheckDirectoryResponse(buffer, offset);
  182. case CommandName.SMB_COM_WRITE_RAW:
  183. return new WriteRawInterimResponse(buffer, offset);
  184. case CommandName.SMB_COM_WRITE_COMPLETE:
  185. return new WriteRawFinalResponse(buffer, offset);
  186. case CommandName.SMB_COM_SET_INFORMATION2:
  187. return new SetInformation2Response(buffer, offset);
  188. case CommandName.SMB_COM_LOCKING_ANDX:
  189. return new LockingAndXResponse(buffer, offset);
  190. case CommandName.SMB_COM_TRANSACTION:
  191. {
  192. if (wordCount * 2 == TransactionInterimResponse.ParametersLength)
  193. {
  194. return new TransactionInterimResponse(buffer, offset);
  195. }
  196. else
  197. {
  198. return new TransactionResponse(buffer, offset);
  199. }
  200. }
  201. case CommandName.SMB_COM_ECHO:
  202. return new EchoResponse(buffer, offset);
  203. case CommandName.SMB_COM_OPEN_ANDX:
  204. {
  205. if (wordCount * 2 == OpenAndXResponse.ParametersLength)
  206. {
  207. throw new NotImplementedException();
  208. }
  209. else if (wordCount * 2 == OpenAndXResponseExtended.ParametersLength)
  210. {
  211. throw new NotImplementedException();
  212. }
  213. else
  214. {
  215. throw new InvalidRequestException(); ;
  216. }
  217. }
  218. case CommandName.SMB_COM_READ_ANDX:
  219. return new ReadAndXResponse(buffer, offset, isUnicode);
  220. case CommandName.SMB_COM_WRITE_ANDX:
  221. return new WriteAndXResponse(buffer, offset);
  222. case CommandName.SMB_COM_TRANSACTION2:
  223. {
  224. if (wordCount * 2 == Transaction2InterimResponse.ParametersLength)
  225. {
  226. return new Transaction2InterimResponse(buffer, offset);
  227. }
  228. else
  229. {
  230. return new Transaction2Response(buffer, offset);
  231. }
  232. }
  233. case CommandName.SMB_COM_FIND_CLOSE2:
  234. return new FindClose2Response(buffer, offset);
  235. case CommandName.SMB_COM_TREE_DISCONNECT:
  236. return new TreeDisconnectResponse(buffer, offset);
  237. case CommandName.SMB_COM_NEGOTIATE:
  238. {
  239. if (wordCount * 2 == NegotiateResponse.ParametersLength)
  240. {
  241. return new NegotiateResponse(buffer, offset, isUnicode);
  242. }
  243. else if (wordCount * 2 == NegotiateResponseExtended.ParametersLength)
  244. {
  245. return new NegotiateResponseExtended(buffer, offset);
  246. }
  247. else
  248. {
  249. throw new InvalidRequestException();
  250. }
  251. }
  252. case CommandName.SMB_COM_SESSION_SETUP_ANDX:
  253. if (wordCount * 2 == SessionSetupAndXResponse.ParametersLength)
  254. {
  255. return new SessionSetupAndXResponse(buffer, offset, isUnicode);
  256. }
  257. else if (wordCount * 2 == SessionSetupAndXResponseExtended.ParametersLength)
  258. {
  259. return new SessionSetupAndXResponseExtended(buffer, offset, isUnicode);
  260. }
  261. else
  262. {
  263. throw new InvalidRequestException(); ;
  264. }
  265. case CommandName.SMB_COM_LOGOFF_ANDX:
  266. return new LogoffAndXResponse(buffer, offset);
  267. case CommandName.SMB_COM_TREE_CONNECT_ANDX:
  268. return new TreeConnectAndXResponse(buffer, offset, isUnicode);
  269. case CommandName.SMB_COM_NT_TRANSACT:
  270. {
  271. if (wordCount * 2 == NTTransactInterimResponse.ParametersLength)
  272. {
  273. return new NTTransactInterimResponse(buffer, offset);
  274. }
  275. else
  276. {
  277. return new NTTransactResponse(buffer, offset);
  278. }
  279. }
  280. case CommandName.SMB_COM_NT_CREATE_ANDX:
  281. return new NTCreateAndXResponse(buffer, offset);
  282. default:
  283. throw new NotImplementedException("SMB Command 0x" + commandName.ToString("X"));
  284. }
  285. }
  286. public static implicit operator List<SMB1Command>(SMB1Command command)
  287. {
  288. List<SMB1Command> result = new List<SMB1Command>();
  289. result.Add(command);
  290. return result;
  291. }
  292. }
  293. }