Files
simulation_core/lib/Packet.Net/PacketDotNet/Ieee80211/BlockAcknowledgmentFrame.cs

304 lines
11 KiB
C#
Raw Normal View History

/*
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 <http://www.gnu.org/licenses/>.
*/
/*
* Copyright 2012 Alan Rushforth <alan.rushforth@gmail.com>
*/
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
{
/// <summary>
/// Format of the 802.11 block acknowledgment frame.
/// http://en.wikipedia.org/wiki/Block_acknowledgement
/// </summary>
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;
}
}
/// <summary>
/// Receiver address
/// </summary>
public PhysicalAddress ReceiverAddress {get; set;}
/// <summary>
/// Transmitter address
/// </summary>
public PhysicalAddress TransmitterAddress { get; set; }
/// <summary>
/// Gets or sets the block ack request control bytes.
/// </summary>
/// <value>
/// The block ack request control bytes.
/// </value>
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);
}
}
/// <summary>
/// Block acknowledgment control field
/// </summary>
public BlockAcknowledgmentControlField BlockAcknowledgmentControl
{
get;
set;
}
/// <summary>
/// Gets or sets the block ack starting sequence control.
/// </summary>
/// <value>
/// The block ack starting sequence control.
/// </value>
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;
/// <summary>
/// Gets or sets the block ack bitmap used to indicate the receive status of the MPDUs.
/// </summary>
/// <value>
/// The block ack bitmap.
/// </value>
/// <exception cref='ArgumentException'>
/// 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.
/// </exception>
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());
}
}
/// <summary>
/// Length of the frame
/// </summary>
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;
}
/// <summary>
/// Constructor
/// </summary>
/// <param name="bas">
/// A <see cref="ByteArraySegment"/>
/// </param>
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;
}
/// <summary>
/// Initializes a new instance of the <see cref="PacketDotNet.Ieee80211.BlockAcknowledgmentFrame"/> class.
/// </summary>
/// <param name='TransmitterAddress'>
/// Transmitter address.
/// </param>
/// <param name='ReceiverAddress'>
/// Receiver address.
/// </param>
/// <param name='BlockAckBitmap'>
/// The Block ack bitmap signalling the receive status of the MSDUs.
/// </param>
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;
}
/// <summary>
/// Writes the current packet properties to the backing ByteArraySegment.
/// </summary>
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;
}
/// <summary>
/// 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().
/// </summary>
/// <returns>
/// The address string.
/// </returns>
protected override String GetAddressString()
{
return String.Format("RA {0} TA {1}", ReceiverAddress, TransmitterAddress);
}
}
}
}