mirror of
https://github.com/danbulant/Cosmos
synced 2026-05-19 20:39:01 +00:00
300 lines
10 KiB
C#
300 lines
10 KiB
C#
using System;
|
|
|
|
namespace Cosmos.System.Network.IPv4
|
|
{
|
|
/// <summary>
|
|
/// ICMPPacket class. See also: <seealso cref="IPPacket"/>.
|
|
/// </summary>
|
|
internal class ICMPPacket : IPPacket
|
|
{
|
|
/// <summary>
|
|
/// Packet type.
|
|
/// </summary>
|
|
protected byte icmpType;
|
|
/// <summary>
|
|
/// Packet code.
|
|
/// </summary>
|
|
protected byte icmpCode;
|
|
/// <summary>
|
|
/// Packet CRC.
|
|
/// </summary>
|
|
protected ushort icmpCRC;
|
|
/// <summary>
|
|
/// Received reply.
|
|
/// </summary>
|
|
public static ICMPEchoReply recvd_reply;
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPPacket"/> class.
|
|
/// </summary>
|
|
/// <param name="packetData">Packet data.</param>
|
|
/// <exception cref="ArgumentException">Thrown if packetData is invalid.</exception>
|
|
internal static void ICMPHandler(byte[] packetData)
|
|
{
|
|
NetworkStack.debugger.Send("ICMP Handler called");
|
|
ICMPPacket icmp_packet = new ICMPPacket(packetData);
|
|
switch (icmp_packet.ICMP_Type)
|
|
{
|
|
case 0:
|
|
recvd_reply = new ICMPEchoReply(packetData);
|
|
NetworkStack.debugger.Send("Received ICMP Echo reply from " + recvd_reply.SourceIP.ToString());
|
|
break;
|
|
case 8:
|
|
ICMPEchoRequest request = new ICMPEchoRequest(packetData);
|
|
NetworkStack.debugger.Send("Received " + request.ToString());
|
|
ICMPEchoReply reply = new ICMPEchoReply(request);
|
|
NetworkStack.debugger.Send("Sending ICMP Echo reply to " + reply.DestinationIP.ToString());
|
|
OutgoingBuffer.AddPacket(reply);
|
|
NetworkStack.Update();
|
|
break;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Work around to make VMT scanner include the initFields method
|
|
/// </summary>
|
|
public static void VMTInclude()
|
|
{
|
|
new ICMPPacket();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPPacket"/> class.
|
|
/// </summary>
|
|
internal ICMPPacket()
|
|
: base()
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPPacket"/> class.
|
|
/// </summary>
|
|
/// <param name="rawData">Raw data.</param>
|
|
internal ICMPPacket(byte[] rawData)
|
|
: base(rawData)
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Init ICMPPacket fields.1
|
|
/// </summary>
|
|
/// <exception cref="ArgumentException">Thrown if RawData is invalid or null.</exception>
|
|
protected override void initFields()
|
|
{
|
|
//Sys.Console.WriteLine("ICMPPacket.initFields() called;");
|
|
base.initFields();
|
|
icmpType = RawData[DataOffset];
|
|
icmpCode = RawData[DataOffset + 1];
|
|
icmpCRC = (ushort)((RawData[DataOffset + 2] << 8) | RawData[DataOffset + 3]);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPPacket"/> class.
|
|
/// </summary>
|
|
/// <param name="source">Source address.</param>
|
|
/// <param name="dest">Destination address.</param>
|
|
/// <param name="type">Type.</param>
|
|
/// <param name="code">Code.</param>
|
|
/// <param name="id">ID.</param>
|
|
/// <param name="seq">SEQ.</param>
|
|
/// <param name="icmpDataSize">Data size.</param>
|
|
/// <exception cref="ArgumentException">Thrown if RawData is invalid or null.</exception>
|
|
internal ICMPPacket(Address source, Address dest, byte type, byte code, ushort id, ushort seq, ushort icmpDataSize)
|
|
: base(icmpDataSize, 1, source, dest, 0x00)
|
|
{
|
|
RawData[DataOffset] = type;
|
|
RawData[DataOffset + 1] = code;
|
|
RawData[DataOffset + 2] = 0x00;
|
|
RawData[DataOffset + 3] = 0x00;
|
|
RawData[DataOffset + 4] = (byte)((id >> 8) & 0xFF);
|
|
RawData[DataOffset + 5] = (byte)((id >> 0) & 0xFF);
|
|
RawData[DataOffset + 6] = (byte)((seq >> 8) & 0xFF);
|
|
RawData[DataOffset + 7] = (byte)((seq >> 0) & 0xFF);
|
|
|
|
icmpCRC = CalcICMPCRC((ushort)(icmpDataSize + 8));
|
|
RawData[DataOffset + 2] = (byte)((icmpCRC >> 8) & 0xFF);
|
|
RawData[DataOffset + 3] = (byte)((icmpCRC >> 0) & 0xFF);
|
|
initFields();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Calculate ICMP CRC3.
|
|
/// </summary>
|
|
/// <param name="length">Lenght.</param>
|
|
/// <returns></returns>
|
|
protected ushort CalcICMPCRC(ushort length)
|
|
{
|
|
return CalcOcCRC(DataOffset, length);
|
|
}
|
|
|
|
internal byte ICMP_Type => icmpType;
|
|
internal byte ICMP_Code => icmpCode;
|
|
internal ushort ICMP_CRC => icmpCRC;
|
|
internal ushort ICMP_DataLength => (ushort)(DataLength - 8);
|
|
|
|
internal byte[] GetICMPData()
|
|
{
|
|
byte[] data = new byte[ICMP_DataLength];
|
|
|
|
for (int b = 0; b < ICMP_DataLength; b++)
|
|
{
|
|
data[b] = RawData[DataOffset + 8 + b];
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
public override string ToString()
|
|
{
|
|
return "ICMP Packet Src=" + SourceIP + ", Dest=" + DestinationIP + ", Type=" + icmpType + ", Code=" + icmpCode;
|
|
}
|
|
}
|
|
|
|
internal class ICMPEchoRequest : ICMPPacket
|
|
{
|
|
protected ushort icmpID;
|
|
protected ushort icmpSequence;
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPEchoRequest"/> class.
|
|
/// </summary>
|
|
internal ICMPEchoRequest()
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPEchoRequest"/> class.
|
|
/// </summary>
|
|
/// <param name="rawData">Raw data.</param>
|
|
internal ICMPEchoRequest(byte[] rawData)
|
|
: base(rawData)
|
|
{
|
|
}
|
|
|
|
internal ICMPEchoRequest(Address source, Address dest, ushort id, ushort sequence)
|
|
: base(source, dest, 8, 0, id, sequence, 40)
|
|
{
|
|
for (int b = 8; b < ICMP_DataLength; b++)
|
|
{
|
|
RawData[DataOffset + b] = (byte)b;
|
|
}
|
|
|
|
RawData[DataOffset + 2] = 0x00;
|
|
RawData[DataOffset + 3] = 0x00;
|
|
icmpCRC = CalcICMPCRC((ushort)(ICMP_DataLength + 8));
|
|
RawData[DataOffset + 2] = (byte)((icmpCRC >> 8) & 0xFF);
|
|
RawData[DataOffset + 3] = (byte)((icmpCRC >> 0) & 0xFF);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Work around to make VMT scanner include the initFields method
|
|
/// </summary>
|
|
public new static void VMTInclude()
|
|
{
|
|
new ICMPEchoRequest();
|
|
}
|
|
|
|
protected override void initFields()
|
|
{
|
|
//Sys.Console.WriteLine("ICMPEchoRequest.initFields() called;");
|
|
base.initFields();
|
|
icmpID = (ushort)((RawData[DataOffset + 4] << 8) | RawData[DataOffset + 5]);
|
|
icmpSequence = (ushort)((RawData[DataOffset + 6] << 8) | RawData[DataOffset + 7]);
|
|
}
|
|
|
|
internal ushort ICMP_ID => icmpID;
|
|
internal ushort ICMP_Sequence => icmpSequence;
|
|
|
|
/// <summary>
|
|
/// To string.
|
|
/// </summary>
|
|
/// <returns>string value.</returns>
|
|
public override string ToString()
|
|
{
|
|
return "ICMP Echo Request Src=" + SourceIP + ", Dest=" + DestinationIP + ", ID=" + icmpID + ", Sequence=" + icmpSequence;
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// ICMPEchoReply class. See also: <seealso cref="ICMPPacket"/>.
|
|
/// </summary>
|
|
internal class ICMPEchoReply : ICMPPacket
|
|
{
|
|
protected ushort icmpID;
|
|
protected ushort icmpSequence;
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPEchoReply"/> class.
|
|
/// </summary>
|
|
internal ICMPEchoReply()
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPEchoReply"/> class.
|
|
/// </summary>
|
|
/// <param name="rawData">Raw data.</param>
|
|
internal ICMPEchoReply(byte[] rawData)
|
|
: base(rawData)
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Work around to make VMT scanner include the initFields method
|
|
/// </summary>
|
|
public new static void VMTInclude()
|
|
{
|
|
new ICMPEchoReply();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Init ICMPEchoReply fields.
|
|
/// </summary>
|
|
/// <exception cref="ArgumentException">Thrown if RawData is invalid or null.</exception>
|
|
protected override void initFields()
|
|
{
|
|
//Sys.Console.WriteLine("ICMPEchoReply.initFields() called;");
|
|
base.initFields();
|
|
icmpID = (ushort)((RawData[DataOffset + 4] << 8) | RawData[DataOffset + 5]);
|
|
icmpSequence = (ushort)((RawData[DataOffset + 6] << 8) | RawData[DataOffset + 7]);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create new inctanse of the <see cref="ICMPEchoReply"/> class.
|
|
/// </summary>
|
|
/// <param name="request">ICMP echo request.</param>
|
|
/// <exception cref="ArgumentException">Thrown if RawData is invalid or null.</exception>
|
|
internal ICMPEchoReply(ICMPEchoRequest request)
|
|
: base(request.DestinationIP, request.SourceIP, 0, 0,
|
|
request.ICMP_ID, request.ICMP_Sequence, (ushort)(request.ICMP_DataLength + 8))
|
|
{
|
|
for (int b = 0; b < ICMP_DataLength; b++)
|
|
{
|
|
RawData[DataOffset + 8 + b] = request.RawData[DataOffset + 8 + b];
|
|
}
|
|
|
|
RawData[DataOffset + 2] = 0x00;
|
|
RawData[DataOffset + 3] = 0x00;
|
|
icmpCRC = CalcICMPCRC((ushort)(ICMP_DataLength + 8));
|
|
RawData[DataOffset + 2] = (byte)((icmpCRC >> 8) & 0xFF);
|
|
RawData[DataOffset + 3] = (byte)((icmpCRC >> 0) & 0xFF);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get ICMP ID.
|
|
/// </summary>
|
|
internal UInt16 ICMP_ID => icmpID;
|
|
/// <summary>
|
|
/// Get ICMP sequence.
|
|
/// </summary>
|
|
internal UInt16 ICMP_Sequence => icmpSequence;
|
|
|
|
/// <summary>
|
|
/// To string.
|
|
/// </summary>
|
|
/// <returns>string value.</returns>
|
|
public override string ToString()
|
|
{
|
|
return "ICMP Echo Reply Src=" + SourceIP + ", Dest=" + DestinationIP + ", ID=" + icmpID + ", Sequence=" + icmpSequence;
|
|
}
|
|
}
|
|
}
|