once.hpp 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540
  1. #ifndef BOOST_THREAD_PTHREAD_ONCE_HPP
  2. #define BOOST_THREAD_PTHREAD_ONCE_HPP
  3. // once.hpp
  4. //
  5. // (C) Copyright 2007-8 Anthony Williams
  6. // (C) Copyright 2011-2012 Vicente J. Botet Escriba
  7. //
  8. // Distributed under the Boost Software License, Version 1.0. (See
  9. // accompanying file LICENSE_1_0.txt or copy at
  10. // http://www.boost.org/LICENSE_1_0.txt)
  11. #include <boost/thread/detail/config.hpp>
  12. #include <boost/thread/detail/move.hpp>
  13. #include <boost/thread/detail/invoke.hpp>
  14. #include <boost/thread/pthread/pthread_mutex_scoped_lock.hpp>
  15. #include <boost/thread/detail/delete.hpp>
  16. #include <boost/core/no_exceptions_support.hpp>
  17. #include <boost/bind.hpp>
  18. #include <boost/assert.hpp>
  19. #include <boost/config/abi_prefix.hpp>
  20. #include <boost/cstdint.hpp>
  21. #include <pthread.h>
  22. #include <csignal>
  23. namespace boost
  24. {
  25. struct once_flag;
  26. #define BOOST_ONCE_INITIAL_FLAG_VALUE 0
  27. namespace thread_detail
  28. {
  29. typedef boost::uint32_t uintmax_atomic_t;
  30. #define BOOST_THREAD_DETAIL_UINTMAX_ATOMIC_C2(value) value##u
  31. #define BOOST_THREAD_DETAIL_UINTMAX_ATOMIC_MAX_C BOOST_THREAD_DETAIL_UINTMAX_ATOMIC_C2(~0)
  32. }
  33. #ifdef BOOST_THREAD_PROVIDES_ONCE_CXX11
  34. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) && !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  35. template<typename Function, class ...ArgTypes>
  36. inline void call_once(once_flag& flag, BOOST_THREAD_RV_REF(Function) f, BOOST_THREAD_RV_REF(ArgTypes)... args);
  37. #else
  38. template<typename Function>
  39. inline void call_once(once_flag& flag, Function f);
  40. template<typename Function, typename T1>
  41. inline void call_once(once_flag& flag, Function f, T1 p1);
  42. template<typename Function, typename T1, typename T2>
  43. inline void call_once(once_flag& flag, Function f, T1 p1, T2 p2);
  44. template<typename Function, typename T1, typename T2, typename T3>
  45. inline void call_once(once_flag& flag, Function f, T1 p1, T2 p2, T3 p3);
  46. #endif
  47. struct once_flag
  48. {
  49. BOOST_THREAD_NO_COPYABLE(once_flag)
  50. BOOST_CONSTEXPR once_flag() BOOST_NOEXCEPT
  51. : epoch(BOOST_ONCE_INITIAL_FLAG_VALUE)
  52. {}
  53. private:
  54. volatile thread_detail::uintmax_atomic_t epoch;
  55. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) && !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  56. template<typename Function, class ...ArgTypes>
  57. friend void call_once(once_flag& flag, BOOST_THREAD_RV_REF(Function) f, BOOST_THREAD_RV_REF(ArgTypes)... args);
  58. #else
  59. template<typename Function>
  60. friend void call_once(once_flag& flag, Function f);
  61. template<typename Function, typename T1>
  62. friend void call_once(once_flag& flag, Function f, T1 p1);
  63. template<typename Function, typename T1, typename T2>
  64. friend void call_once(once_flag& flag, Function f, T1 p1, T2 p2);
  65. template<typename Function, typename T1, typename T2, typename T3>
  66. friend void call_once(once_flag& flag, Function f, T1 p1, T2 p2, T3 p3);
  67. #endif
  68. };
  69. #define BOOST_ONCE_INIT once_flag()
  70. #else // BOOST_THREAD_PROVIDES_ONCE_CXX11
  71. struct once_flag
  72. {
  73. volatile thread_detail::uintmax_atomic_t epoch;
  74. };
  75. #define BOOST_ONCE_INIT {BOOST_ONCE_INITIAL_FLAG_VALUE}
  76. #endif // BOOST_THREAD_PROVIDES_ONCE_CXX11
  77. #if defined BOOST_THREAD_PROVIDES_INVOKE
  78. #define BOOST_THREAD_INVOKE_RET_VOID detail::invoke
  79. #define BOOST_THREAD_INVOKE_RET_VOID_CALL
  80. #elif defined BOOST_THREAD_PROVIDES_INVOKE_RET
  81. #define BOOST_THREAD_INVOKE_RET_VOID detail::invoke<void>
  82. #define BOOST_THREAD_INVOKE_RET_VOID_CALL
  83. #else
  84. #define BOOST_THREAD_INVOKE_RET_VOID boost::bind
  85. #define BOOST_THREAD_INVOKE_RET_VOID_CALL ()
  86. #endif
  87. namespace thread_detail
  88. {
  89. BOOST_THREAD_DECL uintmax_atomic_t& get_once_per_thread_epoch();
  90. BOOST_THREAD_DECL extern uintmax_atomic_t once_global_epoch;
  91. BOOST_THREAD_DECL extern pthread_mutex_t once_epoch_mutex;
  92. BOOST_THREAD_DECL extern pthread_cond_t once_epoch_cv;
  93. }
  94. // Based on Mike Burrows fast_pthread_once algorithm as described in
  95. // http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2007/n2444.html
  96. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) && !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  97. template<typename Function, class ...ArgTypes>
  98. inline void call_once(once_flag& flag, BOOST_THREAD_RV_REF(Function) f, BOOST_THREAD_RV_REF(ArgTypes)... args)
  99. {
  100. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  101. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  102. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  103. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  104. if(epoch<this_thread_epoch)
  105. {
  106. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  107. while(flag.epoch<=being_initialized)
  108. {
  109. if(flag.epoch==uninitialized_flag)
  110. {
  111. flag.epoch=being_initialized;
  112. BOOST_TRY
  113. {
  114. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  115. BOOST_THREAD_INVOKE_RET_VOID(
  116. thread_detail::decay_copy(boost::forward<Function>(f)),
  117. thread_detail::decay_copy(boost::forward<ArgTypes>(args))...
  118. ) BOOST_THREAD_INVOKE_RET_VOID_CALL;
  119. }
  120. BOOST_CATCH (...)
  121. {
  122. flag.epoch=uninitialized_flag;
  123. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  124. BOOST_RETHROW
  125. }
  126. BOOST_CATCH_END
  127. flag.epoch=--thread_detail::once_global_epoch;
  128. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  129. }
  130. else
  131. {
  132. while(flag.epoch==being_initialized)
  133. {
  134. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  135. }
  136. }
  137. }
  138. this_thread_epoch=thread_detail::once_global_epoch;
  139. }
  140. }
  141. #else
  142. template<typename Function>
  143. inline void call_once(once_flag& flag, Function f)
  144. {
  145. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  146. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  147. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  148. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  149. if(epoch<this_thread_epoch)
  150. {
  151. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  152. while(flag.epoch<=being_initialized)
  153. {
  154. if(flag.epoch==uninitialized_flag)
  155. {
  156. flag.epoch=being_initialized;
  157. BOOST_TRY
  158. {
  159. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  160. f();
  161. }
  162. BOOST_CATCH (...)
  163. {
  164. flag.epoch=uninitialized_flag;
  165. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  166. BOOST_RETHROW
  167. }
  168. BOOST_CATCH_END
  169. flag.epoch=--thread_detail::once_global_epoch;
  170. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  171. }
  172. else
  173. {
  174. while(flag.epoch==being_initialized)
  175. {
  176. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  177. }
  178. }
  179. }
  180. this_thread_epoch=thread_detail::once_global_epoch;
  181. }
  182. }
  183. template<typename Function, typename T1>
  184. inline void call_once(once_flag& flag, Function f, T1 p1)
  185. {
  186. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  187. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  188. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  189. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  190. if(epoch<this_thread_epoch)
  191. {
  192. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  193. while(flag.epoch<=being_initialized)
  194. {
  195. if(flag.epoch==uninitialized_flag)
  196. {
  197. flag.epoch=being_initialized;
  198. BOOST_TRY
  199. {
  200. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  201. BOOST_THREAD_INVOKE_RET_VOID(f,p1) BOOST_THREAD_INVOKE_RET_VOID_CALL;
  202. }
  203. BOOST_CATCH (...)
  204. {
  205. flag.epoch=uninitialized_flag;
  206. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  207. BOOST_RETHROW
  208. }
  209. BOOST_CATCH_END
  210. flag.epoch=--thread_detail::once_global_epoch;
  211. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  212. }
  213. else
  214. {
  215. while(flag.epoch==being_initialized)
  216. {
  217. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  218. }
  219. }
  220. }
  221. this_thread_epoch=thread_detail::once_global_epoch;
  222. }
  223. }
  224. template<typename Function, typename T1, typename T2>
  225. inline void call_once(once_flag& flag, Function f, T1 p1, T2 p2)
  226. {
  227. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  228. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  229. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  230. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  231. if(epoch<this_thread_epoch)
  232. {
  233. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  234. while(flag.epoch<=being_initialized)
  235. {
  236. if(flag.epoch==uninitialized_flag)
  237. {
  238. flag.epoch=being_initialized;
  239. BOOST_TRY
  240. {
  241. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  242. BOOST_THREAD_INVOKE_RET_VOID(f,p1, p2) BOOST_THREAD_INVOKE_RET_VOID_CALL;
  243. }
  244. BOOST_CATCH (...)
  245. {
  246. flag.epoch=uninitialized_flag;
  247. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  248. BOOST_RETHROW
  249. }
  250. BOOST_CATCH_END
  251. flag.epoch=--thread_detail::once_global_epoch;
  252. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  253. }
  254. else
  255. {
  256. while(flag.epoch==being_initialized)
  257. {
  258. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  259. }
  260. }
  261. }
  262. this_thread_epoch=thread_detail::once_global_epoch;
  263. }
  264. }
  265. template<typename Function, typename T1, typename T2, typename T3>
  266. inline void call_once(once_flag& flag, Function f, T1 p1, T2 p2, T3 p3)
  267. {
  268. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  269. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  270. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  271. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  272. if(epoch<this_thread_epoch)
  273. {
  274. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  275. while(flag.epoch<=being_initialized)
  276. {
  277. if(flag.epoch==uninitialized_flag)
  278. {
  279. flag.epoch=being_initialized;
  280. BOOST_TRY
  281. {
  282. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  283. BOOST_THREAD_INVOKE_RET_VOID(f,p1, p2, p3) BOOST_THREAD_INVOKE_RET_VOID_CALL;
  284. }
  285. BOOST_CATCH (...)
  286. {
  287. flag.epoch=uninitialized_flag;
  288. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  289. BOOST_RETHROW
  290. }
  291. BOOST_CATCH_END
  292. flag.epoch=--thread_detail::once_global_epoch;
  293. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  294. }
  295. else
  296. {
  297. while(flag.epoch==being_initialized)
  298. {
  299. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  300. }
  301. }
  302. }
  303. this_thread_epoch=thread_detail::once_global_epoch;
  304. }
  305. }
  306. template<typename Function>
  307. inline void call_once(once_flag& flag, BOOST_THREAD_RV_REF(Function) f)
  308. {
  309. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  310. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  311. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  312. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  313. if(epoch<this_thread_epoch)
  314. {
  315. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  316. while(flag.epoch<=being_initialized)
  317. {
  318. if(flag.epoch==uninitialized_flag)
  319. {
  320. flag.epoch=being_initialized;
  321. BOOST_TRY
  322. {
  323. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  324. f();
  325. }
  326. BOOST_CATCH (...)
  327. {
  328. flag.epoch=uninitialized_flag;
  329. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  330. BOOST_RETHROW
  331. }
  332. BOOST_CATCH_END
  333. flag.epoch=--thread_detail::once_global_epoch;
  334. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  335. }
  336. else
  337. {
  338. while(flag.epoch==being_initialized)
  339. {
  340. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  341. }
  342. }
  343. }
  344. this_thread_epoch=thread_detail::once_global_epoch;
  345. }
  346. }
  347. template<typename Function, typename T1>
  348. inline void call_once(once_flag& flag, BOOST_THREAD_RV_REF(Function) f, BOOST_THREAD_RV_REF(T1) p1)
  349. {
  350. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  351. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  352. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  353. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  354. if(epoch<this_thread_epoch)
  355. {
  356. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  357. while(flag.epoch<=being_initialized)
  358. {
  359. if(flag.epoch==uninitialized_flag)
  360. {
  361. flag.epoch=being_initialized;
  362. BOOST_TRY
  363. {
  364. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  365. BOOST_THREAD_INVOKE_RET_VOID(
  366. thread_detail::decay_copy(boost::forward<Function>(f)),
  367. thread_detail::decay_copy(boost::forward<T1>(p1))
  368. ) BOOST_THREAD_INVOKE_RET_VOID_CALL;
  369. }
  370. BOOST_CATCH (...)
  371. {
  372. flag.epoch=uninitialized_flag;
  373. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  374. BOOST_RETHROW
  375. }
  376. BOOST_CATCH_END
  377. flag.epoch=--thread_detail::once_global_epoch;
  378. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  379. }
  380. else
  381. {
  382. while(flag.epoch==being_initialized)
  383. {
  384. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  385. }
  386. }
  387. }
  388. this_thread_epoch=thread_detail::once_global_epoch;
  389. }
  390. }
  391. template<typename Function, typename T1, typename T2>
  392. inline void call_once(once_flag& flag, BOOST_THREAD_RV_REF(Function) f, BOOST_THREAD_RV_REF(T1) p1, BOOST_THREAD_RV_REF(T2) p2)
  393. {
  394. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  395. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  396. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  397. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  398. if(epoch<this_thread_epoch)
  399. {
  400. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  401. while(flag.epoch<=being_initialized)
  402. {
  403. if(flag.epoch==uninitialized_flag)
  404. {
  405. flag.epoch=being_initialized;
  406. BOOST_TRY
  407. {
  408. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  409. BOOST_THREAD_INVOKE_RET_VOID(
  410. thread_detail::decay_copy(boost::forward<Function>(f)),
  411. thread_detail::decay_copy(boost::forward<T1>(p1)),
  412. thread_detail::decay_copy(boost::forward<T1>(p2))
  413. ) BOOST_THREAD_INVOKE_RET_VOID_CALL;
  414. }
  415. BOOST_CATCH (...)
  416. {
  417. flag.epoch=uninitialized_flag;
  418. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  419. BOOST_RETHROW
  420. }
  421. BOOST_CATCH_END
  422. flag.epoch=--thread_detail::once_global_epoch;
  423. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  424. }
  425. else
  426. {
  427. while(flag.epoch==being_initialized)
  428. {
  429. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  430. }
  431. }
  432. }
  433. this_thread_epoch=thread_detail::once_global_epoch;
  434. }
  435. }
  436. template<typename Function, typename T1, typename T2, typename T3>
  437. inline void call_once(once_flag& flag, BOOST_THREAD_RV_REF(Function) f, BOOST_THREAD_RV_REF(T1) p1, BOOST_THREAD_RV_REF(T2) p2, BOOST_THREAD_RV_REF(T3) p3)
  438. {
  439. static thread_detail::uintmax_atomic_t const uninitialized_flag=BOOST_ONCE_INITIAL_FLAG_VALUE;
  440. static thread_detail::uintmax_atomic_t const being_initialized=uninitialized_flag+1;
  441. thread_detail::uintmax_atomic_t const epoch=flag.epoch;
  442. thread_detail::uintmax_atomic_t& this_thread_epoch=thread_detail::get_once_per_thread_epoch();
  443. if(epoch<this_thread_epoch)
  444. {
  445. pthread::pthread_mutex_scoped_lock lk(&thread_detail::once_epoch_mutex);
  446. while(flag.epoch<=being_initialized)
  447. {
  448. if(flag.epoch==uninitialized_flag)
  449. {
  450. flag.epoch=being_initialized;
  451. BOOST_TRY
  452. {
  453. pthread::pthread_mutex_scoped_unlock relocker(&thread_detail::once_epoch_mutex);
  454. BOOST_THREAD_INVOKE_RET_VOID(
  455. thread_detail::decay_copy(boost::forward<Function>(f)),
  456. thread_detail::decay_copy(boost::forward<T1>(p1)),
  457. thread_detail::decay_copy(boost::forward<T1>(p2)),
  458. thread_detail::decay_copy(boost::forward<T1>(p3))
  459. ) BOOST_THREAD_INVOKE_RET_VOID_CALL;
  460. }
  461. BOOST_CATCH (...)
  462. {
  463. flag.epoch=uninitialized_flag;
  464. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  465. BOOST_RETHROW
  466. }
  467. BOOST_CATCH_END
  468. flag.epoch=--thread_detail::once_global_epoch;
  469. BOOST_VERIFY(!pthread_cond_broadcast(&thread_detail::once_epoch_cv));
  470. }
  471. else
  472. {
  473. while(flag.epoch==being_initialized)
  474. {
  475. BOOST_VERIFY(!pthread_cond_wait(&thread_detail::once_epoch_cv,&thread_detail::once_epoch_mutex));
  476. }
  477. }
  478. }
  479. this_thread_epoch=thread_detail::once_global_epoch;
  480. }
  481. }
  482. #endif
  483. }
  484. #include <boost/config/abi_suffix.hpp>
  485. #endif