pcap.codec.ip.Ip4 Maven / Gradle / Ivy
/** This code is licenced under the GPL version 2. */
package pcap.codec.ip;
import java.util.Arrays;
import java.util.Objects;
import pcap.codec.AbstractPacket;
import pcap.codec.Packet;
import pcap.codec.TransportLayer;
import pcap.common.annotation.Inclubating;
import pcap.common.memory.Memory;
import pcap.common.net.Inet4Address;
import pcap.common.util.Strings;
import pcap.common.util.Validate;
/**
* @see Wikipedia
* @see RFC
* @author Ardika Rommy Sanjaya
*/
@Inclubating
public class Ip4 extends Ip {
private final Header header;
private final Packet payload;
private final Builder builder;
private Ip4(final Builder builder) {
this.header = new Header(builder);
this.payloadBuffer = builder.payloadBuffer;
if (this.payloadBuffer != null
&& this.payloadBuffer.readerIndex() < this.payloadBuffer.writerIndex()) {
this.payload =
TransportLayer.valueOf(this.header.payloadType().value()).newInstance(this.payloadBuffer);
} else {
payload = null;
}
this.builder = builder;
}
public static final Ip4 newPacket(Memory buffer) {
return new Builder().build(buffer);
}
@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 final class Header extends AbstractPacketHeader {
public static final int IPV4_HEADER_LENGTH = 20;
private final byte headerLength;
private final byte diffServ;
private final byte expCon;
private final short totalLength;
private final short identification;
private final byte flags;
private final short fragmentOffset;
private final byte ttl;
private final TransportLayer protocol;
private final short checksum;
private final Inet4Address sourceAddress;
private final Inet4Address destinationAddress;
private final byte[] options;
private final Builder builder;
protected Header(final Builder builder) {
super((byte) 0x04);
this.headerLength = builder.headerLength;
this.diffServ = builder.diffServ;
this.expCon = builder.expCon;
this.totalLength = builder.totalLength;
this.identification = builder.identification;
this.flags = builder.flags;
this.fragmentOffset = builder.fragmentOffset;
this.ttl = builder.ttl;
this.protocol = builder.protocol;
this.checksum = builder.checksum;
this.sourceAddress = builder.sourceAddress;
this.destinationAddress = builder.destinationAddress;
this.options = builder.options;
this.buffer = resetIndex(builder.buffer, length());
this.builder = builder;
}
private static short calculateChecksum(Memory buffer, int headerLength, int offset) {
int index = offset;
int accumulation = 0;
for (int i = 0; i < headerLength * 2; ++i) {
if (i == 5) {
accumulation += 0;
} else {
accumulation += 0xFFFF & buffer.getShort(index);
}
index += 2;
}
accumulation = (accumulation >> 16 & 0xFFFF) + (accumulation & 0xFFFF);
return (short) (~accumulation & 0xFFFF);
}
/**
* Header length.
*
* @return return header length.
*/
public int headerLength() {
return headerLength & 0xF;
}
/**
* Diff serv.
*
* @return returns diff serv.
*/
public int diffServ() {
return diffServ & 0x3F;
}
/**
* Exp con.
*
* @return returns exp con.
*/
public int expCon() {
return expCon & 0x3;
}
/**
* Total length.
*
* @return returns total length.
*/
public int totalLength() {
return totalLength & 0xFFFF;
}
/**
* Identification.
*
* @return returns identification.
*/
public int identification() {
return identification & 0xFFFF;
}
/**
* Flags.
*
* @return returns flags.
*/
public int flags() {
return flags & 0x7;
}
/**
* Fragment offset.
*
* @return returns frament offset.
*/
public int fragmentOffset() {
return fragmentOffset & 0x1FFF;
}
/**
* TTL.
*
* @return returns TTL.
*/
public int ttl() {
return ttl & 0xFF;
}
/**
* Next protocol type.
*
* @return returns {@link TransportLayer}.
*/
public TransportLayer protocol() {
return protocol;
}
/**
* Checksum.
*
* @return returns checksum.
*/
public int checksum() {
return checksum & 0xFFFF;
}
/**
* Source address.
*
* @return returns source address.
*/
public Inet4Address sourceAddress() {
return sourceAddress;
}
/**
* Destination address.
*
* @return returns destination address.
*/
public Inet4Address destinationAddress() {
return destinationAddress;
}
/**
* Get options.
*
* @return returns options.
*/
public byte[] options() {
byte[] options = new byte[this.options.length];
System.arraycopy(this.options, 0, options, 0, options.length);
return options;
}
/**
* Check whether checksum is valid.
*
* @return returns true if checksum is valid, false otherwise.
*/
public boolean isValidChecksum() {
return checksum == calculateChecksum(buffer, headerLength, 0);
}
@Override
public TransportLayer payloadType() {
return protocol;
}
@Override
public int length() {
return IPV4_HEADER_LENGTH + ((options == null) ? 0 : options.length);
}
@Override
public Memory buffer() {
if (buffer == null) {
buffer = ALLOCATOR.allocate(length());
buffer.writeByte(((super.version & 0xF) << 4 | headerLength & 0xF));
buffer.writeByte((((diffServ << 2) & 0x3F) | expCon & 0x3));
buffer.writeShort(totalLength);
buffer.writeShort(identification);
buffer.writeShort((flags & 0x7) << 13 | fragmentOffset & 0x1FFF);
buffer.writeByte(ttl);
buffer.writeByte(protocol.value());
buffer.writeShort(checksum & 0xFFFF);
buffer.writeBytes(sourceAddress.address());
buffer.writeBytes(destinationAddress.address());
if (options != null && options.length > 0 && headerLength > 5) {
buffer.writeBytes(options);
}
}
return buffer;
}
@Override
public Builder builder() {
return builder;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Header header = (Header) o;
return headerLength == header.headerLength
&& diffServ == header.diffServ
&& expCon == header.expCon
&& totalLength == header.totalLength
&& identification == header.identification
&& flags == header.flags
&& fragmentOffset == header.fragmentOffset
&& ttl == header.ttl
&& checksum == header.checksum
&& protocol.equals(header.protocol)
&& sourceAddress.equals(header.sourceAddress)
&& destinationAddress.equals(header.destinationAddress)
&& Arrays.equals(options, header.options);
}
@Override
public int hashCode() {
int result =
Objects.hash(
headerLength,
diffServ,
expCon,
totalLength,
identification,
flags,
fragmentOffset,
ttl,
protocol,
checksum,
sourceAddress,
destinationAddress);
result = 31 * result + Arrays.hashCode(options);
return result;
}
@Override
public String toString() {
return Strings.toStringBuilder(this)
.add("version", version())
.add("headerLength", headerLength() & 0xF)
.add("diffServ", diffServ() & 0x3F)
.add("expCon", expCon() & 0x3)
.add("totalLength", totalLength() & 0xFFFF)
.add("identification", identification() & 0xFFFF)
.add("flags", flags() & 0x7)
.add("fragmentOffset", fragmentOffset() & 0x1FFF)
.add("ttl", ttl() & 0xFFFF)
.add("protocol", protocol())
.add("checksum", checksum() & 0xFFFF)
.add("sourceAddress", sourceAddress())
.add("destinationAddress", destinationAddress())
.add("options", Arrays.toString(options()))
.add("validChecksum", isValidChecksum())
.toString();
}
}
public static final class Builder extends AbstractPaketBuilder {
private byte headerLength;
private byte diffServ;
private byte expCon;
private short totalLength;
private short identification;
private byte flags;
private short fragmentOffset;
private byte ttl;
private TransportLayer protocol = TransportLayer.TCP;
private short checksum;
private Inet4Address sourceAddress = Inet4Address.ZERO;
private Inet4Address destinationAddress = Inet4Address.ZERO;
private byte[] options = new byte[0];
private Memory buffer;
private Memory payloadBuffer;
/** A helper field. */
private boolean calculateChecksum;
/**
* Header length.
*
* @param headerLength header length.
* @return returns this {@link Builder}.
*/
public Builder headerLength(final int headerLength) {
this.headerLength = (byte) (headerLength & 0xF);
return this;
}
/**
* Diff serv.
*
* @param diffServ diff serv.
* @return returns this {@link Builder}.
*/
public Builder diffServ(final int diffServ) {
this.diffServ = (byte) (diffServ & 0x3F);
return this;
}
/**
* Exp con.
*
* @param expCon exp con.
* @return returns this {@link Builder}.
*/
public Builder expCon(final int expCon) {
this.expCon = (byte) (expCon & 0x3);
return this;
}
/**
* Total length.
*
* @param totalLength total length.
* @return returns this {@link Builder}.
*/
public Builder totalLength(final int totalLength) {
this.totalLength = (short) (totalLength & 0xFFFF);
return this;
}
/**
* Identification.
*
* @param identification identification.
* @return returns {@link Builder}.
*/
public Builder identification(final int identification) {
this.identification = (short) (identification & 0xFFFF);
return this;
}
/**
* Flags.
*
* @param flags flags.
* @return returns {@link Builder}.
*/
public Builder flags(final int flags) {
this.flags = (byte) (flags & 0x7);
return this;
}
/**
* Frament offset.
*
* @param fragmentOffset fragment offset.
* @return returns this {@link Builder}.
*/
public Builder fragmentOffset(final int fragmentOffset) {
this.fragmentOffset = (short) (fragmentOffset & 0x1FFF);
return this;
}
/**
* TTL.
*
* @param ttl ttl.
* @return returns this {@link Builder}.
*/
public Builder ttl(final int ttl) {
this.ttl = (byte) (ttl & 0xFF);
return this;
}
/**
* Protocol.
*
* @param protocol protocol.
* @return returns this {@link Builder}.
*/
public Builder protocol(TransportLayer protocol) {
this.protocol = protocol;
return this;
}
/**
* Checksum.
*
* @param checksum checksum.
* @return returns {@link Builder}.
*/
public Builder checksum(final int checksum) {
this.checksum = (short) (checksum & 0xFFFF);
return this;
}
/**
* Source address.
*
* @param sourceAddress source address.
* @return returns this {@link Builder}.
*/
public Builder sourceAddress(final Inet4Address sourceAddress) {
this.sourceAddress = sourceAddress;
return this;
}
/**
* Destination address.
*
* @param destinationAddress destination address.
* @return returns this {@link Builder}.
*/
public Builder destinationAddress(final Inet4Address destinationAddress) {
this.destinationAddress = destinationAddress;
return this;
}
/**
* Calculate checksum.
*
* @param calculateChecksum true for calculating checksum, false otherwise.
* @return returns this {@link Builder}.
*/
public Builder calculateChecksum(boolean calculateChecksum) {
this.calculateChecksum = calculateChecksum;
return this;
}
/**
* Add options.
*
* @param options options.
* @return returns this {@link Builder} object.
*/
public Builder options(final byte[] options) {
this.options = new byte[options.length];
System.arraycopy(options, 0, this.options, 0, this.options.length);
return this;
}
@Override
public Builder payload(AbstractPacket packet) {
this.payloadBuffer = packet.buffer();
return this;
}
@Override
public Ip4 build() {
if (calculateChecksum) {
if (buffer == null) {
buffer = new Ip4(this).buffer();
}
checksum(Ip4.Header.calculateChecksum(buffer, headerLength, 0));
buffer.setShort(10, this.checksum);
}
return new Ip4(this);
}
@Override
public Ip4 build(final Memory buffer) {
resetIndex(buffer);
this.headerLength = (byte) (buffer.readByte() & 0xF);
byte tmp = buffer.readByte();
this.diffServ = (byte) ((tmp >> 2) & 0x3F);
this.expCon = (byte) (tmp & 0x3);
this.totalLength = buffer.readShort();
this.identification = buffer.readShort();
short sscratch = buffer.readShort();
this.flags = (byte) (sscratch >> 13 & 0x7);
this.fragmentOffset = (short) (sscratch & 0x1FFF);
this.ttl = buffer.readByte();
this.protocol = TransportLayer.valueOf(buffer.readByte());
this.checksum = (short) (buffer.readShort() & 0xFFFF);
byte[] ipv4Buffer;
ipv4Buffer = new byte[Inet4Address.IPV4_ADDRESS_LENGTH];
buffer.readBytes(ipv4Buffer);
this.sourceAddress = Inet4Address.valueOf(ipv4Buffer);
ipv4Buffer = new byte[Inet4Address.IPV4_ADDRESS_LENGTH];
buffer.readBytes(ipv4Buffer);
this.destinationAddress = Inet4Address.valueOf(ipv4Buffer);
if (headerLength > 5) {
int optionsLength = (headerLength - 5) * 4;
this.options = new byte[optionsLength];
buffer.readBytes(options);
} else {
options = new byte[0];
}
this.buffer = buffer;
this.payloadBuffer = buffer.slice();
return new Ip4(this);
}
@Override
public Builder reset() {
return reset(readerIndex, Header.IPV4_HEADER_LENGTH);
}
@Override
public Builder reset(int offset, int length) {
if (buffer != null) {
Validate.notIllegalArgument(offset + length <= buffer.capacity());
Validate.notIllegalArgument((headerLength & 0xF) >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument((diffServ & 0x3F) >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument((expCon & 0x3) >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument((totalLength & 0xFFFF) >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument((identification & 0xFFFF) >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument((flags & 0x7) >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument((fragmentOffset & 0x1FFF) >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument((ttl & 0xFF) >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument(protocol != null, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument(checksum >= 0, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument(sourceAddress != null, ILLEGAL_HEADER_EXCEPTION);
Validate.notIllegalArgument(destinationAddress != null, ILLEGAL_HEADER_EXCEPTION);
int index = offset;
buffer.setByte(index, (((0x4 & 0xF) << 4) | this.headerLength & 0xF));
index += 1;
int tmp = ((diffServ << 2) & 0x3F) | (expCon & 0x3);
buffer.setByte(index, tmp);
index += 1;
buffer.setShort(index, totalLength);
index += 2;
buffer.setShort(index, identification);
index += 2;
buffer.setShort(index, (flags & 0x7) << 13 | fragmentOffset & 0x1FFF);
index += 2;
buffer.setByte(index, ttl);
index += 1;
buffer.setByte(index, protocol.value());
index += 1;
buffer.setShort(index, checksum);
index += 2;
buffer.setBytes(index, sourceAddress.address());
index += Inet4Address.IPV4_ADDRESS_LENGTH;
buffer.setBytes(index, destinationAddress.address());
index += Inet4Address.IPV4_ADDRESS_LENGTH;
buffer.setBytes(index, options);
}
return this;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy