Logo AND Algorithmique Numérique Distribuée

Public GIT Repository
add Exec::setBound (and what lies beneath)
[simgrid.git] / src / kernel / activity / CommImpl.cpp
1 /* Copyright (c) 2007-2017. 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/kernel/activity/CommImpl.hpp"
7
8 #include "simgrid/modelchecker.h"
9 #include "src/mc/mc_replay.hpp"
10 #include "src/simix/smx_network_private.hpp"
11 #include "src/surf/surf_interface.hpp"
12
13 XBT_LOG_EXTERNAL_DEFAULT_CATEGORY(simix_network);
14
15 simgrid::kernel::activity::CommImpl::CommImpl(e_smx_comm_type_t _type) : type(_type)
16 {
17   state    = SIMIX_WAITING;
18   src_data = nullptr;
19   dst_data = nullptr;
20   XBT_DEBUG("Create comm activity %p", this);
21 }
22
23 simgrid::kernel::activity::CommImpl::~CommImpl()
24 {
25   XBT_DEBUG("Really free communication %p", this);
26
27   cleanupSurf();
28
29   if (detached && state != SIMIX_DONE) {
30     /* the communication has failed and was detached:
31      * we have to free the buffer */
32     if (clean_fun)
33       clean_fun(src_buff);
34     src_buff = nullptr;
35   }
36
37   if (mbox)
38     mbox->remove(this);
39 }
40
41 void simgrid::kernel::activity::CommImpl::suspend()
42 {
43   /* FIXME: shall we suspend also the timeout synchro? */
44   if (surfAction_)
45     surfAction_->suspend();
46   /* in the other case, the action will be suspended on creation, in SIMIX_comm_start() */
47 }
48
49 void simgrid::kernel::activity::CommImpl::resume()
50 {
51   /*FIXME: check what happen with the timeouts */
52   if (surfAction_)
53     surfAction_->resume();
54   /* in the other case, the synchro were not really suspended yet, see SIMIX_comm_suspend() and SIMIX_comm_start() */
55 }
56
57 void simgrid::kernel::activity::CommImpl::cancel()
58 {
59   /* if the synchro is a waiting state means that it is still in a mbox */
60   /* so remove from it and delete it */
61   if (state == SIMIX_WAITING) {
62     mbox->remove(this);
63     state = SIMIX_CANCELED;
64   } else if (not MC_is_active() /* when running the MC there are no surf actions */
65              && not MC_record_replay_is_active() && (state == SIMIX_READY || state == SIMIX_RUNNING)) {
66
67     surfAction_->cancel();
68   }
69 }
70
71 /**  @brief get the amount remaining from the communication */
72 double simgrid::kernel::activity::CommImpl::remains()
73 {
74   return surfAction_->getRemains();
75 }
76
77 /** @brief This is part of the cleanup process, probably an internal command */
78 void simgrid::kernel::activity::CommImpl::cleanupSurf()
79 {
80   if (surfAction_) {
81     surfAction_->unref();
82     surfAction_ = nullptr;
83   }
84
85   if (src_timeout) {
86     src_timeout->unref();
87     src_timeout = nullptr;
88   }
89
90   if (dst_timeout) {
91     dst_timeout->unref();
92     dst_timeout = nullptr;
93   }
94 }
95
96 void simgrid::kernel::activity::CommImpl::post()
97 {
98   /* Update synchro state */
99   if (src_timeout && src_timeout->getState() == simgrid::surf::Action::State::done)
100     state = SIMIX_SRC_TIMEOUT;
101   else if (dst_timeout && dst_timeout->getState() == simgrid::surf::Action::State::done)
102     state = SIMIX_DST_TIMEOUT;
103   else if (src_timeout && src_timeout->getState() == simgrid::surf::Action::State::failed)
104     state = SIMIX_SRC_HOST_FAILURE;
105   else if (dst_timeout && dst_timeout->getState() == simgrid::surf::Action::State::failed)
106     state = SIMIX_DST_HOST_FAILURE;
107   else if (surfAction_ && surfAction_->getState() == simgrid::surf::Action::State::failed) {
108     state = SIMIX_LINK_FAILURE;
109   } else
110     state = SIMIX_DONE;
111
112   XBT_DEBUG("SIMIX_post_comm: comm %p, state %d, src_proc %p, dst_proc %p, detached: %d", this, (int)state, src_proc,
113             dst_proc, detached);
114
115   /* destroy the surf actions associated with the Simix communication */
116   cleanupSurf();
117
118   /* if there are simcalls associated with the synchro, then answer them */
119   if (not simcalls.empty()) {
120     SIMIX_comm_finish(this);
121   }
122 }