参考资料:http://blog.csdn.net/leo115/article/details/7784349
徐雷鸣:NS与网络模拟
MFlood类中有recv函数
先检测是否是自己生成的,如果是自己发的,且转发的次数为0,那么就是自己生成的。
ip header的source addr,index_是自己的ip地址,比较这两个可以判断是否自己发的
common header中的num_forwards可以得到转发的次数
如果是自己生成的将common header的size加上IP包的长度,将ip header的ttl设为最大,将flood的seq_加1,然后转发包。
然后检测是否是路由环路,如果是自己发的,且转发次数不为0,那么就是一个路由环路,丢弃
然后检测ttl是否为0,如果是丢弃
如果不是上面所有的情况,那么解析包,即寻找下一跳地址。// Packet Reception Routines void MFlood::recv(Packet *p, Handler*) { struct hdr_cmn *ch = HDR_CMN(p); struct hdr_ip *ih = HDR_IP(p); struct hdr_mflood *fh = HDR_MFLOOD(p); assert(initialized()); if((ih->saddr() == index_) && (ch->num_forwards() == 0)) { // Must be a packet I'm originating... ch->size() += IP_HDR_LEN; // Add the IP Header ih->ttl_ = NETWORK_DIAMETER; fh->seq_ = myseq_++; forward((MFlood_RTEntry*)1,p,0); return; } else if(ih->saddr() == index_) { // I received a packet that I sent. Probably a routing loop. drop(p, DROP_RTR_ROUTE_LOOP); return; } else { // Packet I'm forwarding... if(--ih->ttl_ == 0) { // Check the TTL. If it is zero, then discard. drop(p, DROP_RTR_TTL); return; } } rt_resolve(p); }
// Route Handling Functions void MFlood::rt_resolve(Packet *p) { struct hdr_cmn *ch = HDR_CMN(p); struct hdr_ip *ih = HDR_IP(p); struct hdr_mflood *fh = HDR_MFLOOD(p); MFlood_RTEntry* rt; rt = rtable_.rt_lookup(ih->saddr()); if(rt == NULL) { rt = new MFlood_RTEntry(ih->saddr(), fh->seq_); LIST_INSERT_HEAD(&rtable_.rthead,rt,rt_link); //printf("%.8f %d,no uptarget,\n",NOW,index_); forward(rt,p,FORWARD_DELAY); //printf("%.8f %d,no rt,so forward.rt_seq:%d,pkt seq:%d\n",NOW,index_,rt->max_seqno,fh->seq_); rtable_.rt_print(); } // else if(rt->seq_ < fh->seq_ ) else if(rt->isNewSeq(fh->seq_) ) {
else if(rt->isNewSeq(fh->seq_) ) { //printf("%.8f %d,no uptarget,\n",NOW,index_); forward(rt, p, FORWARD_DELAY); // rt->seq_ = fh->seq_; rt->addSeq(fh->seq_); //printf("%.8f %d,rt seq too small,so forward,rt_seq:%d,packet seq:%d.\n",NOW,index_,rt->max_seqno,fh->seq_); rtable_.rt_print(); } else { drop(p, "LOWSEQ"); } }
下面解释一下整个路由表的实现,发现自己背离了扩散法的原理去读代码,结果读的一头雾水,当重新了解了一下扩散法的原理之后,代码也就清晰了,汗颜啊!而且,徐雷鸣的书中有专门介绍MFlood的,里面也讲的比较清楚,早看的话,就能少走些弯路了,不过自己想出来的也挺有成就感的:
MFlood类下会有路由表rtable_,而路由表类下又有一个路由表链表(由路由表项元素组成的链表)rthead,而路由表项类下会有变量src_,这就是源地址,每个表项对应一个源地址,对应一个序列号seqno数组rt_seqnos[REM_SEQ_COUNT]。
即每个路由表项下会有一个源地址,一个序列号数组,这就是扩散法的路由表需要的内容,而普通路由算法的话,应该会有目的地址,下一跳地址等。
另外为了防止路由表项无限扩大,设置了max_seqno变量(计算机网络书中讲到),如果收到的分组的源地址对应的seqno大于max,那么肯定是新的;如果小于max,mflood中还设置了一个min_seqno,用于丢弃太久之前的包,这里可能seqno回卷的话就不对了啊,这里每太想明白???
关于路由表项链表的操作也没太看明白,下面这个链表表项第二项是上一个下一个元素的地址,链表不是上一个元素和下一个元素的指针吗???
#define LIST_ENTRY(type) \ struct { \ type*le_next; /* next element */ \ type**le_prev; /* address of previous nextelement */ \ }
一开始一直看不明白下面是什么意思,怎么还把基类的构造函数搬出来了,C++继承那块一直没仔细看,原来继承不会继承基类的构造和析构函数,在派生类构造函数时,也要执行基类的构造函数,当派生类中内嵌其他类的时候,也要将这些类构造。
MFlood::MFlood(nsaddr_t id) : Agent(PT_MFLOOD) { index_ = id; logtarget = 0; myseq_ = 0; }徐雷鸣书中对C++和Tcl之间关系的解释: