NDRWriter.cs 5.2 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 NDRWriter
  20. {
  21. private MemoryStream m_stream = new MemoryStream();
  22. private int m_depth;
  23. private List<INDRStructure> m_deferredStructures = new List<INDRStructure>();
  24. private Dictionary<uint, INDRStructure> m_referentToInstance = new Dictionary<uint, INDRStructure>();
  25. private uint m_nextReferentID = 0x00020000;
  26. public void BeginStructure()
  27. {
  28. m_depth++;
  29. }
  30. /// <summary>
  31. /// Add embedded pointer deferred structure (referent) writer
  32. /// </summary>
  33. private void AddDeferredStructure(INDRStructure structure)
  34. {
  35. m_deferredStructures.Add(structure);
  36. }
  37. public void EndStructure()
  38. {
  39. m_depth--;
  40. // 14.3.12.3 - Algorithm for Deferral of Referents
  41. // Representations of (embedded) pointer referents are ordered according to a left-to-right, depth-first traversal of the embedding construction.
  42. // referent representations for the embedded construction are further deferred to a position in the octet stream that
  43. // follows the representation of the embedding construction. The set of referent representations for the embedded construction
  44. // is inserted among the referent representations for any pointers in the embedding construction, according to the order of elements or
  45. // members in the embedding construction
  46. if (m_depth == 0)
  47. {
  48. // Make a copy of all the deferred structures, additional deferred structures will be inserted to m_deferredStructures
  49. // as we process the existing list
  50. List<INDRStructure> deferredStructures = new List<INDRStructure>(m_deferredStructures);
  51. m_deferredStructures.Clear();
  52. // Write all deferred types:
  53. foreach (INDRStructure deferredStructure in deferredStructures)
  54. {
  55. deferredStructure.Write(this);
  56. }
  57. }
  58. }
  59. public void WriteUnicodeString(string value)
  60. {
  61. NDRUnicodeString unicodeString = new NDRUnicodeString(value);
  62. unicodeString.Write(this);
  63. }
  64. public void WriteStructure(INDRStructure structure)
  65. {
  66. structure.Write(this);
  67. }
  68. public void WriteTopLevelUnicodeStringPointer(string value)
  69. {
  70. if (value == null)
  71. {
  72. WriteUInt32(0);
  73. return;
  74. }
  75. // Note: We do not bother searching for existing values
  76. uint referentID = GetNextReferentID();
  77. WriteUInt32(referentID);
  78. NDRUnicodeString unicodeString = new NDRUnicodeString(value);
  79. unicodeString.Write(this);
  80. m_referentToInstance.Add(referentID, unicodeString);
  81. }
  82. // 14.3.12.1 Embedded Full Pointers
  83. public void WriteEmbeddedStructureFullPointer(INDRStructure structure)
  84. {
  85. if (structure == null)
  86. {
  87. WriteUInt32(0); // null
  88. return;
  89. }
  90. else
  91. {
  92. // Note: We do not bother searching for existing values
  93. uint referentID = GetNextReferentID();
  94. WriteUInt32(referentID);
  95. AddDeferredStructure(structure);
  96. m_referentToInstance.Add(referentID, structure);
  97. }
  98. }
  99. // 14.2.2 - Alignment of Primitive Types
  100. public void WriteUInt16(ushort value)
  101. {
  102. uint padding = (uint)(2 - (m_stream.Position % 2)) % 2;
  103. m_stream.Position += padding;
  104. LittleEndianWriter.WriteUInt16(m_stream, value);
  105. }
  106. // 14.2.2 - Alignment of Primitive Types
  107. public void WriteUInt32(uint value)
  108. {
  109. uint padding = (uint)(4 - (m_stream.Position % 4)) % 4;
  110. m_stream.Position += padding;
  111. LittleEndianWriter.WriteUInt32(m_stream, value);
  112. }
  113. public byte[] GetBytes()
  114. {
  115. byte[] buffer = new byte[m_stream.Length];
  116. m_stream.Seek(0, SeekOrigin.Begin);
  117. m_stream.Read(buffer, 0, buffer.Length);
  118. return buffer;
  119. }
  120. private uint GetNextReferentID()
  121. {
  122. uint result = m_nextReferentID;
  123. m_nextReferentID++;
  124. return result;
  125. }
  126. }
  127. }