/*
* ackretrysender.{cc,hh} -- element buffers packets until acknowledged
* Douglas S. J. De Couto
*
* Copyright (c) 2002 Massachusetts Institute of Technology
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, subject to the conditions
* listed in the Click LICENSE file. These conditions include: you must
* preserve this copyright notice, and you cannot mention the copyright
* holders in advertising related to the Software without their permission.
* The Software is provided WITHOUT ANY WARRANTY, EXPRESS OR IMPLIED. This
* notice is a summary of the Click LICENSE file; the license in that file is
* legally binding.
*/
#include <click/config.h>
#include <click/glue.hh>
#include <clicknet/ether.h>
#include <click/confparse.hh>
#include <click/packet.hh>
#include <click/error.hh>
#include <click/standard/scheduleinfo.hh>
#include <click/straccum.hh>
#include "ackretrysender.hh"
#include "ackresponder.hh"
CLICK_DECLS
ACKRetrySender::ACKRetrySender()
: _timeout(0), _max_tries(0),
_num_tries(0), _history_length(500),
_waiting_packet(0), _verbose(true),
_timer(this), _task(this)
{
}
ACKRetrySender::~ACKRetrySender()
{
}
void
ACKRetrySender::push(int port, Packet *p)
{
assert(port == 1);
check();
if (!_waiting_packet) {
// we aren't waiting for ACK
if (_verbose)
click_chatter("ACKRetrySender %s: got unexpected ACK", name().c_str());
p->kill();
return;
}
click_ether *e_ack = (click_ether *) p->data();
if (ntohs(e_ack->ether_type) != ETHERTYPE_GRID_ACK) {
click_chatter("ACKRetrySender %d: got non-ACK packet on second input", name().c_str());
p->kill();
return;
}
// was this response for the packet we have?
click_ether *e_waiting = (click_ether *) _waiting_packet->data();
if (memcmp(e_ack->ether_shost, e_waiting->ether_dhost, 6) ||
memcmp(e_ack->ether_dhost, e_waiting->ether_shost, 6)) {
// no, it wasn't for our packet...
if (_verbose)
click_chatter("ACKRetrySender %s: got ACK for wrong packet", name().c_str());
p->kill();
return;
}
// ahhh, ACK was for us.
_history.push_back(tx_result_t(_waiting_packet->timestamp_anno(),
_num_tries, true));
while (_history.size() > (int) _history_length)
_history.pop_front();
_waiting_packet->kill();
_waiting_packet = 0;
_num_tries = 0;
_timer.unschedule();
p->kill();
check();
}
bool
ACKRetrySender::run_task()
{
check();
_task.fast_reschedule();
if (_waiting_packet)
return true;
Packet *p = input(0).pull();
if (!p)
return true;
if (_max_tries > 1) {
_waiting_packet = p->clone();
_num_tries = 1;
_timer.schedule_after_msec(_timeout);
}
check();
output(0).push(p);
return true;
}
int
ACKRetrySender::configure(Vector<String> &conf, ErrorHandler *errh)
{
_max_tries = 16;
_timeout = 10;
_verbose = true;
_history_length = 500;
int res = cp_va_parse(conf, this, errh,
cpKeywords,
"MAX_TRIES", cpUnsigned, "max retries per packet (> 0)", &_max_tries,
"TIMEOUT", cpUnsigned, "time between retried (> 0 msecs)", &_timeout,
"VERBOSE", cpBool, "noisy?", &_verbose,
"HISTORY_LEN", cpUnsigned, "max number of packets for which to remember retry history", &_history_length,
cpEnd);
if (res < 0)
return res;
if (_timeout == 0)
return errh->error("TIMEOUT must be > 0");
if (_max_tries == 0)
return errh->error("MAX_TRIES must be > 0");
return 0;
}
int
ACKRetrySender::initialize(ErrorHandler *errh)
{
_timer.initialize(this);
ScheduleInfo::join_scheduler(this, &_task, errh);
check();
return 0;
}
void
ACKRetrySender::run_timer(Timer *)
{
assert(_waiting_packet && !_timer.scheduled());
Packet *p = _waiting_packet;
if (_num_tries >= _max_tries) {
_history.push_back(tx_result_t(p->timestamp_anno(), _num_tries, false));
while (_history.size() > (int) _history_length)
_history.pop_front();
_waiting_packet->kill();
_waiting_packet = p = 0;
_num_tries = 0;
}
else {
_timer.schedule_after_msec(_timeout);
_waiting_packet = p->clone();
_num_tries++;
}
check();
if (p)
output(0).push(p);
}
void
ACKRetrySender::check()
{
// check() should be called *before* most pushes() from element
// functions, as each push may call back into the element.
// if there is a packet waiting, the timeout timer should be running
assert(_waiting_packet ? _timer.scheduled() : !_timer.scheduled());
// no packet has been sent more than the max number of times
assert(_num_tries <= _max_tries);
// any waiting packet has been sent at least once
assert(_waiting_packet ? _num_tries >= 1 : _num_tries == 0);
}
void
ACKRetrySender::add_handlers()
{
add_read_handler("history", print_history, 0);
add_read_handler("summary", print_summary, 0);
add_write_handler("clear", clear_history, 0);
}
String
ACKRetrySender::print_history(Element *e, void *)
{
ACKRetrySender *a = (ACKRetrySender *) e;
StringAccum s;
for (ACKRetrySender::HistQ::const_iterator i = a->_history.begin();
i != a->_history.end(); i++)
s << i->pkt_time << "\t" << i->num_tx << "\t"
<< (i->success ? "succ" : "fail") << "\n";
return s.take_string();
}
String
ACKRetrySender::print_summary(Element *e, void *)
{
ACKRetrySender *a = (ACKRetrySender *) e;
unsigned num_succ = 0;
unsigned num_fail = 0;
unsigned sum_tx = 0;
unsigned max_tx = 0;
unsigned min_tx = 0;
int n = a->_history.size();
for (ACKRetrySender::HistQ::const_iterator i = a->_history.begin();
i != a->_history.end(); i++) {
if (sum_tx == 0)
max_tx = min_tx = i->num_tx;
else {
max_tx = max_tx > i->num_tx ? max_tx : i->num_tx;
min_tx = min_tx < i->num_tx ? min_tx : i->num_tx;
}
if (i->success)
num_succ++;
else
num_fail++;
sum_tx += i->num_tx;
}
unsigned txc = 0;
if (n > 0)
txc = (1000 * sum_tx) / n;
StringAccum s;
s << "packets: " << n << "\n"
<< "success: " << num_succ << "\n"
<< "fail: " << num_fail << "\n"
<< "min_txc: " << min_tx << "\n"
<< "max_txc: " << max_tx << "\n"
<< "avg_txc: " << cp_unparse_real10(txc, 3) << "\n";
return s.take_string();
}
int
ACKRetrySender::clear_history(const String &, Element *e, void *, ErrorHandler *)
{
ACKRetrySender *a = (ACKRetrySender *) e;
a->_history.clear();
return 0;
}
#include <click/dequeue.cc>
template class DEQueue<ACKRetrySender::tx_result_t>;
CLICK_ENDDECLS
EXPORT_ELEMENT(ACKRetrySender)
syntax highlighted by Code2HTML, v. 0.9.1