12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697 |
- /* Copyright (C) 2014-2017 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
- *
- * You can redistribute this program and/or modify it under the terms of
- * the GNU Lesser Public License as published by the Free Software Foundation,
- * either version 3 of the License, or (at your option) any later version.
- */
- using System;
- using System.Collections.Generic;
- using System.Text;
- using Utilities;
- namespace SMBLibrary.RPC
- {
- /// <summary>
- /// See DCE 1.1: Remote Procedure Call, Chapter 12.6 - Connection-oriented RPC PDUs
- /// </summary>
- public abstract class RPCPDU
- {
- public const int CommonFieldsLength = 16;
- // The common header fields, which appear in all (connection oriented) PDU types:
- public byte VersionMajor; // rpc_vers
- public byte VersionMinor; // rpc_vers_minor
- public PacketTypeName PacketType;
- public PacketFlags Flags;
- public DataRepresentationFormat DataRepresentation;
- protected ushort FragmentLength; // The length of the entire PDU
- public ushort AuthLength;
- public uint CallID;
- public RPCPDU()
- {
- VersionMajor = 5;
- VersionMinor = 0;
- }
- public RPCPDU(byte[] buffer, int offset)
- {
- VersionMajor = ByteReader.ReadByte(buffer, offset + 0);
- VersionMinor = ByteReader.ReadByte(buffer, offset + 1);
- PacketType = (PacketTypeName)ByteReader.ReadByte(buffer, offset + 2);
- Flags = (PacketFlags)ByteReader.ReadByte(buffer, offset + 3);
- DataRepresentation = new DataRepresentationFormat(buffer, offset + 4);
- FragmentLength = LittleEndianConverter.ToUInt16(buffer, offset + 8);
- AuthLength = LittleEndianConverter.ToUInt16(buffer, offset + 10);
- CallID = LittleEndianConverter.ToUInt32(buffer, offset + 12);
- }
- public abstract byte[] GetBytes();
- public void WriteCommonFieldsBytes(byte[] buffer)
- {
- ByteWriter.WriteByte(buffer, 0, VersionMajor);
- ByteWriter.WriteByte(buffer, 1, VersionMinor);
- ByteWriter.WriteByte(buffer, 2, (byte)PacketType);
- ByteWriter.WriteByte(buffer, 3, (byte)Flags);
- DataRepresentation.WriteBytes(buffer, 4);
- LittleEndianWriter.WriteUInt16(buffer, 8, (ushort)Length);
- LittleEndianWriter.WriteUInt16(buffer, 10, AuthLength);
- LittleEndianWriter.WriteUInt32(buffer, 12, CallID);
- }
- /// <summary>
- /// The length of the entire PDU
- /// </summary>
- public abstract int Length
- {
- get;
- }
- public static RPCPDU GetPDU(byte[] buffer, int offset)
- {
- PacketTypeName packetType = (PacketTypeName)ByteReader.ReadByte(buffer, 2);
- switch (packetType)
- {
- case PacketTypeName.Request:
- return new RequestPDU(buffer, offset);
- case PacketTypeName.Response:
- return new ResponsePDU(buffer, offset);
- case PacketTypeName.Fault:
- return new FaultPDU(buffer, offset);
- case PacketTypeName.Bind:
- return new BindPDU(buffer, offset);
- case PacketTypeName.BindAck:
- return new BindAckPDU(buffer, offset);
- default:
- throw new NotImplementedException();
- }
- }
- public static ushort GetPDULength(byte[] buffer, int offset)
- {
- ushort fragmentLength = LittleEndianConverter.ToUInt16(buffer, offset + 8);
- return fragmentLength;
- }
- }
- }
|