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

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

/*_##########################################################################
  _##
  _##  Copyright (C) 2016  Pcap4J.org
  _##
  _##########################################################################
*/

package org.pcap4j.packet;

import org.pcap4j.packet.RadiotapPacket.RadiotapData;
import org.pcap4j.util.ByteArrays;

/**
 * Radiotap MCS field.
 *
 * @see Radiotap
 * @author Kaito Yamada
 * @since pcap4j 1.6.5
 */
public final class RadiotapDataMcs implements RadiotapData {

  /** */
  private static final long serialVersionUID = 8914690461479810322L;

  private static final int LENGTH = 3;

  private final boolean bandwidthKnown;
  private final boolean mcsIndexKnown;
  private final boolean guardIntervalKnown;
  private final boolean htFormatKnown;
  private final boolean fecTypeKnown;
  private final boolean stbcKnown;
  private final boolean nessKnown;
  private final boolean nessMsb;
  private final Bandwidth bandwidth;
  private final boolean shortGuardInterval;
  private final HtFormat htFormat;
  private final RadiotapFecType fecType;
  private final byte numStbcStreams;
  private final boolean nessLsb;
  private final byte mcsRateIndex;

  /**
   * 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 RadiotapMcs object.
   * @throws IllegalRawDataException if parsing the raw data fails.
   */
  public static RadiotapDataMcs newInstance(byte[] rawData, int offset, int length)
      throws IllegalRawDataException {
    ByteArrays.validateBounds(rawData, offset, length);
    return new RadiotapDataMcs(rawData, offset, length);
  }

  private RadiotapDataMcs(byte[] rawData, int offset, int length) throws IllegalRawDataException {
    if (length < LENGTH) {
      StringBuilder sb = new StringBuilder(200);
      sb.append("The data is too short to build a RadiotapMcs (")
          .append(LENGTH)
          .append(" bytes). data: ")
          .append(ByteArrays.toHexString(rawData, " "))
          .append(", offset: ")
          .append(offset)
          .append(", length: ")
          .append(length);
      throw new IllegalRawDataException(sb.toString());
    }

    this.bandwidthKnown = (rawData[offset] & 0x01) != 0;
    this.mcsIndexKnown = (rawData[offset] & 0x02) != 0;
    this.guardIntervalKnown = (rawData[offset] & 0x04) != 0;
    this.htFormatKnown = (rawData[offset] & 0x08) != 0;
    this.fecTypeKnown = (rawData[offset] & 0x10) != 0;
    this.stbcKnown = (rawData[offset] & 0x20) != 0;
    this.nessKnown = (rawData[offset] & 0x40) != 0;
    this.nessMsb = (rawData[offset] & 0x80) != 0;
    switch (rawData[offset + 1] & 0x03) {
      case 0:
        this.bandwidth = Bandwidth.BW_20;
        break;
      case 1:
        this.bandwidth = Bandwidth.BW_40;
        break;
      case 2:
        this.bandwidth = Bandwidth.BW_20L;
        break;
      case 3:
        this.bandwidth = Bandwidth.BW_20U;
        break;
      default:
        throw new AssertionError("Never get here.");
    }
    this.shortGuardInterval = (rawData[offset + 1] & 0x04) != 0;
    switch (rawData[offset + 1] & 0x08) {
      case 0:
        this.htFormat = HtFormat.MIXED;
        break;
      default:
        this.htFormat = HtFormat.GREENFIELD;
    }
    switch (rawData[offset + 1] & 0x10) {
      case 0:
        this.fecType = RadiotapFecType.BCC;
        break;
      default:
        this.fecType = RadiotapFecType.LDPC;
    }
    this.numStbcStreams = (byte) ((rawData[offset + 1] & 0x60) >> 5);
    this.nessLsb = (rawData[offset + 1] & 0x80) != 0;
    this.mcsRateIndex = rawData[offset + 2];
  }

  private RadiotapDataMcs(Builder builder) {
    if (builder == null
        || builder.bandwidth == null
        || builder.htFormat == null
        || builder.fecType == null) {
      StringBuilder sb = new StringBuilder();
      sb.append("builder: ")
          .append(builder)
          .append(" builder.bandwidth: ")
          .append(builder.bandwidth)
          .append(" builder.htFormat: ")
          .append(builder.htFormat)
          .append(" builder.fecType: ")
          .append(builder.fecType);
      throw new NullPointerException(sb.toString());
    }
    if ((builder.numStbcStreams & 0xFC) != 0) {
      throw new IllegalArgumentException(
          "(builder.numStbcStreams & 0xFC) must be 0. builder.numStbcStreams: "
              + builder.numStbcStreams);
    }

    this.bandwidthKnown = builder.bandwidthKnown;
    this.mcsIndexKnown = builder.mcsIndexKnown;
    this.guardIntervalKnown = builder.guardIntervalKnown;
    this.htFormatKnown = builder.htFormatKnown;
    this.fecTypeKnown = builder.fecTypeKnown;
    this.stbcKnown = builder.stbcKnown;
    this.nessKnown = builder.nessKnown;
    this.nessMsb = builder.nessMsb;
    this.bandwidth = builder.bandwidth;
    this.shortGuardInterval = builder.shortGuardInterval;
    this.htFormat = builder.htFormat;
    this.fecType = builder.fecType;
    this.numStbcStreams = builder.numStbcStreams;
    this.nessLsb = builder.nessLsb;
    this.mcsRateIndex = builder.mcsRateIndex;
  }

  /** @return true if the bandwidth is known; false otherwise. */
  public boolean isBandwidthKnown() {
    return bandwidthKnown;
  }

  /** @return true if the MCS index is known; false otherwise. */
  public boolean isMcsIndexKnown() {
    return mcsIndexKnown;
  }

  /** @return true if the guard interval is known; false otherwise. */
  public boolean isGuardIntervalKnown() {
    return guardIntervalKnown;
  }

  /** @return true if the HT format is known; false otherwise. */
  public boolean isHtFormatKnown() {
    return htFormatKnown;
  }

  /** @return true if the FEC type is known; false otherwise. */
  public boolean isFecTypeKnown() {
    return fecTypeKnown;
  }

  /** @return true if the STBC is known; false otherwise. */
  public boolean isStbcKnown() {
    return stbcKnown;
  }

  /** @return true if the Ness is known; false otherwise. */
  public boolean isNessKnown() {
    return nessKnown;
  }

  /** @return true if the MSB of Ness is 1; false otherwise. */
  public boolean getNessMsb() {
    return nessMsb;
  }

  /** @return bandwidth */
  public Bandwidth getBandwidth() {
    return bandwidth;
  }

  /** @return true if the guard interval is short; false otherwise. */
  public boolean isShortGuardInterval() {
    return shortGuardInterval;
  }

  /** @return htFormat */
  public HtFormat getHtFormat() {
    return htFormat;
  }

  /** @return fecType */
  public RadiotapFecType getFecType() {
    return fecType;
  }

  /** @return numStbcStreams */
  public byte getNumStbcStreams() {
    return numStbcStreams;
  }

  /** @return numStbcStreams */
  public int getNumStbcStreamsAsInt() {
    return numStbcStreams;
  }

  /** @return true if LSB of Ness is 1; false otherwise. */
  public boolean getNessLsb() {
    return nessLsb;
  }

  /** @return mcsRateIndex */
  public byte getMcsRateIndex() {
    return mcsRateIndex;
  }

  /** @return mcsRateIndex */
  public int getMcsRateIndexAsInt() {
    return mcsRateIndex & 0xFF;
  }

  @Override
  public int length() {
    return LENGTH;
  }

  @Override
  public byte[] getRawData() {
    byte[] data = new byte[LENGTH];

    if (bandwidthKnown) {
      data[0] |= 0x01;
    }
    if (mcsIndexKnown) {
      data[0] |= 0x02;
    }
    if (guardIntervalKnown) {
      data[0] |= 0x04;
    }
    if (htFormatKnown) {
      data[0] |= 0x08;
    }
    if (fecTypeKnown) {
      data[0] |= 0x10;
    }
    if (stbcKnown) {
      data[0] |= 0x20;
    }
    if (nessKnown) {
      data[0] |= 0x40;
    }
    if (nessMsb) {
      data[0] |= 0x80;
    }

    data[1] = (byte) bandwidth.value;
    if (shortGuardInterval) {
      data[1] |= 0x04;
    }
    if (htFormat == HtFormat.GREENFIELD) {
      data[1] |= 0x08;
    }
    if (fecType == RadiotapFecType.LDPC) {
      data[1] |= 0x10;
    }
    data[1] |= numStbcStreams << 5;
    if (nessLsb) {
      data[1] |= 0x80;
    }

    data[2] = mcsRateIndex;

    return data;
  }

  /** @return a new Builder object populated with this object's fields. */
  public Builder getBuilder() {
    return new Builder(this);
  }

  @Override
  public String toString() {
    return toString("");
  }

  @Override
  public String toString(String indent) {
    StringBuilder sb = new StringBuilder();
    String ls = System.getProperty("line.separator");

    sb.append(indent)
        .append("MCS: ")
        .append(ls)
        .append(indent)
        .append("  bandwidth known: ")
        .append(bandwidthKnown)
        .append(ls)
        .append(indent)
        .append("  MCS index known: ")
        .append(mcsIndexKnown)
        .append(ls)
        .append(indent)
        .append("  guard interval known: ")
        .append(guardIntervalKnown)
        .append(ls)
        .append(indent)
        .append("  HT format known: ")
        .append(htFormatKnown)
        .append(ls)
        .append(indent)
        .append("  FEC type known: ")
        .append(fecTypeKnown)
        .append(ls)
        .append(indent)
        .append("  STBC known: ")
        .append(stbcKnown)
        .append(ls)
        .append(indent)
        .append("  Ness known: ")
        .append(nessKnown)
        .append(ls)
        .append(indent)
        .append("  Ness data known: ")
        .append(nessMsb)
        .append(ls)
        .append(indent)
        .append("  bandwidth: ")
        .append(bandwidth)
        .append(ls)
        .append(indent)
        .append("  short guard interval: ")
        .append(shortGuardInterval)
        .append(ls)
        .append(indent)
        .append("  HT format: ")
        .append(htFormat)
        .append(ls)
        .append(indent)
        .append("  FEC type: ")
        .append(fecType)
        .append(ls)
        .append(indent)
        .append("  Number of STBC streams: ")
        .append(numStbcStreams)
        .append(ls)
        .append(indent)
        .append("  Ness: ")
        .append(nessLsb)
        .append(ls)
        .append(indent)
        .append("  MCS rate index: ")
        .append(getMcsRateIndexAsInt())
        .append(ls);

    return sb.toString();
  }

  @Override
  public int hashCode() {
    final int prime = 31;
    int result = 1;
    result = prime * result + bandwidth.hashCode();
    result = prime * result + (bandwidthKnown ? 1231 : 1237);
    result = prime * result + fecType.hashCode();
    result = prime * result + (fecTypeKnown ? 1231 : 1237);
    result = prime * result + (guardIntervalKnown ? 1231 : 1237);
    result = prime * result + htFormat.hashCode();
    result = prime * result + (htFormatKnown ? 1231 : 1237);
    result = prime * result + (mcsIndexKnown ? 1231 : 1237);
    result = prime * result + mcsRateIndex;
    result = prime * result + (nessLsb ? 1231 : 1237);
    result = prime * result + (nessMsb ? 1231 : 1237);
    result = prime * result + (nessKnown ? 1231 : 1237);
    result = prime * result + numStbcStreams;
    result = prime * result + (shortGuardInterval ? 1231 : 1237);
    result = prime * result + (stbcKnown ? 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;
    RadiotapDataMcs other = (RadiotapDataMcs) obj;
    if (bandwidth != other.bandwidth) return false;
    if (bandwidthKnown != other.bandwidthKnown) return false;
    if (fecType != other.fecType) return false;
    if (fecTypeKnown != other.fecTypeKnown) return false;
    if (guardIntervalKnown != other.guardIntervalKnown) return false;
    if (htFormat != other.htFormat) return false;
    if (htFormatKnown != other.htFormatKnown) return false;
    if (mcsIndexKnown != other.mcsIndexKnown) return false;
    if (mcsRateIndex != other.mcsRateIndex) return false;
    if (nessLsb != other.nessLsb) return false;
    if (nessMsb != other.nessMsb) return false;
    if (nessKnown != other.nessKnown) return false;
    if (numStbcStreams != other.numStbcStreams) return false;
    if (shortGuardInterval != other.shortGuardInterval) return false;
    if (stbcKnown != other.stbcKnown) return false;
    return true;
  }

  /**
   * @author Kaito Yamada
   * @since pcap4j 1.6.5
   */
  public static final class Builder {

    private boolean bandwidthKnown;
    private boolean mcsIndexKnown;
    private boolean guardIntervalKnown;
    private boolean htFormatKnown;
    private boolean fecTypeKnown;
    private boolean stbcKnown;
    private boolean nessKnown;
    private boolean nessMsb;
    private Bandwidth bandwidth;
    private boolean shortGuardInterval;
    private HtFormat htFormat;
    private RadiotapFecType fecType;
    private byte numStbcStreams;
    private boolean nessLsb;
    private byte mcsRateIndex;

    /** */
    public Builder() {}

    private Builder(RadiotapDataMcs obj) {
      this.bandwidthKnown = obj.bandwidthKnown;
      this.mcsIndexKnown = obj.mcsIndexKnown;
      this.guardIntervalKnown = obj.guardIntervalKnown;
      this.htFormatKnown = obj.htFormatKnown;
      this.fecTypeKnown = obj.fecTypeKnown;
      this.stbcKnown = obj.stbcKnown;
      this.nessKnown = obj.nessKnown;
      this.nessMsb = obj.nessMsb;
      this.bandwidth = obj.bandwidth;
      this.shortGuardInterval = obj.shortGuardInterval;
      this.htFormat = obj.htFormat;
      this.fecType = obj.fecType;
      this.numStbcStreams = obj.numStbcStreams;
      this.nessLsb = obj.nessLsb;
      this.mcsRateIndex = obj.mcsRateIndex;
    }

    /**
     * @param bandwidthKnown bandwidthKnown
     * @return this Builder object for method chaining.
     */
    public Builder bandwidthKnown(boolean bandwidthKnown) {
      this.bandwidthKnown = bandwidthKnown;
      return this;
    }

    /**
     * @param mcsIndexKnown mcsIndexKnown
     * @return this Builder object for method chaining.
     */
    public Builder mcsIndexKnown(boolean mcsIndexKnown) {
      this.mcsIndexKnown = mcsIndexKnown;
      return this;
    }

    /**
     * @param guardIntervalKnown guardIntervalKnown
     * @return this Builder object for method chaining.
     */
    public Builder guardIntervalKnown(boolean guardIntervalKnown) {
      this.guardIntervalKnown = guardIntervalKnown;
      return this;
    }

    /**
     * @param htFormatKnown htFormatKnown
     * @return this Builder object for method chaining.
     */
    public Builder htFormatKnown(boolean htFormatKnown) {
      this.htFormatKnown = htFormatKnown;
      return this;
    }

    /**
     * @param fecTypeKnown fecTypeKnown
     * @return this Builder object for method chaining.
     */
    public Builder fecTypeKnown(boolean fecTypeKnown) {
      this.fecTypeKnown = fecTypeKnown;
      return this;
    }

    /**
     * @param stbcKnown stbcKnown
     * @return this Builder object for method chaining.
     */
    public Builder stbcKnown(boolean stbcKnown) {
      this.stbcKnown = stbcKnown;
      return this;
    }

    /**
     * @param nessKnown nessKnown
     * @return this Builder object for method chaining.
     */
    public Builder nessKnown(boolean nessKnown) {
      this.nessKnown = nessKnown;
      return this;
    }

    /**
     * @param nessMsb nessMsb
     * @return this Builder object for method chaining.
     */
    public Builder nessMsb(boolean nessMsb) {
      this.nessMsb = nessMsb;
      return this;
    }

    /**
     * @param bandwidth bandwidth
     * @return this Builder object for method chaining.
     */
    public Builder bandwidth(Bandwidth bandwidth) {
      this.bandwidth = bandwidth;
      return this;
    }

    /**
     * @param shortGuardInterval shortGuardInterval
     * @return this Builder object for method chaining.
     */
    public Builder shortGuardInterval(boolean shortGuardInterval) {
      this.shortGuardInterval = shortGuardInterval;
      return this;
    }

    /**
     * @param htFormat htFormat
     * @return this Builder object for method chaining.
     */
    public Builder htFormat(HtFormat htFormat) {
      this.htFormat = htFormat;
      return this;
    }

    /**
     * @param fecType fecType
     * @return this Builder object for method chaining.
     */
    public Builder fecType(RadiotapFecType fecType) {
      this.fecType = fecType;
      return this;
    }

    /**
     * @param numStbcStreams numStbcStreams
     * @return this Builder object for method chaining.
     */
    public Builder numStbcStreams(byte numStbcStreams) {
      this.numStbcStreams = numStbcStreams;
      return this;
    }

    /**
     * @param nessLsb nessLsb
     * @return this Builder object for method chaining.
     */
    public Builder nessLsb(boolean nessLsb) {
      this.nessLsb = nessLsb;
      return this;
    }

    /**
     * @param mcsRateIndex mcsRateIndex
     * @return this Builder object for method chaining.
     */
    public Builder mcsRateIndex(byte mcsRateIndex) {
      this.mcsRateIndex = mcsRateIndex;
      return this;
    }

    /** @return a new RadiotapMcs object. */
    public RadiotapDataMcs build() {
      return new RadiotapDataMcs(this);
    }
  }

  /**
   * @author Kaito Yamada
   * @since pcap4j 1.6.5
   */
  public static enum Bandwidth {

    /** 20 */
    BW_20(0, "20"),

    /** 40 */
    BW_40(1, "40"),

    /** 20L */
    BW_20L(2, "20L"),

    /** 20U */
    BW_20U(3, "20U");

    private final int value;
    private final String name;

    private Bandwidth(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();
      sb.append(value).append(" (").append(name).append(")");
      return sb.toString();
    }
  }

  /**
   * @author Kaito Yamada
   * @since pcap4j 1.6.5
   */
  public static enum HtFormat {

    /** mixed */
    MIXED(0),

    /** greenfield */
    GREENFIELD(1);

    private final int value;

    private HtFormat(int value) {
      this.value = value;
    }

    /** @return value */
    public int getValue() {
      return value;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy