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

通讯编程

开发平台:

Visual C++

  1. /* -*- Mode:C++; c-basic-offset:8; tab-width:8; indent-tabs-mode:t -*- */
  2. /*
  3.  * Copyright (c) 1996 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 MASH Research
  17.  *  Group at the University of California Berkeley.
  18.  * 4. Neither the name of the University nor of the Research Group may be
  19.  *    used 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. #ifndef lint
  35. static const char rcsid[] =
  36.     "@(#) $Header: /cvsroot/nsnam/ns-2/classifier/classifier.cc,v 1.42 2005/09/26 09:12:46 lacage Exp $";
  37. #endif
  38. #include <stdlib.h>
  39. #include "config.h"
  40. #include "classifier.h"
  41. #include "packet.h"
  42. static class ClassifierClass : public TclClass {
  43. public:
  44. ClassifierClass() : TclClass("Classifier") {}
  45. TclObject* create(int, const char*const*) {
  46. return (new Classifier());
  47. }
  48. } class_classifier;
  49. Classifier::Classifier() : 
  50. slot_(0), nslot_(0), maxslot_(-1), shift_(0), mask_(0xffffffff), nsize_(0)
  51. {
  52. default_target_ = 0;
  53. bind("offset_", &offset_);
  54. bind("shift_", &shift_);
  55. bind("mask_", &mask_);
  56. }
  57. int Classifier::classify(Packet *p)
  58. {
  59. return (mshift(*((int*) p->access(offset_))));
  60. }
  61. Classifier::~Classifier()
  62. {
  63. delete [] slot_;
  64. }
  65. void Classifier::set_table_size(int nn)
  66. {
  67. nsize_ = nn;
  68. }
  69. void Classifier::alloc(int slot)
  70. {
  71. NsObject** old = slot_;
  72. int n = nslot_;
  73. if (old == 0) 
  74. if (nsize_ != 0) {
  75. //printf("classifier %x set to %d....%dth visitn", this, nsize_, i++);
  76. nslot_ = nsize_;
  77. }
  78. else {
  79. //printf("classifier %x set to 32....%dth visitn", this, j++);
  80. nslot_ = 32;
  81. }
  82. while (nslot_ <= slot) 
  83. nslot_ <<= 1;
  84. slot_ = new NsObject*[nslot_];
  85. memset(slot_, 0, nslot_ * sizeof(NsObject*));
  86. for (int i = 0; i < n; ++i)
  87. slot_[i] = old[i];
  88. delete [] old;
  89. }
  90. void Classifier::install(int slot, NsObject* p)
  91. {
  92. if (slot >= nslot_)
  93. alloc(slot);
  94. slot_[slot] = p;
  95. if (slot >= maxslot_)
  96. maxslot_ = slot;
  97. }
  98. void Classifier::clear(int slot)
  99. {
  100. slot_[slot] = 0;
  101. if (slot == maxslot_) {
  102. while (--maxslot_ >= 0 && slot_[maxslot_] == 0)
  103. ;
  104. }
  105. }
  106. int Classifier::allocPort (NsObject *nullagent)
  107. {
  108. return getnxt (nullagent);
  109. }
  110. int Classifier::getnxt(NsObject *nullagent)
  111. {
  112. int i;
  113. for (i=0; i < nslot_; i++)
  114. if (slot_[i]==0 || slot_[i]==nullagent)
  115. return i;
  116. i=nslot_;
  117. alloc(nslot_);
  118. return i;
  119. }
  120. /*
  121.  * objects only ever see "packet" events, which come either
  122.  * from an incoming link or a local agent (i.e., packet source).
  123.  */
  124. void Classifier::recv(Packet* p, Handler*h)
  125. {
  126. NsObject* node = find(p);
  127. if (node == NULL) {
  128. /*
  129.  * XXX this should be "dropped" somehow.  Right now,
  130.  * these events aren't traced.
  131.  */
  132. Packet::free(p);
  133. return;
  134. }
  135. node->recv(p,h);
  136. }
  137. /*
  138.  * perform the mapping from packet to object
  139.  * perform upcall if no mapping
  140.  */
  141. NsObject* Classifier::find(Packet* p)
  142. {
  143. NsObject* node = NULL;
  144. int cl = classify(p);
  145. if (cl < 0 || cl >= nslot_ || (node = slot_[cl]) == 0) { 
  146. if (default_target_) 
  147. return default_target_;
  148. /*
  149.  * Sigh.  Can't pass the pkt out to tcl because it's
  150.  * not an object.
  151.  */
  152. Tcl::instance().evalf("%s no-slot %ld", name(), cl);
  153. if (cl == TWICE) {
  154. /*
  155.  * Try again.  Maybe callback patched up the table.
  156.  */
  157. cl = classify(p);
  158. if (cl < 0 || cl >= nslot_ || (node = slot_[cl]) == 0)
  159. return (NULL);
  160. }
  161. }
  162. return (node);
  163. }
  164. int Classifier::install_next(NsObject *node) {
  165. int slot = maxslot_ + 1;
  166. install(slot, node);
  167. return (slot);
  168. }
  169. int Classifier::command(int argc, const char*const* argv)
  170. {
  171. Tcl& tcl = Tcl::instance();
  172. if(argc == 2) {
  173.                 if (strcmp(argv[1], "defaulttarget") == 0) {
  174.                         if (default_target_ != 0)
  175.                                 tcl.result(default_target_->name());
  176.                         return (TCL_OK);
  177.                 }
  178.         } else if (argc == 3) {
  179. /*
  180.  * $classifier alloc-port nullagent
  181.  */
  182. if (strcmp(argv[1],"alloc-port") == 0) {
  183. int slot;
  184. NsObject* nullagent =
  185. (NsObject*)TclObject::lookup(argv[2]);
  186. slot = getnxt(nullagent);
  187. tcl.resultf("%u",slot);
  188. return(TCL_OK);
  189. }
  190. /*
  191.  * $classifier clear $slot
  192.  */
  193. if (strcmp(argv[1], "clear") == 0) {
  194. int slot = atoi(argv[2]);
  195. clear(slot);
  196. return (TCL_OK);
  197. }
  198. /*
  199.  * $classifier installNext $node
  200.  */
  201. if (strcmp(argv[1], "installNext") == 0) {
  202. //int slot = maxslot_ + 1;
  203. NsObject* node = (NsObject*)TclObject::lookup(argv[2]);
  204. if (node == NULL) {
  205. tcl.resultf("Classifier::installNext attempt "
  206.     "to install non-object %s into classifier", argv[2]);
  207. return TCL_ERROR;
  208. };
  209. int slot = install_next(node);
  210. tcl.resultf("%u", slot);
  211. return TCL_OK;
  212. }
  213. /*
  214.  * $classifier slot snum
  215.  * returns the name of the object in slot # snum
  216.  */
  217. if (strcmp(argv[1], "slot") == 0) {
  218. int slot = atoi(argv[2]);
  219. if (slot >= 0 && slot < nslot_ && slot_[slot] != NULL) {
  220. tcl.resultf("%s", slot_[slot]->name());
  221. return TCL_OK;
  222. }
  223. tcl.resultf("Classifier: no object at slot %d", slot);
  224. return (TCL_ERROR);
  225. }
  226. /*
  227.  * $classifier findslot $node
  228.  * finds the slot containing $node
  229.  */
  230. if (strcmp(argv[1], "findslot") == 0) {
  231. int slot = 0;
  232. NsObject* node = (NsObject*)TclObject::lookup(argv[2]);
  233. if (node == NULL) {
  234. return (TCL_ERROR);
  235. }
  236. while (slot < nslot_) {
  237. // check if the slot is empty (xuanc, 1/14/02) 
  238. // fix contributed by Frank A. Zdarsky 
  239. // <frank.zdarsky@kom.tu-darmstadt.de>
  240. if (slot_[slot] && 
  241.     strcmp(slot_[slot]->name(), argv[2]) == 0){
  242. tcl.resultf("%u", slot);
  243. return (TCL_OK);
  244. }
  245. slot++;
  246. }
  247. tcl.result("-1");
  248. return (TCL_OK);
  249. }
  250. if (strcmp(argv[1], "defaulttarget") == 0) {
  251. default_target_=(NsObject*)TclObject::lookup(argv[2]);
  252. if (default_target_ == 0)
  253. return TCL_ERROR;
  254. return TCL_OK;
  255. }
  256. } else if (argc == 4) {
  257. /*
  258.  * $classifier install $slot $node
  259.  */
  260. if (strcmp(argv[1], "install") == 0) {
  261. int slot = atoi(argv[2]);
  262. NsObject* node = (NsObject*)TclObject::lookup(argv[3]);
  263. install(slot, node);
  264. return (TCL_OK);
  265. }
  266. }
  267. return (NsObject::command(argc, argv));
  268. }