]> AND Private Git Repository - loba.git/blob - communicator.cpp
Logo AND Algorithmique Numérique Distribuée

Private GIT Repository
f3de918962815d0c0822fc6aa9f42f7b1b767390
[loba.git] / communicator.cpp
1 #include "communicator.h"
2
3 #include <algorithm>
4 #include <tr1/functional>
5 #include <sstream>
6 #include <msg/msg.h>
7 #include <xbt/log.h>
8 #include "simgrid_features.h"
9 #include "misc.h"
10 #include "options.h"
11
12 XBT_LOG_EXTERNAL_DEFAULT_CATEGORY(comm);
13
14 std::string message::to_string()
15 {
16     static const char* str[] = { "INFO", "CREDIT", "LOAD",
17                                  "CTRL_CLOSE", "DATA_CLOSE" };
18     std::ostringstream oss;
19     oss << str[type] << " (" << amount << ")";
20     return oss.str();
21 }
22
23 const int communicator::send_count_before_flush = 128;
24
25 communicator::communicator()
26     : host((hostdata* )MSG_host_get_data(MSG_host_self()))
27     , send_counter(0)
28     , ctrl_task(NULL)
29     , ctrl_comm(NULL)
30     , ctrl_close_is_last(false)
31     , data_task(NULL)
32     , data_comm(NULL)
33     , data_close_is_last(false)
34 {
35 }
36
37 communicator::~communicator()
38 {
39     if (ctrl_comm)
40         WARN0("ctrl_comm is pending!");
41     if (data_comm)
42         WARN0("data_comm is pending!");
43     if (!sent_comm.empty())
44         WARN2("lost %ld send communication%s!",
45               (long )sent_comm.size(), ESSE(sent_comm.size()));
46 }
47
48 void communicator::listen()
49 {
50     ctrl_comm = MSG_task_irecv(&ctrl_task, get_ctrl_mbox());
51     data_comm = MSG_task_irecv(&data_task, get_data_mbox());
52 }
53
54 void communicator::send(const char* dest, message* msg)
55 {
56     double msg_size = sizeof *msg;
57     if (msg->get_type() == message::LOAD)
58         msg_size += opt::comm_cost(msg->get_amount());
59     m_task_t task = MSG_task_create("message", 0.0, msg_size, msg);    
60     msg_comm_t comm = MSG_task_isend(task, dest);
61     sent_comm.push_back(comm);
62
63     if (++send_counter >= send_count_before_flush) {
64         flush(false);
65         send_counter = 0;
66     }
67 }
68
69 bool communicator::recv(message*& msg, m_host_t& from, bool wait)
70 {
71     bool restart;
72     msg = NULL;
73
74     do {
75         if (ctrl_comm && comm_test_n_destroy(ctrl_comm)) {
76             msg = (message* )MSG_task_get_data(ctrl_task);
77             from = MSG_task_get_source(ctrl_task);
78             MSG_task_destroy(ctrl_task);
79             ctrl_task = NULL;
80             ctrl_comm =
81                 (!ctrl_close_is_last || msg->get_type() != message::CTRL_CLOSE)
82                 ? MSG_task_irecv(&ctrl_task, get_ctrl_mbox())
83                 : NULL;
84
85         } else if (data_comm && comm_test_n_destroy(data_comm)) {
86             msg = (message* )MSG_task_get_data(data_task);
87             from = MSG_task_get_source(data_task);
88             MSG_task_destroy(data_task);
89             data_task = NULL;
90             data_comm =
91                 (!data_close_is_last || msg->get_type() != message::DATA_CLOSE)
92                 ? MSG_task_irecv(&data_task, get_data_mbox())
93                 : NULL;
94         }
95
96         restart = wait && !msg && (ctrl_comm || data_comm);
97         if (restart) {
98             xbt_dynar_t comms = xbt_dynar_new(sizeof(msg_comm_t), NULL);
99             if (ctrl_comm)
100                 xbt_dynar_push(comms, &ctrl_comm);
101             if (data_comm)
102                 xbt_dynar_push(comms, &data_comm);
103             MSG_comm_waitany(comms);
104             xbt_dynar_free(&comms);
105         }
106     } while (restart);
107
108     return msg != NULL;
109 }
110
111 void communicator::flush(bool wait)
112 {
113     using namespace std::tr1;
114     using namespace std::tr1::placeholders;
115
116     sent_comm.remove_if(comm_test_n_destroy);
117     if (wait && !sent_comm.empty()) {
118         xbt_dynar_t comms = xbt_dynar_new(sizeof(msg_comm_t), NULL);
119         while (!sent_comm.empty()) {
120             std::for_each(sent_comm.begin(), sent_comm.end(),
121                           bind(xbt_dynar_push,
122                                comms, bind(misc::address<msg_comm_t>(), _1)));
123             MSG_comm_waitany(comms);
124             xbt_dynar_reset(comms);
125             sent_comm.remove_if(comm_test_n_destroy);
126         }
127         xbt_dynar_free(&comms);
128     }
129 }
130
131 void communicator::next_close_on_ctrl_is_last()
132 {
133     ctrl_close_is_last = true;
134 }
135
136 void communicator::next_close_on_data_is_last()
137 {
138     data_close_is_last = true;
139 }
140
141 bool communicator::comm_test_n_destroy(msg_comm_t comm)
142 {
143     if (MSG_comm_test(comm)) {
144         MSG_comm_destroy(comm);
145         return true;
146     } else
147         return false;
148 }
149
150 // Local variables:
151 // mode: c++
152 // End: