📄 diffusion.cc
字号:
/*
* diffusion.cc
* Copyright (C) 2000 by the University of Southern California
* $Id: diffusion.cc,v 1.14 2006/03/10 12:25:28 mahrenho Exp $
*
* 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.
*
*
* The copyright of this module includes the following
* linking-with-specific-other-licenses addition:
*
* In addition, as a special exception, the copyright holders of
* this module give you permission to combine (via static or
* dynamic linking) this module 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 module 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 module
* 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.
*
*/
//
// $Header: /cvsroot/nsnam/ns-2/diffusion/diffusion.cc,v 1.14 2006/03/10 12:25:28 mahrenho Exp $
/****************************************************************/
/* diffusion.cc : Chalermek Intanagonwiwat (USC/ISI) 05/18/99 */
/****************************************************************/
// Important Note: Work still in progress !
#include <assert.h>
#include <math.h>
#include <stdio.h>
#include <signal.h>
#include <float.h>
#include <tcl.h>
#include <stdlib.h>
#include "diff_header.h"
#include "agent.h"
#include "tclcl.h"
#include "ip.h"
#include "config.h"
#include "packet.h"
#include "trace.h"
#include "random.h"
#include "classifier.h"
#include "node.h"
#include "diffusion.h"
#include "iflist.h"
#include "hash_table.h"
#include "arp.h"
#include "mac.h"
#include "ll.h"
#include "dsr/path.h"
#include "god.h"
#include "routing_table.h"
/* Callback helper */
void XmitFailedCallback(Packet *pkt, void *data)
{
DiffusionAgent *agent = (DiffusionAgent *)data; // cast of trust
agent->xmitFailed(pkt);
}
char *MsgStr[]= {"", "INTEREST", "DATA", "DATA_READY", "DATA_REQUEST",
"POS_REINFORCE", "NEG_REINFORCE", "INHIBIT", "TX_FAILED",
"DATA_STOP", "TX_FAILED", "ENG_LXK_"};
int hdr_cdiff::offset_;
static class DiffHeaderClass : public PacketHeaderClass {
public:
DiffHeaderClass() : PacketHeaderClass("PacketHeader/Diffusion",
sizeof(hdr_cdiff)) {
bind_offset(&hdr_cdiff::offset_);
}
} class_diffhdr;
static class DiffusionClass : public TclClass {
public:
DiffusionClass() : TclClass("Agent/Diffusion") {}
TclObject* create(int , const char*const* ) {
return(new DiffusionAgent());
}
} class_diffusion;
void ArpBufferTimer::expire(Event *e)
{
a_->ArpBufferCheck();
resched(ARP_BUFFER_CHECK + ARP_BUFFER_CHECK *
(double) ((long) e>>5 & 0xff) /256.0);
}
void SendBufTimer::expire(Event *e)
{
a_->SendBufferCheck();
resched(SEND_BUFFER_CHECK + SEND_BUFFER_CHECK * (double) ((long) e>>5 & 0xff)/256.0);
}
void DiffusionAgent::DataForSink(Packet *pkt)
{
hdr_cdiff *dfh = HDR_CDIFF(pkt);
unsigned int dtype = dfh->data_type;
Agent_List *cur_agent;
Packet *cur_pkt;
hdr_cdiff *cur_dfh;
hdr_ip *cur_iph;
// Give it to sink agents on this node. Don't care if wireless.
for (cur_agent= (routing_table[dtype]).sink; cur_agent != NULL;
cur_agent= AGENT_NEXT(cur_agent) ) {
cur_pkt = pkt->copy();
cur_iph = HDR_IP(cur_pkt);
cur_iph->dst_ = AGT_ADDR(cur_agent);
cur_dfh = HDR_CDIFF(cur_pkt);
cur_dfh->forward_agent_id = here_;
cur_dfh->num_next = 1;
cur_dfh->next_nodes[0] = NODE_ADDR(cur_agent);
send_to_dmux(cur_pkt, 0);
}
}
Packet *DiffusionAgent::prepare_message(unsigned int dtype, ns_addr_t to_addr,
int msg_type)
{
Packet *pkt;
hdr_cdiff *dfh;
hdr_ip *iph;
pkt = create_packet();
dfh = HDR_CDIFF(pkt);
iph = HDR_IP(pkt);
dfh->mess_type = msg_type;
dfh->pk_num = pk_count;
pk_count++;
dfh->sender_id = here_;
dfh->data_type = dtype;
dfh->forward_agent_id = here_;
dfh->ts_ = NOW;
dfh->num_next = 1;
dfh->next_nodes[0] = to_addr.addr_;
iph->src_ = here_;
iph->dst_ = to_addr;
return pkt;
}
DiffusionAgent::DiffusionAgent() : Agent(PT_DIFF), arp_buf_timer(this),
send_buf_timer(this)
{
POS_REINF_ = true;
NEG_REINF_ = true;
overhead = 0;
pk_count = 0;
target_ = 0;
node = NULL;
tracetarget = NULL;
}
void DiffusionAgent::recv(Packet* packet, Handler*)
{
hdr_cdiff* dfh = HDR_CDIFF(packet);
// Packet Hash Table is used to keep info about experienced pkts.
Pkt_Hash_Entry *hashPtr= PktTable.GetHash(dfh->sender_id, dfh->pk_num);
// Received this packet before ?
if (hashPtr != NULL) {
consider_old(packet);
return;
}
// Never receive it before ? Put in hash table.
PktTable.put_in_hash(dfh);
// Take action for a new pkt.
consider_new(packet);
}
void DiffusionAgent::consider_old(Packet *pkt)
{
Packet::free(pkt);
}
void DiffusionAgent::consider_new(Packet *pkt)
{
Packet::free(pkt);
}
void DiffusionAgent::Terminate()
{
#ifdef DEBUG_OUTPUT
printf("Diffusion node %d : terminates (overhead %d)\n",
THIS_NODE, overhead);
printf("node %d: remaining energy %f, initial energy %f\n", THIS_NODE,
node->energy_model()->energy(),
node->energy_model()->initialenergy() );
Print_IOlist();
#endif
}
void DiffusionAgent::Print_IOlist()
{
}
void DiffusionAgent::Start()
{
arp_buf_timer.sched(ARP_BUFFER_CHECK + ARP_BUFFER_CHECK *
Random::uniform(1.0));
send_buf_timer.sched(SEND_BUFFER_CHECK + SEND_BUFFER_CHECK *
Random::uniform(1.0));
}
int DiffusionAgent::command(int argc, const char*const* argv)
{
Tcl& tcl = Tcl::instance();
if (argc == 2) {
if (strcasecmp(argv[1], "reset-state")==0) {
reset();
return TCL_OK;
}
if (strcasecmp(argv[1], "reset")==0) {
return Agent::command(argc, argv);
}
if (strcasecmp(argv[1], "start")==0) {
Start();
return TCL_OK;
}
if (strcasecmp(argv[1], "stop")==0) {
return TCL_OK;
}
if (strcasecmp(argv[1], "terminate")==0) {
Terminate();
return TCL_OK;
}
if (strcasecmp(argv[1], "stop-source")==0) {
StopSource();
return TCL_OK;
}
if (strcasecmp(argv[1], "enable-pos") == 0) {
POS_REINF_ = true;
return TCL_OK;
}
if (strcasecmp(argv[1], "enable-neg") == 0) {
NEG_REINF_ = true;
return TCL_OK;
}
if (strcasecmp(argv[1], "disable-pos") == 0) {
POS_REINF_ = false;
return TCL_OK;
}
if (strcasecmp(argv[1], "disable-neg") == 0) {
NEG_REINF_ = false;
return TCL_OK;
}
if (strcasecmp(argv[1], "enable-eng-lxk") == 0) {
ENG_LXK_ = true;
return TCL_OK;
}
if (strcasecmp(argv[1], "disable-eng-lxk") == 0) {
ENG_LXK_ = false;
return TCL_OK;
}
} else if (argc == 3) {
if (strcasecmp(argv[1], "on-node")==0) {
node = (Node *)tcl.lookup(argv[2]);
return TCL_OK;
}
if (strcasecmp(argv[1], "add-ll") == 0) {
TclObject *obj;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -