/* This file is part of PacketDotNet PacketDotNet is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. PacketDotNet is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with PacketDotNet. If not, see . */ /* * Copyright 2012 Alan Rushforth */ using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Net.NetworkInformation; using PacketDotNet.Utils; using MiscUtil.Conversion; namespace PacketDotNet { namespace Ieee80211 { /// /// Format of the 802.11 block acknowledgment frame. /// http://en.wikipedia.org/wiki/Block_acknowledgement /// public class BlockAcknowledgmentFrame : MacFrame { private class BlockAcknowledgmentField { public readonly static int BlockAckRequestControlLength = 2; public readonly static int BlockAckStartingSequenceControlLength = 2; public readonly static int BlockAckRequestControlPosition; public readonly static int BlockAckStartingSequenceControlPosition; public readonly static int BlockAckBitmapPosition; static BlockAcknowledgmentField() { BlockAckRequestControlPosition = MacFields.DurationIDPosition + MacFields.DurationIDLength + (2 * MacFields.AddressLength); BlockAckStartingSequenceControlPosition = BlockAckRequestControlPosition + BlockAckRequestControlLength; BlockAckBitmapPosition = BlockAckStartingSequenceControlPosition + BlockAckStartingSequenceControlLength; } } /// /// Receiver address /// public PhysicalAddress ReceiverAddress {get; set;} /// /// Transmitter address /// public PhysicalAddress TransmitterAddress { get; set; } /// /// Gets or sets the block ack request control bytes. /// /// /// The block ack request control bytes. /// private UInt16 BlockAckRequestControlBytes { get { if(header.Length >= (BlockAcknowledgmentField.BlockAckRequestControlPosition + BlockAcknowledgmentField.BlockAckRequestControlLength)) { return EndianBitConverter.Little.ToUInt16(header.Bytes, header.Offset + BlockAcknowledgmentField.BlockAckRequestControlPosition); } else { return 0; } } set { EndianBitConverter.Little.CopyBytes(value, header.Bytes, header.Offset + BlockAcknowledgmentField.BlockAckRequestControlPosition); } } /// /// Block acknowledgment control field /// public BlockAcknowledgmentControlField BlockAcknowledgmentControl { get; set; } /// /// Gets or sets the block ack starting sequence control. /// /// /// The block ack starting sequence control. /// public UInt16 BlockAckStartingSequenceControl {get; set;} private UInt16 BlockAckStartingSequenceControlBytes { get { if(header.Length >= (BlockAcknowledgmentField.BlockAckStartingSequenceControlPosition + BlockAcknowledgmentField.BlockAckStartingSequenceControlLength)) { return EndianBitConverter.Little.ToUInt16 (header.Bytes, header.Offset + BlockAcknowledgmentField.BlockAckStartingSequenceControlPosition); } else { return 0; } } set { EndianBitConverter.Little.CopyBytes (value, header.Bytes, header.Offset + BlockAcknowledgmentField.BlockAckStartingSequenceControlPosition); } } private byte[] blockAckBitmap; /// /// Gets or sets the block ack bitmap used to indicate the receive status of the MPDUs. /// /// /// The block ack bitmap. /// /// /// Is thrown when the bitmap is of an incorrect lenght. The bitmap must be either 8 or 64 btyes longs depending on whether or not /// it is compressed. /// public Byte[] BlockAckBitmap { get { return blockAckBitmap; } set { if (value.Length == 8) { BlockAcknowledgmentControl.CompressedBitmap = true; } else if (value.Length == 64) { BlockAcknowledgmentControl.CompressedBitmap = false; } else { throw new ArgumentException ("Invalid BlockAckBitmap size. Must be either 8 or 64 bytes long."); } blockAckBitmap = value; } } private Byte[] BlockAckBitmapBytes { get { Byte[] bitmap = new Byte[GetBitmapLength ()]; if(header.Length >= (BlockAcknowledgmentField.BlockAckBitmapPosition + GetBitmapLength())) { Array.Copy (header.Bytes, (BlockAcknowledgmentField.BlockAckBitmapPosition), bitmap, 0, GetBitmapLength ()); } return bitmap; } set { Array.Copy (BlockAckBitmap, 0, header.Bytes, BlockAcknowledgmentField.BlockAckBitmapPosition, GetBitmapLength()); } } /// /// Length of the frame /// override public int FrameSize { get { return (MacFields.FrameControlLength + MacFields.DurationIDLength + (MacFields.AddressLength * 2) + BlockAcknowledgmentField.BlockAckRequestControlLength + BlockAcknowledgmentField.BlockAckStartingSequenceControlLength + GetBitmapLength()); } } private int GetBitmapLength() { return BlockAcknowledgmentControl.CompressedBitmap ? 8 : 64; } /// /// Constructor /// /// /// A /// public BlockAcknowledgmentFrame (ByteArraySegment bas) { header = new ByteArraySegment (bas); FrameControl = new FrameControlField (FrameControlBytes); Duration = new DurationField (DurationBytes); ReceiverAddress = GetAddress (0); TransmitterAddress = GetAddress (1); BlockAcknowledgmentControl = new BlockAcknowledgmentControlField (BlockAckRequestControlBytes); BlockAckStartingSequenceControl = BlockAckStartingSequenceControlBytes; BlockAckBitmap = BlockAckBitmapBytes; header.Length = FrameSize; } /// /// Initializes a new instance of the class. /// /// /// Transmitter address. /// /// /// Receiver address. /// /// /// The Block ack bitmap signalling the receive status of the MSDUs. /// public BlockAcknowledgmentFrame (PhysicalAddress TransmitterAddress, PhysicalAddress ReceiverAddress, Byte[] BlockAckBitmap) { this.FrameControl = new FrameControlField (); this.Duration = new DurationField (); this.ReceiverAddress = ReceiverAddress; this.TransmitterAddress = TransmitterAddress; this.BlockAcknowledgmentControl = new BlockAcknowledgmentControlField (); this.BlockAckBitmap = BlockAckBitmap; this.FrameControl.SubType = FrameControlField.FrameSubTypes.ControlBlockAcknowledgment; } /// /// Writes the current packet properties to the backing ByteArraySegment. /// public override void UpdateCalculatedValues () { if ((header == null) || (header.Length > (header.BytesLength - header.Offset)) || (header.Length < FrameSize)) { header = new ByteArraySegment (new Byte[FrameSize]); } this.FrameControlBytes = this.FrameControl.Field; this.DurationBytes = this.Duration.Field; SetAddress (0, ReceiverAddress); SetAddress (1, TransmitterAddress); this.BlockAckRequestControlBytes = this.BlockAcknowledgmentControl.Field; this.BlockAckStartingSequenceControlBytes = this.BlockAckStartingSequenceControl; BlockAckBitmapBytes = BlockAckBitmap; header.Length = FrameSize; } /// /// Returns a string with a description of the addresses used in the packet. /// This is used as a compoent of the string returned by ToString(). /// /// /// The address string. /// protected override String GetAddressString() { return String.Format("RA {0} TA {1}", ReceiverAddress, TransmitterAddress); } } } }