ChangeNotifyHelper.cs 3.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970
  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.SMB2;
  10. using Utilities;
  11. namespace SMBLibrary.Server.SMB2
  12. {
  13. internal class ChangeNotifyHelper
  14. {
  15. internal static SMB2Command GetChangeNotifyInterimResponse(ChangeNotifyRequest request, ISMBShare share, SMB2ConnectionState state)
  16. {
  17. SMB2Session session = state.GetSession(request.Header.SessionID);
  18. OpenFileObject openFile = session.GetOpenFileObject(request.FileId);
  19. bool watchTree = (request.Flags & ChangeNotifyFlags.WatchTree) > 0;
  20. SMB2AsyncContext context = state.CreateAsyncContext(request.FileId, state, request.Header.SessionID, request.Header.TreeID);
  21. // We have to make sure that we don't send an interim response after the final response.
  22. lock (context)
  23. {
  24. NTStatus status = share.FileStore.NotifyChange(out context.IORequest, openFile.Handle, request.CompletionFilter, watchTree, (int)request.OutputBufferLength, OnNotifyChangeCompleted, context);
  25. if (status == NTStatus.STATUS_PENDING)
  26. {
  27. state.LogToServer(Severity.Verbose, "NotifyChange: Monitoring of '{0}{1}' started. AsyncID: {2}.", share.Name, openFile.Path, context.AsyncID);
  28. }
  29. // [MS-SMB2] If the underlying object store does not support change notifications, the server MUST fail this request with STATUS_NOT_SUPPORTED
  30. ErrorResponse response = new ErrorResponse(request.CommandName, status);
  31. // Windows 7 / 8 / 10 will infinitely retry sending ChangeNotify requests if the response does not have SMB2_FLAGS_ASYNC_COMMAND set.
  32. // Note: NoRemoteChangeNotify can be set in the registry to prevent the client from sending ChangeNotify requests altogether.
  33. response.Header.IsAsync = true;
  34. response.Header.AsyncID = context.AsyncID;
  35. return response;
  36. }
  37. }
  38. private static void OnNotifyChangeCompleted(NTStatus status, byte[] buffer, object context)
  39. {
  40. SMB2AsyncContext asyncContext = (SMB2AsyncContext)context;
  41. // Wait until the interim response has been sent
  42. lock (asyncContext)
  43. {
  44. SMB2ConnectionState connection = asyncContext.Connection;
  45. connection.RemoveAsyncContext(asyncContext);
  46. SMB2Session session = connection.GetSession(asyncContext.SessionID);
  47. if (session != null)
  48. {
  49. ISMBShare share = session.GetConnectedTree(asyncContext.TreeID);
  50. OpenFileObject openFile = session.GetOpenFileObject(asyncContext.FileID);
  51. if (share != null && openFile != null)
  52. {
  53. connection.LogToServer(Severity.Verbose, "NotifyChange: Monitoring of '{0}{1}' completed. NTStatus: {2}. AsyncID: {3}", share.Name, openFile.Path, status, asyncContext.AsyncID);
  54. }
  55. }
  56. ChangeNotifyResponse response = new ChangeNotifyResponse();
  57. response.Header.Status = status;
  58. response.Header.IsAsync = true;
  59. response.Header.AsyncID = asyncContext.AsyncID;
  60. response.Header.SessionID = asyncContext.SessionID;
  61. response.OutputBuffer = buffer;
  62. SMBServer.EnqueueResponse(connection, response);
  63. }
  64. }
  65. }
  66. }