NS2中对TCP数据包和ACK包的TCP Sink类的主要实现代码详尽剖析--吐血放送

来源:互联网 发布:淘宝女鞋店铺 编辑:程序博客网 时间:2024/04/28 20:14

NS2中对TCP数据包和ACK包的TCP Sink类的主要实现代码详尽剖析,限于个人水平,如有错误请留言指出!

TcpSink类的recv()方法:

void TcpSink::recv(Packet* pkt, Handler*){int numToDeliver;int numBytes = hdr_cmn::access(pkt)->size();//接收到的包的大小// number of bytes in the packet just receivedhdr_tcp *th = hdr_tcp::access(pkt);//定义接收到的包头位指针/* W.N. Check if packet is from previous incarnation */if (th->ts() < lastreset_) {//说明该包是无效的包// Remove packet and do nothingPacket::free(pkt);//删除该包return;}acker_->update_ts(th->seqno(),th->ts(),ts_echo_rfc1323_);//更新接收端确认器,更新内容:包的序列号、到达时间,相应的时间戳// update the timestamp to echo      numToDeliver = acker_->update(th->seqno(), numBytes);//把更新的序列号和字节数,给变量numToDeliver用于计算recv窗口// update the recv window; figure out how many in-order-bytes// (if any) can be removed from the window and handed to the// applicationif (numToDeliver) {//应用程序对变量numToDeliver处理bytes_ += numToDeliver;recvBytes(numToDeliver);}// send any packets to the application      ack(pkt);//回应该包的ACK// ACK the packetPacket::free(pkt);// remove it from the system}


TcpSink类的ack()方法:

void TcpSink::ack(Packet* opkt){Packet* npkt = allocpkt();//opkt是指刚接收到的数据包,npkt是即将构建的该数据包的对应ACK包// opkt is the "old" packet that was received// npkt is the "new" packet being constructed (for the ACK)double now = Scheduler::instance().clock();//获取当前时间用于ACK包hdr_tcp *otcp = hdr_tcp::access(opkt);//接收到的数据包的TCP包头指针hdr_ip *oiph = hdr_ip::access(opkt);//接收到的数据包的IP包头指针hdr_tcp *ntcp = hdr_tcp::access(npkt);//将要构建的ACK包的TCP包头指针if (qs_enabled_) {//如果可以进行快速启动,进行以下相关处理// QuickStart code from Srikanth Sundarrajan.hdr_qs *oqsh = hdr_qs::access(opkt);hdr_qs *nqsh = hdr_qs::access(npkt);        if (otcp->seqno() == 0 && oqsh->flag() == QS_REQUEST) {                nqsh->flag() = QS_RESPONSE;                nqsh->ttl() = (oiph->ttl() - oqsh->ttl()) % 256;                nqsh->rate() = oqsh->rate();         }        else {                nqsh->flag() = QS_DISABLE;        }}// get the tcp headersntcp->seqno() = acker_->Seqno();//序列号填充ACK???// get the cumulative sequence number to put in the ACK; this is just the left edge of the receive window - 1ntcp->ts() = now;//将时间填充到ACK的TCP包头// timestamp the packetif (ts_echo_bugfix_)  /* TCP/IP Illustrated, Vol. 2, pg. 870 */  //以下对是否启用时间戳的处理ntcp->ts_echo() = acker_->ts_to_echo();elsentcp->ts_echo() = otcp->ts();// echo the original's time stamphdr_ip* oip = hdr_ip::access(opkt);//接收到的数据包的IP包头指针hdr_ip* nip = hdr_ip::access(npkt);//用于构建的ACK包的IP包头指针// get the ip headersnip->flowid() = oip->flowid();//接收到的数据包的IP包的ID赋给构建的ACK包的IP包头// copy the flow idhdr_flags* of = hdr_flags::access(opkt);//接收到的数据包的包标记指针hdr_flags* nf = hdr_flags::access(npkt);//构建的ACK包的包标记指针hdr_flags* sf; //接收端已经存储的包标记指针,主要用于对延迟包的处理,无需详细解释if (save_ != NULL)sf = hdr_flags::access(save_);else sf = 0;// Look at delayed packet being acked. if ( (sf != 0 && sf->cong_action()) || of->cong_action() ) //如果接收端有已经存储的包标记且该标记的拥塞响应位为真或者刚收到的新数据包的拥塞响应位为真,后一种情况是研究重点// Sender has responsed to congestion. acker_->update_ecn_unacked(0);//确认器将未确认的拥塞指示置为已经确认(即没有没确认,有点拗口,但就是这样理解)if ( (sf != 0 && sf->ect() && sf->ce())  || (of->ect() && of->ce()) )//如果接收端有已经存储的包标记且该标记的ECT位和CE位都为真或者刚收到的数据包的标记的ECT位和CE位都为真,也就是说该包到达接收端的过程中经历量拥塞,后一种情况是研究重点// New report of congestion.  acker_->update_ecn_unacked(1);//确认器将未确认的拥塞指示置为未确认if ( (sf != 0 && sf->ect()) || of->ect() )//如果接收端有已经存储的包标记且该包标记的ECT位为真或者刚收到的新数据包的标记的ECT位为真,此时并没有考虑CE位的真假,后者为研究重点// Set EcnEcho bit.  nf->ecnecho() = acker_->ecn_unacked();//将确认器的已经存在的位赋值给即将构建的ACK包的标记位ECN-Echoif (!of->ect() && of->ecnecho() || (sf != 0 && !sf->ect() && sf->ecnecho()) ) {//如果刚收到的新的数据包的标记的ECT位为假且ECN-Echo位为真或者接收端有已经存储的包标记且该标记的ECT位位假且ECN-Echo位为真,简而言之,该包到达接收端过程中没有经历拥塞,同时对响应的带有ECE位的ACK做出了响应 // This is the negotiation for ECN-capability. // We are not checking for of->cong_action() also.  // In this respect, this does not conform to the specifications in the internet draft nf->ecnecho() = 1;//直接将即将构建的ACK包的标记的ECN-Echo位置1if (ecn_syn_) //如果是TCP连接刚建立时nf->ect() = 1; //将即将构建的ACK包的标记的ECT位置1,不是研究重点}        //下面两行是对非对称的TCP连接的特殊处理,无需研究acker_->append_ack(hdr_cmn::access(npkt),ntcp, otcp->seqno());add_to_ack(npkt);// the above function is used in TcpAsymSink        // Andrei Gurtov 用于记录序列号        acker_->last_ack_sent_ = ntcp->seqno();        // printf("ACK %d ts %f\n", ntcp->seqno(), ntcp->ts_echo());send(npkt, 0);//发送该ACK包// send it}