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

通讯编程

开发平台:

Visual C++

  1. /*
  2.  * Copyright (c) 2003  International Computer Science Institute
  3.  * All rights reserved.
  4.  *
  5.  * Redistribution and use in source and binary forms, with or without
  6.  * modification, are permitted provided that the following conditions
  7.  * are met:
  8.  * 1. Redistributions of source code must retain the above copyright
  9.  *    notice, this list of conditions and the following disclaimer.
  10.  * 2. Redistributions in binary form must reproduce the above copyright
  11.  *    notice, this list of conditions and the following disclaimer in the
  12.  *    documentation and/or other materials provided with the distribution.
  13.  * 3. All advertising materials mentioning features or use of this software
  14.  *    must display the following acknowledgement:
  15.  *      This product includes software developed by ICIR, the ICSI
  16.  *      Center for Internet Research (ICSI: the International Computer
  17.  *      Science Institute).
  18.  * 4. Neither the name of ICIR nor of ICSI 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 ICSI 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 ICSI 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.  */
  35. #include "delayer.h"
  36. Delayer::Delayer() : Connector(), last_sent(-100000), alloc_int(NULL), 
  37. alloc_len(NULL),  alloc_free(1), at_(*this), 
  38. spike_int(NULL), spike_len(NULL), spike_free(1),
  39. st_(*this), target_free(1), th_ (*this), prev_h_(NULL), pkt_(NULL) {
  40. }
  41.  
  42. void Delayer::recv(Packet* p, Handler* h) {
  43. double now = Scheduler::instance().clock(); 
  44. if (pkt_ != NULL) {
  45. printf("delayer not empty!n");
  46. exit(1);
  47.   }
  48. prev_h_ = h;
  49. pkt_ = p;
  50.         // if queue has been empty then trigger channel allocation delay
  51.         if (alloc_len && now - last_sent > alloc_int->value()) {        
  52.                         alloc_free = 0;
  53.                         at_.resched(alloc_len->value());
  54.                         return;
  55.         }
  56. try_send();
  57. }
  58. void Delayer::try_send() {
  59. double now = Scheduler::instance().clock(); 
  60. if (debug_)
  61.         printf("now %f last_sent %f alloc_free %d target_free %d, spike_free %dn", 
  62. now, last_sent, alloc_free, target_free, spike_free);
  63. if (!target_free || !alloc_free || !spike_free)
  64. return;
  65. if (pkt_ && target_) {
  66.                 target_->recv(pkt_, &th_);
  67. last_sent = Scheduler::instance().clock(); 
  68. target_free = 0;
  69. pkt_ = NULL;
  70. }
  71. if (prev_h_)
  72. prev_h_->handle(&e);
  73. }
  74. int Delayer::command(int argc, const char*const* argv) {
  75.         // explicitly block the queue to model a delay spike
  76.         if (argc == 2 && !strcmp(argv[1],"block") 
  77. && !spike_len 
  78. ) {
  79.         spike_free = 0;
  80.          return (TCL_OK);
  81.         }
  82.         if (argc == 2 && !strcmp(argv[1],"unblock") 
  83. // && !spike_len
  84. ) {
  85.                 spike_free = 1;
  86.                 try_send();
  87.                 return (TCL_OK);
  88.         }
  89.         // set distributions for channel allocation delay
  90.         if (argc == 4 && !strcmp(argv[1],"alloc")) {
  91.                 alloc_int = (RandomVariable *)TclObject::lookup(argv[2]);
  92.                 alloc_len = (RandomVariable *)TclObject::lookup(argv[3]);
  93.                 return (TCL_OK);
  94.         }
  95.         // set distributions for delay spikes
  96.         if (argc == 4 && !strcmp(argv[1],"spike")) {
  97.                 spike_int = (RandomVariable *)TclObject::lookup(argv[2]);
  98.                 spike_len = (RandomVariable *)TclObject::lookup(argv[3]);
  99. st_.sched(getNextSpikeInt());
  100.                 return (TCL_OK);
  101.         }
  102.         return Connector::command(argc, argv);
  103. }
  104. void SpikeTimer::expire(Event*) {
  105. // printf("SpikeHandler, spike_free %dn", delayer_.getSpike());
  106.    if (delayer_.getSpike()) {
  107.             delayer_.takeSpike();
  108.                 resched(delayer_.getNextSpikeLen());
  109.                return;
  110.     }
  111.     delayer_.freeSpike();
  112.     resched(delayer_.getNextSpikeInt());
  113.     delayer_.try_send();
  114. }
  115. void AllocTimer::expire(Event*) {
  116. delayer_.freeAlloc(); 
  117. delayer_.try_send();
  118. // printf("AllocHandlern");
  119. }
  120. void TargetHandler::handle(Event*) {
  121. delayer_.freeTarget(); 
  122. delayer_.try_send();
  123. // printf("TargetHandlern");
  124. }