SMBServer.SMB2.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337
  1. /* Copyright (C) 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 SMBLibrary.NetBios;
  10. using SMBLibrary.Server.SMB2;
  11. using SMBLibrary.SMB2;
  12. using Utilities;
  13. namespace SMBLibrary.Server
  14. {
  15. public partial class SMBServer
  16. {
  17. // Key is the persistent portion of the FileID
  18. private Dictionary<ulong, OpenFileObject> m_globalOpenFiles = new Dictionary<ulong, OpenFileObject>();
  19. private static ulong m_nextPersistentFileID = 1; // A numeric value that uniquely identifies the open handle to a file or a pipe within the scope of all opens granted by the server
  20. private ulong? AllocatePersistentFileID()
  21. {
  22. for (ulong offset = 0; offset < UInt64.MaxValue; offset++)
  23. {
  24. ulong persistentID = (ulong)(m_nextPersistentFileID + offset);
  25. if (persistentID == 0 || persistentID == 0xFFFFFFFFFFFFFFFF)
  26. {
  27. continue;
  28. }
  29. if (!m_globalOpenFiles.ContainsKey(persistentID))
  30. {
  31. m_nextPersistentFileID = (ulong)(persistentID + 1);
  32. return persistentID;
  33. }
  34. }
  35. return null;
  36. }
  37. private void ProcessSMB2RequestChain(List<SMB2Command> requestChain, ref ConnectionState state)
  38. {
  39. List<SMB2Command> responseChain = new List<SMB2Command>();
  40. FileID? fileID = null;
  41. foreach (SMB2Command request in requestChain)
  42. {
  43. if (request.Header.IsRelatedOperations && fileID.HasValue)
  44. {
  45. SetRequestFileID(request, fileID.Value);
  46. }
  47. SMB2Command response = ProcessSMB2Command(request, ref state);
  48. if (response != null)
  49. {
  50. UpdateSMB2Header(response, request);
  51. responseChain.Add(response);
  52. if (!request.Header.IsRelatedOperations)
  53. {
  54. fileID = GetResponseFileID(response);
  55. }
  56. }
  57. }
  58. if (responseChain.Count > 0)
  59. {
  60. TrySendResponseChain(state, responseChain);
  61. }
  62. }
  63. /// <summary>
  64. /// May return null
  65. /// </summary>
  66. private SMB2Command ProcessSMB2Command(SMB2Command command, ref ConnectionState state)
  67. {
  68. if (state.Dialect == SMBDialect.NotSet)
  69. {
  70. if (command is NegotiateRequest)
  71. {
  72. NegotiateRequest request = (NegotiateRequest)command;
  73. SMB2Command response = NegotiateHelper.GetNegotiateResponse(request, m_securityProvider, state, m_serverGuid, m_serverStartTime);
  74. if (state.Dialect != SMBDialect.NotSet)
  75. {
  76. state = new SMB2ConnectionState(state, AllocatePersistentFileID);
  77. m_connectionManager.AddConnection(state);
  78. }
  79. return response;
  80. }
  81. else
  82. {
  83. // [MS-SMB2] If the request being received is not an SMB2 NEGOTIATE Request [..]
  84. // and Connection.NegotiateDialect is 0xFFFF or 0x02FF, the server MUST
  85. // disconnect the connection.
  86. state.LogToServer(Severity.Debug, "Invalid Connection State for command {0}", command.CommandName.ToString());
  87. state.ClientSocket.Close();
  88. return null;
  89. }
  90. }
  91. else if (command is NegotiateRequest)
  92. {
  93. // [MS-SMB2] If Connection.NegotiateDialect is 0x0202, 0x0210, 0x0300, 0x0302, or 0x0311,
  94. // the server MUST disconnect the connection.
  95. state.LogToServer(Severity.Debug, "Rejecting NegotiateRequest. NegotiateDialect is already set");
  96. state.ClientSocket.Close();
  97. return null;
  98. }
  99. else
  100. {
  101. return ProcessSMB2Command(command, (SMB2ConnectionState)state);
  102. }
  103. }
  104. private SMB2Command ProcessSMB2Command(SMB2Command command, SMB2ConnectionState state)
  105. {
  106. if (command is SessionSetupRequest)
  107. {
  108. return SessionSetupHelper.GetSessionSetupResponse((SessionSetupRequest)command, m_securityProvider, state);
  109. }
  110. else if (command is EchoRequest)
  111. {
  112. return new EchoResponse();
  113. }
  114. else
  115. {
  116. SMB2Session session = state.GetSession(command.Header.SessionID);
  117. if (session == null)
  118. {
  119. return new ErrorResponse(command.CommandName, NTStatus.STATUS_USER_SESSION_DELETED);
  120. }
  121. if (command is TreeConnectRequest)
  122. {
  123. return TreeConnectHelper.GetTreeConnectResponse((TreeConnectRequest)command, state, m_services, m_shares);
  124. }
  125. else if (command is LogoffRequest)
  126. {
  127. m_securityProvider.DeleteSecurityContext(ref session.SecurityContext.AuthenticationContext);
  128. state.RemoveSession(command.Header.SessionID);
  129. return new LogoffResponse();
  130. }
  131. else
  132. {
  133. // Cancel requests can have an ASYNC header (TreeID will not be present)
  134. if (command is CancelRequest)
  135. {
  136. if (command.Header.IsAsync && command.Header.AsyncID == 0)
  137. {
  138. ErrorResponse response = new ErrorResponse(command.CommandName, NTStatus.STATUS_CANCELLED);
  139. response.Header.IsAsync = true;
  140. return response;
  141. }
  142. // [MS-SMB2] If a request is not found, the server MUST stop processing for this cancel request. No response is sent.
  143. return null;
  144. }
  145. ISMBShare share = session.GetConnectedTree(command.Header.TreeID);
  146. if (share == null)
  147. {
  148. return new ErrorResponse(command.CommandName, NTStatus.STATUS_NETWORK_NAME_DELETED);
  149. }
  150. if (command is TreeDisconnectRequest)
  151. {
  152. session.RemoveConnectedTree(command.Header.TreeID);
  153. return new TreeDisconnectResponse();
  154. }
  155. else if (command is CreateRequest)
  156. {
  157. return CreateHelper.GetCreateResponse((CreateRequest)command, share, state);
  158. }
  159. else if (command is QueryInfoRequest)
  160. {
  161. return QueryInfoHelper.GetQueryInfoResponse((QueryInfoRequest)command, share, state);
  162. }
  163. else if (command is SetInfoRequest)
  164. {
  165. return SetInfoHelper.GetSetInfoResponse((SetInfoRequest)command, share, state);
  166. }
  167. else if (command is QueryDirectoryRequest)
  168. {
  169. return QueryDirectoryHelper.GetQueryDirectoryResponse((QueryDirectoryRequest)command, share, state);
  170. }
  171. else if (command is ReadRequest)
  172. {
  173. return ReadWriteResponseHelper.GetReadResponse((ReadRequest)command, share, state);
  174. }
  175. else if (command is WriteRequest)
  176. {
  177. return ReadWriteResponseHelper.GetWriteResponse((WriteRequest)command, share, state);
  178. }
  179. else if (command is FlushRequest)
  180. {
  181. FlushRequest request = (FlushRequest)command;
  182. OpenFileObject openFile = session.GetOpenFileObject(request.FileId.Persistent);
  183. if (openFile == null)
  184. {
  185. return new ErrorResponse(request.CommandName, NTStatus.STATUS_FILE_CLOSED);
  186. }
  187. NTStatus status = share.FileStore.FlushFileBuffers(openFile.Handle);
  188. if (status != NTStatus.STATUS_SUCCESS)
  189. {
  190. return new ErrorResponse(request.CommandName, status);
  191. }
  192. return new FlushResponse();
  193. }
  194. else if (command is CloseRequest)
  195. {
  196. return CloseHelper.GetCloseResponse((CloseRequest)command, share, state);
  197. }
  198. else if (command is IOCtlRequest)
  199. {
  200. return IOCtlHelper.GetIOCtlResponse((IOCtlRequest)command, share, state);
  201. }
  202. else if (command is ChangeNotifyRequest)
  203. {
  204. // [MS-SMB2] If the underlying object store does not support change notifications, the server MUST fail this request with STATUS_NOT_SUPPORTED
  205. ErrorResponse response = new ErrorResponse(command.CommandName, NTStatus.STATUS_NOT_SUPPORTED);
  206. // Windows 7 / 8 / 10 will infinitely retry sending ChangeNotify requests if the response does not have SMB2_FLAGS_ASYNC_COMMAND set.
  207. // Note: NoRemoteChangeNotify can be set in the registry to prevent the client from sending ChangeNotify requests altogether.
  208. response.Header.IsAsync = true;
  209. return response;
  210. }
  211. }
  212. }
  213. return new ErrorResponse(command.CommandName, NTStatus.STATUS_NOT_SUPPORTED);
  214. }
  215. private static void TrySendResponse(ConnectionState state, SMB2Command response)
  216. {
  217. SessionMessagePacket packet = new SessionMessagePacket();
  218. packet.Trailer = response.GetBytes();
  219. TrySendPacket(state, packet);
  220. state.LogToServer(Severity.Verbose, "SMB2 response sent: {0}, Packet length: {1}", response.CommandName.ToString(), packet.Length);
  221. }
  222. private static void TrySendResponseChain(ConnectionState state, List<SMB2Command> responseChain)
  223. {
  224. byte[] sessionKey = null;
  225. if (state is SMB2ConnectionState)
  226. {
  227. // Note: multiple sessions MAY be multiplexed on the same connection, so theoretically
  228. // we could have compounding unrelated requests from different sessions.
  229. // In practice however this is not a real problem.
  230. ulong sessionID = responseChain[0].Header.SessionID;
  231. if (sessionID != 0)
  232. {
  233. SMB2Session session = ((SMB2ConnectionState)state).GetSession(sessionID);
  234. if (session != null)
  235. {
  236. sessionKey = session.SessionKey;
  237. }
  238. }
  239. }
  240. SessionMessagePacket packet = new SessionMessagePacket();
  241. packet.Trailer = SMB2Command.GetCommandChainBytes(responseChain, sessionKey);
  242. TrySendPacket(state, packet);
  243. state.LogToServer(Severity.Verbose, "SMB2 response chain sent: Response count: {0}, First response: {1}, Packet length: {2}", responseChain.Count, responseChain[0].CommandName.ToString(), packet.Length);
  244. }
  245. private static void UpdateSMB2Header(SMB2Command response, SMB2Command request)
  246. {
  247. response.Header.MessageID = request.Header.MessageID;
  248. response.Header.CreditCharge = request.Header.CreditCharge;
  249. response.Header.Credits = Math.Max((ushort)1, request.Header.Credits);
  250. response.Header.IsRelatedOperations = request.Header.IsRelatedOperations;
  251. // [MS-SMB2] The server SHOULD sign the message [..] if the request was signed by the client,
  252. // and the response is not an interim response to an asynchronously processed request.
  253. bool isInterimResponse = (request.Header.IsAsync && request.Header.Status == NTStatus.STATUS_PENDING);
  254. response.Header.IsSigned = request.Header.IsSigned && !isInterimResponse;
  255. response.Header.Reserved = request.Header.Reserved;
  256. if (response.Header.SessionID == 0)
  257. {
  258. response.Header.SessionID = request.Header.SessionID;
  259. }
  260. if (response.Header.TreeID == 0)
  261. {
  262. response.Header.TreeID = request.Header.TreeID;
  263. }
  264. }
  265. private static void SetRequestFileID(SMB2Command command, FileID fileID)
  266. {
  267. if (command is ChangeNotifyRequest)
  268. {
  269. ((ChangeNotifyRequest)command).FileId = fileID;
  270. }
  271. else if (command is CloseRequest)
  272. {
  273. ((CloseRequest)command).FileId = fileID;
  274. }
  275. else if (command is FlushRequest)
  276. {
  277. ((FlushRequest)command).FileId = fileID;
  278. }
  279. else if (command is IOCtlRequest)
  280. {
  281. ((IOCtlRequest)command).FileId = fileID;
  282. }
  283. else if (command is LockRequest)
  284. {
  285. ((LockRequest)command).FileId = fileID;
  286. }
  287. else if (command is QueryDirectoryRequest)
  288. {
  289. ((QueryDirectoryRequest)command).FileId = fileID;
  290. }
  291. else if (command is QueryInfoRequest)
  292. {
  293. ((QueryInfoRequest)command).FileId = fileID;
  294. }
  295. else if (command is ReadRequest)
  296. {
  297. ((ReadRequest)command).FileId = fileID;
  298. }
  299. else if (command is SetInfoRequest)
  300. {
  301. ((SetInfoRequest)command).FileId = fileID;
  302. }
  303. else if (command is WriteRequest)
  304. {
  305. ((WriteRequest)command).FileId = fileID;
  306. }
  307. }
  308. private static FileID? GetResponseFileID(SMB2Command command)
  309. {
  310. if (command is CreateResponse)
  311. {
  312. return ((CreateResponse)command).FileId;
  313. }
  314. else if (command is IOCtlResponse)
  315. {
  316. return ((IOCtlResponse)command).FileId;
  317. }
  318. return null;
  319. }
  320. }
  321. }