All Downloads are FREE. Search and download functionalities are using the official Maven repository.

pcap.codec.icmp.Icmp4 Maven / Gradle / Ivy

There is a newer version: 1.5.1
Show newest version
/** This code is licenced under the GPL version 2. */
package pcap.codec.icmp;

import pcap.codec.AbstractPacket;
import pcap.codec.Packet;
import pcap.codec.icmp.icmp4.*;
import pcap.common.annotation.Inclubating;
import pcap.common.memory.Memory;
import pcap.common.util.NamedNumber;
import pcap.common.util.Strings;
import pcap.common.util.Validate;

/** @author Ardika Rommy Sanjaya */
@Inclubating
public class Icmp4 extends AbstractPacket {

  static {
    try {
      Class.forName(Icmp4DestinationUnreachable.class.getName());
      Class.forName(Icmp4EchoReply.class.getName());
      Class.forName(Icmp4EchoRequest.class.getName());
      Class.forName(Icmp4ParameterProblem.class.getName());
      Class.forName(Icmp4RedirectMessage.class.getName());
      Class.forName(Icmp4RouterAdvertisement.class.getName());
      Class.forName(Icmp4RouterSolicitation.class.getName());
      Class.forName(Icmp4TimeExceeded.class.getName());
      Class.forName(Icmp4Timestamp.class.getName());
      Class.forName(Icmp4TimestampReply.class.getName());
    } catch (ClassNotFoundException e) {
      //
    }
  }

  private final Header header;
  private final Packet payload;
  private final Builder builder;

  private Icmp4(Builder builder) {
    this.header = new Header(builder);
    this.payloadBuffer = builder.payloadBuffer;
    if (this.payloadBuffer != null
        && this.payloadBuffer.readerIndex() < this.payloadBuffer.writerIndex()) {
      this.payload =
          Icmp.IcmpTypeAndCode.valueOf(this.header.payloadType().value().byteValue())
              .newInstance(this.payloadBuffer);
    } else {
      this.payload = null;
    }
    this.builder = builder;
  }

  @Override
  public Header header() {
    return header;
  }

  @Override
  public Packet payload() {
    return payload;
  }

  @Override
  public Builder builder() {
    return builder;
  }

  @Override
  public Memory buffer() {
    return header().buffer();
  }

  @Override
  public String toString() {
    return Strings.toStringBuilder(this)
        .add("header", header)
        .add("payload", payload != null ? payload.getClass().getSimpleName() : "(None)")
        .toString();
  }

  public static class Header extends Icmp.AbstractPacketHeader {

    private final Builder builder;

    private Header(Builder builder) {
      typeAndCode = builder.typeAndCode;
      checksum = builder.checksum;
      buffer = resetIndex(builder.buffer, length());
      this.builder = builder;
    }

    @SuppressWarnings("TypeParameterUnusedInFormals")
    @Override
    public  T payloadType() {
      return (T) typeAndCode;
    }

    @Override
    public Builder builder() {
      return builder;
    }

    @Override
    public String toString() {
      return Strings.toStringBuilder(this)
          .add("typeAndCode", typeAndCode)
          .add("checksum", checksum)
          .toString();
    }
  }

  public static class Builder extends Icmp.AbstractPacketBuilder {

    private Memory buffer;
    private Memory payloadBuffer;

    private boolean calculateChecksum;

    private Builder calculateChecksum(boolean caculateChecksum) {
      this.calculateChecksum = caculateChecksum;
      return this;
    }

    @Override
    public Packet build() {
      return new Icmp4(this);
    }

    @Override
    public Packet build(Memory buffer) {
      resetIndex(buffer);
      byte type = buffer.readByte();
      byte code = buffer.readByte();
      super.typeAndCode = Icmp.findIcmpTypeAndCode(type, code, Icmp.IcmpTypeAndCode.ICMP4_REGISTRY);
      super.checksum = buffer.readShort();
      if (calculateChecksum) {
        int index = 0;
        int accumulation = 0;
        for (int i = 0; i < buffer.capacity() / 2; ++i) {
          accumulation += 0xffff & buffer.getShort(index);
          index += 2;
        }
        // pad to an even number of shorts
        if (buffer.capacity() % 2 > 0) {
          accumulation += (buffer.getByte(index) & 0xff) << 8;
          index++;
        }
        accumulation = (accumulation >> 16 & 0xffff) + (accumulation & 0xffff);
        short checksum = (short) (~accumulation & 0xffff);
        super.checksum = buffer.readShort();
        if (checksum != super.checksum) {
          // invalid checksum
          this.checksum = 0;
        }
      }
      this.buffer = buffer;
      this.payloadBuffer = buffer.slice();
      return new Icmp4(this);
    }

    @Override
    public Builder reset() {
      return reset(readerIndex, Header.ICMP_HEADER_LENGTH);
    }

    @Override
    public Builder reset(int offset, int length) {
      if (buffer != null) {
        Validate.notIllegalArgument(offset + length <= buffer.capacity());
        Validate.notIllegalArgument(typeAndCode != null, ILLEGAL_HEADER_EXCEPTION);
        Validate.notIllegalArgument(checksum >= 0, ILLEGAL_HEADER_EXCEPTION);
        int index = offset;
        buffer.setByte(index, typeAndCode.type());
        index += 1;
        buffer.setByte(index, typeAndCode.code());
        index += 1;
        buffer.setShort(index, checksum);
      }
      return this;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy