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

org.pcap4j.packet.Dot11LinkAdaptationControl Maven / Gradle / Ivy

There is a newer version: 2.0.0-alpha.6
Show newest version
/*_##########################################################################
  _##
  _##  Copyright (C) 2016  Pcap4J.org
  _##
  _##########################################################################
*/

package org.pcap4j.packet;

import java.io.Serializable;

import org.pcap4j.util.ByteArrays;

/**
 * Link Adaptation Control field of an IEEE802.11 frame.
 * 
 * {@code
 *      0        1        2        3        4        5        6        7
 * +--------+--------+--------+--------+--------+--------+--------+--------+
 * |VHT_MFB |  TRQ   |                MAI                |       MFSI      |
 * +--------+--------+--------+--------+--------+--------+--------+--------+
 * |  MFSI  |                           MFB/ASELC                          |
 * +--------+--------+--------+--------+--------+--------+--------+--------+
 * }
 * 
* * @see IEEE802.11 * @author Kaito Yamada * @since pcap4j 1.7.0 */ public final class Dot11LinkAdaptationControl implements Serializable { /** * */ private static final long serialVersionUID = 7735461000002622072L; /** * ASELI */ public static final byte ASELI = 14; private final boolean vhtMfb; private final boolean trq; private final boolean aseli; private final Mai mai; private final byte mfsi; private final Byte mfb; private final Aselc aselc; /** * A static factory method. * This method validates the arguments by {@link ByteArrays#validateBounds(byte[], int, int)}, * which may throw exceptions undocumented here. * * @param rawData rawData * @param offset offset * @param length length * @return a new Dot11LinkAdaptationControl object. * @throws IllegalRawDataException if parsing the raw data fails. */ public static Dot11LinkAdaptationControl newInstance( byte[] rawData, int offset, int length ) throws IllegalRawDataException { ByteArrays.validateBounds(rawData, offset, length); return new Dot11LinkAdaptationControl(rawData, offset, length); } private Dot11LinkAdaptationControl( byte[] rawData, int offset, int length ) throws IllegalRawDataException { if (length < 2) { StringBuilder sb = new StringBuilder(200); sb.append("The data is too short to build a Dot11LinkAdaptationControl (") .append(2) .append(" bytes). data: ") .append(ByteArrays.toHexString(rawData, " ")) .append(", offset: ") .append(offset) .append(", length: ") .append(length); throw new IllegalRawDataException(sb.toString()); } byte first = rawData[offset]; byte second = rawData[offset + 1]; this.vhtMfb = (first & 0x01) != 0; this.trq = (first & 0x02) != 0; this.aseli = ((first >> 2) & 0x0F) == ASELI; if (aseli) { this.mai = null; } else { this.mai = new Mai((first & 0x04) != 0, (byte) ((first >> 3) & 0x07)); } this.mfsi = (byte) (((first >> 6) & 0x03) | ((second & 0x01) << 2)); byte mfbAselc = (byte) ((second >> 1) & 0x7F); this.mfb = mfbAselc; this.aselc = new Aselc(mfbAselc); } private Dot11LinkAdaptationControl(Builder builder) { if (builder == null) { throw new NullPointerException("builder is null."); } if ((builder.maiOrAseli & 0xF0) != 0) { throw new IllegalArgumentException( "(builder.maiOrAseli & 0xF0) must be zero. builder.maiOrAseli: " + builder.maiOrAseli ); } if ((builder.mfsi & 0xF8) != 0) { throw new IllegalArgumentException( "(builder.mfsi & 0xF8) must be zero. builder.mfsi: " + builder.mfsi ); } if ((builder.mfbOrAselc & 0x80) != 0) { throw new IllegalArgumentException( "(builder.mfbOrAselc & 0x80) must be zero. builder.mfbOrAselc: " + builder.mfbOrAselc ); } this.vhtMfb = builder.vhtMfb; this.trq = builder.trq; this.aseli = builder.maiOrAseli == ASELI; if (aseli) { this.mai = null; } else { this.mai = new Mai(builder.maiOrAseli); } this.mfsi = builder.mfsi; this.mfb = builder.mfbOrAselc; this.aselc = new Aselc(builder.mfbOrAselc); } /** * @return true if the VHT_MFB field is set to 1; false otherwise. */ public boolean isVhtMfb() { return vhtMfb; } /** * @return true if the TRQ field is set to 1; false otherwise. */ public boolean isTrq() { return trq; } /** * @return true if the MAI field is set to 14 (ASELI); false otherwise. * @see #ASELI */ public boolean isAselIndicated() { return aseli; } /** * @return a {@link Mai} object if {@link #isAselIndicated() isAselIndicated} returns false; * otherwise null. */ public Mai getMai() { return mai; } /** * @return mfsi */ public byte getMfsi() { return mfsi; } /** * @return mfsi */ public int getMfsiAsInt() { return mfsi; } /** * @return the value of MFB if {@link #isAselIndicated() isAselIndicated} returns false; * otherwise null. */ public Byte getMfb() { if (aseli) { return null; } else { return mfb; } } /** * @return the value of MFB if {@link #isAselIndicated() isAselIndicated} returns false; * otherwise null. */ public Integer getMfbAsInteger() { if (aseli) { return null; } else { return (int) mfb; } } /** * @return an {@link Aselc} object if {@link #isAselIndicated() isAselIndicated} returns true; * otherwise null. */ public Aselc getAselc() { if (aseli) { return aselc; } else { return null; } } /** * * @return a new Builder object populated with this object's fields. */ public Builder getBuilder() { return new Builder(this); } /** * @return the raw data. */ public byte[] getRawData() { byte[] data = new byte[2]; byte maiOrAseli = aseli ? ASELI : mai.getRawData(); data[0] = (byte) ((mfsi << 6) | maiOrAseli << 2); if (trq) { data[0] |= 0x02; } if (vhtMfb) { data[0] |= 0x01; } data[1] = (byte) (mfb << 1); if ((mfsi & 0x04) != 0) { data[1] |= 0x01; } return data; } /** * @return length */ public int length() { return 2; } @Override public String toString() { StringBuilder sb = new StringBuilder(250); sb.append("[VHT_MFB: ") .append(vhtMfb) .append(", TRQ: ") .append(trq) .append(", ASELI: ") .append(aseli); if (!aseli) { sb.append(", MAI: ") .append(mai); } sb.append(", MFSI: ") .append(mfsi); if (aseli) { sb.append(", ASELC: ") .append(aselc); } else { sb.append(", MFB: ") .append(mfb); } sb.append("]"); return sb.toString(); } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((mai == null) ? 0 : mai.hashCode()); result = prime * result + mfb.hashCode(); result = prime * result + mfsi; result = prime * result + (aseli ? 1231 : 1237); result = prime * result + (vhtMfb ? 1231 : 1237); result = prime * result + (trq ? 1231 : 1237); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; Dot11LinkAdaptationControl other = (Dot11LinkAdaptationControl) obj; if (mai == null) { if (other.mai != null) return false; } else if (!mai.equals(other.mai)) return false; if (!mfb.equals(other.mfb)) return false; if (mfsi != other.mfsi) return false; if (aseli != other.aseli) return false; if (vhtMfb != other.vhtMfb) return false; if (trq != other.trq) return false; return true; } /** * MAI subfield * *
   *   0   1   2   3
   * +---+---+---+---+
   * |MRQ|    MSI    |
   * +---+---+---+---+
   * 
* * @author Kaito Yamada * @since pcap4j 1.7.0 */ public static final class Mai implements Serializable { /** * */ private static final long serialVersionUID = -7417614720576047794L; private final boolean mrq; private final byte msi; /** * @param mrq mrq * @param msi msi */ public Mai(boolean mrq, byte msi) { if (msi < 0 || msi > 6) { throw new IllegalArgumentException("msi must be between 0 and 6 but is actually: " + msi); } this.mrq = mrq; this.msi = msi; } /** * @param rawData the raw data which the MRQ is encoded at the LSB. */ public Mai(byte rawData) { this.mrq = (rawData & 0x01) != 0; this.msi = (byte) ((rawData >> 1) & 0x07); } /** * @return mrq */ public boolean isMrq() { return mrq; } /** * @return msi */ public byte getMsi() { return msi; } /** * @return the raw data */ public byte getRawData() { if (mrq) { return (byte) ((msi << 1) | 1); } else { return (byte) (msi << 1); } } @Override public String toString() { StringBuilder sb = new StringBuilder(250); sb.append("[MRQ: ") .append(mrq) .append(", MSI: ") .append(msi) .append("]"); return sb.toString(); } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + (mrq ? 1231 : 1237); result = prime * result + msi; return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; Mai other = (Mai) obj; if (mrq != other.mrq) return false; if (msi != other.msi) return false; return true; } } /** * ASELC subfield *
   *    0    1    2    3    4    5
   * +----+----+----+----+----+----+----+
   * | ASEL Command |     ASEL Data     |
   * +----+----+----+----+----+----+----+
   * 
* * @author Kaito Yamada * @since pcap4j 1.7.0 */ public static final class Aselc implements Serializable { /** * */ private static final long serialVersionUID = -5404846090809709793L; private final AselCommand command; private final byte data; /** * @param command command * @param data data */ public Aselc(AselCommand command, byte data) { if (command == null) { throw new IllegalArgumentException("command is null."); } if ((data & 0xF0) != 0) { throw new IllegalArgumentException("(data & 0xF0) must be zero. data: " + data); } this.command = command; this.data = data; } /** * @param rawData the raw data which the ASEL Command is encoded in 3 bits from the LSB. */ public Aselc(byte rawData) { this.command = AselCommand.getInstance(rawData & 0x07); this.data = (byte) ((rawData >> 3) & 0x0F); } /** * @return command */ public AselCommand getCommand() { return command; } /** * @return data */ public byte getData() { return data; } /** * @return the raw data. */ public byte getRawData() { return (byte) ((command.value << 4) | (data)); } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + command.hashCode(); result = prime * result + data; return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; Aselc other = (Aselc) obj; if (command != other.command) return false; if (data != other.data) return false; return true; } @Override public String toString() { StringBuilder sb = new StringBuilder(250); sb.append("[ASEL Command: ") .append(command) .append(", ASEL Data: ") .append(data) .append("]"); return sb.toString(); } } /** * @author Kaito Yamada * @since pcap4j 1.7.0 */ public static enum AselCommand { /** * Transmit Antenna Selection Sounding Indication (TXASSI): 0 */ TXASSI(0, "TXASSI"), /** * Transmit Antenna Selection Sounding Request (TXASSR) or Transmit ASEL Sounding Resumption: 1 */ TXASSR(1, "TXASSR"), /** * Receive Antenna Selection Sounding Indication (RXASSI): 2 */ RXASSI(2, "RXASSI"), /** * Receive Antenna Selection Sounding Request (RXASSR): 3 */ RXASSR(3, "RXASSR"), /** * Sounding Label: 4 */ SOUNDING_LABEL(4, "Sounding Label"), /** * No Feedback Due to ASEL Training Failure or Stale Feedback: 5 */ NO_FEEDBACK(5, "No Feedback"), /** * Transmit Antenna Selection Sounding Indication requesting feedback of explicit CSI * (TXASSI-CSI): 6 */ TXASSI_CSI(6, "TXASSI-CSI"), /** * Reserved: 7 */ SEVEN(7, "Reserved"); private final int value; private final String name; private AselCommand(int value, String name) { this.value = value; this.name = name; } /** * @return value */ public int getValue() { return value; } /** * @return name */ public String getName() { return name; } @Override public String toString() { StringBuilder sb = new StringBuilder(50); sb.append(value) .append(" (") .append(name) .append(")"); return sb.toString(); } /** * @param value value * @return the AselCommand object the value of which is the given value. */ public static AselCommand getInstance(int value) { for (AselCommand com: values()) { if (com.value == value) { return com; } } throw new IllegalArgumentException("Invalid value: " + value); } } /** * @author Kaito Yamada * @since pcap4j 1.7.0 */ public static final class Builder { private boolean vhtMfb; private boolean trq; private byte maiOrAseli; private byte mfsi; private byte mfbOrAselc; /** * */ public Builder() {} private Builder(Dot11LinkAdaptationControl obj) { this.vhtMfb = obj.vhtMfb; this.trq = obj.trq; this.maiOrAseli = obj.mai.getRawData(); this.mfsi = obj.mfsi; this.mfbOrAselc = obj.mfb; } /** * @param vhtMfb vhtMfb * @return this Builder object for method chaining. */ public Builder vhtMfb(boolean vhtMfb) { this.vhtMfb = vhtMfb; return this; } /** * @param trq trq * @return this Builder object for method chaining. */ public Builder trq(boolean trq) { this.trq = trq; return this; } /** * @param maiOrAseli maiOrAseli * @return this Builder object for method chaining. * @see #ASELI */ public Builder maiOrAseli(byte maiOrAseli) { this.maiOrAseli = maiOrAseli; return this; } /** * @param maiOrAseli maiOrAseli * @return this Builder object for method chaining. */ public Builder maiOrAseli(Mai maiOrAseli) { this.maiOrAseli = maiOrAseli.getRawData(); return this; } /** * @param mfsi mfsi * @return this Builder object for method chaining. */ public Builder mfsi(byte mfsi) { this.mfsi = mfsi; return this; } /** * @param mfbOrAselc mfbOrAselc * @return this Builder object for method chaining. */ public Builder mfbOrAselc(byte mfbOrAselc) { this.mfbOrAselc = mfbOrAselc; return this; } /** * @param mfbOrAselc mfbOrAselc * @return this Builder object for method chaining. */ public Builder mfbOrAselc(Aselc mfbOrAselc) { this.mfbOrAselc = mfbOrAselc.getRawData(); return this; } /** * * @return a new Dot11LinkAdaptationControl object. */ public Dot11LinkAdaptationControl build() { return new Dot11LinkAdaptationControl(this); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy