Logo AND Algorithmique Numérique Distribuée

Public GIT Repository
cosmetics
[simgrid.git] / src / surf / ns3 / ns3_simulator.cc
1 /* Copyright (c) 2007-2016. The SimGrid Team. All rights reserved.          */
2
3 /* This program is free software; you can redistribute it and/or modify it
4  * under the terms of the license (GNU LGPL) which comes with this package. */
5
6 #include "src/surf/ns3/ns3_simulator.h"
7 #include "xbt/dict.h"
8 #include "xbt/log.h"
9 #include "xbt/sysdep.h"
10
11 xbt_dict_t flowFromSock = xbt_dict_new_homogeneous([](void *p) {
12   delete (SgFlow*)p;
13 }); // ns3::sock -> SgFlow
14
15 static void receive_callback(ns3::Ptr<ns3::Socket> socket);
16 static void send_callback(ns3::Ptr<ns3::Socket> sock, uint32_t txSpace);
17 static void datasent_callback(ns3::Ptr<ns3::Socket> socket, uint32_t dataSent);
18
19 XBT_LOG_EXTERNAL_DEFAULT_CATEGORY(ns3);
20
21 SgFlow::SgFlow(uint32_t totalBytes, simgrid::surf::NetworkNS3Action* action)
22 {
23   totalBytes_ = totalBytes;
24   remaining_  = totalBytes;
25   action_     = action;
26 }
27
28 static SgFlow* getFlowFromSocket(ns3::Ptr<ns3::Socket> socket)
29 {
30   return (SgFlow*)xbt_dict_get_or_null(flowFromSock, transformSocketPtr(socket));
31 }
32
33 static void receive_callback(ns3::Ptr<ns3::Socket> socket)
34 {
35   SgFlow* flow = getFlowFromSocket(socket);
36
37   if (flow->finished_ == false) {
38     flow->finished_ = true;
39     XBT_DEBUG("recv_cb of F[%p, %p, %d]", flow, flow->action_, flow->totalBytes_);
40     XBT_DEBUG("Stop simulator at %f seconds", ns3::Simulator::Now().GetSeconds());
41     ns3::Simulator::Stop(ns3::Seconds(0.0));
42     ns3::Simulator::Run();
43   }
44 }
45
46 static void send_callback(ns3::Ptr<ns3::Socket> sock, uint32_t txSpace)
47 {
48   SgFlow* flow = getFlowFromSocket(sock);
49
50   if (flow->remaining_ == 0) // all data was already buffered (and socket was already closed)
51     return;
52
53   uint8_t* data = (uint8_t*)malloc(sizeof(uint8_t) * txSpace);
54
55   while (flow->bufferedBytes_ < flow->totalBytes_ && sock->GetTxAvailable() > 0) {
56
57     uint32_t toWrite = std::min({flow->remaining_, txSpace, sock->GetTxAvailable()});
58     int amountSent   = sock->Send(data, toWrite, 0);
59
60     if (amountSent < 0)
61       return;
62     flow->bufferedBytes_ += amountSent;
63     flow->remaining_ -= amountSent;
64
65     XBT_DEBUG("send_cb of F[%p, %p, %d] (%d/%d) %d buffered", flow, flow->action_, flow->totalBytes_, flow->remaining_,
66               flow->totalBytes_, amountSent);
67   }
68   free(data);
69
70   if ((flow->bufferedBytes_) >= flow->totalBytes_)
71     sock->Close();
72 }
73
74 static void datasent_callback(ns3::Ptr<ns3::Socket> socket, uint32_t dataSent)
75 {
76   SgFlow* flow = getFlowFromSocket(socket);
77   flow->sentBytes_ += dataSent;
78   XBT_DEBUG("datasent_cb of F[%p, %p, %d] %d sent", flow, flow->action_, flow->totalBytes_, dataSent);
79 }
80
81 static void normalClose_callback(ns3::Ptr<ns3::Socket> socket)
82 {
83   SgFlow* flow = getFlowFromSocket(socket);
84   XBT_DEBUG("normalClose_cb of F[%p, %p, %d]", flow, flow->action_, flow->totalBytes_);
85   receive_callback(socket);
86 }
87
88 static void errorClose_callback(ns3::Ptr<ns3::Socket> socket)
89 {
90   SgFlow* flow = getFlowFromSocket(socket);
91   XBT_DEBUG("errorClose_cb of F[%p, %p, %d]", flow, flow->action_, flow->totalBytes_);
92   xbt_die("NS3: a socket was closed anormally");
93 }
94
95 static void succeededConnect_callback(ns3::Ptr<ns3::Socket> socket)
96 {
97   SgFlow* flow = getFlowFromSocket(socket);
98   XBT_DEBUG("succeededConnect_cb of F[%p, %p, %d]", flow, flow->action_, flow->totalBytes_);
99 }
100
101 static void failedConnect_callback(ns3::Ptr<ns3::Socket> socket)
102 {
103   SgFlow* mysocket = getFlowFromSocket(socket);
104   XBT_DEBUG("failedConnect_cb of F[%p, %p, %d]", mysocket, mysocket->action_, mysocket->totalBytes_);
105   xbt_die("NS3: a socket failed to connect");
106 }
107
108 void StartFlow(ns3::Ptr<ns3::Socket> sock, const char* to, uint16_t port_number)
109 {
110   ns3::InetSocketAddress serverAddr(to, port_number);
111
112   sock->Connect(serverAddr);
113   sock->SetSendCallback(MakeCallback(&send_callback));
114   sock->SetRecvCallback(MakeCallback(&receive_callback));
115   sock->SetDataSentCallback(MakeCallback(&datasent_callback));
116   sock->SetConnectCallback(MakeCallback(&succeededConnect_callback), MakeCallback(&failedConnect_callback));
117   sock->SetCloseCallbacks(MakeCallback(&normalClose_callback), MakeCallback(&errorClose_callback));
118
119   SgFlow* flow = getFlowFromSocket(sock);
120   XBT_DEBUG("startFlow_cb of F[%p, %p, %d] dest=%s port=%d", flow, flow->action_, flow->totalBytes_, to, port_number);
121 }