|
|
|
// 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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|