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

通讯编程

开发平台:

Visual C++

  1. //
  2. // gear_sender.cc : Gear Sender Main File
  3. // author         : Fabio Silva
  4. //
  5. // Copyright (C) 2000-2003 by the University of Southern California
  6. // $Id: gear_sender.cc,v 1.5 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_sender.hh"
  43. #include <unistd.h>
  44. #ifdef NS_DIFFUSION
  45. static class GearSenderAppClass : public TclClass {
  46. public:
  47.   GearSenderAppClass() : TclClass("Application/DiffApp/GearSenderApp") {}
  48.   TclObject * create(int argc, const char*const* argv) {
  49.     return (new GearSenderApp());
  50.   }
  51. } class_gear_sender_app_class;
  52. void GearSendDataTimer::expire(Event *e) {
  53.   a_->send();
  54. }
  55. void GearSenderApp::send()
  56. {
  57.   struct timeval tmv;
  58.   int retval;
  59.   
  60.   // Send data if we have active subscriptions
  61.   if ((num_subscriptions_ > 0) || using_push_)
  62.     {
  63.       // Update time in the packet
  64.       GetTime(&tmv);
  65.       lastEventTime_->seconds_ = tmv.tv_sec;
  66.       lastEventTime_->useconds_ = tmv.tv_usec;
  67.       
  68.       // Send data probe
  69.       DiffPrint(DEBUG_ALWAYS, "Node%d: Sending Data %dn", ((DiffusionRouting *)dr_)->getNodeId(), last_seq_sent_);
  70.       retval = dr_->send(pubHandle_, &data_attr_);
  71.       
  72.       // Update counter
  73.       last_seq_sent_++;
  74.       counterAttr_->setVal(last_seq_sent_);
  75.     }
  76.   // re-schedule the timer 
  77.   sdt_.resched(SEND_DATA_INTERVAL);
  78. }
  79. int GearSenderApp::command(int argc, const char*const* argv) {
  80.   if (argc == 2) {
  81.     if (strcmp(argv[1], "publish") == 0) {
  82.       run();
  83.       return TCL_OK;
  84.     }
  85.   }
  86.   else if (argc >= 6) {
  87.     // TCL API $app push-pull-option <push/pull> <point/region> <co-od1> <co-od2> <co-od3> <co-od4>
  88.     if (strcmp(argv[1], "push-pull-options") == 0) {
  89.       if (strcmp(argv[2], "push") == 0) 
  90. using_push_ = true;
  91.       else
  92. using_push_ = false;
  93.       if (strcmp(argv[3], "point") == 0) 
  94. {
  95.   using_points_ = true;
  96.   if (argv[4] != NULL && argv[5] != NULL) 
  97.     {
  98.       lat_pt_ = atoi(argv[4]);
  99.       long_pt_ = atoi(argv[5]);
  100.     }
  101. }
  102.       else
  103. {
  104.   using_points_ = false;
  105.   if (argv[4] != NULL && argv[5] != NULL && argv[6] != NULL && argv[7] != NULL)
  106.     {
  107.       lat_min_ = atoi(argv[4]);
  108.       lat_max_ = atoi(argv[5]);
  109.       long_min_ = atoi(argv[6]);
  110.       long_max_ = atoi(argv[7]);
  111.     }
  112. }
  113.       return TCL_OK;
  114.     }
  115.   }
  116.   return DiffApp::command(argc, argv);
  117. }
  118. #endif //NS_DIFFUSION
  119. void GearSenderReceive::recv(NRAttrVec *data, NR::handle my_handle)
  120. {
  121.   app_->recv(data, my_handle);
  122. }
  123. void GearSenderApp::recv(NRAttrVec *data, NR::handle my_handle)
  124. {
  125.   NRSimpleAttribute<int> *nrclass = NULL;
  126.   nrclass = NRClassAttr.find(data);
  127.   if (!nrclass){
  128.     DiffPrint(DEBUG_ALWAYS, "Received a BAD packet !n");
  129.     return;
  130.   }
  131.   switch (nrclass->getVal()){
  132.   case NRAttribute::INTEREST_CLASS:
  133.     DiffPrint(DEBUG_ALWAYS, "Received an Interest message !n");
  134.     num_subscriptions_++;
  135.     break;
  136.   case NRAttribute::DISINTEREST_CLASS:
  137.     DiffPrint(DEBUG_ALWAYS, "Received a Disinterest message !n");
  138.     num_subscriptions_--;
  139.     break;
  140.   default:
  141.     DiffPrint(DEBUG_ALWAYS, "Received an unknown message (%d)!n", nrclass->getVal());
  142.     break;
  143.   }
  144. }
  145. handle GearSenderApp::setupSubscription()
  146. {
  147.   NRAttrVec attrs;
  148.   attrs.push_back(NRClassAttr.make(NRAttribute::NE,
  149.    NRAttribute::DATA_CLASS));
  150.   attrs.push_back(NRScopeAttr.make(NRAttribute::IS,
  151.    NRAttribute::NODE_LOCAL_SCOPE));
  152.   attrs.push_back(GearTargetAttr.make(NRAttribute::EQ, "F117A"));
  153.   attrs.push_back(LatitudeAttr.make(NRAttribute::IS, lat_pt_));
  154.   attrs.push_back(LongitudeAttr.make(NRAttribute::IS, long_pt_));
  155.   handle h = dr_->subscribe(&attrs, mr_);
  156.   ClearAttrs(&attrs);
  157.   return h;
  158. }
  159. handle GearSenderApp::setupPublication()
  160. {
  161.   NRAttrVec attrs;
  162.   attrs.push_back(NRClassAttr.make(NRAttribute::IS,
  163.    NRAttribute::DATA_CLASS));
  164.   // Use push or pull semantics
  165.   if (using_push_){
  166.     attrs.push_back(NRScopeAttr.make(NRAttribute::IS,
  167.      NRAttribute::GLOBAL_SCOPE));
  168.     // Insert point or region coordinates
  169.     if (using_points_){
  170.       attrs.push_back(LatitudeAttr.make(NRAttribute::EQ, lat_pt_));
  171.       attrs.push_back(LongitudeAttr.make(NRAttribute::EQ, long_pt_));
  172.     }
  173.     else{
  174.       attrs.push_back(LatitudeAttr.make(NRAttribute::GE, lat_min_));
  175.       attrs.push_back(LatitudeAttr.make(NRAttribute::LE, lat_max_));
  176.       attrs.push_back(LongitudeAttr.make(NRAttribute::GE, long_min_));
  177.       attrs.push_back(LongitudeAttr.make(NRAttribute::LE, long_max_));
  178.     }
  179.   }
  180.   else{
  181.     attrs.push_back(NRScopeAttr.make(NRAttribute::IS,
  182.      NRAttribute::NODE_LOCAL_SCOPE));
  183.     // Insert point coordinates
  184.     attrs.push_back(LatitudeAttr.make(NRAttribute::IS, lat_pt_));
  185.     attrs.push_back(LongitudeAttr.make(NRAttribute::IS, long_pt_));
  186.   }
  187.   attrs.push_back(GearTargetAttr.make(NRAttribute::IS, "F117A"));
  188.   handle h = dr_->publish(&attrs);
  189.   ClearAttrs(&attrs);
  190.   return h;
  191. }
  192. void GearSenderApp::run()
  193. {
  194.   struct timeval tmv;
  195. #ifndef NS_DIFFUSION
  196.   int retval;
  197. #endif // !NS_DIFFUSION
  198.   
  199. #ifdef INTERACTIVE
  200.   char input;
  201.   fd_set FDS;
  202. #endif // INTERATIVE
  203.   // Setup publication and subscription
  204.   if (!using_push_)
  205.     subHandle_ = setupSubscription();
  206.   pubHandle_ = setupPublication();
  207.   // Create time attribute
  208.   GetTime(&tmv);
  209.   lastEventTime_ = new EventTime;
  210.   lastEventTime_->seconds_ = tmv.tv_sec;
  211.   lastEventTime_->useconds_ = tmv.tv_usec;
  212.   timeAttr_ = GearTimeAttr.make(NRAttribute::IS, (void *) &lastEventTime_,
  213.     sizeof(EventTime));
  214.   data_attr_.push_back(timeAttr_);
  215.   // Change pointer to point to attribute's payload
  216.   delete lastEventTime_;
  217.   lastEventTime_ = (EventTime *) timeAttr_->getVal();
  218.   // Create counter attribute
  219.   counterAttr_ = GearCounterAttr.make(NRAttribute::IS, last_seq_sent_);
  220.   data_attr_.push_back(counterAttr_);
  221. #ifndef NS_DIFFUSION
  222.   // Main thread will send ping probes
  223.   while(1){
  224. #ifdef INTERACTIVE
  225.     FD_SET(0, &FDS);
  226.     fprintf(stdout, "Press <Enter> to send a ping probe...");
  227.     fflush(NULL);
  228.     select(1, &FDS, NULL, NULL, NULL);
  229.     input = getc(stdin);
  230. #else
  231.     sleep(SEND_DATA_INTERVAL);
  232. #endif // INTERACTIVE
  233.     // Send data packet if we have active subscriptions
  234.     if ((num_subscriptions_ > 0) || using_push_){
  235.       // Update time in the packet
  236.       GetTime(&tmv);
  237.       lastEventTime_->seconds_ = tmv.tv_sec;
  238.       lastEventTime_->useconds_ = tmv.tv_usec;
  239.       // Send data probe
  240.       DiffPrint(DEBUG_ALWAYS, "Node%d: Sending Data %dn", ((DiffusionRouting *)dr_)->getNodeId(), last_seq_sent_);
  241.       retval = dr_->send(pubHandle_, &data_attr_);
  242.       // Update counter
  243.       last_seq_sent_++;
  244.       counterAttr_->setVal(last_seq_sent_);
  245.     }
  246.   }
  247. #else
  248.   send();
  249. #endif // !NS_DIFFUSION
  250. }
  251. void GearSenderApp::usage(char *s){
  252.   DiffPrint(DEBUG_ALWAYS, "Usage: %s [-d debug] [-p port] [-s] [-r] [-h]nn", s);
  253.   DiffPrint(DEBUG_ALWAYS, "t-d - Sets debug level (0-10)n");
  254.   DiffPrint(DEBUG_ALWAYS, "t-p - Uses port 'port' to talk to diffusionn");
  255.   DiffPrint(DEBUG_ALWAYS, "t-s - Uses push semantics (default: pull)n");
  256.   DiffPrint(DEBUG_ALWAYS, "t-r - Uses regions (default: points)n");
  257.   DiffPrint(DEBUG_ALWAYS, "t-h - Prints this informationn");
  258.   DiffPrint(DEBUG_ALWAYS, "n");
  259.   exit(0);
  260. }
  261. void GearSenderApp::parseCommandLine(int argc, char **argv)
  262. {
  263.   u_int16_t diff_port = DEFAULT_DIFFUSION_PORT;
  264.   int debug_level;
  265.   int opt;
  266.   config_file_ = NULL;
  267.   using_points_ = true;
  268.   using_push_ = false;
  269.   opterr = 0;
  270.   while (1){
  271.     opt = getopt(argc, argv, "srhd:p:");
  272.     switch (opt){
  273.     case 'p':
  274.       diff_port = (u_int16_t) atoi(optarg);
  275.       if ((diff_port < 1024) || (diff_port >= 65535)){
  276. DiffPrint(DEBUG_ALWAYS, "Error: Diffusion port must be between 1024 and 65535 !n");
  277. exit(-1);
  278.       }
  279.       break;
  280.     case 'h':
  281.       usage(argv[0]);
  282.       break;
  283.     case 'd':
  284.       debug_level = atoi(optarg);
  285.       if (debug_level < 1 || debug_level > 10){
  286. DiffPrint(DEBUG_ALWAYS, "Error: Debug level outside range or missing !n");
  287. usage(argv[0]);
  288.       }
  289.       global_debug_level = debug_level;
  290.       break;
  291.     case 's':
  292.       using_push_ = true;
  293.       break;
  294.     case 'r':
  295.       using_points_ = false;
  296.       break;
  297.     case '?':
  298.       DiffPrint(DEBUG_ALWAYS,
  299. "Error: %c isn't a valid option or its parameter is missing !n", optopt);
  300.       usage(argv[0]);
  301.       break;
  302.     case ':':
  303.       DiffPrint(DEBUG_ALWAYS, "Parameter missing !n");
  304.       usage(argv[0]);
  305.       break;
  306.     }
  307.     if (opt == -1)
  308.       break;
  309.   }
  310.   diffusion_port_ = diff_port;
  311. }
  312. void GearSenderApp::readGeographicCoordinates()
  313. {
  314.   char *gear_coord_env;
  315.   if (using_points_ || !using_push_){
  316.     // Read latitude coordinate
  317.     gear_coord_env = getenv("lat_pt");
  318.     if (!gear_coord_env){
  319.       DiffPrint(DEBUG_ALWAYS, "Cannot read lat_pt !n");
  320.       exit(-1);
  321.     }
  322.     lat_pt_ = atoi(gear_coord_env);
  323.     // Read longitude coordinate
  324.     gear_coord_env = getenv("long_pt");
  325.     if (!gear_coord_env){
  326.       DiffPrint(DEBUG_ALWAYS, "Cannot read long_pt !n");
  327.       exit(-1);
  328.     }
  329.     long_pt_ = atoi(gear_coord_env);
  330.   }
  331.   else{
  332.     // Read latitude_min value
  333.     gear_coord_env = getenv("lat_min");
  334.     if (!gear_coord_env){
  335.       DiffPrint(DEBUG_ALWAYS, "Cannot read lat_min !n");
  336.       exit(-1);
  337.     }
  338.     lat_min_ = atoi(gear_coord_env);
  339.     // Read latitude_max value
  340.     gear_coord_env = getenv("lat_max");
  341.     if (!gear_coord_env){
  342.       DiffPrint(DEBUG_ALWAYS, "Cannot read lat_max !n");
  343.       exit(-1);
  344.     }
  345.     lat_max_ = atoi(gear_coord_env);
  346.     // Read longitude_min value
  347.     gear_coord_env = getenv("long_min");
  348.     if (!gear_coord_env){
  349.       DiffPrint(DEBUG_ALWAYS, "Cannot read long_min !n");
  350.       exit(-1);
  351.     }
  352.     long_min_ = atoi(gear_coord_env);
  353.     // Read longitude_max value
  354.     gear_coord_env = getenv("long_max");
  355.     if (!gear_coord_env){
  356.       DiffPrint(DEBUG_ALWAYS, "Cannot read long_max !n");
  357.       exit(-1);
  358.     }
  359.     long_max_ = atoi(gear_coord_env);
  360.   }
  361. }
  362. #ifdef NS_DIFFUSION
  363. GearSenderApp::GearSenderApp() : sdt_(this)
  364. #else
  365. GearSenderApp::GearSenderApp(int argc, char **argv)
  366. #endif //NS_DIFFUSION
  367. {
  368.   last_seq_sent_ = 0;
  369.   num_subscriptions_ = 0;
  370.   mr_ = new GearSenderReceive(this);
  371. #ifndef NS_DIFFUSION
  372.   parseCommandLine(argc, argv);
  373.   readGeographicCoordinates();
  374.   dr_ = NR::createNR(diffusion_port_);
  375. #endif //!NS_DIFFUSION
  376. }
  377. #ifndef USE_SINGLE_ADDRESS_SPACE
  378. int main(int argc, char **argv)
  379. {
  380.   GearSenderApp *app;
  381.   app = new GearSenderApp(argc, argv);
  382.   app->run();
  383.   return 0;
  384. }
  385. #endif //!USE_SINGLE_ADDRESS_SPACE