SessionSetupAndXResponseExtended.cs 3.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  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. public class SessionSetupAndXResponseExtended : SMBAndXCommand
  14. {
  15. public const int ParametersLength = 8;
  16. // Parameters:
  17. //CommandName AndXCommand;
  18. //byte AndXReserved;
  19. //ushort AndXOffset;
  20. public SessionSetupAction Action;
  21. private ushort SecurityBlobLength;
  22. // Data:
  23. public byte[] SecurityBlob;
  24. public string NativeOS; // SMB_STRING (If Unicode, this field MUST be aligned to start on a 2-byte boundary from the start of the SMB header)
  25. public string NativeLanMan; // SMB_STRING (this field WILL be aligned to start on a 2-byte boundary from the start of the SMB header)
  26. public SessionSetupAndXResponseExtended() : base()
  27. {
  28. SecurityBlob = new byte[0];
  29. NativeOS = String.Empty;
  30. NativeLanMan = String.Empty;
  31. }
  32. public SessionSetupAndXResponseExtended(byte[] buffer, int offset, bool isUnicode) : base(buffer, offset, isUnicode)
  33. {
  34. Action = (SessionSetupAction)LittleEndianConverter.ToUInt16(this.SMBParameters, 4);
  35. SecurityBlobLength = LittleEndianConverter.ToUInt16(this.SMBParameters, 6);
  36. SecurityBlob = ByteReader.ReadBytes(this.SMBData, 0, SecurityBlobLength);
  37. int dataOffset = SecurityBlob.Length;
  38. if (isUnicode)
  39. {
  40. // A Unicode string MUST be aligned to a 16-bit boundary with respect to the beginning of the SMB Header.
  41. // Note: SMBData starts at an odd offset.
  42. int padding = (SecurityBlobLength + 1) % 2;
  43. dataOffset += padding;
  44. }
  45. NativeOS = SMB1Helper.ReadSMBString(this.SMBData, ref dataOffset, isUnicode);
  46. NativeLanMan = SMB1Helper.ReadSMBString(this.SMBData, ref dataOffset, isUnicode);
  47. }
  48. public override byte[] GetBytes(bool isUnicode)
  49. {
  50. ushort securityBlobLength = (ushort)SecurityBlob.Length;
  51. this.SMBParameters = new byte[ParametersLength];
  52. LittleEndianWriter.WriteUInt16(this.SMBParameters, 4, (ushort)Action);
  53. LittleEndianWriter.WriteUInt16(this.SMBParameters, 6, securityBlobLength);
  54. int padding = 0;
  55. if (isUnicode)
  56. {
  57. // A Unicode string MUST be aligned to a 16-bit boundary with respect to the beginning of the SMB Header.
  58. // Note: SMBData starts at an odd offset.
  59. padding = (1 + securityBlobLength) % 2;
  60. this.SMBData = new byte[SecurityBlob.Length + padding + NativeOS.Length * 2 + NativeLanMan.Length * 2 + 4];
  61. }
  62. else
  63. {
  64. this.SMBData = new byte[SecurityBlob.Length + NativeOS.Length + NativeLanMan.Length + 2];
  65. }
  66. int offset = 0;
  67. ByteWriter.WriteBytes(this.SMBData, ref offset, SecurityBlob);
  68. offset += padding;
  69. SMB1Helper.WriteSMBString(this.SMBData, ref offset, isUnicode, NativeOS);
  70. SMB1Helper.WriteSMBString(this.SMBData, ref offset, isUnicode, NativeLanMan);
  71. return base.GetBytes(isUnicode);
  72. }
  73. public override CommandName CommandName
  74. {
  75. get
  76. {
  77. return CommandName.SMB_COM_SESSION_SETUP_ANDX;
  78. }
  79. }
  80. }
  81. }