/* 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 2010 Evan Plaice */ using System; using MiscUtil.Conversion; using PacketDotNet.Utils; namespace PacketDotNet.Tcp { /// /// SACK (Selective Ack) Option /// Provides a means for a receiver to notify the sender about /// all the segments that have arrived successfully. /// Used to cut down on the number of unnecessary re-transmissions. /// /// /// References: /// http://datatracker.ietf.org/doc/rfc2018/ /// http://datatracker.ietf.org/doc/rfc2883/ /// public class SACK : Option { #region Constructors /// /// Creates a SACK (Selective Ack) Option /// /// /// A /// /// /// A /// /// /// A /// public SACK(byte[] bytes, int offset, int length) : base(bytes, offset, length) { } #endregion #region Properties /// /// Contains an array of SACK (Selective Ack) Blocks /// public ushort[] SACKBlocks { get { int numOfBlocks = (Length - SACKBlocksFieldOffset) / BlockLength; ushort[] blocks = new ushort[numOfBlocks]; int offset = 0; for(int i = 0; i < numOfBlocks; i++) { offset = SACKBlocksFieldOffset + (i * BlockLength); blocks[i] = EndianBitConverter.Big.ToUInt16(Bytes, offset); } return blocks; } } #endregion #region Methods /// /// Returns the Option info as a string /// /// /// A /// public override string ToString() { string output = "[" + Kind.ToString() + ": "; for(int i = 0; i < SACKBlocks.Length; i++) { output += "Block" + i + "=" + SACKBlocks[i].ToString() + " "; } output.TrimEnd(); output += "]"; return output; } #endregion #region Members // the length (in bytes) of a SACK block const int BlockLength = 2; // the offset (in bytes) of the ScaleFactor Field const int SACKBlocksFieldOffset = 2; #endregion } }