SessionSetupAndXRequest.cs 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. /* Copyright (C) 2014 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
  2. *
  3. * You can redistribute this program and/or modify it under the terms of
  4. * the GNU Lesser Public License as published by the Free Software Foundation,
  5. * either version 3 of the License, or (at your option) any later version.
  6. */
  7. using System;
  8. using System.Collections.Generic;
  9. using System.Text;
  10. using Utilities;
  11. namespace SMBLibrary.SMB1
  12. {
  13. /// <summary>
  14. /// SMB_COM_SESSION_SETUP_ANDX Request
  15. /// </summary>
  16. public class SessionSetupAndXRequest : SMBAndXCommand
  17. {
  18. public const int ParametersLength = 26;
  19. // Parameters:
  20. public ushort MaxBufferSize;
  21. public ushort MaxMpxCount;
  22. public ushort VcNumber;
  23. public uint SessionKey;
  24. private ushort OEMPasswordLength;
  25. private ushort UnicodePasswordLength;
  26. public uint Reserved;
  27. public ServerCapabilities Capabilities;
  28. // Data:
  29. public byte[] OEMPassword;
  30. public byte[] UnicodePassword;
  31. // Padding
  32. public string AccountName; // SMB_STRING (If Unicode, this field MUST be aligned to start on a 2-byte boundary from the start of the SMB header)
  33. public string PrimaryDomain; // SMB_STRING (this field WILL be aligned to start on a 2-byte boundary from the start of the SMB header)
  34. public string NativeOS; // SMB_STRING (this field WILL be aligned to start on a 2-byte boundary from the start of the SMB header)
  35. public string NativeLanMan; // SMB_STRING (this field WILL be aligned to start on a 2-byte boundary from the start of the SMB header)
  36. public SessionSetupAndXRequest(): base()
  37. {
  38. }
  39. public SessionSetupAndXRequest(byte[] buffer, int offset, bool isUnicode) : base(buffer, offset, isUnicode)
  40. {
  41. MaxBufferSize = LittleEndianConverter.ToUInt16(this.SMBParameters, 4);
  42. MaxMpxCount = LittleEndianConverter.ToUInt16(this.SMBParameters, 6);
  43. VcNumber = LittleEndianConverter.ToUInt16(this.SMBParameters, 8);
  44. SessionKey = LittleEndianConverter.ToUInt32(this.SMBParameters, 10);
  45. OEMPasswordLength = LittleEndianConverter.ToUInt16(this.SMBParameters, 14);
  46. UnicodePasswordLength = LittleEndianConverter.ToUInt16(this.SMBParameters, 16);
  47. Reserved = LittleEndianConverter.ToUInt32(this.SMBParameters, 18);
  48. Capabilities = (ServerCapabilities)LittleEndianConverter.ToUInt32(this.SMBParameters, 22);
  49. OEMPassword = ByteReader.ReadBytes(this.SMBData, 0, OEMPasswordLength);
  50. UnicodePassword = ByteReader.ReadBytes(this.SMBData, OEMPasswordLength, UnicodePasswordLength);
  51. int dataOffset = OEMPasswordLength + UnicodePasswordLength;
  52. if (isUnicode)
  53. {
  54. // A Unicode string MUST be aligned to a 16-bit boundary with respect to the beginning of the SMB Header.
  55. // Note: SMBData starts at an odd offset.
  56. int padding = (OEMPasswordLength + UnicodePasswordLength + 1) % 2;
  57. dataOffset += padding;
  58. }
  59. AccountName = SMB1Helper.ReadSMBString(this.SMBData, ref dataOffset, isUnicode);
  60. PrimaryDomain = SMB1Helper.ReadSMBString(this.SMBData, ref dataOffset, isUnicode);
  61. NativeOS = SMB1Helper.ReadSMBString(this.SMBData, ref dataOffset, isUnicode);
  62. NativeLanMan = SMB1Helper.ReadSMBString(this.SMBData, ref dataOffset, isUnicode);
  63. }
  64. public override byte[] GetBytes(bool isUnicode)
  65. {
  66. OEMPasswordLength = (ushort)OEMPassword.Length;
  67. UnicodePasswordLength = (ushort)UnicodePassword.Length;
  68. this.SMBParameters = new byte[ParametersLength];
  69. LittleEndianWriter.WriteUInt16(this.SMBParameters, 4, MaxBufferSize);
  70. LittleEndianWriter.WriteUInt16(this.SMBParameters, 6, MaxMpxCount);
  71. LittleEndianWriter.WriteUInt16(this.SMBParameters, 8, VcNumber);
  72. LittleEndianWriter.WriteUInt32(this.SMBParameters, 10, SessionKey);
  73. LittleEndianWriter.WriteUInt16(this.SMBParameters, 14, OEMPasswordLength);
  74. LittleEndianWriter.WriteUInt16(this.SMBParameters, 16, UnicodePasswordLength);
  75. LittleEndianWriter.WriteUInt32(this.SMBParameters, 18, Reserved);
  76. int padding = 0;
  77. if (isUnicode)
  78. {
  79. // A Unicode string MUST be aligned to a 16-bit boundary with respect to the beginning of the SMB Header.
  80. // Note: SMBData starts at an odd offset.
  81. padding = (OEMPasswordLength + UnicodePasswordLength + 1) % 2;
  82. this.SMBData = new byte[OEMPassword.Length + UnicodePassword.Length + padding + (AccountName.Length + 1) * 2 + (PrimaryDomain.Length + 1) * 2 + (NativeOS.Length + 1) * 2 + (NativeLanMan.Length + 1) * 2];
  83. }
  84. else
  85. {
  86. this.SMBData = new byte[OEMPassword.Length + UnicodePassword.Length + AccountName.Length + 1 + PrimaryDomain.Length + 1 + NativeOS.Length + 1 + NativeLanMan.Length + 1];
  87. }
  88. int offset = 0;
  89. ByteWriter.WriteBytes(this.SMBData, ref offset, OEMPassword);
  90. ByteWriter.WriteBytes(this.SMBData, ref offset, UnicodePassword);
  91. offset += padding;
  92. SMB1Helper.WriteSMBString(this.SMBData, ref offset, isUnicode, AccountName);
  93. SMB1Helper.WriteSMBString(this.SMBData, ref offset, isUnicode, PrimaryDomain);
  94. SMB1Helper.WriteSMBString(this.SMBData, ref offset, isUnicode, NativeOS);
  95. SMB1Helper.WriteSMBString(this.SMBData, ref offset, isUnicode, NativeLanMan);
  96. return base.GetBytes(isUnicode);
  97. }
  98. public override CommandName CommandName
  99. {
  100. get
  101. {
  102. return CommandName.SMB_COM_SESSION_SETUP_ANDX;
  103. }
  104. }
  105. }
  106. }