NDRParser.cs 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  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.IO;
  10. using System.Runtime.InteropServices;
  11. using System.Text;
  12. using Utilities;
  13. namespace SMBLibrary.RPC
  14. {
  15. /// <summary>
  16. /// NDR - Native Data Representation
  17. /// See DCE 1.1: Remote Procedure Call, Chapter 14 - Transfer Syntax NDR
  18. /// </summary>
  19. public class NDRParser
  20. {
  21. private byte[] m_buffer;
  22. private int m_offset;
  23. private int m_depth;
  24. private List<INDRStructure> m_deferredStructures = new List<INDRStructure>();
  25. private Dictionary<uint, INDRStructure> m_referentToInstance = new Dictionary<uint, INDRStructure>();
  26. public NDRParser(byte[] buffer)
  27. {
  28. m_buffer = buffer;
  29. m_offset = 0;
  30. m_depth = 0;
  31. }
  32. public void BeginStructure()
  33. {
  34. m_depth++;
  35. }
  36. /// <summary>
  37. /// Add embedded pointer deferred structure (referent) parser
  38. /// </summary>
  39. private void AddDeferredStructure(INDRStructure structure)
  40. {
  41. m_deferredStructures.Add(structure);
  42. }
  43. public void EndStructure()
  44. {
  45. m_depth--;
  46. // 14.3.12.3 - Algorithm for Deferral of Referents
  47. // Representations of (embedded) pointer referents are ordered according to a left-to-right, depth-first traversal of the embedding construction.
  48. // referent representations for the embedded construction are further deferred to a position in the octet stream that
  49. // follows the representation of the embedding construction. The set of referent representations for the embedded construction
  50. // is inserted among the referent representations for any pointers in the embedding construction, according to the order of elements or
  51. // members in the embedding construction
  52. if (m_depth == 0)
  53. {
  54. // Make a copy of all the deferred structures, additional deferred structures will be inserted to m_deferredStructures
  55. // as we process the existing list
  56. List<INDRStructure> deferredStructures = new List<INDRStructure>(m_deferredStructures);
  57. m_deferredStructures.Clear();
  58. // Read all deferred types:
  59. foreach (INDRStructure deferredStructure in deferredStructures)
  60. {
  61. deferredStructure.Read(this);
  62. }
  63. }
  64. }
  65. public string ReadUnicodeString()
  66. {
  67. NDRUnicodeString unicodeString = new NDRUnicodeString(this);
  68. return unicodeString.Value;
  69. }
  70. public void ReadStructure(INDRStructure structure)
  71. {
  72. structure.Read(this);
  73. }
  74. // 14.3.11.1 - Top-level Full Pointers
  75. public string ReadTopLevelUnicodeStringPointer()
  76. {
  77. uint referentID = ReadUInt32();
  78. if (referentID == 0)
  79. {
  80. return null;
  81. }
  82. if (m_referentToInstance.ContainsKey(referentID))
  83. {
  84. NDRUnicodeString unicodeString = (NDRUnicodeString)m_referentToInstance[referentID];
  85. return unicodeString.Value;
  86. }
  87. else
  88. {
  89. NDRUnicodeString unicodeString = new NDRUnicodeString(this);
  90. m_referentToInstance.Add(referentID, unicodeString);
  91. return unicodeString.Value;
  92. }
  93. }
  94. public void ReadEmbeddedStructureFullPointer(ref NDRUnicodeString structure)
  95. {
  96. ReadEmbeddedStructureFullPointer<NDRUnicodeString>(ref structure);
  97. }
  98. public void ReadEmbeddedStructureFullPointer<T>(ref T structure) where T : INDRStructure, new ()
  99. {
  100. uint referentID = ReadUInt32();
  101. if (referentID != 0) // not null
  102. {
  103. if (structure == null)
  104. {
  105. structure = new T();
  106. }
  107. AddDeferredStructure(structure);
  108. }
  109. else
  110. {
  111. structure = default(T);
  112. }
  113. }
  114. // 14.2.2 - Alignment of Primitive Types
  115. public uint ReadUInt16()
  116. {
  117. m_offset += (2 - (m_offset % 2)) % 2;
  118. return LittleEndianReader.ReadUInt16(m_buffer, ref m_offset);
  119. }
  120. // 14.2.2 - Alignment of Primitive Types
  121. public uint ReadUInt32()
  122. {
  123. m_offset += (4 - (m_offset % 4)) % 4;
  124. return LittleEndianReader.ReadUInt32(m_buffer, ref m_offset);
  125. }
  126. }
  127. }