graph_manager.hpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. /**
  2. * @file tests/multithreading/simple/graph_manager.cpp
  3. * @author The ARTIS Development Team
  4. * See the AUTHORS or Authors.txt file
  5. */
  6. /*
  7. * ARTIS - the multimodeling and simulation environment
  8. * This file is a part of the ARTIS environment
  9. *
  10. * Copyright (C) 2013-2019 ULCO http://www.univ-littoral.fr
  11. *
  12. * This program is free software: you can redistribute it and/or modify
  13. * it under the terms of the GNU General Public License as published by
  14. * the Free Software Foundation, either version 3 of the License, or
  15. * (at your option) any later version.
  16. *
  17. * This program is distributed in the hope that it will be useful,
  18. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  19. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  20. * GNU General Public License for more details.
  21. *
  22. * You should have received a copy of the GNU General Public License
  23. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  24. */
  25. #ifndef TESTS_MULTITHREADING_SIMPLE_GRAPH_MANAGER_HPP
  26. #define TESTS_MULTITHREADING_SIMPLE_GRAPH_MANAGER_HPP
  27. #include <tests/multithreading/simple/models.hpp>
  28. #include <artis-star/kernel/pdevs/multithreading/Coordinator.hpp>
  29. #include <artis-star/kernel/pdevs/GraphManager.hpp>
  30. #include <artis-star/kernel/pdevs/Simulator.hpp>
  31. namespace artis {
  32. namespace tests {
  33. namespace multithreading {
  34. namespace simple {
  35. class LinksGraphManager :
  36. public artis::pdevs::GraphManager<artis::common::DoubleTime, GeneratorParameters> {
  37. public:
  38. enum submodels {
  39. LINK_1, LINK_2
  40. };
  41. struct inputs {
  42. enum values {
  43. IN_1, IN_2
  44. };
  45. };
  46. struct outputs {
  47. enum values {
  48. OUT_1, OUT_2
  49. };
  50. };
  51. LinksGraphManager(
  52. artis::common::Coordinator<artis::common::DoubleTime> *coordinator,
  53. const GeneratorParameters &parameters,
  54. const artis::common::NoParameters &graph_parameters)
  55. :
  56. artis::pdevs::GraphManager<artis::common::DoubleTime, GeneratorParameters>(
  57. coordinator, parameters, graph_parameters),
  58. _link_1("link_1", artis::common::NoParameters()),
  59. _link_2("link_2", artis::common::NoParameters()) {
  60. add_child(LINK_1, &_link_1);
  61. add_child(LINK_2, &_link_2);
  62. coordinator->input_ports({{inputs::IN_1, "in_1"},
  63. {inputs::IN_2, "in_2"}});
  64. coordinator->output_ports({{outputs::OUT_1, "out_1"},
  65. {outputs::OUT_2, "out_2"}});
  66. in({coordinator, inputs::IN_1})
  67. >> out({&_link_1, Link::inputs::IN});
  68. in({coordinator, inputs::IN_2})
  69. >> out({&_link_2, Link::inputs::IN});
  70. out({&_link_1, Link::outputs::OUT})
  71. >> out({coordinator, outputs::OUT_1});
  72. out({&_link_2, Link::outputs::OUT})
  73. >> out({coordinator, outputs::OUT_2});
  74. }
  75. ~LinksGraphManager() override = default;
  76. common::DoubleTime::type
  77. lookahead(const common::DoubleTime::type &t) const override {
  78. std::vector<double> lookaheads = {_link_1.lookahead(t),
  79. _link_2.lookahead(t)};
  80. return *std::min(lookaheads.begin(), lookaheads.end());
  81. }
  82. void init() {}
  83. void start(common::DoubleTime::type /* t */) {}
  84. void transition(const common::Models<common::DoubleTime> & /* receivers */,
  85. artis::common::DoubleTime::type /* t */) {}
  86. private:
  87. artis::pdevs::Simulator<artis::common::DoubleTime, Link> _link_1;
  88. artis::pdevs::Simulator<artis::common::DoubleTime, Link> _link_2;
  89. };
  90. class SimpleGraphManager :
  91. public artis::pdevs::GraphManager<artis::common::DoubleTime, GeneratorParameters> {
  92. public:
  93. enum submodels {
  94. GENERATOR, LINK_1_2, LINK_3_4, LINK_5_6, LINK_7_8, COUNTER
  95. };
  96. SimpleGraphManager(
  97. artis::common::Coordinator<artis::common::DoubleTime> *coordinator,
  98. const GeneratorParameters &parameters,
  99. const artis::common::NoParameters &graph_parameters)
  100. :
  101. artis::pdevs::GraphManager<artis::common::DoubleTime, GeneratorParameters>(
  102. coordinator, parameters, graph_parameters),
  103. _generator("generator", parameters),
  104. _link_1_2("link_1_2", parameters, artis::common::NoParameters()),
  105. _link_3_4("link_3_4", parameters, artis::common::NoParameters()),
  106. _link_5_6("link_5_6", parameters, artis::common::NoParameters()),
  107. _link_7_8("link_7_8", parameters, artis::common::NoParameters()),
  108. _counter("counter", artis::common::NoParameters()) {
  109. add_child(GENERATOR, &_generator);
  110. add_child(LINK_1_2, &_link_1_2);
  111. add_child(LINK_3_4, &_link_3_4);
  112. add_child(LINK_5_6, &_link_5_6);
  113. add_child(LINK_7_8, &_link_7_8);
  114. add_child(COUNTER, &_counter);
  115. out({&_generator, Generator::outputs::OUT})
  116. >> in({&_link_1_2, LinksGraphManager::inputs::IN_1});
  117. out({&_generator, Generator::outputs::OUT + 1})
  118. >> in({&_link_1_2, LinksGraphManager::inputs::IN_2});
  119. out({&_generator, Generator::outputs::OUT + 2})
  120. >> in({&_link_3_4, LinksGraphManager::inputs::IN_1});
  121. out({&_generator, Generator::outputs::OUT + 3})
  122. >> in({&_link_3_4, LinksGraphManager::inputs::IN_2});
  123. out({&_generator, Generator::outputs::OUT + 4})
  124. >> in({&_link_5_6, LinksGraphManager::inputs::IN_1});
  125. out({&_generator, Generator::outputs::OUT + 5})
  126. >> in({&_link_5_6, LinksGraphManager::inputs::IN_2});
  127. out({&_generator, Generator::outputs::OUT + 6})
  128. >> in({&_link_7_8, LinksGraphManager::inputs::IN_1});
  129. out({&_generator, Generator::outputs::OUT + 7})
  130. >> in({&_link_7_8, LinksGraphManager::inputs::IN_2});
  131. out({&_link_1_2, LinksGraphManager::outputs::OUT_1})
  132. >> in({&_counter, Counter::inputs::IN});
  133. out({&_link_1_2, LinksGraphManager::outputs::OUT_2})
  134. >> in({&_counter, Counter::inputs::IN});
  135. out({&_link_3_4, LinksGraphManager::outputs::OUT_1})
  136. >> in({&_counter, Counter::inputs::IN});
  137. out({&_link_3_4, LinksGraphManager::outputs::OUT_2})
  138. >> in({&_counter, Counter::inputs::IN});
  139. out({&_link_5_6, LinksGraphManager::outputs::OUT_1})
  140. >> in({&_counter, Counter::inputs::IN});
  141. out({&_link_5_6, LinksGraphManager::outputs::OUT_2})
  142. >> in({&_counter, Counter::inputs::IN});
  143. out({&_link_7_8, LinksGraphManager::outputs::OUT_1})
  144. >> in({&_counter, Counter::inputs::IN});
  145. out({&_link_7_8, LinksGraphManager::outputs::OUT_2})
  146. >> in({&_counter, Counter::inputs::IN});
  147. }
  148. ~SimpleGraphManager() override = default;
  149. common::DoubleTime::type
  150. lookahead(const common::DoubleTime::type &t) const override {
  151. std::vector<double> lookaheads = {_generator.lookahead(t),
  152. _link_1_2.lookahead(t),
  153. _link_3_4.lookahead(t),
  154. _link_5_6.lookahead(t),
  155. _link_7_8.lookahead(t),
  156. _counter.lookahead(t)};
  157. return *std::min(lookaheads.begin(), lookaheads.end());
  158. }
  159. void init() {
  160. _link_1_2.set_sender(
  161. (dynamic_cast< artis::pdevs::multithreading::Coordinator<
  162. artis::common::DoubleTime,
  163. SimpleGraphManager,
  164. GeneratorParameters,
  165. artis::common::NoParameters> *>(this->coordinator()))->get_sender());
  166. _link_3_4.set_sender(
  167. (dynamic_cast< artis::pdevs::multithreading::Coordinator<
  168. artis::common::DoubleTime,
  169. SimpleGraphManager,
  170. GeneratorParameters,
  171. artis::common::NoParameters> *>(this->coordinator()))->get_sender());
  172. _link_5_6.set_sender(
  173. (dynamic_cast< artis::pdevs::multithreading::Coordinator<
  174. artis::common::DoubleTime,
  175. SimpleGraphManager,
  176. GeneratorParameters,
  177. artis::common::NoParameters> *>(this->coordinator()))->get_sender());
  178. _link_7_8.set_sender(
  179. (dynamic_cast< artis::pdevs::multithreading::Coordinator<
  180. artis::common::DoubleTime,
  181. SimpleGraphManager,
  182. GeneratorParameters,
  183. artis::common::NoParameters> *>(this->coordinator()))->get_sender());
  184. }
  185. void start(common::DoubleTime::type t) {
  186. _link_1_2.get_sender().send(
  187. artis::pdevs::multithreading::start_message<
  188. artis::common::DoubleTime>(t));
  189. _link_3_4.get_sender().send(
  190. artis::pdevs::multithreading::start_message<
  191. artis::common::DoubleTime>(t));
  192. _link_5_6.get_sender().send(
  193. artis::pdevs::multithreading::start_message<
  194. artis::common::DoubleTime>(t));
  195. _link_7_8.get_sender().send(
  196. artis::pdevs::multithreading::start_message<
  197. artis::common::DoubleTime>(t));
  198. }
  199. void transition(const common::Models<common::DoubleTime> &receivers,
  200. artis::common::DoubleTime::type t) {
  201. std::for_each(receivers.begin(), receivers.end(),
  202. [this, t](const common::Model<common::DoubleTime> *model) {
  203. if (not model->is_atomic()) {
  204. if (model == &_link_1_2) {
  205. _link_1_2.get_sender().send(
  206. artis::pdevs::multithreading::transition_message<
  207. artis::common::DoubleTime>(t));
  208. } else if (model == &_link_3_4) {
  209. _link_3_4.get_sender().send(
  210. artis::pdevs::multithreading::transition_message<
  211. artis::common::DoubleTime>(t));
  212. } else if (model == &_link_5_6) {
  213. _link_5_6.get_sender().send(
  214. artis::pdevs::multithreading::transition_message<
  215. artis::common::DoubleTime>(t));
  216. } else if (model == &_link_7_8) {
  217. _link_7_8.get_sender().send(
  218. artis::pdevs::multithreading::transition_message<
  219. artis::common::DoubleTime>(t));
  220. }
  221. }
  222. });
  223. }
  224. private:
  225. artis::pdevs::Simulator<artis::common::DoubleTime, Generator, GeneratorParameters> _generator;
  226. /*
  227. artis::pdevs::Coordinator<artis::common::DoubleTime, LinksGraphManager, GeneratorParameters, artis::common::NoParameters> _link_1_2;
  228. artis::pdevs::Coordinator<artis::common::DoubleTime, LinksGraphManager, GeneratorParameters, artis::common::NoParameters> _link_3_4;
  229. artis::pdevs::Coordinator<artis::common::DoubleTime, LinksGraphManager, GeneratorParameters, artis::common::NoParameters> _link_5_6;
  230. artis::pdevs::Coordinator<artis::common::DoubleTime, LinksGraphManager, GeneratorParameters, artis::common::NoParameters> _link_7_8;
  231. */
  232. artis::pdevs::multithreading::Coordinator<artis::common::DoubleTime,
  233. LinksGraphManager,
  234. GeneratorParameters,
  235. artis::common::NoParameters> _link_1_2;
  236. artis::pdevs::multithreading::Coordinator<artis::common::DoubleTime,
  237. LinksGraphManager,
  238. GeneratorParameters,
  239. artis::common::NoParameters> _link_3_4;
  240. artis::pdevs::multithreading::Coordinator<artis::common::DoubleTime,
  241. LinksGraphManager,
  242. GeneratorParameters,
  243. artis::common::NoParameters> _link_5_6;
  244. artis::pdevs::multithreading::Coordinator<artis::common::DoubleTime,
  245. LinksGraphManager,
  246. GeneratorParameters,
  247. artis::common::NoParameters> _link_7_8;
  248. artis::pdevs::Simulator<artis::common::DoubleTime, Counter> _counter;
  249. };
  250. }
  251. }
  252. }
  253. } // namespace artis tests multithreading simple
  254. #endif