/* Copyright (C) 2014 Tal Aloni . 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.Text; using Utilities; namespace SMBLibrary.SMB1 { /// /// SMB_COM_NT_TRANSACT Response /// public class NTTransactResponse : SMB1Command { public const int FixedSMBParametersLength = 36; // Parameters: public byte[] Reserved1; // 3 bytes public uint TotalParameterCount; public uint TotalDataCount; //uint ParameterCount; //uint ParameterOffset; //uint ParameterDisplacement; //uint DataCount; //uint DataOffset; //uint DataDisplacement; //byte SetupCount; // In 2-byte words public byte[] Setup; // Data: // Padding (alignment to 4 byte boundary) public byte[] TransParameters; // Trans_Parameters // Padding (alignment to 4 byte boundary) public byte[] TransData; // Trans_Data public NTTransactResponse() : base() { Reserved1 = new byte[3]; } public NTTransactResponse(byte[] buffer, int offset) : base(buffer, offset, false) { int readOffset = 0; Reserved1 = ByteReader.ReadBytes(this.SMBParameters, ref readOffset, 3); TotalParameterCount = LittleEndianReader.ReadUInt32(this.SMBParameters, ref readOffset); TotalDataCount = LittleEndianReader.ReadUInt32(this.SMBParameters, ref readOffset); uint parameterCount = LittleEndianReader.ReadUInt32(this.SMBParameters, ref readOffset); uint parameterOffset = LittleEndianReader.ReadUInt32(this.SMBParameters, ref readOffset); uint parameterDisplacement = LittleEndianReader.ReadUInt32(this.SMBParameters, ref readOffset); uint dataCount = LittleEndianReader.ReadUInt32(this.SMBParameters, ref readOffset); uint dataOffset = LittleEndianReader.ReadUInt32(this.SMBParameters, ref readOffset); uint dataDisplacement = LittleEndianReader.ReadUInt32(this.SMBParameters, ref readOffset); byte setupCount = ByteReader.ReadByte(this.SMBParameters, ref readOffset); Setup = ByteReader.ReadBytes(this.SMBParameters, ref offset, setupCount * 2); TransParameters = ByteReader.ReadBytes(buffer, (int)parameterOffset, (int)parameterCount); TransData = ByteReader.ReadBytes(buffer, (int)dataOffset, (int)dataCount); } public override byte[] GetBytes(bool isUnicode) { byte setupCount = (byte)(Setup.Length / 2); uint parameterCount = (ushort)TransParameters.Length; uint dataCount = (ushort)TransData.Length; uint parameterDisplacement = 0; uint dataDisplacement = 0; // WordCount + ByteCount are additional 3 bytes uint parameterOffset = (ushort)(SMB1Header.Length + 3 + (FixedSMBParametersLength + Setup.Length)); int padding1 = (int)(4 - (parameterOffset % 4)) % 4; parameterOffset += (ushort)padding1; uint dataOffset = (ushort)(parameterOffset + parameterCount); int padding2 = (int)(4 - (dataOffset % 4)) % 4; dataOffset += (ushort)padding2; this.SMBParameters = new byte[FixedSMBParametersLength + Setup.Length]; int writeOffset = 0; ByteWriter.WriteBytes(this.SMBParameters, ref writeOffset, Reserved1, 3); LittleEndianWriter.WriteUInt32(this.SMBParameters, ref writeOffset, TotalParameterCount); LittleEndianWriter.WriteUInt32(this.SMBParameters, ref writeOffset, TotalDataCount); LittleEndianWriter.WriteUInt32(this.SMBParameters, ref writeOffset, parameterCount); LittleEndianWriter.WriteUInt32(this.SMBParameters, ref writeOffset, parameterOffset); LittleEndianWriter.WriteUInt32(this.SMBParameters, ref writeOffset, parameterDisplacement); LittleEndianWriter.WriteUInt32(this.SMBParameters, ref writeOffset, dataCount); LittleEndianWriter.WriteUInt32(this.SMBParameters, ref writeOffset, dataOffset); LittleEndianWriter.WriteUInt32(this.SMBParameters, ref writeOffset, dataDisplacement); ByteWriter.WriteByte(this.SMBParameters, ref writeOffset, setupCount); ByteWriter.WriteBytes(this.SMBParameters, ref writeOffset, Setup); this.SMBData = new byte[parameterCount + dataCount + padding1 + padding2]; ByteWriter.WriteBytes(this.SMBData, padding1, TransParameters); ByteWriter.WriteBytes(this.SMBData, (int)(padding1 + parameterCount + padding2), TransData); return base.GetBytes(isUnicode); } public override CommandName CommandName { get { return CommandName.SMB_COM_NT_TRANSACT; } } public static int CalculateMessageSize(int setupLength, int trans2ParametersLength, int trans2DataLength) { int parameterOffset = SMB1Header.Length + 3 + (FixedSMBParametersLength + setupLength); int padding1 = (4 - (parameterOffset % 4)) % 4; parameterOffset += padding1; int dataOffset = (parameterOffset + trans2ParametersLength); int padding2 = (4 - (dataOffset % 4)) % 4; int messageParametersLength = FixedSMBParametersLength + setupLength; int messageDataLength = trans2ParametersLength + trans2DataLength + padding1 + padding2; // WordCount + ByteCount are additional 3 bytes return SMB1Header.Length + messageParametersLength + messageDataLength + 3; } } }