1 // Copyright (C) 2005-2006 Douglas Gregor <doug.gregor@gmail.com>
2
3 // Use, modification and distribution is subject to the Boost Software
4 // License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
5 // http://www.boost.org/LICENSE_1_0.txt)
6
7 // Message Passing Interface 1.1 -- 7.1.1. Environmental Inquiries
8 #include <boost/mpi/environment.hpp>
9 #include <boost/mpi/exception.hpp>
10 #include <boost/mpi/detail/mpi_datatype_cache.hpp>
11 #include <boost/core/uncaught_exceptions.hpp>
12 #include <cassert>
13 #include <string>
14 #include <exception>
15 #include <stdexcept>
16 #include <ostream>
17
18 namespace boost { namespace mpi {
19 namespace threading {
operator >>(std::istream & in,level & l)20 std::istream& operator>>(std::istream& in, level& l)
21 {
22 std::string tk;
23 in >> tk;
24 if (!in.bad()) {
25 if (tk == "single") {
26 l = single;
27 } else if (tk == "funneled") {
28 l = funneled;
29 } else if (tk == "serialized") {
30 l = serialized;
31 } else if (tk == "multiple") {
32 l = multiple;
33 } else {
34 in.setstate(std::ios::badbit);
35 }
36 }
37 return in;
38 }
39
operator <<(std::ostream & out,level l)40 std::ostream& operator<<(std::ostream& out, level l)
41 {
42 switch(l) {
43 case single:
44 out << "single";
45 break;
46 case funneled:
47 out << "funneled";
48 break;
49 case serialized:
50 out << "serialized";
51 break;
52 case multiple:
53 out << "multiple";
54 break;
55 default:
56 out << "<level error>[" << int(l) << ']';
57 out.setstate(std::ios::badbit);
58 break;
59 }
60 return out;
61 }
62
63 } // namespace threading
64
65 #ifdef BOOST_MPI_HAS_NOARG_INITIALIZATION
environment(bool abrt)66 environment::environment(bool abrt)
67 : i_initialized(false),
68 abort_on_exception(abrt)
69 {
70 if (!initialized()) {
71 BOOST_MPI_CHECK_RESULT(MPI_Init, (0, 0));
72 i_initialized = true;
73 }
74
75 #if (2 <= MPI_VERSION)
76 MPI_Comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
77 #else
78 MPI_Errhandler_set(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
79 #endif
80 }
81
environment(threading::level mt_level,bool abrt)82 environment::environment(threading::level mt_level, bool abrt)
83 : i_initialized(false),
84 abort_on_exception(abrt)
85 {
86 // It is not clear that we can pass null in MPI_Init_thread.
87 int dummy_thread_level = 0;
88 if (!initialized()) {
89 BOOST_MPI_CHECK_RESULT(MPI_Init_thread,
90 (0, 0, int(mt_level), &dummy_thread_level ));
91 i_initialized = true;
92 }
93
94 #if (2 <= MPI_VERSION)
95 MPI_Comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
96 #else
97 MPI_Errhandler_set(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
98 #endif
99 }
100 #endif
101
environment(int & argc,char ** & argv,bool abrt)102 environment::environment(int& argc, char** &argv, bool abrt)
103 : i_initialized(false),
104 abort_on_exception(abrt)
105 {
106 if (!initialized()) {
107 BOOST_MPI_CHECK_RESULT(MPI_Init, (&argc, &argv));
108 i_initialized = true;
109 }
110
111 #if (2 <= MPI_VERSION)
112 MPI_Comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
113 #else
114 MPI_Errhandler_set(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
115 #endif
116 }
117
environment(int & argc,char ** & argv,threading::level mt_level,bool abrt)118 environment::environment(int& argc, char** &argv, threading::level mt_level,
119 bool abrt)
120 : i_initialized(false),
121 abort_on_exception(abrt)
122 {
123 // It is not clear that we can pass null in MPI_Init_thread.
124 int dummy_thread_level = 0;
125 if (!initialized()) {
126 BOOST_MPI_CHECK_RESULT(MPI_Init_thread,
127 (&argc, &argv, int(mt_level), &dummy_thread_level));
128 i_initialized = true;
129 }
130
131 #if (2 <= MPI_VERSION)
132 MPI_Comm_set_errhandler(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
133 #else
134 MPI_Errhandler_set(MPI_COMM_WORLD, MPI_ERRORS_RETURN);
135 #endif
136 }
137
~environment()138 environment::~environment()
139 {
140 if (i_initialized) {
141 if (boost::core::uncaught_exceptions() > 0 && abort_on_exception) {
142 abort(-1);
143 } else if (!finalized()) {
144 detail::mpi_datatype_cache().clear();
145 BOOST_MPI_CHECK_RESULT(MPI_Finalize, ());
146 }
147 }
148 }
149
abort(int errcode)150 void environment::abort(int errcode)
151 {
152 BOOST_MPI_CHECK_RESULT(MPI_Abort, (MPI_COMM_WORLD, errcode));
153 }
154
initialized()155 bool environment::initialized()
156 {
157 int flag;
158 BOOST_MPI_CHECK_RESULT(MPI_Initialized, (&flag));
159 return flag != 0;
160 }
161
finalized()162 bool environment::finalized()
163 {
164 int flag;
165 BOOST_MPI_CHECK_RESULT(MPI_Finalized, (&flag));
166 return flag != 0;
167 }
168
max_tag()169 int environment::max_tag()
170 {
171 int* max_tag_value;
172 int found = 0;
173
174 #if (2 <= MPI_VERSION)
175 BOOST_MPI_CHECK_RESULT(MPI_Comm_get_attr,
176 (MPI_COMM_WORLD, MPI_TAG_UB, &max_tag_value, &found));
177 #else
178 BOOST_MPI_CHECK_RESULT(MPI_Attr_get,
179 (MPI_COMM_WORLD, MPI_TAG_UB, &max_tag_value, &found));
180 #endif
181 assert(found != 0);
182 return *max_tag_value - num_reserved_tags;
183 }
184
collectives_tag()185 int environment::collectives_tag()
186 {
187 return max_tag() + 1;
188 }
189
host_rank()190 optional<int> environment::host_rank()
191 {
192 int* host;
193 int found = 0;
194
195 #if (2 <= MPI_VERSION)
196 BOOST_MPI_CHECK_RESULT(MPI_Comm_get_attr,
197 (MPI_COMM_WORLD, MPI_HOST, &host, &found));
198 #else
199 BOOST_MPI_CHECK_RESULT(MPI_Attr_get,
200 (MPI_COMM_WORLD, MPI_HOST, &host, &found));
201 #endif
202 if (!found || *host == MPI_PROC_NULL)
203 return optional<int>();
204 else
205 return *host;
206 }
207
io_rank()208 optional<int> environment::io_rank()
209 {
210 int* io;
211 int found = 0;
212
213 #if (2 <= MPI_VERSION)
214 BOOST_MPI_CHECK_RESULT(MPI_Comm_get_attr,
215 (MPI_COMM_WORLD, MPI_IO, &io, &found));
216 #else
217 BOOST_MPI_CHECK_RESULT(MPI_Attr_get,
218 (MPI_COMM_WORLD, MPI_IO, &io, &found));
219 #endif
220 if (!found || *io == MPI_PROC_NULL)
221 return optional<int>();
222 else
223 return *io;
224 }
225
processor_name()226 std::string environment::processor_name()
227 {
228 char name[MPI_MAX_PROCESSOR_NAME];
229 int len;
230
231 BOOST_MPI_CHECK_RESULT(MPI_Get_processor_name, (name, &len));
232 return std::string(name, len);
233 }
234
thread_level()235 threading::level environment::thread_level()
236 {
237 int level;
238
239 BOOST_MPI_CHECK_RESULT(MPI_Query_thread, (&level));
240 return static_cast<threading::level>(level);
241 }
242
is_main_thread()243 bool environment::is_main_thread()
244 {
245 int isit;
246
247 BOOST_MPI_CHECK_RESULT(MPI_Is_thread_main, (&isit));
248 return static_cast<bool>(isit);
249 }
250
version()251 std::pair<int, int> environment::version()
252 {
253 int major, minor;
254 BOOST_MPI_CHECK_RESULT(MPI_Get_version, (&major, &minor));
255 return std::make_pair(major, minor);
256 }
257
258 } } // end namespace boost::mpi
259