jellyfin-server/Emby.Common.Implementations/IO/SharpCifs/Smb/SmbComNegotiateResponse.cs

180 lines
7.9 KiB
C#
Raw Normal View History

2017-04-02 00:36:06 +00:00
// 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;
using SharpCifs.Util.Sharpen;
namespace SharpCifs.Smb
{
2017-06-21 06:46:57 +00:00
internal class SmbComNegotiateResponse : ServerMessageBlock
{
internal int DialectIndex;
2017-04-02 00:36:06 +00:00
2017-06-21 06:46:57 +00:00
internal SmbTransport.ServerData Server;
2017-04-02 00:36:06 +00:00
2017-06-21 06:46:57 +00:00
internal SmbComNegotiateResponse(SmbTransport.ServerData server)
{
this.Server = server;
}
2017-04-02 00:36:06 +00:00
2017-06-21 06:46:57 +00:00
internal override int WriteParameterWordsWireFormat(byte[] dst, int dstIndex)
{
return 0;
}
2017-04-02 00:36:06 +00:00
2017-06-21 06:46:57 +00:00
internal override int WriteBytesWireFormat(byte[] dst, int dstIndex)
{
return 0;
}
2017-04-02 00:36:06 +00:00
2017-06-21 06:46:57 +00:00
internal override int ReadParameterWordsWireFormat(byte[] buffer, int bufferIndex)
{
int start = bufferIndex;
DialectIndex = ReadInt2(buffer, bufferIndex);
bufferIndex += 2;
if (DialectIndex > 10)
{
return bufferIndex - start;
}
Server.SecurityMode = buffer[bufferIndex++] & unchecked(0xFF);
Server.Security = Server.SecurityMode & unchecked(0x01);
Server.EncryptedPasswords
= (Server.SecurityMode & unchecked(0x02)) == unchecked(0x02);
Server.SignaturesEnabled
= (Server.SecurityMode & unchecked(0x04)) == unchecked(0x04);
Server.SignaturesRequired
= (Server.SecurityMode & unchecked(0x08)) == unchecked(0x08);
Server.MaxMpxCount = ReadInt2(buffer, bufferIndex);
bufferIndex += 2;
Server.MaxNumberVcs = ReadInt2(buffer, bufferIndex);
bufferIndex += 2;
Server.MaxBufferSize = ReadInt4(buffer, bufferIndex);
bufferIndex += 4;
Server.MaxRawSize = ReadInt4(buffer, bufferIndex);
bufferIndex += 4;
Server.SessionKey = ReadInt4(buffer, bufferIndex);
bufferIndex += 4;
Server.Capabilities = ReadInt4(buffer, bufferIndex);
bufferIndex += 4;
Server.ServerTime = ReadTime(buffer, bufferIndex);
bufferIndex += 8;
Server.ServerTimeZone = ReadInt2(buffer, bufferIndex);
bufferIndex += 2;
Server.EncryptionKeyLength = buffer[bufferIndex++] & unchecked(0xFF);
return bufferIndex - start;
}
2017-04-02 00:36:06 +00:00
2017-06-21 06:46:57 +00:00
internal override int ReadBytesWireFormat(byte[] buffer, int bufferIndex)
{
int start = bufferIndex;
if ((Server.Capabilities & SmbConstants.CapExtendedSecurity) == 0)
{
Server.EncryptionKey = new byte[Server.EncryptionKeyLength];
Array.Copy(buffer,
bufferIndex,
Server.EncryptionKey,
0,
Server.EncryptionKeyLength);
bufferIndex += Server.EncryptionKeyLength;
if (ByteCount > Server.EncryptionKeyLength)
{
int len = 0;
// TODO: we can use new string routine here
try
{
2017-04-02 00:36:06 +00:00
if ((Flags2 & SmbConstants.Flags2Unicode) == SmbConstants.Flags2Unicode)
2017-06-21 06:46:57 +00:00
{
while (buffer[bufferIndex + len] != unchecked(unchecked(0x00))
|| buffer[bufferIndex + len + 1] != unchecked(unchecked(0x00)))
{
len += 2;
if (len > 256)
{
throw new RuntimeException("zero termination not found");
}
}
Server.OemDomainName = Runtime.GetStringForBytes(buffer,
bufferIndex,
len,
SmbConstants.UniEncoding);
}
else
{
while (buffer[bufferIndex + len] != unchecked(unchecked(0x00)))
{
len++;
if (len > 256)
{
throw new RuntimeException("zero termination not found");
}
}
Server.OemDomainName = Runtime.GetStringForBytes(buffer,
bufferIndex,
len,
SmbConstants.OemEncoding);
}
}
catch (UnsupportedEncodingException uee)
{
if (Log.Level > 1)
{
Runtime.PrintStackTrace(uee, Log);
}
}
bufferIndex += len;
}
else
{
Server.OemDomainName = "";
}
}
else
{
Server.Guid = new byte[16];
Array.Copy(buffer, bufferIndex, Server.Guid, 0, 16);
Server.OemDomainName = "";
}
// ignore SPNEGO token for now ...
return bufferIndex - start;
}
2017-04-02 00:36:06 +00:00
2017-06-21 06:46:57 +00:00
public override string ToString()
{
return "SmbComNegotiateResponse["
+ base.ToString()
+ ",wordCount=" + WordCount
+ ",dialectIndex=" + DialectIndex
+ ",securityMode=0x" + Hexdump.ToHexString(Server.SecurityMode, 1)
+ ",security=" + (Server.Security == SmbConstants.SecurityShare
? "share"
: "user")
+ ",encryptedPasswords=" + Server.EncryptedPasswords
+ ",maxMpxCount=" + Server.MaxMpxCount
+ ",maxNumberVcs=" + Server.MaxNumberVcs
+ ",maxBufferSize=" + Server.MaxBufferSize
+ ",maxRawSize=" + Server.MaxRawSize
+ ",sessionKey=0x" + Hexdump.ToHexString(Server.SessionKey, 8)
+ ",capabilities=0x" + Hexdump.ToHexString(Server.Capabilities, 8)
+ ",serverTime=" + Extensions.CreateDate(Server.ServerTime)
+ ",serverTimeZone=" + Server.ServerTimeZone
+ ",encryptionKeyLength=" + Server.EncryptionKeyLength
+ ",byteCount=" + ByteCount
+ ",oemDomainName=" + Server.OemDomainName + "]";
}
}
2017-04-02 00:36:06 +00:00
}