/* -*- Mode:C++; c-basic-offset:8; tab-width:8; indent-tabs-mode:t -*- */
/*
* Copyright (c) Xerox Corporation 1997. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at your
* option) any later version.
*
* 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.,
* 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*
* Linking this file statically or dynamically with other modules is making
* a combined work based on this file. Thus, the terms and conditions of
* the GNU General Public License cover the whole combination.
*
* In addition, as a special exception, the copyright holders of this file
* give you permission to combine this file with free software programs or
* libraries that are released under the GNU LGPL and with code included in
* the standard release of ns-2 under the Apache 2.0 license or under
* otherwise-compatible licenses with advertising requirements (or modified
* versions of such code, with unchanged license). You may copy and
* distribute such a system following the terms of the GNU GPL for this
* file and the licenses of the other code concerned, provided that you
* include the source code of that other code when and as the GNU GPL
* requires distribution of source code.
*
* Note that people who make modified versions of this file are not
* obligated to grant this special exception for their modified versions;
* it is their choice whether to do so. The GNU General Public License
* gives permission to release a modified version without this exception;
* this exception also makes it possible to release a modified version
* which carries forward this exception.
*/
/* Token Bucket filter which has 3 parameters :
a. Token Generation rate
b. Token bucket depth
c. Max. Queue Length (a finite length would allow this to be used as policer as packets are dropped after queue gets full)
*/
#include "connector.h"
#include "packet.h"
#include "queue.h"
#include "tbf.h"
TBF::TBF() :tokens_(0),tbf_timer_(this), init_(1)
{
q_=new PacketQueue();
bind_bw("rate_",&rate_);
bind("bucket_",&bucket_);
bind("qlen_",&qlen_);
}
TBF::~TBF()
{
if (q_->length() != 0) {
//Clear all pending timers
tbf_timer_.cancel();
//Free up the packetqueue
for (Packet *p=q_->head();p!=0;p=p->next_)
Packet::free(p);
}
delete q_;
}
void TBF::recv(Packet *p, Handler *)
{
//start with a full bucket
if (init_) {
tokens_=bucket_;
lastupdatetime_ = Scheduler::instance().clock();
init_=0;
}
hdr_cmn *ch=hdr_cmn::access(p);
//enque packets appropriately if a non-zero q already exists
if (q_->length() !=0) {
if (q_->length() < qlen_) {
q_->enque(p);
return;
}
drop(p);
return;
}
double tok;
tok = getupdatedtokens();
int pktsize = ch->size()<<3;
if (tokens_ >=pktsize) {
target_->recv(p);
tokens_-=pktsize;
}
else {
if (qlen_!=0) {
q_->enque(p);
tbf_timer_.resched((pktsize-tokens_)/rate_);
}
else {
drop(p);
}
}
}
double TBF::getupdatedtokens(void)
{
double now=Scheduler::instance().clock();
tokens_ += (now-lastupdatetime_)*rate_;
if (tokens_ > bucket_)
tokens_=bucket_;
lastupdatetime_ = Scheduler::instance().clock();
return tokens_;
}
void TBF::timeout(int)
{
if (q_->length() == 0) {
fprintf (stderr,"ERROR in tbf\n");
abort();
}
Packet *p=q_->deque();
double tok;
tok = getupdatedtokens();
hdr_cmn *ch=hdr_cmn::access(p);
int pktsize = ch->size()<<3;
//We simply send the packet here without checking if we have enough tokens
//because the timer is supposed to fire at the right time
target_->recv(p);
tokens_-=pktsize;
if (q_->length() !=0 ) {
p=q_->head();
hdr_cmn *ch=hdr_cmn::access(p);
pktsize = ch->size()<<3;
tbf_timer_.resched((pktsize-tokens_)/rate_);
}
}
void TBF_Timer::expire(Event* /*e*/)
{
tbf_->timeout(0);
}
static class TBFClass : public TclClass {
public:
TBFClass() : TclClass ("TBF") {}
TclObject* create(int,const char*const*) {
return (new TBF());
}
}class_tbf;
syntax highlighted by Code2HTML, v. 0.9.1