RPCPDU.cs 3.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. /* Copyright (C) 2014-2017 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.RPC
  12. {
  13. /// <summary>
  14. /// See DCE 1.1: Remote Procedure Call, Chapter 12.6 - Connection-oriented RPC PDUs
  15. /// </summary>
  16. public abstract class RPCPDU
  17. {
  18. public const int CommonFieldsLength = 16;
  19. // The common header fields, which appear in all (connection oriented) PDU types:
  20. public byte VersionMajor; // rpc_vers
  21. public byte VersionMinor; // rpc_vers_minor
  22. public PacketTypeName PacketType;
  23. public PacketFlags Flags;
  24. public DataRepresentationFormat DataRepresentation;
  25. protected ushort FragmentLength; // The length of the entire PDU
  26. public ushort AuthLength;
  27. public uint CallID;
  28. public RPCPDU()
  29. {
  30. VersionMajor = 5;
  31. VersionMinor = 0;
  32. }
  33. public RPCPDU(byte[] buffer, int offset)
  34. {
  35. VersionMajor = ByteReader.ReadByte(buffer, offset + 0);
  36. VersionMinor = ByteReader.ReadByte(buffer, offset + 1);
  37. PacketType = (PacketTypeName)ByteReader.ReadByte(buffer, offset + 2);
  38. Flags = (PacketFlags)ByteReader.ReadByte(buffer, offset + 3);
  39. DataRepresentation = new DataRepresentationFormat(buffer, offset + 4);
  40. FragmentLength = LittleEndianConverter.ToUInt16(buffer, offset + 8);
  41. AuthLength = LittleEndianConverter.ToUInt16(buffer, offset + 10);
  42. CallID = LittleEndianConverter.ToUInt32(buffer, offset + 12);
  43. }
  44. public abstract byte[] GetBytes();
  45. public void WriteCommonFieldsBytes(byte[] buffer)
  46. {
  47. ByteWriter.WriteByte(buffer, 0, VersionMajor);
  48. ByteWriter.WriteByte(buffer, 1, VersionMinor);
  49. ByteWriter.WriteByte(buffer, 2, (byte)PacketType);
  50. ByteWriter.WriteByte(buffer, 3, (byte)Flags);
  51. DataRepresentation.WriteBytes(buffer, 4);
  52. LittleEndianWriter.WriteUInt16(buffer, 8, (ushort)Length);
  53. LittleEndianWriter.WriteUInt16(buffer, 10, AuthLength);
  54. LittleEndianWriter.WriteUInt32(buffer, 12, CallID);
  55. }
  56. /// <summary>
  57. /// The length of the entire PDU
  58. /// </summary>
  59. public abstract int Length
  60. {
  61. get;
  62. }
  63. public static RPCPDU GetPDU(byte[] buffer, int offset)
  64. {
  65. PacketTypeName packetType = (PacketTypeName)ByteReader.ReadByte(buffer, 2);
  66. switch (packetType)
  67. {
  68. case PacketTypeName.Request:
  69. return new RequestPDU(buffer, offset);
  70. case PacketTypeName.Response:
  71. return new ResponsePDU(buffer, offset);
  72. case PacketTypeName.Fault:
  73. return new FaultPDU(buffer, offset);
  74. case PacketTypeName.Bind:
  75. return new BindPDU(buffer, offset);
  76. case PacketTypeName.BindAck:
  77. return new BindAckPDU(buffer, offset);
  78. default:
  79. throw new NotImplementedException();
  80. }
  81. }
  82. public static ushort GetPDULength(byte[] buffer, int offset)
  83. {
  84. ushort fragmentLength = LittleEndianConverter.ToUInt16(buffer, offset + 8);
  85. return fragmentLength;
  86. }
  87. }
  88. }