LockingHelper.cs 4.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576
  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 LockingHelper
  14. {
  15. internal static List<SMB1Command> GetLockingAndXResponse(SMB1Header header, LockingAndXRequest request, ISMBShare share, SMB1ConnectionState state)
  16. {
  17. SMB1Session session = state.GetSession(header.UID);
  18. OpenFileObject openFile = session.GetOpenFileObject(request.FID);
  19. if (openFile == null)
  20. {
  21. state.LogToServer(Severity.Verbose, "Locking failed. Invalid FID. (UID: {0}, TID: {1}, FID: {2})", header.UID, header.TID, request.FID);
  22. header.Status = NTStatus.STATUS_INVALID_HANDLE;
  23. return new ErrorResponse(request.CommandName);
  24. }
  25. if ((request.TypeOfLock & LockType.CHANGE_LOCKTYPE) > 0)
  26. {
  27. // [MS-CIFS] Windows NT Server does not support the CHANGE_LOCKTYPE flag of TypeOfLock.
  28. state.LogToServer(Severity.Verbose, "Locking failed. CHANGE_LOCKTYPE is not supported.");
  29. header.Status = NTStatus.STATUS_NOT_IMPLEMENTED;
  30. return new ErrorResponse(request.CommandName);
  31. }
  32. if (request.Unlocks.Count == 0 && request.Locks.Count == 0)
  33. {
  34. // [MS-CIFS] If NumberOfRequestedUnlocks and NumberOfRequestedLocks are both zero [..] the server MUST NOT send an SMB_COM_LOCKING_ANDX Response.
  35. return new List<SMB1Command>();
  36. }
  37. // [MS-CIFS] If the CANCEL_LOCK bit is set, Windows NT servers cancel only the first lock request range listed in the lock array.
  38. for(int lockIndex = 0; lockIndex < request.Unlocks.Count; lockIndex++)
  39. {
  40. LockingRange lockingRange = request.Unlocks[lockIndex];
  41. header.Status = share.FileStore.UnlockFile(openFile.Handle, (long)lockingRange.ByteOffset, (long)lockingRange.LengthInBytes);
  42. if (header.Status != NTStatus.STATUS_SUCCESS)
  43. {
  44. state.LogToServer(Severity.Verbose, "Locking: Unlocking '{0}{1}' failed. Offset: {2}, Length: {3}. NTStatus: {4}.", share.Name, openFile.Path, lockingRange.ByteOffset, lockingRange.LengthInBytes, header.Status);
  45. return new ErrorResponse(request.CommandName);
  46. }
  47. state.LogToServer(Severity.Verbose, "Locking: Unlocking '{0}{1}' succeeded. Offset: {2}, Length: {3}.", share.Name, openFile.Path, lockingRange.ByteOffset, lockingRange.LengthInBytes);
  48. }
  49. for (int lockIndex = 0; lockIndex < request.Locks.Count; lockIndex++)
  50. {
  51. LockingRange lockingRange = request.Locks[lockIndex];
  52. bool exclusiveLock = (request.TypeOfLock & LockType.SHARED_LOCK) == 0;
  53. header.Status = share.FileStore.LockFile(openFile.Handle, (long)lockingRange.ByteOffset, (long)lockingRange.LengthInBytes, exclusiveLock);
  54. if (header.Status != NTStatus.STATUS_SUCCESS)
  55. {
  56. state.LogToServer(Severity.Verbose, "Locking: Locking '{0}{1}' failed. Offset: {2}, Length: {3}. NTStatus: {4}.", share.Name, openFile.Path, lockingRange.ByteOffset, lockingRange.LengthInBytes, header.Status);
  57. // [MS-CIFS] This client request is atomic. If the area to be locked is already locked or the
  58. // lock request otherwise fails, no other ranges specified in the client request are locked.
  59. for (int index = 0; index < lockIndex; index++)
  60. {
  61. share.FileStore.UnlockFile(openFile.Handle, (long)request.Locks[index].ByteOffset, (long)request.Locks[index].LengthInBytes);
  62. }
  63. return new ErrorResponse(request.CommandName);
  64. }
  65. state.LogToServer(Severity.Verbose, "Locking: Locking '{0}{1}' succeeded. Offset: {2}, Length: {3}.", share.Name, openFile.Path, lockingRange.ByteOffset, lockingRange.LengthInBytes);
  66. }
  67. return new LockingAndXResponse();
  68. }
  69. }
  70. }