123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464 |
- /* Copyright (C) 2014-2017 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
- *
- * You can redistribute this program and/or modify it under the terms of
- * the GNU Lesser Public License as published by the Free Software Foundation,
- * either version 3 of the License, or (at your option) any later version.
- */
- using System;
- using System.Collections.Generic;
- using System.IO;
- using SMBLibrary.Services;
- using Utilities;
- namespace SMBLibrary.Server
- {
- /// <summary>
- /// Helper class to access the FileSystemShare / NamedPipeShare in an NT-like manner dictated by the SMB protocol
- /// </summary>
- public partial class NTFileSystemHelper
- {
- public const int BytesPerSector = 512;
- public const int ClusterSize = 4096;
- public static NTStatus CreateFile(out FileSystemEntry entry, FileSystemShare share, string userName, string path, CreateDisposition createDisposition, CreateOptions createOptions, AccessMask desiredAccess, ConnectionState state)
- {
- bool hasWriteAccess = share.HasWriteAccess(userName);
- IFileSystem fileSystem = share.FileSystem;
- bool forceDirectory = (createOptions & CreateOptions.FILE_DIRECTORY_FILE) > 0;
- bool forceFile = (createOptions & CreateOptions.FILE_NON_DIRECTORY_FILE) > 0;
- if (forceDirectory & (createDisposition != CreateDisposition.FILE_CREATE &&
- createDisposition != CreateDisposition.FILE_OPEN &&
- createDisposition != CreateDisposition.FILE_OPEN_IF))
- {
- entry = null;
- return NTStatus.STATUS_INVALID_PARAMETER;
- }
- // Windows will try to access named streams (alternate data streams) regardless of the FILE_NAMED_STREAMS flag, we need to prevent this behaviour.
- if (path.Contains(":"))
- {
- // Windows Server 2003 will return STATUS_OBJECT_NAME_NOT_FOUND
- entry = null;
- return NTStatus.STATUS_NO_SUCH_FILE;
- }
- entry = fileSystem.GetEntry(path);
- if (createDisposition == CreateDisposition.FILE_OPEN)
- {
- if (entry == null)
- {
- return NTStatus.STATUS_OBJECT_PATH_NOT_FOUND;
- }
- if (entry.IsDirectory && forceFile)
- {
- return NTStatus.STATUS_FILE_IS_A_DIRECTORY;
- }
- if (!entry.IsDirectory && forceDirectory)
- {
- // Not sure if that's the correct response
- return NTStatus.STATUS_OBJECT_NAME_COLLISION;
- }
- }
- else if (createDisposition == CreateDisposition.FILE_CREATE)
- {
- if (entry != null)
- {
- // File already exists, fail the request
- state.LogToServer(Severity.Debug, "CreateFile: File '{0}' already exist", path);
- return NTStatus.STATUS_OBJECT_NAME_COLLISION;
- }
- if (!hasWriteAccess)
- {
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- try
- {
- if (forceDirectory)
- {
- state.LogToServer(Severity.Information, "CreateFile: Creating directory '{0}'", path);
- entry = fileSystem.CreateDirectory(path);
- }
- else
- {
- state.LogToServer(Severity.Information, "CreateFile: Creating file '{0}'", path);
- entry = fileSystem.CreateFile(path);
- }
- }
- catch (IOException ex)
- {
- ushort errorCode = IOExceptionHelper.GetWin32ErrorCode(ex);
- if (errorCode == (ushort)Win32Error.ERROR_SHARING_VIOLATION)
- {
- state.LogToServer(Severity.Debug, "CreateFile: Error creating '{0}'. Sharing violation.", path);
- return NTStatus.STATUS_SHARING_VIOLATION;
- }
- else
- {
- state.LogToServer(Severity.Debug, "CreateFile: Error creating '{0}'. Data Error.", path);
- return NTStatus.STATUS_DATA_ERROR;
- }
- }
- catch (UnauthorizedAccessException)
- {
- state.LogToServer(Severity.Debug, "CreateFile: Error creating '{0}'. Access Denied.", path);
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- }
- else if (createDisposition == CreateDisposition.FILE_OPEN_IF ||
- createDisposition == CreateDisposition.FILE_OVERWRITE ||
- createDisposition == CreateDisposition.FILE_OVERWRITE_IF ||
- createDisposition == CreateDisposition.FILE_SUPERSEDE)
- {
- entry = fileSystem.GetEntry(path);
- if (entry == null)
- {
- if (createDisposition == CreateDisposition.FILE_OVERWRITE)
- {
- return NTStatus.STATUS_OBJECT_PATH_NOT_FOUND;
- }
- if (!hasWriteAccess)
- {
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- try
- {
- if (forceDirectory)
- {
- state.LogToServer(Severity.Information, "CreateFile: Creating directory '{0}'", path);
- entry = fileSystem.CreateDirectory(path);
- }
- else
- {
- state.LogToServer(Severity.Information, "CreateFile: Creating file '{0}'", path);
- entry = fileSystem.CreateFile(path);
- }
- }
- catch (IOException ex)
- {
- ushort errorCode = IOExceptionHelper.GetWin32ErrorCode(ex);
- if (errorCode == (ushort)Win32Error.ERROR_SHARING_VIOLATION)
- {
- return NTStatus.STATUS_SHARING_VIOLATION;
- }
- else
- {
- return NTStatus.STATUS_DATA_ERROR;
- }
- }
- catch (UnauthorizedAccessException)
- {
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- }
- else
- {
- if (createDisposition == CreateDisposition.FILE_OVERWRITE ||
- createDisposition == CreateDisposition.FILE_OVERWRITE_IF ||
- createDisposition == CreateDisposition.FILE_SUPERSEDE)
- {
- if (!hasWriteAccess)
- {
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- // Truncate the file
- try
- {
- Stream temp = fileSystem.OpenFile(path, FileMode.Truncate, FileAccess.ReadWrite, FileShare.ReadWrite);
- temp.Close();
- }
- catch (IOException ex)
- {
- ushort errorCode = IOExceptionHelper.GetWin32ErrorCode(ex);
- if (errorCode == (ushort)Win32Error.ERROR_SHARING_VIOLATION)
- {
- return NTStatus.STATUS_SHARING_VIOLATION;
- }
- else
- {
- return NTStatus.STATUS_DATA_ERROR;
- }
- }
- catch (UnauthorizedAccessException)
- {
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- }
- }
- }
- else
- {
- return NTStatus.STATUS_INVALID_PARAMETER;
- }
- FileAccess fileAccess = ToFileAccess(desiredAccess.File);
- if (!hasWriteAccess && (fileAccess == FileAccess.Write || fileAccess == FileAccess.ReadWrite))
- {
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- return NTStatus.STATUS_SUCCESS;
- }
- public static FileAccess ToFileAccess(FileAccessMask desiredAccess)
- {
- if ((desiredAccess & FileAccessMask.GENERIC_ALL) > 0 ||
- ((desiredAccess & FileAccessMask.FILE_READ_DATA) > 0 && (desiredAccess & FileAccessMask.FILE_WRITE_DATA) > 0) ||
- ((desiredAccess & FileAccessMask.FILE_READ_DATA) > 0 && (desiredAccess & FileAccessMask.FILE_APPEND_DATA) > 0))
- {
- return FileAccess.ReadWrite;
- }
- else if ((desiredAccess & FileAccessMask.GENERIC_WRITE) > 0 ||
- (desiredAccess & FileAccessMask.FILE_WRITE_DATA) > 0 ||
- (desiredAccess & FileAccessMask.FILE_APPEND_DATA) > 0)
- {
- return FileAccess.Write;
- }
- else if ((desiredAccess & FileAccessMask.FILE_READ_DATA) > 0)
- {
- return FileAccess.Read;
- }
- else
- {
- return (FileAccess)0;
- }
- }
- public static FileShare ToFileShare(ShareAccess shareAccess)
- {
- if ((shareAccess & ShareAccess.FILE_SHARE_READ) > 0 && (shareAccess & ShareAccess.FILE_SHARE_WRITE) > 0)
- {
- return FileShare.ReadWrite;
- }
- else if ((shareAccess & ShareAccess.FILE_SHARE_WRITE) > 0)
- {
- return FileShare.Write;
- }
- else if ((shareAccess & ShareAccess.FILE_SHARE_READ) > 0)
- {
- return FileShare.Read;
- }
- else if ((shareAccess & ShareAccess.FILE_SHARE_DELETE) > 0)
- {
- return FileShare.Delete;
- }
- else
- {
- return FileShare.None;
- }
- }
- public static NTStatus OpenFile(out Stream stream, IFileSystem fileSystem, string path, FileAccess fileAccess, ShareAccess shareAccess, bool buffered, ConnectionState state)
- {
- stream = null;
- FileShare fileShare = NTFileSystemHelper.ToFileShare(shareAccess);
- state.LogToServer(Severity.Verbose, "OpenFile: Opening '{0}', Access={1}, Share={2}, Buffered={3}", path, fileAccess, fileShare, buffered);
- try
- {
- stream = fileSystem.OpenFile(path, FileMode.Open, fileAccess, fileShare);
- }
- catch (DirectoryNotFoundException)
- {
- state.LogToServer(Severity.Debug, "OpenFile: Cannot open '{0}'. Directory not found.", path);
- return NTStatus.STATUS_NO_SUCH_FILE;
- }
- catch (FileNotFoundException)
- {
- state.LogToServer(Severity.Debug, "OpenFile: Cannot open '{0}'. File not found.", path);
- return NTStatus.STATUS_NO_SUCH_FILE;
- }
- catch (IOException ex)
- {
- ushort errorCode = IOExceptionHelper.GetWin32ErrorCode(ex);
- if (errorCode == (ushort)Win32Error.ERROR_SHARING_VIOLATION)
- {
- state.LogToServer(Severity.Debug, "OpenFile: Cannot open '{0}'. Sharing violation.", path);
- return NTStatus.STATUS_SHARING_VIOLATION;
- }
- else
- {
- state.LogToServer(Severity.Debug, "OpenFile: Cannot open '{0}'. Data Error.", path);
- return NTStatus.STATUS_DATA_ERROR;
- }
- }
- catch (UnauthorizedAccessException)
- {
- state.LogToServer(Severity.Debug, "OpenFile: Cannot open '{0}'. Access Denied.", path);
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- if (buffered)
- {
- stream = new PrefetchedStream(stream);
- }
- return NTStatus.STATUS_SUCCESS;
- }
- public static NTStatus ReadFile(out byte[] data, OpenFileObject openFile, long offset, int maxCount, ConnectionState state)
- {
- data = null;
- string openFilePath = openFile.Path;
- Stream stream = openFile.Stream;
- if (stream is RPCPipeStream)
- {
- data = new byte[maxCount];
- int bytesRead = stream.Read(data, 0, maxCount);
- if (bytesRead < maxCount)
- {
- // EOF, we must trim the response data array
- data = ByteReader.ReadBytes(data, 0, bytesRead);
- }
- return NTStatus.STATUS_SUCCESS;
- }
- else // File
- {
- if (stream == null || !stream.CanRead)
- {
- state.LogToServer(Severity.Debug, "ReadFile: Cannot read '{0}', Invalid Operation.", openFilePath);
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- int bytesRead;
- try
- {
- stream.Seek(offset, SeekOrigin.Begin);
- data = new byte[maxCount];
- bytesRead = stream.Read(data, 0, maxCount);
- }
- catch (IOException ex)
- {
- ushort errorCode = IOExceptionHelper.GetWin32ErrorCode(ex);
- if (errorCode == (ushort)Win32Error.ERROR_SHARING_VIOLATION)
- {
- // Returning STATUS_SHARING_VIOLATION is undocumented but apparently valid
- state.LogToServer(Severity.Debug, "ReadFile: Cannot read '{0}'. Sharing Violation.", openFilePath);
- return NTStatus.STATUS_SHARING_VIOLATION;
- }
- else
- {
- state.LogToServer(Severity.Debug, "ReadFile: Cannot read '{0}'. Data Error.", openFilePath);
- return NTStatus.STATUS_DATA_ERROR;
- }
- }
- catch (ArgumentOutOfRangeException)
- {
- state.LogToServer(Severity.Debug, "ReadFile: Cannot read '{0}'. Offset Out Of Range.", openFilePath);
- return NTStatus.STATUS_DATA_ERROR;
- }
- catch (UnauthorizedAccessException)
- {
- state.LogToServer(Severity.Debug, "ReadFile: Cannot read '{0}', Access Denied.", openFilePath);
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- if (bytesRead < maxCount)
- {
- // EOF, we must trim the response data array
- data = ByteReader.ReadBytes(data, 0, bytesRead);
- }
- return NTStatus.STATUS_SUCCESS;
- }
- }
- public static NTStatus WriteFile(out int numberOfBytesWritten, OpenFileObject openFile, long offset, byte[] data, ConnectionState state)
- {
- numberOfBytesWritten = 0;
- string openFilePath = openFile.Path;
- Stream stream = openFile.Stream;
- if (stream is RPCPipeStream)
- {
- stream.Write(data, 0, data.Length);
- numberOfBytesWritten = data.Length;
- return NTStatus.STATUS_SUCCESS;
- }
- else // File
- {
- if (stream == null || !stream.CanWrite)
- {
- state.LogToServer(Severity.Debug, "WriteFile: Cannot write '{0}'. Invalid Operation.", openFilePath);
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- try
- {
- stream.Seek(offset, SeekOrigin.Begin);
- stream.Write(data, 0, data.Length);
- numberOfBytesWritten = data.Length;
- return NTStatus.STATUS_SUCCESS;
- }
- catch (IOException ex)
- {
- ushort errorCode = IOExceptionHelper.GetWin32ErrorCode(ex);
- if (errorCode == (ushort)Win32Error.ERROR_DISK_FULL)
- {
- state.LogToServer(Severity.Debug, "WriteFile: Cannot write '{0}'. Disk Full.", openFilePath);
- return NTStatus.STATUS_DISK_FULL;
- }
- else if (errorCode == (ushort)Win32Error.ERROR_SHARING_VIOLATION)
- {
- state.LogToServer(Severity.Debug, "WriteFile: Cannot write '{0}'. Sharing Violation.", openFilePath);
- // Returning STATUS_SHARING_VIOLATION is undocumented but apparently valid
- return NTStatus.STATUS_SHARING_VIOLATION;
- }
- else
- {
- state.LogToServer(Severity.Debug, "WriteFile: Cannot write '{0}'. Data Error.", openFilePath);
- return NTStatus.STATUS_DATA_ERROR;
- }
- }
- catch (ArgumentOutOfRangeException)
- {
- state.LogToServer(Severity.Debug, "WriteFile: Cannot write '{0}'. Offset Out Of Range.", openFilePath);
- return NTStatus.STATUS_DATA_ERROR;
- }
- catch (UnauthorizedAccessException)
- {
- state.LogToServer(Severity.Debug, "WriteFile: Cannot write '{0}'. Access Denied.", openFilePath);
- // The user may have tried to write to a readonly file
- return NTStatus.STATUS_ACCESS_DENIED;
- }
- }
- }
- /// <summary>
- /// Will return a virtual allocation size, assuming 4096 bytes per cluster
- /// </summary>
- public static ulong GetAllocationSize(ulong size)
- {
- return (ulong)Math.Ceiling((double)size / ClusterSize) * ClusterSize;
- }
- public static string GetShortName(string fileName)
- {
- string fileNameWithoutExt = System.IO.Path.GetFileNameWithoutExtension(fileName);
- string extension = System.IO.Path.GetExtension(fileName);
- if (fileNameWithoutExt.Length > 8 || extension.Length > 4)
- {
- if (fileNameWithoutExt.Length > 8)
- {
- fileNameWithoutExt = fileNameWithoutExt.Substring(0, 8);
- }
- if (extension.Length > 4)
- {
- extension = extension.Substring(0, 4);
- }
- return fileNameWithoutExt + extension;
- }
- else
- {
- return fileName;
- }
- }
- }
- }
|