RPCPipeStream.cs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. /* Copyright (C) 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.IO;
  10. using SMBLibrary.RPC;
  11. namespace SMBLibrary.Services
  12. {
  13. public class RPCPipeStream : Stream
  14. {
  15. private RemoteService m_service;
  16. private List<MemoryStream> m_outputStreams; // A stream for each message in order to support message mode named pipe
  17. private int? m_maxTransmitFragmentSize;
  18. public RPCPipeStream(RemoteService service)
  19. {
  20. m_service = service;
  21. m_outputStreams = new List<MemoryStream>();
  22. }
  23. public override int Read(byte[] buffer, int offset, int count)
  24. {
  25. if (m_outputStreams.Count > 0)
  26. {
  27. int result = m_outputStreams[0].Read(buffer, offset, count);
  28. if (m_outputStreams[0].Position == m_outputStreams[0].Length)
  29. {
  30. m_outputStreams.RemoveAt(0);
  31. }
  32. return result;
  33. }
  34. else
  35. {
  36. return 0;
  37. }
  38. }
  39. public override void Write(byte[] buffer, int offset, int count)
  40. {
  41. int lengthOfPDUs = 0;
  42. do
  43. {
  44. RPCPDU rpcRequest = RPCPDU.GetPDU(buffer, offset);
  45. ProcessRPCRequest(rpcRequest);
  46. lengthOfPDUs += rpcRequest.FragmentLength;
  47. }
  48. while (lengthOfPDUs < count);
  49. }
  50. private void ProcessRPCRequest(RPCPDU rpcRequest)
  51. {
  52. if (rpcRequest is BindPDU)
  53. {
  54. BindAckPDU bindAckPDU = RemoteServiceHelper.GetRPCBindResponse((BindPDU)rpcRequest, m_service);
  55. m_maxTransmitFragmentSize = bindAckPDU.MaxTransmitFragmentSize;
  56. Append(bindAckPDU.GetBytes());
  57. }
  58. else if (rpcRequest is RequestPDU)
  59. {
  60. // if BindPDU was not received, we ignore any subsequent RPC packets
  61. if (m_maxTransmitFragmentSize.HasValue)
  62. {
  63. List<ResponsePDU> responsePDUs = RemoteServiceHelper.GetRPCResponse((RequestPDU)rpcRequest, m_service, m_maxTransmitFragmentSize.Value);
  64. foreach (ResponsePDU responsePDU in responsePDUs)
  65. {
  66. Append(responsePDU.GetBytes());
  67. }
  68. }
  69. }
  70. else
  71. {
  72. throw new NotImplementedException("Unsupported RPC Packet Type");
  73. }
  74. }
  75. private void Append(byte[] buffer)
  76. {
  77. MemoryStream stream = new MemoryStream(buffer);
  78. m_outputStreams.Add(stream);
  79. }
  80. public override void Flush()
  81. {
  82. }
  83. public override void Close()
  84. {
  85. }
  86. public override long Seek(long offset, SeekOrigin origin)
  87. {
  88. throw new NotSupportedException();
  89. }
  90. public override void SetLength(long value)
  91. {
  92. throw new NotSupportedException();
  93. }
  94. public override bool CanSeek
  95. {
  96. get
  97. {
  98. return false;
  99. }
  100. }
  101. public override bool CanRead
  102. {
  103. get
  104. {
  105. return true;
  106. }
  107. }
  108. public override bool CanWrite
  109. {
  110. get
  111. {
  112. return true;
  113. }
  114. }
  115. public override long Length
  116. {
  117. get
  118. {
  119. // Stream.Length only works on Stream implementations where seeking is available.
  120. throw new NotSupportedException();
  121. }
  122. }
  123. public override long Position
  124. {
  125. get
  126. {
  127. throw new NotSupportedException();
  128. }
  129. set
  130. {
  131. throw new NotSupportedException();
  132. }
  133. }
  134. /// <summary>
  135. /// The length of the first message available in the pipe
  136. /// </summary>
  137. public int MessageLength
  138. {
  139. get
  140. {
  141. if (m_outputStreams.Count > 0)
  142. {
  143. return (int)m_outputStreams[0].Length;
  144. }
  145. else
  146. {
  147. return 0;
  148. }
  149. }
  150. }
  151. }
  152. }