gk.cc
上传用户:rrhhcc
上传日期:2015-12-11
资源大小:54129k
文件大小:6k
源码类别:

通讯编程

开发平台:

Visual C++

  1. /* -*-  Mode:C++; c-basic-offset:4; tab-width:4; indent-tabs-mode:t -*- */
  2. /*
  3.  * Copyright (c) 1994 Regents of the University of California.
  4.  * All rights reserved.
  5.  *
  6.  * Redistribution and use in source and binary forms, with or without
  7.  * modification, are permitted provided that the following conditions
  8.  * are met:
  9.  * 1. Redistributions of source code must retain the above copyright
  10.  *    notice, this list of conditions and the following disclaimer.
  11.  * 2. Redistributions in binary form must reproduce the above copyright
  12.  *    notice, this list of conditions and the following disclaimer in the
  13.  *    documentation and/or other materials provided with the distribution.
  14.  * 3. All advertising materials mentioning features or use of this software
  15.  *    must display the following acknowledgement:
  16.  *  This product includes software developed by the Computer Systems
  17.  *  Engineering Group at Lawrence Berkeley Laboratory.
  18.  * 4. Neither the name of the University nor of the Laboratory may be used
  19.  *    to endorse or promote products derived from this software without
  20.  *    specific prior written permission.
  21.  *
  22.  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  23.  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  24.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  25.  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  26.  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  27.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  28.  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  29.  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  30.  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  31.  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  32.  * SUCH DAMAGE.
  33.  */
  34. /* Marking scheme proposed by Gibbens and Kelly in "Resource pricing
  35.    and the evolution of Internet"
  36.    Central Idea:
  37.    ------------
  38.    The link maintains a virtual queue with the same arrivals as the
  39.    real queue. However, the capacity of the virtual queue is smaller
  40.    than the capacity of the real queue, i.e., set the capacity of the
  41.    virtual queue, ctilde = ecnlim_ * c_. The buffer is also scaled by
  42.    the same factor, i,e B_{vq} = ecnlim_ * qlim_. 
  43.    When the VQ drops a packet, mark ALL the packets in the real queue
  44.    as well as all the INCOMING packets till the VQ becomes empty
  45.    again.
  46.    Variables:
  47.    ----------
  48.    ecnlim_ : Fraction of the buffer size and the capacity that the VQ
  49.              has.
  50.  
  51.    mark_flag: Indicates that the VQ overflowed and that all outgoing
  52.               packets has to be marked.
  53. */ 
  54. #include "flags.h"
  55. #include "delay.h"
  56. #include "gk.h"
  57. #include "math.h"
  58. static class GKClass : public TclClass {
  59. public:
  60.    GKClass() : TclClass("Queue/GK") {}
  61. TclObject* create(int argc, const char*const* argv) {
  62. if (argc==5)
  63. return (new GK(argv[4]));
  64. else
  65. return (new GK("Drop"));
  66. }
  67. } class_gk;
  68. GK::GK(const char * trace):link_(NULL), EDTrace(NULL), tchan_(0)
  69. {
  70. q_ = new PacketQueue;
  71. pq_ = q_;
  72. bind_bool("drop_front_", &drop_front_);
  73. bind("ecnlim_", &ecnlim_); 
  74. bind("mean_pktsize_", &mean_pktsize_); 
  75. bind("curq_", &curq_); 
  76. vq_len = 0.0;
  77. prev_time = 0.0;
  78. mark_flag = 0;
  79. }
  80. int GK::command(int argc, const char*const* argv) {
  81. Tcl& tcl = Tcl::instance();
  82.   if (argc == 3) {
  83.   if (strcmp(argv[1], "link") == 0) {
  84.   LinkDelay* del = (LinkDelay*)TclObject::lookup(argv[2]);
  85.   if (del == 0) {
  86.   return(TCL_ERROR);
  87.   }
  88.   // set capacity now
  89. link_ = del;
  90.   c_ = del->bandwidth();
  91. c_ = c_ / (8.0 * mean_pktsize_);
  92.   return (TCL_OK);
  93.   }
  94.   if (!strcmp(argv[1], "packetqueue-attach")) {
  95.   delete q_;
  96.   if (!(q_ = (PacketQueue*) TclObject::lookup(argv[2])))
  97.   return (TCL_ERROR);
  98.   else {
  99.   pq_ = q_;
  100.   return (TCL_OK);
  101.   }
  102.   }
  103. // attach a file for variable tracing
  104. if (strcmp(argv[1], "attach") == 0) {
  105. int mode;
  106. const char* id = argv[2];
  107. tchan_ = Tcl_GetChannel(tcl.interp(), (char*)id, &mode);
  108. if (tchan_ == 0) {
  109. tcl.resultf("Vq: trace: can't attach %s for writing", id);
  110. return (TCL_ERROR);
  111. }
  112. return (TCL_OK);
  113. }
  114.   }
  115.   return Queue::command(argc, argv);
  116. }
  117. void GK::enque(Packet* p)
  118. {
  119.   q_->enque(p);
  120.    
  121.   curr_time = Scheduler::instance().clock();
  122.   /*Whenever a packet is enqueued, the actual length of the
  123.     virtual queue is determined */
  124.  
  125.   if(curr_time > prev_time){
  126.   deque_vq();
  127.   }
  128.   /* Add the packet to the VQ */
  129.   vq_len = vq_len + 1.0;
  130.   /* If the VQ overflows, set flag so that all packets may be marked
  131.  till the VQ hits zero again. */
  132.   if(vq_len > (ecnlim_ * qlim_)){
  133.   mark_flag = 1; // Indicates that all outgoing packets has to be marked
  134.   vq_len = vq_len - 1.0;
  135.   }
  136.   
  137.   if (q_->length() >= qlim_) {
  138. if (drop_front_) { /* remove from head of queue */
  139.    Packet *pp = q_->deque();
  140.    drop(pp);
  141. } else {
  142.    q_->remove(p);
  143.    drop(p);
  144. }
  145.   }
  146.   curq_ = q_->length(); 
  147. }
  148. Packet* GK::deque()
  149. {
  150.   /* Check the status of the virtual queue. We do this to update the
  151.  mark_flag.  */
  152.   curr_time = Scheduler::instance().clock();
  153.   deque_vq();
  154.   /* If the Real queue has packets and the mark_flag is set, mark the
  155.  outgoing packet. */
  156.   if((q_->length() > 0) && (mark_flag == 1)){
  157. Packet *pp = q_->deque();
  158. hdr_flags* hf = hdr_flags::access(pp);
  159. if(hf->ect() == 1)  // ECN capable flow
  160. hf->ce() = 1; // Mark the TCP Flow;
  161. return pp;
  162.   }
  163.   else return q_->deque();
  164. }
  165. /* This procedure updates the VQ */
  166. void GK::deque_vq(){
  167.   if(vq_len > 0.0){ 
  168. vq_len = vq_len - (ecnlim_ * c_ * (curr_time - prev_time));
  169. prev_time = curr_time;
  170. /* If the VQ hits zero, unset mark_flag */
  171. if(vq_len <= 0.0){
  172.   vq_len = 0.0;
  173.   mark_flag = 0;
  174. }
  175.   }
  176. }
  177. /*
  178.  * Routine called by TracedVar facility when variables change values.
  179.  * Currently used to trace value of 
  180.  * the instantaneous queue size seen by arriving packets.
  181.  * Note that the tracing of each var must be enabled in tcl to work.
  182.  */
  183. void GK::trace(TracedVar* v)
  184. {
  185. char wrk[500];
  186. const char *p;
  187. if ((p = strstr(v->name(), "curq")) == NULL) {
  188. fprintf(stderr, "Vq:unknown trace var %sn", v->name());
  189. return;
  190. }
  191. if (tchan_) {
  192. int n;
  193. double t = Scheduler::instance().clock();
  194. // XXX: be compatible with nsv1 RED trace entries
  195. if (*p == 'c') {
  196. sprintf(wrk, "Q %g %d", t, int(*((TracedInt*) v)));
  197. } else {
  198. sprintf(wrk, "%c %g %g", *p, t, double(*((TracedDouble*) v)));
  199. }
  200. n = strlen(wrk);
  201. wrk[n] = 'n'; 
  202. wrk[n+1] = 0;
  203. (void)Tcl_Write(tchan_, wrk, n+1);
  204. }
  205. return; 
  206. }