123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268 |
- /* 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 SMBLibrary.SMB1;
- using Utilities;
- namespace SMBLibrary.Client
- {
- public class SMB1FileStore : INTFileStore
- {
- private SMB1Client m_client;
- private ushort m_treeID;
- public SMB1FileStore(SMB1Client client, ushort treeID)
- {
- m_client = client;
- m_treeID = treeID;
- }
- public NTStatus CreateFile(out object handle, out FileStatus fileStatus, string path, AccessMask desiredAccess, FileAttributes fileAttributes, ShareAccess shareAccess, CreateDisposition createDisposition, CreateOptions createOptions, SecurityContext securityContext)
- {
- handle = null;
- fileStatus = FileStatus.FILE_DOES_NOT_EXIST;
- NTCreateAndXRequest request = new NTCreateAndXRequest();
- request.FileName = path;
- request.DesiredAccess = desiredAccess;
- request.ExtFileAttributes = ToExtendedFileAttributes(fileAttributes);
- request.ShareAccess = shareAccess;
- request.CreateDisposition = createDisposition;
- request.CreateOptions = createOptions;
- request.ImpersonationLevel = ImpersonationLevel.SEC_IMPERSONATE;
- TrySendMessage(request);
- SMB1Message reply = m_client.WaitForMessage(CommandName.SMB_COM_NT_CREATE_ANDX);
- if (reply != null)
- {
- if (reply.Commands[0] is NTCreateAndXResponse)
- {
- NTCreateAndXResponse response = reply.Commands[0] as NTCreateAndXResponse;
- handle = response.FID;
- fileStatus = ToFileStatus(response.CreateDisposition);
- return reply.Header.Status;
- }
- else if (reply.Commands[0] is ErrorResponse)
- {
- return reply.Header.Status;
- }
- }
- return NTStatus.STATUS_INVALID_SMB;
- }
- public NTStatus CloseFile(object handle)
- {
- CloseRequest request = new CloseRequest();
- request.FID = (ushort)handle;
- TrySendMessage(request);
- SMB1Message reply = m_client.WaitForMessage(CommandName.SMB_COM_CLOSE);
- if (reply != null)
- {
- return reply.Header.Status;
- }
- return NTStatus.STATUS_INVALID_SMB;
- }
- public NTStatus ReadFile(out byte[] data, object handle, long offset, int maxCount)
- {
- data = null;
- ReadAndXRequest request = new ReadAndXRequest();
- request.FID = (ushort)handle;
- request.Offset = (ulong)offset;
- request.MaxCountLarge = (uint)maxCount;
- TrySendMessage(request);
- SMB1Message reply = m_client.WaitForMessage(CommandName.SMB_COM_READ_ANDX);
- if (reply != null)
- {
- if (reply.Header.Status == NTStatus.STATUS_SUCCESS && reply.Commands[0] is ReadAndXResponse)
- {
- data = ((ReadAndXResponse)reply.Commands[0]).Data;
- }
- return reply.Header.Status;
- }
- return NTStatus.STATUS_INVALID_SMB;
- }
- public NTStatus WriteFile(out int numberOfBytesWritten, object handle, long offset, byte[] data)
- {
- numberOfBytesWritten = 0;
- WriteAndXRequest request = new WriteAndXRequest();
- request.FID = (ushort)handle;
- request.Offset = (ulong)offset;
- request.Data = data;
- TrySendMessage(request);
- SMB1Message reply = m_client.WaitForMessage(CommandName.SMB_COM_WRITE_ANDX);
- if (reply != null)
- {
- if (reply.Header.Status == NTStatus.STATUS_SUCCESS && reply.Commands[0] is WriteAndXResponse)
- {
- numberOfBytesWritten = (int)((WriteAndXResponse)reply.Commands[0]).Count;
- }
- return reply.Header.Status;
- }
- return NTStatus.STATUS_INVALID_SMB;
- }
- public NTStatus FlushFileBuffers(object handle)
- {
- throw new NotImplementedException();
- }
- public NTStatus LockFile(object handle, long byteOffset, long length, bool exclusiveLock)
- {
- throw new NotImplementedException();
- }
- public NTStatus UnlockFile(object handle, long byteOffset, long length)
- {
- throw new NotImplementedException();
- }
- public NTStatus QueryDirectory(out List<QueryDirectoryFileInformation> result, object handle, string fileName, FileInformationClass informationClass)
- {
- throw new NotImplementedException();
- }
- public NTStatus GetFileInformation(out FileInformation result, object handle, FileInformationClass informationClass)
- {
- throw new NotImplementedException();
- }
- public NTStatus SetFileInformation(object handle, FileInformation information)
- {
- throw new NotImplementedException();
- }
- public NTStatus GetFileSystemInformation(out FileSystemInformation result, FileSystemInformationClass informationClass)
- {
- throw new NotImplementedException();
- }
- public NTStatus NotifyChange(out object ioRequest, object handle, NotifyChangeFilter completionFilter, bool watchTree, int outputBufferSize, OnNotifyChangeCompleted onNotifyChangeCompleted, object context)
- {
- throw new NotImplementedException();
- }
- public NTStatus Cancel(object ioRequest)
- {
- throw new NotImplementedException();
- }
- public NTStatus DeviceIOControl(object handle, uint ctlCode, byte[] input, out byte[] output, int maxOutputLength)
- {
- if ((IoControlCode)ctlCode == IoControlCode.FSCTL_PIPE_TRANSCEIVE)
- {
- return FsCtlPipeTranscieve(handle, input, out output, maxOutputLength);
- }
- output = null;
- NTTransactIOCTLRequest subcommand = new NTTransactIOCTLRequest();
- subcommand.FID = (ushort)handle;
- subcommand.FunctionCode = ctlCode;
- subcommand.IsFsctl = true;
- subcommand.Data = input;
- NTTransactRequest request = new NTTransactRequest();
- request.Function = subcommand.SubcommandName;
- request.Setup = subcommand.GetSetup();
- request.TransParameters = subcommand.GetParameters(true);
- request.TransData = subcommand.GetData();
- request.TotalDataCount = (uint)request.TransData.Length;
- request.TotalParameterCount = (uint)request.TransParameters.Length;
- request.MaxParameterCount = 0;
- request.MaxDataCount = (uint)maxOutputLength;
- TrySendMessage(request);
- SMB1Message reply = m_client.WaitForMessage(CommandName.SMB_COM_NT_TRANSACT);
- if (reply != null)
- {
- if (reply.Header.Status == NTStatus.STATUS_SUCCESS && reply.Commands[0] is NTTransactResponse)
- {
- NTTransactResponse response = (NTTransactResponse)reply.Commands[0];
- NTTransactIOCTLResponse subcommandResponse = new NTTransactIOCTLResponse(response.Setup, response.TransData);
- output = subcommandResponse.Data;
- }
- return reply.Header.Status;
- }
- return NTStatus.STATUS_INVALID_SMB;
- }
- public NTStatus FsCtlPipeTranscieve(object handle, byte[] input, out byte[] output, int maxOutputLength)
- {
- output = null;
- TransactionTransactNamedPipeRequest subcommand = new TransactionTransactNamedPipeRequest();
- subcommand.FID = (ushort)handle;
- subcommand.WriteData = input;
- TransactionRequest request = new TransactionRequest();
- request.Setup = subcommand.GetSetup();
- request.TransParameters = subcommand.GetParameters(true);
- request.TransData = subcommand.GetData();
- request.TotalDataCount = (ushort)request.TransData.Length;
- request.TotalParameterCount = (ushort)request.TransParameters.Length;
- request.MaxParameterCount = 0;
- request.MaxDataCount = (ushort)maxOutputLength;
- request.Name = @"\PIPE\";
- TrySendMessage(request);
- SMB1Message reply = m_client.WaitForMessage(CommandName.SMB_COM_TRANSACTION);
- if (reply != null)
- {
- if (reply.Header.Status == NTStatus.STATUS_SUCCESS && reply.Commands[0] is TransactionResponse)
- {
- TransactionResponse response = (TransactionResponse)reply.Commands[0];
- TransactionTransactNamedPipeResponse subcommandResponse = new TransactionTransactNamedPipeResponse(response.TransData);
- output = subcommandResponse.ReadData;
- }
- return reply.Header.Status;
- }
- return NTStatus.STATUS_INVALID_SMB;
- }
- private static ExtendedFileAttributes ToExtendedFileAttributes(FileAttributes fileAttributes)
- {
- // We only return flags that can be used with NtCreateFile
- ExtendedFileAttributes extendedFileAttributes = ExtendedFileAttributes.ReadOnly |
- ExtendedFileAttributes.Hidden |
- ExtendedFileAttributes.System |
- ExtendedFileAttributes.Archive |
- ExtendedFileAttributes.Normal |
- ExtendedFileAttributes.Temporary |
- ExtendedFileAttributes.Offline |
- ExtendedFileAttributes.Encrypted;
- return (extendedFileAttributes & (ExtendedFileAttributes)fileAttributes);
- }
- private void TrySendMessage(SMB1Command request)
- {
- m_client.TrySendMessage(request, m_treeID);
- }
- private static FileStatus ToFileStatus(CreateDisposition createDisposition)
- {
- switch (createDisposition)
- {
- case CreateDisposition.FILE_SUPERSEDE:
- return FileStatus.FILE_SUPERSEDED;
- case CreateDisposition.FILE_OPEN:
- return FileStatus.FILE_OPENED;
- case CreateDisposition.FILE_CREATE:
- return FileStatus.FILE_CREATED;
- case CreateDisposition.FILE_OPEN_IF:
- return FileStatus.FILE_OVERWRITTEN;
- case CreateDisposition.FILE_OVERWRITE:
- return FileStatus.FILE_EXISTS;
- case CreateDisposition.FILE_OVERWRITE_IF:
- return FileStatus.FILE_DOES_NOT_EXIST;
- default:
- return FileStatus.FILE_OPENED;
- }
- }
- }
- }
|