You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
jellyfin/Emby.Common.Implementations/IO/SharpCifs/Smb/SmbComTransaction.cs

350 lines
12 KiB

// This code is derived from jcifs smb client library <jcifs at samba dot org>
// Ported by J. Arturo <webmaster at komodosoft dot net>
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Lesser General Public
// License as published by the Free Software Foundation; either
// version 2.1 of the License, or (at your option) any later version.
//
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
// Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
using System;
using SharpCifs.Util;
namespace SharpCifs.Smb
{
internal abstract class SmbComTransaction : ServerMessageBlock
{
private static readonly int DefaultMaxDataCount
= Config.GetInt("jcifs.smb.client.transaction_buf_size", TransactionBufSize) - 512;
private const int PrimarySetupOffset = 61;
private const int SecondaryParameterOffset = 51;
private const int DisconnectTid = unchecked(0x01);
private const int OneWayTransaction = unchecked(0x02);
private const int PaddingSize = 2;
private int _flags = unchecked(0x00);
private int _fid;
private int _pad;
private int _pad1;
private bool _hasMore = true;
private bool _isPrimary = true;
private int _bufParameterOffset;
private int _bufDataOffset;
internal const int TransactionBufSize = unchecked(0xFFFF);
internal const byte Trans2FindFirst2 = unchecked(unchecked(0x01));
internal const byte Trans2FindNext2 = unchecked(unchecked(0x02));
internal const byte Trans2QueryFsInformation = unchecked(unchecked(0x03));
internal const byte Trans2QueryPathInformation = unchecked(unchecked(0x05));
internal const byte Trans2GetDfsReferral = unchecked(unchecked(0x10));
internal const byte Trans2SetFileInformation = unchecked(unchecked(0x08));
internal const int NetShareEnum = unchecked(0x0000);
internal const int NetServerEnum2 = unchecked(0x0068);
internal const int NetServerEnum3 = unchecked(0x00D7);
internal const byte TransPeekNamedPipe = unchecked(unchecked(0x23));
internal const byte TransWaitNamedPipe = unchecked(unchecked(0x53));
internal const byte TransCallNamedPipe = unchecked(unchecked(0x54));
internal const byte TransTransactNamedPipe = unchecked(unchecked(0x26));
protected internal int primarySetupOffset;
protected internal int secondaryParameterOffset;
protected internal int ParameterCount;
protected internal int ParameterOffset;
protected internal int ParameterDisplacement;
protected internal int DataCount;
protected internal int DataOffset;
protected internal int DataDisplacement;
internal int TotalParameterCount;
internal int TotalDataCount;
internal int MaxParameterCount;
internal int MaxDataCount = DefaultMaxDataCount;
internal byte MaxSetupCount;
internal int Timeout = 0;
internal int SetupCount = 1;
internal byte SubCommand;
internal string Name = string.Empty;
internal int MaxBufferSize;
internal byte[] TxnBuf;
public SmbComTransaction()
{
// relative to headerStart
// set in SmbTransport.sendTransaction() before nextElement called
MaxParameterCount = 1024;
primarySetupOffset = PrimarySetupOffset;
secondaryParameterOffset = SecondaryParameterOffset;
}
internal override void Reset()
{
base.Reset();
_isPrimary = _hasMore = true;
}
internal virtual void Reset(int key, string lastName)
{
Reset();
}
public virtual bool MoveNext()
{
return _hasMore;
}
public virtual object Current()
{
if (_isPrimary)
{
_isPrimary = false;
ParameterOffset = primarySetupOffset + (SetupCount * 2) + 2;
if (Command != SmbComNtTransact)
{
if (Command == SmbComTransaction && IsResponse() == false)
{
ParameterOffset += StringWireLength(Name, ParameterOffset);
}
}
else
{
if (Command == SmbComNtTransact)
{
ParameterOffset += 2;
}
}
_pad = ParameterOffset % PaddingSize;
_pad = _pad == 0 ? 0 : PaddingSize - _pad;
ParameterOffset += _pad;
TotalParameterCount = WriteParametersWireFormat(TxnBuf, _bufParameterOffset);
_bufDataOffset = TotalParameterCount;
// data comes right after data
int available = MaxBufferSize - ParameterOffset;
ParameterCount = Math.Min(TotalParameterCount, available);
available -= ParameterCount;
DataOffset = ParameterOffset + ParameterCount;
_pad1 = DataOffset % PaddingSize;
_pad1 = _pad1 == 0 ? 0 : PaddingSize - _pad1;
DataOffset += _pad1;
TotalDataCount = WriteDataWireFormat(TxnBuf, _bufDataOffset);
DataCount = Math.Min(TotalDataCount, available);
}
else
{
if (Command != SmbComNtTransact)
{
Command = SmbComTransactionSecondary;
}
else
{
Command = SmbComNtTransactSecondary;
}
// totalParameterCount and totalDataCount are set ok from primary
ParameterOffset = SecondaryParameterOffset;
if ((TotalParameterCount - ParameterDisplacement) > 0)
{
_pad = ParameterOffset % PaddingSize;
_pad = _pad == 0 ? 0 : PaddingSize - _pad;
ParameterOffset += _pad;
}
// caclulate parameterDisplacement before calculating new parameterCount
ParameterDisplacement += ParameterCount;
int available = MaxBufferSize - ParameterOffset - _pad;
ParameterCount = Math.Min(TotalParameterCount - ParameterDisplacement, available);
available -= ParameterCount;
DataOffset = ParameterOffset + ParameterCount;
_pad1 = DataOffset % PaddingSize;
_pad1 = _pad1 == 0 ? 0 : PaddingSize - _pad1;
DataOffset += _pad1;
DataDisplacement += DataCount;
available -= _pad1;
DataCount = Math.Min(TotalDataCount - DataDisplacement, available);
}
if ((ParameterDisplacement + ParameterCount) >= TotalParameterCount
&& (DataDisplacement + DataCount) >= TotalDataCount)
{
_hasMore = false;
}
return this;
}
internal override int WriteParameterWordsWireFormat(byte[] dst, int dstIndex)
{
int start = dstIndex;
WriteInt2(TotalParameterCount, dst, dstIndex);
dstIndex += 2;
WriteInt2(TotalDataCount, dst, dstIndex);
dstIndex += 2;
if (Command != SmbComTransactionSecondary)
{
WriteInt2(MaxParameterCount, dst, dstIndex);
dstIndex += 2;
WriteInt2(MaxDataCount, dst, dstIndex);
dstIndex += 2;
dst[dstIndex++] = MaxSetupCount;
dst[dstIndex++] = unchecked(unchecked(0x00));
// Reserved1
WriteInt2(_flags, dst, dstIndex);
dstIndex += 2;
WriteInt4(Timeout, dst, dstIndex);
dstIndex += 4;
dst[dstIndex++] = unchecked(unchecked(0x00));
// Reserved2
dst[dstIndex++] = unchecked(unchecked(0x00));
}
WriteInt2(ParameterCount, dst, dstIndex);
dstIndex += 2;
// writeInt2(( parameterCount == 0 ? 0 : parameterOffset ), dst, dstIndex );
WriteInt2(ParameterOffset, dst, dstIndex);
dstIndex += 2;
if (Command == SmbComTransactionSecondary)
{
WriteInt2(ParameterDisplacement, dst, dstIndex);
dstIndex += 2;
}
WriteInt2(DataCount, dst, dstIndex);
dstIndex += 2;
WriteInt2((DataCount == 0 ? 0 : DataOffset), dst, dstIndex);
dstIndex += 2;
if (Command == SmbComTransactionSecondary)
{
WriteInt2(DataDisplacement, dst, dstIndex);
dstIndex += 2;
}
else
{
dst[dstIndex++] = unchecked((byte)SetupCount);
dst[dstIndex++] = unchecked(unchecked(0x00));
// Reserved3
dstIndex += WriteSetupWireFormat(dst, dstIndex);
}
return dstIndex - start;
}
internal override int WriteBytesWireFormat(byte[] dst, int dstIndex)
{
int start = dstIndex;
int p = _pad;
if (Command == SmbComTransaction && IsResponse() == false)
{
dstIndex += WriteString(Name, dst, dstIndex);
}
if (ParameterCount > 0)
{
while (p-- > 0)
{
dst[dstIndex++] = unchecked(unchecked(0x00));
}
// Pad
Array.Copy(TxnBuf, _bufParameterOffset, dst, dstIndex, ParameterCount);
dstIndex += ParameterCount;
}
if (DataCount > 0)
{
p = _pad1;
while (p-- > 0)
{
dst[dstIndex++] = unchecked(unchecked(0x00));
}
// Pad1
Array.Copy(TxnBuf, _bufDataOffset, dst, dstIndex, DataCount);
_bufDataOffset += DataCount;
dstIndex += DataCount;
}
return dstIndex - start;
}
internal override int ReadParameterWordsWireFormat(byte[] buffer, int bufferIndex)
{
return 0;
}
internal override int ReadBytesWireFormat(byte[] buffer, int bufferIndex)
{
return 0;
}
internal abstract int WriteSetupWireFormat(byte[] dst, int dstIndex);
internal abstract int WriteParametersWireFormat(byte[] dst, int dstIndex);
internal abstract int WriteDataWireFormat(byte[] dst, int dstIndex);
internal abstract int ReadSetupWireFormat(byte[] buffer, int bufferIndex, int len);
internal abstract int ReadParametersWireFormat(byte[] buffer, int bufferIndex, int len);
internal abstract int ReadDataWireFormat(byte[] buffer, int bufferIndex, int len);
public override string ToString()
{
return base.ToString()
+ ",totalParameterCount=" + TotalParameterCount
+ ",totalDataCount=" + TotalDataCount
+ ",maxParameterCount=" + MaxParameterCount
+ ",maxDataCount=" + MaxDataCount
+ ",maxSetupCount=" + (int)MaxSetupCount
+ ",flags=0x" + Hexdump.ToHexString(_flags, 2)
+ ",timeout=" + Timeout
+ ",parameterCount=" + ParameterCount
+ ",parameterOffset=" + ParameterOffset
+ ",parameterDisplacement=" + ParameterDisplacement
+ ",dataCount=" + DataCount
+ ",dataOffset=" + DataOffset
+ ",dataDisplacement=" + DataDisplacement
+ ",setupCount=" + SetupCount
+ ",pad=" + _pad
+ ",pad1=" + _pad1;
}
}
}