NotifyChangeHelper.cs 4.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  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.SMB1;
  10. using Utilities;
  11. namespace SMBLibrary.Server.SMB1
  12. {
  13. internal class NotifyChangeHelper
  14. {
  15. internal static void ProcessNTTransactNotifyChangeRequest(SMB1Header header, uint maxParameterCount, NTTransactNotifyChangeRequest subcommand, ISMBShare share, SMB1ConnectionState state)
  16. {
  17. SMB1Session session = state.GetSession(header.UID);
  18. OpenFileObject openFile = session.GetOpenFileObject(subcommand.FID);
  19. SMB1AsyncContext context = state.CreateAsyncContext(header.UID, header.TID, header.PID, header.MID, subcommand.FID, state);
  20. // We wish to make sure that the 'Monitoring started' will appear before the 'Monitoring completed' in the log
  21. lock (context)
  22. {
  23. header.Status = share.FileStore.NotifyChange(out context.IORequest, openFile.Handle, subcommand.CompletionFilter, subcommand.WatchTree, (int)maxParameterCount, OnNotifyChangeCompleted, context);
  24. if (header.Status == NTStatus.STATUS_PENDING)
  25. {
  26. state.LogToServer(Severity.Verbose, "NotifyChange: Monitoring of '{0}{1}' started. PID: {2}. MID: {3}.", share.Name, openFile.Path, context.PID, context.MID);
  27. }
  28. else if (header.Status == NTStatus.STATUS_NOT_SUPPORTED)
  29. {
  30. // [MS-CIFS] If the server does not support the NT_TRANSACT_NOTIFY_CHANGE subcommand, it can return an
  31. // error response with STATUS_NOT_IMPLEMENTED [..] in response to an NT_TRANSACT_NOTIFY_CHANGE Request.
  32. header.Status = NTStatus.STATUS_NOT_IMPLEMENTED;
  33. }
  34. }
  35. }
  36. private static void OnNotifyChangeCompleted(NTStatus status, byte[] buffer, object context)
  37. {
  38. NTTransactNotifyChangeResponse notifyChangeResponse = new NTTransactNotifyChangeResponse();
  39. SMB1AsyncContext asyncContext = (SMB1AsyncContext)context;
  40. // Wait until the 'Monitoring started' will be written to the log
  41. lock (asyncContext)
  42. {
  43. SMB1ConnectionState connection = asyncContext.Connection;
  44. connection.RemoveAsyncContext(asyncContext);
  45. SMB1Session session = connection.GetSession(asyncContext.UID);
  46. if (session != null)
  47. {
  48. OpenFileObject openFile = session.GetOpenFileObject(asyncContext.FileID);
  49. if (openFile != null)
  50. {
  51. connection.LogToServer(Severity.Verbose, "NotifyChange: Monitoring of '{0}{1}' completed. NTStatus: {2}. PID: {3}. MID: {4}.", openFile.ShareName, openFile.Path, status, asyncContext.PID, asyncContext.MID);
  52. }
  53. }
  54. SMB1Header header = new SMB1Header();
  55. header.Command = CommandName.SMB_COM_NT_TRANSACT;
  56. header.Status = status;
  57. header.Flags = HeaderFlags.CaseInsensitive | HeaderFlags.CanonicalizedPaths | HeaderFlags.Reply;
  58. // [MS-CIFS] SMB_FLAGS2_UNICODE SHOULD be set to 1 when the negotiated dialect is NT LANMAN.
  59. // [MS-CIFS] The Windows NT Server implementation of NT_TRANSACT_NOTIFY_CHANGE always returns the names of changed files in Unicode format.
  60. header.Flags2 = HeaderFlags2.Unicode | HeaderFlags2.NTStatusCode;
  61. header.UID = asyncContext.UID;
  62. header.TID = asyncContext.TID;
  63. header.PID = asyncContext.PID;
  64. header.MID = asyncContext.MID;
  65. notifyChangeResponse.FileNotifyInformationBytes = buffer;
  66. byte[] responseSetup = notifyChangeResponse.GetSetup();
  67. byte[] responseParameters = notifyChangeResponse.GetParameters(false);
  68. byte[] responseData = notifyChangeResponse.GetData();
  69. List<SMB1Command> responseList = NTTransactHelper.GetNTTransactResponse(responseSetup, responseParameters, responseData, asyncContext.Connection.MaxBufferSize);
  70. foreach (SMB1Command response in responseList)
  71. {
  72. SMB1Message reply = new SMB1Message();
  73. reply.Header = header;
  74. reply.Commands.Add(response);
  75. SMBServer.EnqueueMessage(asyncContext.Connection, reply);
  76. }
  77. }
  78. }
  79. }
  80. }