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

通讯编程

开发平台:

Visual C++

  1. // 
  2. // events.cc     : Provides the EventQueue
  3. // authors       : Lewis Girod John Heidemann and Fabio Silva
  4. //
  5. // Copyright (C) 2000-2002 by the University of Southern California
  6. // $Id: events.cc,v 1.10 2008/03/27 05:24:46 tom_henderson Exp $
  7. //
  8. // This program is free software; you can redistribute it and/or
  9. // modify it under the terms of the GNU General Public License,
  10. // version 2, as published by the Free Software Foundation.
  11. //
  12. // This program is distributed in the hope that it will be useful,
  13. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  15. // GNU General Public License for more details.
  16. //
  17. // You should have received a copy of the GNU General Public License along
  18. // with this program; if not, write to the Free Software Foundation, Inc.,
  19. // 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
  20. //
  21. // Linking this file statically or dynamically with other modules is making
  22. // a combined work based on this file.  Thus, the terms and conditions of
  23. // the GNU General Public License cover the whole combination.
  24. //
  25. // In addition, as a special exception, the copyright holders of this file
  26. // give you permission to combine this file with free software programs or
  27. // libraries that are released under the GNU LGPL and with code included in
  28. // the standard release of ns-2 under the Apache 2.0 license or under
  29. // otherwise-compatible licenses with advertising requirements (or modified
  30. // versions of such code, with unchanged license).  You may copy and
  31. // distribute such a system following the terms of the GNU GPL for this
  32. // file and the licenses of the other code concerned, provided that you
  33. // include the source code of that other code when and as the GNU GPL
  34. // requires distribution of source code.
  35. //
  36. // Note that people who make modified versions of this file are not
  37. // obligated to grant this special exception for their modified versions;
  38. // it is their choice whether to do so.  The GNU General Public License
  39. // gives permission to release a modified version without this exception;
  40. // this exception also makes it possible to release a modified version
  41. // which carries forward this exception.
  42. #include <stdio.h>
  43. #include <stdlib.h>
  44. #include "events.hh"
  45. //
  46. //  Timeval utility routines
  47. //
  48. int TimevalCmp(struct timeval *x, struct timeval *y)
  49. {
  50.   if (x->tv_sec > y->tv_sec) return 1;
  51.   if (x->tv_sec < y->tv_sec) return -1;
  52.   if (x->tv_usec > y->tv_usec) return 1;
  53.   if (x->tv_usec < y->tv_usec) return -1;
  54.   return 0;
  55. }
  56. void TimevalSub(struct timeval *x, struct timeval *y, struct timeval *tv)
  57. {
  58.   if (TimevalCmp(x, y) < 0) 
  59.     tv->tv_sec = tv->tv_usec = 0;
  60.   else{
  61.     *tv = *x;
  62.     // borrow..
  63.     if (tv->tv_usec < y->tv_usec){
  64.       tv->tv_usec += 1000000;
  65.       tv->tv_sec--;
  66.     }
  67.     // sub
  68.     tv->tv_usec -= y->tv_usec;
  69.     tv->tv_sec -= y->tv_sec;
  70.   }
  71. }
  72. void TimevalAddusecs(struct timeval *tv, int usecs)
  73. {
  74.   tv->tv_usec += usecs;
  75.   if (tv->tv_usec > 1000000){
  76.     tv->tv_sec += tv->tv_usec / 1000000;
  77.     tv->tv_usec = tv->tv_usec % 1000000;
  78.   }
  79. }
  80. //
  81. //  EventQueue Methods
  82. //
  83. int EventQueue::eventCmp(QueueEvent *x, QueueEvent *y)
  84. {
  85.   return TimevalCmp(&(x->tv_), &(y->tv_));
  86. }
  87. void EventQueue::eqAdd(QueueEvent *e)
  88. {
  89.   QueueEvent *ptr = head_;
  90.   QueueEvent *last = ptr;
  91.   while (ptr && (eventCmp(e, ptr) > 0)){
  92.     last = ptr; 
  93.     ptr = ptr->next_;
  94.   }
  95.   if (last == ptr){
  96.     e->next_ = head_;
  97.     head_ = e;
  98.   }
  99.   else{
  100.     e->next_ = ptr;
  101.     last->next_ = e;
  102.   }
  103. }
  104. QueueEvent * EventQueue::eqPop()
  105. {
  106.   QueueEvent *e = head_;
  107.   if (e){
  108.     head_ = head_->next_;
  109.     e->next_ = NULL;
  110.   }
  111.   return e;
  112. }
  113. QueueEvent * EventQueue::eqFindEvent(handle hdl)
  114. {
  115.   return eqFindNextEvent(hdl, head_);
  116. }
  117. QueueEvent * EventQueue::eqFindNextEvent(handle hdl, QueueEvent *e)
  118. {
  119.   while (e){
  120.     if (e->handle_ == hdl)
  121.       return e;
  122.     e = e->next_;
  123.   }
  124.   return e;
  125. }
  126. void EventQueue::eqAddAfter(handle hdl, void *payload, int delay_msec)
  127. {
  128.   QueueEvent *e = new QueueEvent;
  129.   e->handle_ = hdl;
  130.   e->payload_ = payload;
  131.   setDelay(e, delay_msec);
  132.   eqAdd(e);
  133. }
  134. void EventQueue::setDelay(QueueEvent *e, int delay_msec)
  135. {
  136.   GetTime(&(e->tv_));
  137.   TimevalAddusecs(&(e->tv_), delay_msec * 1000);
  138. }
  139. void EventQueue::eqNextTimer(struct timeval *tv)
  140. {
  141.   struct timeval now;
  142.   if (head_){
  143.     GetTime(&now);
  144.     TimevalSub(&(head_->tv_), &now, tv);
  145.   }
  146.   else{
  147.     tv->tv_sec = MAXVALUE;
  148.     tv->tv_usec = 0;
  149.   }
  150. }
  151. int EventQueue::eqTopInPast()
  152. {
  153.   struct timeval now;
  154.   if (head_){
  155.     GetTime(&now);
  156.     return (TimevalCmp(&(head_->tv_), &now) <= 0);
  157.   }
  158.   return 0;
  159. }
  160. void EventQueue::eqPrint()
  161. {
  162.   QueueEvent *e = head_;
  163.   for (; (e); e = e->next_){
  164.     fprintf(stderr, "time = %ld:%06ld, handle = %ldn",
  165.     e->tv_.tv_sec, (long int) e->tv_.tv_usec, e->handle_);
  166.   }
  167. }
  168. bool EventQueue::eqRemoveEvent(QueueEvent *e)
  169. {
  170.   QueueEvent *ce, *pe;
  171.   if (e){
  172.     if (head_ == e){
  173.       head_ = e->next_;
  174.       return true;
  175.     }
  176.     pe = head_;
  177.     ce = head_->next_;
  178.     while (ce){
  179.       if (ce == e){
  180. pe->next_ = e->next_;
  181. return true;
  182.       }
  183.       else{
  184. pe = ce;
  185. ce = ce->next_;
  186.       }
  187.     }
  188.   }
  189.   return false;
  190. }
  191. bool EventQueue::eqRemove(handle hdl)
  192. {
  193.   QueueEvent *e = NULL;
  194.   // Look in the event queue for an event with handle hdl
  195.   e = eqFindEvent(hdl);
  196.   // If found, delete event
  197.   if (e){
  198.     return (eqRemoveEvent(e));
  199.   }
  200.   return false;
  201. }
  202. // Computes a randomized delay, measured in milliseconds. Note that
  203. // most OS timers have granularities on order of 5-15 ms.
  204. // timer[2] = { base_timer, variance }
  205. int EventQueue::randDelay(int timer[2])
  206. {
  207.   return (int) (timer[0] + ((((float) GetRand()) /
  208.      ((float) RAND_MAX)) - 0.5) * timer[1]);
  209. }