1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 |
- /* Copyright (C) 2014-2018 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 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
- protected 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);
- case PacketTypeName.BindNak:
- return new BindNakPDU(buffer, offset);
- default:
- throw new NotImplementedException();
- }
- }
- public static ushort GetPDULength(byte[] buffer, int offset)
- {
- ushort fragmentLength = LittleEndianConverter.ToUInt16(buffer, offset + 8);
- return fragmentLength;
- }
- }
- }
|