org.pcap4j.packet.Dot11ManagementPacket Maven / Gradle / Ivy
/*_##########################################################################
_##
_## Copyright (C) 2016 Pcap4J.org
_##
_##########################################################################
*/
package org.pcap4j.packet;
import static org.pcap4j.util.ByteArrays.*;
import java.nio.ByteOrder;
import java.util.ArrayList;
import java.util.List;
import org.pcap4j.util.ByteArrays;
import org.pcap4j.util.MacAddress;
/**
* IEEE802.11 management frame
*
* @see IEEE802.11
* @author Kaito Yamada
* @since pcap4j 1.7.0
*/
public abstract class Dot11ManagementPacket extends AbstractPacket {
/** */
private static final long serialVersionUID = -3972573868672848666L;
private final Integer fcs;
/**
* @param rawData rawData
* @param offset offset
* @param length length
* @param headerLen headerLen
*/
protected Dot11ManagementPacket(byte[] rawData, int offset, int length, int headerLen) {
int remainingLen = length - headerLen;
if (remainingLen >= 4) {
this.fcs = ByteArrays.getInt(rawData, offset + headerLen, ByteOrder.LITTLE_ENDIAN);
} else {
this.fcs = null;
}
}
/**
* @param builder builder
* @param header header
*/
protected Dot11ManagementPacket(Builder builder, Dot11ManagementHeader header) {
if (builder.correctChecksumAtBuild) {
this.fcs = ByteArrays.calcCrc32Checksum(header.getRawData());
} else {
this.fcs = builder.fcs;
}
}
@Override
public abstract Dot11ManagementHeader getHeader();
/** @return fcs. May be null. */
public Integer getFcs() {
return fcs;
}
@Override
protected int calcLength() {
int length = super.calcLength();
if (fcs != null) {
length += 4;
}
return length;
}
@Override
protected byte[] buildRawData() {
byte[] rawData = super.buildRawData();
if (fcs != null) {
System.arraycopy(
ByteArrays.toByteArray(fcs, ByteOrder.LITTLE_ENDIAN), 0, rawData, rawData.length - 4, 4);
}
return rawData;
}
@Override
protected String buildString() {
StringBuilder sb = new StringBuilder();
sb.append(getHeader().toString());
if (fcs != null) {
String ls = System.getProperty("line.separator");
sb.append("[IEEE802.11 Management Packet FCS]")
.append(ls)
.append(" FCS: 0x")
.append(ByteArrays.toHexString(fcs, ""))
.append(ls);
}
return sb.toString();
}
@Override
public abstract Builder getBuilder();
/** @return true if this FCS is present and valid; false otherwise. */
public boolean hasValidFcs() {
if (fcs == null) {
return false;
}
return ByteArrays.calcCrc32Checksum(getHeader().getRawData()) == fcs.intValue();
}
/**
* @author Kaito Yamada
* @since pcap4j 1.7.0
*/
public abstract static class Builder extends AbstractBuilder
implements ChecksumBuilder {
private Dot11FrameControl frameControl;
private short duration;
private MacAddress address1;
private MacAddress address2;
private MacAddress address3;
private Dot11SequenceControl sequenceControl;
private Dot11HtControl htControl;
private Integer fcs;
private boolean correctChecksumAtBuild;
/** */
public Builder() {}
/** @param packet packet */
protected Builder(Dot11ManagementPacket packet) {
this.frameControl = packet.getHeader().frameControl;
this.duration = packet.getHeader().duration;
this.address1 = packet.getHeader().address1;
this.address2 = packet.getHeader().address2;
this.address3 = packet.getHeader().address3;
this.sequenceControl = packet.getHeader().sequenceControl;
this.htControl = packet.getHeader().htControl;
this.fcs = packet.fcs;
}
/**
* @param frameControl frameControl
* @return this Builder object for method chaining.
*/
public Builder frameControl(Dot11FrameControl frameControl) {
this.frameControl = frameControl;
return this;
}
/**
* @param duration duration
* @return this Builder object for method chaining.
*/
public Builder duration(short duration) {
this.duration = duration;
return this;
}
/**
* @param address1 address1
* @return this Builder object for method chaining.
*/
public Builder address1(MacAddress address1) {
this.address1 = address1;
return this;
}
/**
* @param address2 address2
* @return this Builder object for method chaining.
*/
public Builder address2(MacAddress address2) {
this.address2 = address2;
return this;
}
/**
* @param address3 address3
* @return this Builder object for method chaining.
*/
public Builder address3(MacAddress address3) {
this.address3 = address3;
return this;
}
/**
* @param sequenceControl sequenceControl
* @return this Builder object for method chaining.
*/
public Builder sequenceControl(Dot11SequenceControl sequenceControl) {
this.sequenceControl = sequenceControl;
return this;
}
/**
* @param htControl htControl
* @return this Builder object for method chaining.
*/
public Builder htControl(Dot11HtControl htControl) {
this.htControl = htControl;
return this;
}
/**
* @param fcs fcs
* @return this Builder object for method chaining.
*/
public Builder fcs(Integer fcs) {
this.fcs = fcs;
return this;
}
@Override
public Builder correctChecksumAtBuild(boolean correctChecksumAtBuild) {
this.correctChecksumAtBuild = correctChecksumAtBuild;
return this;
}
/** Call me at the top of {@link #build()}. */
protected void checkForNull() {
if (frameControl == null || address1 == null || address2 == null || address3 == null) {
StringBuilder sb = new StringBuilder();
sb.append("frameControl: ")
.append(frameControl)
.append(" address1: ")
.append(address1)
.append(" address2: ")
.append(address2)
.append(" address3: ")
.append(address3);
throw new NullPointerException(sb.toString());
}
}
@Override
public abstract Dot11ManagementPacket build();
}
/**
* Header of IEEE802.11 management frame
*
*
* 0 15
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Frame Control |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Duration |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | |
* | Address1 |
* | |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | |
* | Address2 |
* | |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | |
* | Address3 |
* | |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | Sequence Control |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
* | HT Control |
* | |
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
*
*
* @see IEEE802.11
* @author Kaito Yamada
* @since pcap4j 1.7.0
*/
public abstract static class Dot11ManagementHeader extends AbstractHeader {
/** */
private static final long serialVersionUID = 615170086003609919L;
private static final int FRAME_CONTROL_OFFSET = 0;
private static final int FRAME_CONTROL_SIZE = SHORT_SIZE_IN_BYTES;
private static final int DURATION_OFFSET = FRAME_CONTROL_OFFSET + FRAME_CONTROL_SIZE;
private static final int DURATION_SIZE = SHORT_SIZE_IN_BYTES;
private static final int ADDRESS1_OFFSET = DURATION_OFFSET + DURATION_SIZE;
private static final int ADDRESS1_SIZE = MacAddress.SIZE_IN_BYTES;
private static final int ADDRESS2_OFFSET = ADDRESS1_OFFSET + ADDRESS1_SIZE;
private static final int ADDRESS2_SIZE = MacAddress.SIZE_IN_BYTES;
private static final int ADDRESS3_OFFSET = ADDRESS2_OFFSET + ADDRESS2_SIZE;
private static final int ADDRESS3_SIZE = MacAddress.SIZE_IN_BYTES;
private static final int SEQUENCE_CONTROL_OFFSET = ADDRESS3_OFFSET + ADDRESS3_SIZE;
private static final int SEQUENCE_CONTROL_SIZE = SHORT_SIZE_IN_BYTES;
private static final int HT_CONTROL_OFFSET = SEQUENCE_CONTROL_OFFSET + SEQUENCE_CONTROL_SIZE;
private static final int HT_CONTROL_SIZE = INT_SIZE_IN_BYTES;
private static final int DOT11_HEADER_MIN_SIZE = HT_CONTROL_OFFSET;
private final Dot11FrameControl frameControl;
private final short duration;
private final MacAddress address1;
private final MacAddress address2;
private final MacAddress address3;
private final Dot11SequenceControl sequenceControl;
private final Dot11HtControl htControl;
/**
* @param rawData rawData
* @param offset offset
* @param length length
* @throws IllegalRawDataException if parsing the raw data fails.
*/
protected Dot11ManagementHeader(byte[] rawData, int offset, int length)
throws IllegalRawDataException {
if (length < DOT11_HEADER_MIN_SIZE) {
StringBuilder sb = new StringBuilder(200);
sb.append("The data is too short to build a Dot11ManagementHeader (")
.append(DOT11_HEADER_MIN_SIZE)
.append(" bytes). data: ")
.append(ByteArrays.toHexString(rawData, " "))
.append(", offset: ")
.append(offset)
.append(", length: ")
.append(length);
throw new IllegalRawDataException(sb.toString());
}
this.frameControl =
Dot11FrameControl.newInstance(rawData, offset + FRAME_CONTROL_OFFSET, length);
this.duration =
ByteArrays.getShort(rawData, offset + DURATION_OFFSET, ByteOrder.LITTLE_ENDIAN);
this.address1 = ByteArrays.getMacAddress(rawData, offset + ADDRESS1_OFFSET);
this.address2 = ByteArrays.getMacAddress(rawData, offset + ADDRESS2_OFFSET);
this.address3 = ByteArrays.getMacAddress(rawData, offset + ADDRESS3_OFFSET);
this.sequenceControl =
Dot11SequenceControl.newInstance(
rawData, offset + SEQUENCE_CONTROL_OFFSET, length - SEQUENCE_CONTROL_OFFSET);
if (frameControl.isOrder()) {
if (length < DOT11_HEADER_MIN_SIZE + HT_CONTROL_SIZE) {
StringBuilder sb = new StringBuilder(200);
sb.append("The data is too short to build a Dot11ManagementHeader (")
.append(DOT11_HEADER_MIN_SIZE + HT_CONTROL_SIZE)
.append(" bytes). data: ")
.append(ByteArrays.toHexString(rawData, " "))
.append(", offset: ")
.append(offset)
.append(", length: ")
.append(length);
throw new IllegalRawDataException(sb.toString());
}
this.htControl =
Dot11HtControl.newInstance(
rawData, offset + HT_CONTROL_OFFSET, length - HT_CONTROL_OFFSET);
} else {
this.htControl = null;
}
}
/** @param builder builder */
protected Dot11ManagementHeader(Builder builder) {
this.frameControl = builder.frameControl;
this.duration = builder.duration;
this.address1 = builder.address1;
this.address2 = builder.address2;
this.address3 = builder.address3;
this.sequenceControl = builder.sequenceControl;
this.htControl = builder.htControl;
}
/** @return frameControl */
public Dot11FrameControl getFrameControl() {
return frameControl;
}
/** @return duration */
public short getDuration() {
return duration;
}
/** @return duration */
public int getDurationAsInt() {
return duration & 0xFFFF;
}
/** @return address1 */
public MacAddress getAddress1() {
return address1;
}
/** @return address2 */
public MacAddress getAddress2() {
return address2;
}
/** @return address3 */
public MacAddress getAddress3() {
return address3;
}
/** @return sequenceControl */
public Dot11SequenceControl getSequenceControl() {
return sequenceControl;
}
/** @return htControl. May be null. */
public Dot11HtControl getHtControl() {
return htControl;
}
@Override
protected List getRawFields() {
List rawFields = new ArrayList();
rawFields.add(frameControl.getRawData());
rawFields.add(ByteArrays.toByteArray(duration, ByteOrder.LITTLE_ENDIAN));
rawFields.add(address1.getAddress());
rawFields.add(address2.getAddress());
rawFields.add(address3.getAddress());
rawFields.add(sequenceControl.getRawData());
if (htControl != null) {
rawFields.add(htControl.getRawData());
}
return rawFields;
}
@Override
public int calcLength() {
if (htControl != null) {
return DOT11_HEADER_MIN_SIZE + HT_CONTROL_SIZE;
} else {
return DOT11_HEADER_MIN_SIZE;
}
}
/** @return the header name. */
protected abstract String getHeaderName();
@Override
protected String buildString() {
StringBuilder sb = new StringBuilder();
String ls = System.getProperty("line.separator");
sb.append("[")
.append(getHeaderName())
.append(" (")
.append(length())
.append(" bytes)]")
.append(ls);
sb.append(" Frame Control:").append(ls).append(frameControl.toString(" "));
sb.append(" Duration: ").append(getDurationAsInt()).append(ls);
sb.append(" Address1: ").append(address1).append(ls);
sb.append(" Address2: ").append(address2).append(ls);
sb.append(" Address3: ").append(address3).append(ls);
sb.append(" Sequence Control: ").append(sequenceControl).append(ls);
if (htControl != null) {
sb.append(" HT Control:").append(ls).append(htControl.toString(" "));
}
return sb.toString();
}
@Override
protected int calcHashCode() {
final int prime = 31;
int result = 17;
result = prime * result + address1.hashCode();
result = prime * result + address2.hashCode();
result = prime * result + address3.hashCode();
result = prime * result + duration;
result = prime * result + frameControl.hashCode();
result = prime * result + (htControl != null ? htControl.hashCode() : 0);
result = prime * result + sequenceControl.hashCode();
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj) return true;
if (getClass() != obj.getClass()) return false;
Dot11ManagementHeader other = (Dot11ManagementHeader) obj;
if (!address1.equals(other.address1)) return false;
if (!address2.equals(other.address2)) return false;
if (!address3.equals(other.address3)) return false;
if (duration != other.duration) return false;
if (!frameControl.equals(other.frameControl)) return false;
if (htControl == null) {
if (other.htControl != null) {
return false;
}
} else if (!htControl.equals(other.htControl)) return false;
if (!sequenceControl.equals(other.sequenceControl)) return false;
return true;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy