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

通讯编程

开发平台:

Visual C++

  1. //
  2. // gear_receiver.cc : Gear Receiver Main File
  3. // author           : Fabio Silva
  4. //
  5. // Copyright (C) 2000-2003 by the University of Southern California
  6. // $Id: gear_receiver.cc,v 1.6 2005/09/13 04:53:46 tomh 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 "gear_receiver.hh"
  43. #ifdef NS_DIFFUSION
  44. static class GearReceiverAppClass : public TclClass {
  45. public:
  46.   GearReceiverAppClass() : TclClass("Application/DiffApp/GearReceiverApp") {}
  47.   TclObject * create(int argc, const char*const* argv) {
  48.     return (new GearReceiverApp());
  49.   }
  50. } class_gear_receiver_app_class;
  51. int GearReceiverApp::command(int argc, const char*const* argv) {
  52.   if (argc == 2) {
  53.     if (strcmp(argv[1], "subscribe") == 0) {
  54.       run();
  55.       return TCL_OK;
  56.     }
  57.   }
  58.   else if (argc >= 6) {
  59.     // TCL API $app push-pull-options <push/pull> <point/region> <co-od1> <co-od2> <co-od3> <co-od4>
  60.     if (strcmp(argv[1], "push-pull-options") == 0) {
  61.       if (strcmp(argv[2], "push") == 0) 
  62. using_push_ = true;
  63.       else
  64. using_push_ = false;
  65.       if (strcmp(argv[3], "point") == 0) 
  66. {
  67.   using_points_ = true;
  68.   if (argv[4] != NULL && argv[5] != NULL) 
  69.     {
  70.       lat_pt_ = atoi(argv[4]);
  71.       long_pt_ = atoi(argv[5]);
  72.     }
  73. }
  74.       else
  75. {
  76.   using_points_ = false;
  77.   if (argv[4] != NULL && argv[5] != NULL && argv[6] != NULL && argv[7] != NULL)
  78.     {
  79.       lat_min_ = atoi(argv[4]);
  80.       lat_max_ = atoi(argv[5]);
  81.       long_min_ = atoi(argv[6]);
  82.       long_max_ = atoi(argv[7]);
  83.     }
  84. }
  85.       return TCL_OK;
  86.     }
  87.   }
  88.   return DiffApp::command(argc, argv);
  89. }
  90. #endif //NS_DIFFUSION
  91. void GearReceiverReceive::recv(NRAttrVec *data, NR::handle my_handle)
  92. {
  93.   app_->recv(data, my_handle);
  94. }
  95. void GearReceiverApp::recv(NRAttrVec *data, NR::handle my_handle)
  96. {
  97.   NRSimpleAttribute<int> *counterAttr = NULL;
  98.   NRSimpleAttribute<void *> *timeAttr = NULL;
  99.   EventTime *probe_event;
  100.   long delay_seconds;
  101.   long delay_useconds;
  102.   float total_delay;
  103.   struct timeval tmv;
  104.   GetTime(&tmv);
  105.   counterAttr = GearCounterAttr.find(data);
  106.   timeAttr = GearTimeAttr.find(data);
  107.   if (!counterAttr || !timeAttr){
  108.     DiffPrint(DEBUG_ALWAYS, "Received a BAD packet !n");
  109.     PrintAttrs(data);
  110.     return;
  111.   }
  112.   // Calculate latency
  113.   probe_event = (EventTime *) timeAttr->getVal();
  114.   delay_seconds = tmv.tv_sec;
  115.   delay_useconds = tmv.tv_usec;
  116.   if ((delay_seconds < probe_event->seconds_) ||
  117.       ((delay_seconds == probe_event->seconds_) &&
  118.        (delay_useconds < probe_event->useconds_))){
  119.     // Time's not synchronized
  120.     delay_seconds = -1;
  121.     delay_useconds = 0;
  122.     DiffPrint(DEBUG_ALWAYS, "Error calculating delay !n");
  123.   }
  124.   else{
  125.     delay_seconds = delay_seconds - probe_event->seconds_;
  126.     if (delay_useconds < probe_event->useconds_){
  127.       delay_seconds--;
  128.       delay_useconds = delay_useconds + 1000000;
  129.     }
  130.     delay_useconds = delay_useconds - probe_event->useconds_;
  131.   }
  132.   total_delay = (float) (1.0 * delay_seconds) + ((float) delay_useconds / 1000000.0);
  133.   // Check if this is the first message received
  134.   if (first_msg_recv_ < 0){
  135.     first_msg_recv_ = counterAttr->getVal();
  136.   }
  137.   // Print output message
  138.   if (last_seq_recv_ >= 0){
  139.     if (counterAttr->getVal() < last_seq_recv_){
  140.       // Multiple sources detected, disabling statistics
  141.       last_seq_recv_ = -1;
  142.       DiffPrint(DEBUG_ALWAYS, "Node%d: Received data %d, total latency = %f!n",
  143. ((DiffusionRouting *)dr_)->getNodeId(),
  144. counterAttr->getVal(), total_delay);
  145.     }
  146.     else{
  147.       last_seq_recv_ = counterAttr->getVal();
  148.       num_msg_recv_++;
  149.       DiffPrint(DEBUG_ALWAYS, "Node%d: Received data: %d, total latency = %f, %% messages received: %f !n",
  150. ((DiffusionRouting *)dr_)->getNodeId(),
  151. last_seq_recv_, total_delay,
  152. (float) ((num_msg_recv_ * 100.00) /
  153.  ((last_seq_recv_ - first_msg_recv_) + 1)));
  154.     }
  155.   }
  156.   else{
  157.     DiffPrint(DEBUG_ALWAYS, "Node%d: Received data %d, total latency = %f !n",
  158.       ((DiffusionRouting *)dr_)->getNodeId(),
  159.       counterAttr->getVal(), total_delay);
  160.   }
  161. }
  162. handle GearReceiverApp::setupSubscription()
  163. {
  164.   NRAttrVec attrs;
  165.   attrs.push_back(NRClassAttr.make(NRAttribute::IS,
  166.    NRAttribute::INTEREST_CLASS));
  167.   // Use push or pull semantics
  168.   if (using_push_){
  169.     attrs.push_back(NRScopeAttr.make(NRAttribute::IS,
  170.      NRAttribute::NODE_LOCAL_SCOPE));
  171.     // Insert point coordinates
  172.     attrs.push_back(LatitudeAttr.make(NRAttribute::IS, lat_pt_));
  173.     attrs.push_back(LongitudeAttr.make(NRAttribute::IS, long_pt_));
  174.   }
  175.   else{
  176.     attrs.push_back(NRScopeAttr.make(NRAttribute::IS,
  177.      NRAttribute::GLOBAL_SCOPE));
  178.     // Insert point or region coordinates
  179.     if (using_points_){
  180.       attrs.push_back(LatitudeAttr.make(NRAttribute::EQ, lat_pt_));
  181.       attrs.push_back(LongitudeAttr.make(NRAttribute::EQ, long_pt_));
  182.     }
  183.     else{
  184.       attrs.push_back(LatitudeAttr.make(NRAttribute::GE, lat_min_));
  185.       attrs.push_back(LatitudeAttr.make(NRAttribute::LE, lat_max_));
  186.       attrs.push_back(LongitudeAttr.make(NRAttribute::GE, long_min_));
  187.       attrs.push_back(LongitudeAttr.make(NRAttribute::LE, long_max_));
  188.     }
  189.   }
  190.   attrs.push_back(GearTargetAttr.make(NRAttribute::IS, "F117A"));
  191.   handle h = dr_->subscribe(&attrs, mr_);
  192.   ClearAttrs(&attrs);
  193.   return h;
  194. }
  195. void GearReceiverApp::run()
  196. {
  197.   subHandle_ = setupSubscription();
  198. #ifndef NS_DIFFUSION
  199.   // Do nothing
  200.   while (1){
  201.     sleep(1000);
  202.   }
  203. #endif // !NS_DIFFUSION
  204. }
  205. void GearReceiverApp::usage(char *s){
  206.   DiffPrint(DEBUG_ALWAYS, "Usage: %s [-d debug] [-p port] [-s] [-r] [-h]nn", s);
  207.   DiffPrint(DEBUG_ALWAYS, "t-d - Sets debug level (0-10)n");
  208.   DiffPrint(DEBUG_ALWAYS, "t-p - Uses port 'port' to talk to diffusionn");
  209.   DiffPrint(DEBUG_ALWAYS, "t-s - Uses push semantics (default: pull)n");
  210.   DiffPrint(DEBUG_ALWAYS, "t-r - Uses regions (default: points)n");
  211.   DiffPrint(DEBUG_ALWAYS, "t-h - Prints this informationn");
  212.   DiffPrint(DEBUG_ALWAYS, "n");
  213.   exit(0);
  214. }
  215. void GearReceiverApp::parseCommandLine(int argc, char **argv)
  216. {
  217.   u_int16_t diff_port = DEFAULT_DIFFUSION_PORT;
  218.   int debug_level;
  219.   int opt;
  220.   config_file_ = NULL;
  221.   using_points_ = true;
  222.   using_push_ = false;
  223.   opterr = 0;
  224.   while (1){
  225.     opt = getopt(argc, argv, "srhd:p:");
  226.     switch (opt){
  227.     case 'p':
  228.       diff_port = (u_int16_t) atoi(optarg);
  229.       if ((diff_port < 1024) || (diff_port >= 65535)){
  230. DiffPrint(DEBUG_ALWAYS, "Error: Diffusion port must be between 1024 and 65535 !n");
  231. exit(-1);
  232.       }
  233.       break;
  234.     case 'h':
  235.       usage(argv[0]);
  236.       break;
  237.     case 'd':
  238.       debug_level = atoi(optarg);
  239.       if (debug_level < 1 || debug_level > 10){
  240. DiffPrint(DEBUG_ALWAYS, "Error: Debug level outside range or missing !n");
  241. usage(argv[0]);
  242.       }
  243.       global_debug_level = debug_level;
  244.       break;
  245.     case 's':
  246.       using_push_ = true;
  247.       break;
  248.     case 'r':
  249.       using_points_ = false;
  250.       break;
  251.     case '?':
  252.       DiffPrint(DEBUG_ALWAYS,
  253. "Error: %c isn't a valid option or its parameter is missing !n", optopt);
  254.       usage(argv[0]);
  255.       break;
  256.     case ':':
  257.       DiffPrint(DEBUG_ALWAYS, "Parameter missing !n");
  258.       usage(argv[0]);
  259.       break;
  260.     }
  261.     if (opt == -1)
  262.       break;
  263.   }
  264.   diffusion_port_ = diff_port;
  265. }
  266. void GearReceiverApp::readGeographicCoordinates()
  267. {
  268.   char *gear_coord_env;
  269.   if (using_points_ || using_push_){
  270.     // Read latitude coordinate
  271.     gear_coord_env = getenv("lat_pt");
  272.     if (!gear_coord_env){
  273.       DiffPrint(DEBUG_ALWAYS, "Cannot read lat_pt !n");
  274.       exit(-1);
  275.     }
  276.     lat_pt_ = atoi(gear_coord_env);
  277.     // Read longitude coordinate
  278.     gear_coord_env = getenv("long_pt");
  279.     if (!gear_coord_env){
  280.       DiffPrint(DEBUG_ALWAYS, "Cannot read long_pt !n");
  281.       exit(-1);
  282.     }
  283.     long_pt_ = atoi(gear_coord_env);
  284.   }
  285.   else{
  286.     // Read latitude_min value
  287.     gear_coord_env = getenv("lat_min");
  288.     if (!gear_coord_env){
  289.       DiffPrint(DEBUG_ALWAYS, "Cannot read lat_min !n");
  290.       exit(-1);
  291.     }
  292.     lat_min_ = atoi(gear_coord_env);
  293.     // Read latitude_max value
  294.     gear_coord_env = getenv("lat_max");
  295.     if (!gear_coord_env){
  296.       DiffPrint(DEBUG_ALWAYS, "Cannot read lat_max !n");
  297.       exit(-1);
  298.     }
  299.     lat_max_ = atoi(gear_coord_env);
  300.     // Read longitude_min value
  301.     gear_coord_env = getenv("long_min");
  302.     if (!gear_coord_env){
  303.       DiffPrint(DEBUG_ALWAYS, "Cannot read long_min !n");
  304.       exit(-1);
  305.     }
  306.     long_min_ = atoi(gear_coord_env);
  307.     // Read longitude_max value
  308.     gear_coord_env = getenv("long_max");
  309.     if (!gear_coord_env){
  310.       DiffPrint(DEBUG_ALWAYS, "Cannot read long_max !n");
  311.       exit(-1);
  312.     }
  313.     long_max_ = atoi(gear_coord_env);
  314.   }
  315. }
  316. #ifdef NS_DIFFUSION
  317. GearReceiverApp::GearReceiverApp()
  318. #else
  319. GearReceiverApp::GearReceiverApp(int argc, char **argv)
  320. #endif // NS_DIFFUSION
  321. {
  322.   last_seq_recv_ = 0;
  323.   num_msg_recv_ = 0;
  324.   first_msg_recv_ = -1;
  325.   mr_ = new GearReceiverReceive(this);
  326. #ifndef NS_DIFFUSION
  327.   parseCommandLine(argc, argv);
  328.   readGeographicCoordinates();
  329.   dr_ = NR::createNR(diffusion_port_);
  330. #endif // !NS_DIFFUSION
  331. }
  332. #ifndef NS_ADDRESS
  333. #ifndef USE_SINGLE_ADDRESS_SPACE 
  334. int main(int argc, char **argv)
  335. {
  336.   GearReceiverApp *app;
  337.   app = new GearReceiverApp(argc, argv);
  338.   app->run();
  339.   return 0;
  340. }
  341. #endif // !USE_SINGLE_ADDRESS_SPACE
  342. #endif // !NS_ADDRESS