RemoteServiceHelper.cs 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  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 SMBLibrary.RPC;
  11. using Utilities;
  12. namespace SMBLibrary.Services
  13. {
  14. public class RemoteServiceHelper
  15. {
  16. // v1 - DCE 1.1: Remote Procedure Call
  17. // v2 - [MS-RPCE] 2.2.4.12 NDR Transfer Syntax Identifier
  18. private static readonly Guid NDRTransferSyntaxIdentifier = new Guid("8A885D04-1CEB-11C9-9FE8-08002B104860");
  19. // v1 - [MS-RPCE] 3.3.1.5.3 - Bind Time Feature Negotiation
  20. // Windows will reject this:
  21. //private static readonly Guid BindTimeFeatureIdentifier1 = new Guid("6CB71C2C-9812-4540-0100-000000000000");
  22. // Windows will return NegotiationResult.NegotiateAck:
  23. private static readonly Guid BindTimeFeatureIdentifier3 = new Guid("6CB71C2C-9812-4540-0300-000000000000");
  24. private static uint m_associationGroupID = 1;
  25. public static BindAckPDU GetRPCBindResponse(BindPDU bindPDU, RemoteService service)
  26. {
  27. BindAckPDU bindAckPDU = new BindAckPDU();
  28. bindAckPDU.Flags = PacketFlags.FirstFragment | PacketFlags.LastFragment;
  29. bindAckPDU.DataRepresentation = bindPDU.DataRepresentation;
  30. bindAckPDU.CallID = bindPDU.CallID;
  31. // See DCE 1.1: Remote Procedure Call - 12.6.3.6
  32. // The client should set the assoc_group_id field either to 0 (zero), to indicate a new association group,
  33. // or to the known value. When the server receives a value of 0, this indicates that the client
  34. // has requested a new association group, and it assigns a server unique value to the group.
  35. if (bindPDU.AssociationGroupID == 0)
  36. {
  37. bindAckPDU.AssociationGroupID = m_associationGroupID;
  38. m_associationGroupID++;
  39. if (m_associationGroupID == 0)
  40. {
  41. m_associationGroupID++;
  42. }
  43. }
  44. else
  45. {
  46. bindAckPDU.AssociationGroupID = bindPDU.AssociationGroupID;
  47. }
  48. bindAckPDU.SecondaryAddress = @"\PIPE\" + service.PipeName;
  49. bindAckPDU.MaxTransmitFragmentSize = bindPDU.MaxReceiveFragmentSize;
  50. bindAckPDU.MaxReceiveFragmentSize = bindPDU.MaxTransmitFragmentSize;
  51. foreach (ContextElement element in bindPDU.ContextList)
  52. {
  53. ResultElement resultElement = new ResultElement();
  54. if (element.AbstractSyntax.InterfaceUUID.Equals(service.InterfaceGuid))
  55. {
  56. int index = IndexOfSupportedTransferSyntax(element.TransferSyntaxList);
  57. if (index >= 0)
  58. {
  59. resultElement.Result = NegotiationResult.Acceptance;
  60. resultElement.TransferSyntax = element.TransferSyntaxList[index];
  61. }
  62. else if (element.TransferSyntaxList.Contains(new SyntaxID(BindTimeFeatureIdentifier3, 1)))
  63. {
  64. // [MS-RPCE] 3.3.1.5.3
  65. // If the server supports bind time feature negotiation, it MUST reply with the result
  66. // field in the p_result_t structure of the bind_ack PDU equal to negotiate_ack.
  67. resultElement.Result = NegotiationResult.NegotiateAck;
  68. resultElement.Reason = RejectionReason.AbstractSyntaxNotSupported;
  69. }
  70. else
  71. {
  72. resultElement.Result = NegotiationResult.ProviderRejection;
  73. resultElement.Reason = RejectionReason.ProposedTransferSyntaxesNotSupported;
  74. }
  75. }
  76. else
  77. {
  78. resultElement.Result = NegotiationResult.ProviderRejection;
  79. resultElement.Reason = RejectionReason.AbstractSyntaxNotSupported;
  80. }
  81. bindAckPDU.ResultList.Add(resultElement);
  82. }
  83. return bindAckPDU;
  84. }
  85. private static int IndexOfSupportedTransferSyntax(List<SyntaxID> syntaxList)
  86. {
  87. List<SyntaxID> supportedTransferSyntaxes = new List<SyntaxID>();
  88. supportedTransferSyntaxes.Add(new SyntaxID(NDRTransferSyntaxIdentifier, 1));
  89. // [MS-RPCE] Version 2.0 data representation protocol:
  90. supportedTransferSyntaxes.Add(new SyntaxID(NDRTransferSyntaxIdentifier, 2));
  91. for(int index = 0; index < syntaxList.Count; index++)
  92. {
  93. if (supportedTransferSyntaxes.Contains(syntaxList[index]))
  94. {
  95. return index;
  96. }
  97. }
  98. return -1;
  99. }
  100. public static List<ResponsePDU> GetRPCResponse(RequestPDU requestPDU, RemoteService service, int maxTransmitFragmentSize)
  101. {
  102. byte[] responseBytes = service.GetResponseBytes(requestPDU.OpNum, requestPDU.Data);
  103. int offset = 0;
  104. List<ResponsePDU> result = new List<ResponsePDU>();
  105. int maxPDUDataLength = maxTransmitFragmentSize - RPCPDU.CommonFieldsLength - ResponsePDU.ResponseFieldsLength;
  106. do
  107. {
  108. ResponsePDU responsePDU = new ResponsePDU();
  109. int pduDataLength = Math.Min(responseBytes.Length - offset, maxPDUDataLength);
  110. responsePDU.DataRepresentation = requestPDU.DataRepresentation;
  111. responsePDU.CallID = requestPDU.CallID;
  112. responsePDU.AllocationHint = (uint)(responseBytes.Length - offset);
  113. responsePDU.Data = ByteReader.ReadBytes(responseBytes, offset, pduDataLength);
  114. if (offset == 0)
  115. {
  116. responsePDU.Flags |= PacketFlags.FirstFragment;
  117. }
  118. if (offset + pduDataLength == responseBytes.Length)
  119. {
  120. responsePDU.Flags |= PacketFlags.LastFragment;
  121. }
  122. result.Add(responsePDU);
  123. offset += pduDataLength;
  124. }
  125. while (offset < responseBytes.Length);
  126. return result;
  127. }
  128. }
  129. }