/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ /* * Copyright (c) 2007 University of Washington * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation; * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "ns3/log.h" #include "ns3/enum.h" #include "ns3/uinteger.h" #include "drop-tail-queue.h" NS_LOG_COMPONENT_DEFINE ("DropTailQueue"); namespace ns3 { NS_OBJECT_ENSURE_REGISTERED (DropTailQueue); TypeId DropTailQueue::GetTypeId (void) { static TypeId tid = TypeId ("ns3::DropTailQueue") .SetParent () .AddConstructor () .AddAttribute ("Mode", "Whether to use Bytes (see MaxBytes) or Packets (see MaxPackets) as the maximum queue size metric.", EnumValue (PACKETS), MakeEnumAccessor (&DropTailQueue::SetMode), MakeEnumChecker (BYTES, "Bytes", PACKETS, "Packets")) .AddAttribute ("MaxPackets", "The maximum number of packets accepted by this DropTailQueue.", UintegerValue (100), MakeUintegerAccessor (&DropTailQueue::m_maxPackets), MakeUintegerChecker ()) .AddAttribute ("MaxBytes", "The maximum number of bytes accepted by this DropTailQueue.", UintegerValue (100 * 65535), MakeUintegerAccessor (&DropTailQueue::m_maxBytes), MakeUintegerChecker ()) ; return tid; } DropTailQueue::DropTailQueue () : Queue (), m_packets () { NS_LOG_FUNCTION_NOARGS (); } DropTailQueue::~DropTailQueue () { NS_LOG_FUNCTION_NOARGS (); } void DropTailQueue::SetMode (enum Mode mode) { NS_LOG_FUNCTION (mode); m_mode = mode; } DropTailQueue::Mode DropTailQueue::GetMode (void) { NS_LOG_FUNCTION_NOARGS (); return m_mode; } bool DropTailQueue::DoEnqueue (Ptr p) { NS_LOG_FUNCTION (this << p); if (m_mode == PACKETS && (m_packets.size () >= m_maxPackets)) { NS_LOG_LOGIC ("Queue full (at max packets) -- droppping pkt"); Drop (p); return false; } if (m_mode == BYTES && (m_bytesInQueue + p->GetSize () >= m_maxBytes)) { NS_LOG_LOGIC ("Queue full (packet would exceed max bytes) -- droppping pkt"); Drop (p); return false; } m_bytesInQueue += p->GetSize (); m_packets.push(p); NS_LOG_LOGIC ("Number packets " << m_packets.size ()); NS_LOG_LOGIC ("Number bytes " << m_bytesInQueue); return true; } Ptr DropTailQueue::DoDequeue (void) { NS_LOG_FUNCTION (this); if (m_packets.empty()) { NS_LOG_LOGIC ("Queue empty"); return 0; } Ptr p = m_packets.front (); m_packets.pop (); m_bytesInQueue -= p->GetSize (); NS_LOG_LOGIC ("Popped " << p); NS_LOG_LOGIC ("Number packets " << m_packets.size ()); NS_LOG_LOGIC ("Number bytes " << m_bytesInQueue); return p; } Ptr DropTailQueue::DoPeek (void) const { NS_LOG_FUNCTION (this); if (m_packets.empty()) { NS_LOG_LOGIC ("Queue empty"); return 0; } Ptr p = m_packets.front (); NS_LOG_LOGIC ("Number packets " << m_packets.size ()); NS_LOG_LOGIC ("Number bytes " << m_bytesInQueue); return p; } } // namespace ns3 #include "ns3/test.h" namespace ns3 { class DropTailQueueTestCase : public TestCase { public: DropTailQueueTestCase (); virtual bool DoRun (void); }; DropTailQueueTestCase::DropTailQueueTestCase () : TestCase ("Sanity check on the drop tail queue implementation") {} bool DropTailQueueTestCase::DoRun (void) { Ptr queue = CreateObject (); NS_TEST_EXPECT_MSG_EQ (queue->SetAttributeFailSafe ("MaxPackets", UintegerValue (3)), true, "Verify that we can actually set the attribute"); Ptr p1, p2, p3, p4; p1 = Create (); p2 = Create (); p3 = Create (); p4 = Create (); NS_TEST_EXPECT_MSG_EQ (queue->GetNPackets (), 0, "There should be no packets in there"); queue->Enqueue (p1); NS_TEST_EXPECT_MSG_EQ (queue->GetNPackets (), 1, "There should be one packet in there"); queue->Enqueue (p2); NS_TEST_EXPECT_MSG_EQ (queue->GetNPackets (), 2, "There should be two packets in there"); queue->Enqueue (p3); NS_TEST_EXPECT_MSG_EQ (queue->GetNPackets (), 3, "There should be three packets in there"); queue->Enqueue (p4); // will be dropped NS_TEST_EXPECT_MSG_EQ (queue->GetNPackets (), 3, "There should be still three packets in there"); Ptr p; p = queue->Dequeue (); NS_TEST_EXPECT_MSG_EQ ((p != 0), true, "I want to remove the first packet"); NS_TEST_EXPECT_MSG_EQ (queue->GetNPackets (), 2, "There should be two packets in there"); NS_TEST_EXPECT_MSG_EQ (p->GetUid (), p1->GetUid (), "was this the first packet ?"); p = queue->Dequeue (); NS_TEST_EXPECT_MSG_EQ ((p != 0), true, "I want to remove the second packet"); NS_TEST_EXPECT_MSG_EQ (queue->GetNPackets (), 1, "There should be one packet in there"); NS_TEST_EXPECT_MSG_EQ (p->GetUid (), p2->GetUid (), "Was this the second packet ?"); p = queue->Dequeue (); NS_TEST_EXPECT_MSG_EQ ((p != 0), true, "I want to remove the third packet"); NS_TEST_EXPECT_MSG_EQ (queue->GetNPackets (), 0, "There should be no packets in there"); NS_TEST_EXPECT_MSG_EQ (p->GetUid (), p3->GetUid (), "Was this the third packet ?"); p = queue->Dequeue (); NS_TEST_EXPECT_MSG_EQ ((p == 0), true, "There are really no packets in there"); return false; } static class DropTailQueueTestSuite : public TestSuite { public: DropTailQueueTestSuite () : TestSuite ("drop-tail-queue", UNIT) { AddTestCase (new DropTailQueueTestCase ()); } } g_dropTailQueueTestSuite; } // namespace ns3