signal_set_service.ipp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669
  1. //
  2. // detail/impl/signal_set_service.ipp
  3. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2003-2019 Christopher M. Kohlhoff (chris at kohlhoff dot com)
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  8. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  9. //
  10. #ifndef BOOST_ASIO_DETAIL_IMPL_SIGNAL_SET_SERVICE_IPP
  11. #define BOOST_ASIO_DETAIL_IMPL_SIGNAL_SET_SERVICE_IPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include <boost/asio/detail/config.hpp>
  16. #include <cstring>
  17. #include <stdexcept>
  18. #include <boost/asio/detail/reactor.hpp>
  19. #include <boost/asio/detail/signal_blocker.hpp>
  20. #include <boost/asio/detail/signal_set_service.hpp>
  21. #include <boost/asio/detail/static_mutex.hpp>
  22. #include <boost/asio/detail/throw_exception.hpp>
  23. #include <boost/asio/detail/push_options.hpp>
  24. namespace boost {
  25. namespace asio {
  26. namespace detail {
  27. struct signal_state
  28. {
  29. // Mutex used for protecting global state.
  30. static_mutex mutex_;
  31. // The read end of the pipe used for signal notifications.
  32. int read_descriptor_;
  33. // The write end of the pipe used for signal notifications.
  34. int write_descriptor_;
  35. // Whether the signal state has been prepared for a fork.
  36. bool fork_prepared_;
  37. // The head of a linked list of all signal_set_service instances.
  38. class signal_set_service* service_list_;
  39. // A count of the number of objects that are registered for each signal.
  40. std::size_t registration_count_[max_signal_number];
  41. };
  42. signal_state* get_signal_state()
  43. {
  44. static signal_state state = {
  45. BOOST_ASIO_STATIC_MUTEX_INIT, -1, -1, false, 0, { 0 } };
  46. return &state;
  47. }
  48. void boost_asio_signal_handler(int signal_number)
  49. {
  50. #if defined(BOOST_ASIO_WINDOWS) \
  51. || defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  52. || defined(__CYGWIN__)
  53. signal_set_service::deliver_signal(signal_number);
  54. #else // defined(BOOST_ASIO_WINDOWS)
  55. // || defined(BOOST_ASIO_WINDOWS_RUNTIME)
  56. // || defined(__CYGWIN__)
  57. int saved_errno = errno;
  58. signal_state* state = get_signal_state();
  59. signed_size_type result = ::write(state->write_descriptor_,
  60. &signal_number, sizeof(signal_number));
  61. (void)result;
  62. errno = saved_errno;
  63. #endif // defined(BOOST_ASIO_WINDOWS)
  64. // || defined(BOOST_ASIO_WINDOWS_RUNTIME)
  65. // || defined(__CYGWIN__)
  66. #if defined(BOOST_ASIO_HAS_SIGNAL) && !defined(BOOST_ASIO_HAS_SIGACTION)
  67. ::signal(signal_number, boost_asio_signal_handler);
  68. #endif // defined(BOOST_ASIO_HAS_SIGNAL) && !defined(BOOST_ASIO_HAS_SIGACTION)
  69. }
  70. #if !defined(BOOST_ASIO_WINDOWS) \
  71. && !defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  72. && !defined(__CYGWIN__)
  73. class signal_set_service::pipe_read_op : public reactor_op
  74. {
  75. public:
  76. pipe_read_op()
  77. : reactor_op(&pipe_read_op::do_perform, pipe_read_op::do_complete)
  78. {
  79. }
  80. static status do_perform(reactor_op*)
  81. {
  82. signal_state* state = get_signal_state();
  83. int fd = state->read_descriptor_;
  84. int signal_number = 0;
  85. while (::read(fd, &signal_number, sizeof(int)) == sizeof(int))
  86. if (signal_number >= 0 && signal_number < max_signal_number)
  87. signal_set_service::deliver_signal(signal_number);
  88. return not_done;
  89. }
  90. static void do_complete(void* /*owner*/, operation* base,
  91. const boost::system::error_code& /*ec*/,
  92. std::size_t /*bytes_transferred*/)
  93. {
  94. pipe_read_op* o(static_cast<pipe_read_op*>(base));
  95. delete o;
  96. }
  97. };
  98. #endif // !defined(BOOST_ASIO_WINDOWS)
  99. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  100. // && !defined(__CYGWIN__)
  101. signal_set_service::signal_set_service(execution_context& context)
  102. : execution_context_service_base<signal_set_service>(context),
  103. scheduler_(boost::asio::use_service<scheduler_impl>(context)),
  104. #if !defined(BOOST_ASIO_WINDOWS) \
  105. && !defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  106. && !defined(__CYGWIN__)
  107. reactor_(boost::asio::use_service<reactor>(context)),
  108. #endif // !defined(BOOST_ASIO_WINDOWS)
  109. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  110. // && !defined(__CYGWIN__)
  111. next_(0),
  112. prev_(0)
  113. {
  114. get_signal_state()->mutex_.init();
  115. #if !defined(BOOST_ASIO_WINDOWS) \
  116. && !defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  117. && !defined(__CYGWIN__)
  118. reactor_.init_task();
  119. #endif // !defined(BOOST_ASIO_WINDOWS)
  120. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  121. // && !defined(__CYGWIN__)
  122. for (int i = 0; i < max_signal_number; ++i)
  123. registrations_[i] = 0;
  124. add_service(this);
  125. }
  126. signal_set_service::~signal_set_service()
  127. {
  128. remove_service(this);
  129. }
  130. void signal_set_service::shutdown()
  131. {
  132. remove_service(this);
  133. op_queue<operation> ops;
  134. for (int i = 0; i < max_signal_number; ++i)
  135. {
  136. registration* reg = registrations_[i];
  137. while (reg)
  138. {
  139. ops.push(*reg->queue_);
  140. reg = reg->next_in_table_;
  141. }
  142. }
  143. scheduler_.abandon_operations(ops);
  144. }
  145. void signal_set_service::notify_fork(execution_context::fork_event fork_ev)
  146. {
  147. #if !defined(BOOST_ASIO_WINDOWS) \
  148. && !defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  149. && !defined(__CYGWIN__)
  150. signal_state* state = get_signal_state();
  151. static_mutex::scoped_lock lock(state->mutex_);
  152. switch (fork_ev)
  153. {
  154. case execution_context::fork_prepare:
  155. {
  156. int read_descriptor = state->read_descriptor_;
  157. state->fork_prepared_ = true;
  158. lock.unlock();
  159. reactor_.deregister_internal_descriptor(read_descriptor, reactor_data_);
  160. reactor_.cleanup_descriptor_data(reactor_data_);
  161. }
  162. break;
  163. case execution_context::fork_parent:
  164. if (state->fork_prepared_)
  165. {
  166. int read_descriptor = state->read_descriptor_;
  167. state->fork_prepared_ = false;
  168. lock.unlock();
  169. reactor_.register_internal_descriptor(reactor::read_op,
  170. read_descriptor, reactor_data_, new pipe_read_op);
  171. }
  172. break;
  173. case execution_context::fork_child:
  174. if (state->fork_prepared_)
  175. {
  176. boost::asio::detail::signal_blocker blocker;
  177. close_descriptors();
  178. open_descriptors();
  179. int read_descriptor = state->read_descriptor_;
  180. state->fork_prepared_ = false;
  181. lock.unlock();
  182. reactor_.register_internal_descriptor(reactor::read_op,
  183. read_descriptor, reactor_data_, new pipe_read_op);
  184. }
  185. break;
  186. default:
  187. break;
  188. }
  189. #else // !defined(BOOST_ASIO_WINDOWS)
  190. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  191. // && !defined(__CYGWIN__)
  192. (void)fork_ev;
  193. #endif // !defined(BOOST_ASIO_WINDOWS)
  194. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  195. // && !defined(__CYGWIN__)
  196. }
  197. void signal_set_service::construct(
  198. signal_set_service::implementation_type& impl)
  199. {
  200. impl.signals_ = 0;
  201. }
  202. void signal_set_service::destroy(
  203. signal_set_service::implementation_type& impl)
  204. {
  205. boost::system::error_code ignored_ec;
  206. clear(impl, ignored_ec);
  207. cancel(impl, ignored_ec);
  208. }
  209. boost::system::error_code signal_set_service::add(
  210. signal_set_service::implementation_type& impl,
  211. int signal_number, boost::system::error_code& ec)
  212. {
  213. // Check that the signal number is valid.
  214. if (signal_number < 0 || signal_number >= max_signal_number)
  215. {
  216. ec = boost::asio::error::invalid_argument;
  217. return ec;
  218. }
  219. signal_state* state = get_signal_state();
  220. static_mutex::scoped_lock lock(state->mutex_);
  221. // Find the appropriate place to insert the registration.
  222. registration** insertion_point = &impl.signals_;
  223. registration* next = impl.signals_;
  224. while (next && next->signal_number_ < signal_number)
  225. {
  226. insertion_point = &next->next_in_set_;
  227. next = next->next_in_set_;
  228. }
  229. // Only do something if the signal is not already registered.
  230. if (next == 0 || next->signal_number_ != signal_number)
  231. {
  232. registration* new_registration = new registration;
  233. #if defined(BOOST_ASIO_HAS_SIGNAL) || defined(BOOST_ASIO_HAS_SIGACTION)
  234. // Register for the signal if we're the first.
  235. if (state->registration_count_[signal_number] == 0)
  236. {
  237. # if defined(BOOST_ASIO_HAS_SIGACTION)
  238. using namespace std; // For memset.
  239. struct sigaction sa;
  240. memset(&sa, 0, sizeof(sa));
  241. sa.sa_handler = boost_asio_signal_handler;
  242. sigfillset(&sa.sa_mask);
  243. if (::sigaction(signal_number, &sa, 0) == -1)
  244. # else // defined(BOOST_ASIO_HAS_SIGACTION)
  245. if (::signal(signal_number, boost_asio_signal_handler) == SIG_ERR)
  246. # endif // defined(BOOST_ASIO_HAS_SIGACTION)
  247. {
  248. # if defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  249. ec = boost::asio::error::invalid_argument;
  250. # else // defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  251. ec = boost::system::error_code(errno,
  252. boost::asio::error::get_system_category());
  253. # endif // defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  254. delete new_registration;
  255. return ec;
  256. }
  257. }
  258. #endif // defined(BOOST_ASIO_HAS_SIGNAL) || defined(BOOST_ASIO_HAS_SIGACTION)
  259. // Record the new registration in the set.
  260. new_registration->signal_number_ = signal_number;
  261. new_registration->queue_ = &impl.queue_;
  262. new_registration->next_in_set_ = next;
  263. *insertion_point = new_registration;
  264. // Insert registration into the registration table.
  265. new_registration->next_in_table_ = registrations_[signal_number];
  266. if (registrations_[signal_number])
  267. registrations_[signal_number]->prev_in_table_ = new_registration;
  268. registrations_[signal_number] = new_registration;
  269. ++state->registration_count_[signal_number];
  270. }
  271. ec = boost::system::error_code();
  272. return ec;
  273. }
  274. boost::system::error_code signal_set_service::remove(
  275. signal_set_service::implementation_type& impl,
  276. int signal_number, boost::system::error_code& ec)
  277. {
  278. // Check that the signal number is valid.
  279. if (signal_number < 0 || signal_number >= max_signal_number)
  280. {
  281. ec = boost::asio::error::invalid_argument;
  282. return ec;
  283. }
  284. signal_state* state = get_signal_state();
  285. static_mutex::scoped_lock lock(state->mutex_);
  286. // Find the signal number in the list of registrations.
  287. registration** deletion_point = &impl.signals_;
  288. registration* reg = impl.signals_;
  289. while (reg && reg->signal_number_ < signal_number)
  290. {
  291. deletion_point = &reg->next_in_set_;
  292. reg = reg->next_in_set_;
  293. }
  294. if (reg != 0 && reg->signal_number_ == signal_number)
  295. {
  296. #if defined(BOOST_ASIO_HAS_SIGNAL) || defined(BOOST_ASIO_HAS_SIGACTION)
  297. // Set signal handler back to the default if we're the last.
  298. if (state->registration_count_[signal_number] == 1)
  299. {
  300. # if defined(BOOST_ASIO_HAS_SIGACTION)
  301. using namespace std; // For memset.
  302. struct sigaction sa;
  303. memset(&sa, 0, sizeof(sa));
  304. sa.sa_handler = SIG_DFL;
  305. if (::sigaction(signal_number, &sa, 0) == -1)
  306. # else // defined(BOOST_ASIO_HAS_SIGACTION)
  307. if (::signal(signal_number, SIG_DFL) == SIG_ERR)
  308. # endif // defined(BOOST_ASIO_HAS_SIGACTION)
  309. {
  310. # if defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  311. ec = boost::asio::error::invalid_argument;
  312. # else // defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  313. ec = boost::system::error_code(errno,
  314. boost::asio::error::get_system_category());
  315. # endif // defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  316. return ec;
  317. }
  318. }
  319. #endif // defined(BOOST_ASIO_HAS_SIGNAL) || defined(BOOST_ASIO_HAS_SIGACTION)
  320. // Remove the registration from the set.
  321. *deletion_point = reg->next_in_set_;
  322. // Remove the registration from the registration table.
  323. if (registrations_[signal_number] == reg)
  324. registrations_[signal_number] = reg->next_in_table_;
  325. if (reg->prev_in_table_)
  326. reg->prev_in_table_->next_in_table_ = reg->next_in_table_;
  327. if (reg->next_in_table_)
  328. reg->next_in_table_->prev_in_table_ = reg->prev_in_table_;
  329. --state->registration_count_[signal_number];
  330. delete reg;
  331. }
  332. ec = boost::system::error_code();
  333. return ec;
  334. }
  335. boost::system::error_code signal_set_service::clear(
  336. signal_set_service::implementation_type& impl,
  337. boost::system::error_code& ec)
  338. {
  339. signal_state* state = get_signal_state();
  340. static_mutex::scoped_lock lock(state->mutex_);
  341. while (registration* reg = impl.signals_)
  342. {
  343. #if defined(BOOST_ASIO_HAS_SIGNAL) || defined(BOOST_ASIO_HAS_SIGACTION)
  344. // Set signal handler back to the default if we're the last.
  345. if (state->registration_count_[reg->signal_number_] == 1)
  346. {
  347. # if defined(BOOST_ASIO_HAS_SIGACTION)
  348. using namespace std; // For memset.
  349. struct sigaction sa;
  350. memset(&sa, 0, sizeof(sa));
  351. sa.sa_handler = SIG_DFL;
  352. if (::sigaction(reg->signal_number_, &sa, 0) == -1)
  353. # else // defined(BOOST_ASIO_HAS_SIGACTION)
  354. if (::signal(reg->signal_number_, SIG_DFL) == SIG_ERR)
  355. # endif // defined(BOOST_ASIO_HAS_SIGACTION)
  356. {
  357. # if defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  358. ec = boost::asio::error::invalid_argument;
  359. # else // defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  360. ec = boost::system::error_code(errno,
  361. boost::asio::error::get_system_category());
  362. # endif // defined(BOOST_ASIO_WINDOWS) || defined(__CYGWIN__)
  363. return ec;
  364. }
  365. }
  366. #endif // defined(BOOST_ASIO_HAS_SIGNAL) || defined(BOOST_ASIO_HAS_SIGACTION)
  367. // Remove the registration from the registration table.
  368. if (registrations_[reg->signal_number_] == reg)
  369. registrations_[reg->signal_number_] = reg->next_in_table_;
  370. if (reg->prev_in_table_)
  371. reg->prev_in_table_->next_in_table_ = reg->next_in_table_;
  372. if (reg->next_in_table_)
  373. reg->next_in_table_->prev_in_table_ = reg->prev_in_table_;
  374. --state->registration_count_[reg->signal_number_];
  375. impl.signals_ = reg->next_in_set_;
  376. delete reg;
  377. }
  378. ec = boost::system::error_code();
  379. return ec;
  380. }
  381. boost::system::error_code signal_set_service::cancel(
  382. signal_set_service::implementation_type& impl,
  383. boost::system::error_code& ec)
  384. {
  385. BOOST_ASIO_HANDLER_OPERATION((scheduler_.context(),
  386. "signal_set", &impl, 0, "cancel"));
  387. op_queue<operation> ops;
  388. {
  389. signal_state* state = get_signal_state();
  390. static_mutex::scoped_lock lock(state->mutex_);
  391. while (signal_op* op = impl.queue_.front())
  392. {
  393. op->ec_ = boost::asio::error::operation_aborted;
  394. impl.queue_.pop();
  395. ops.push(op);
  396. }
  397. }
  398. scheduler_.post_deferred_completions(ops);
  399. ec = boost::system::error_code();
  400. return ec;
  401. }
  402. void signal_set_service::deliver_signal(int signal_number)
  403. {
  404. signal_state* state = get_signal_state();
  405. static_mutex::scoped_lock lock(state->mutex_);
  406. signal_set_service* service = state->service_list_;
  407. while (service)
  408. {
  409. op_queue<operation> ops;
  410. registration* reg = service->registrations_[signal_number];
  411. while (reg)
  412. {
  413. if (reg->queue_->empty())
  414. {
  415. ++reg->undelivered_;
  416. }
  417. else
  418. {
  419. while (signal_op* op = reg->queue_->front())
  420. {
  421. op->signal_number_ = signal_number;
  422. reg->queue_->pop();
  423. ops.push(op);
  424. }
  425. }
  426. reg = reg->next_in_table_;
  427. }
  428. service->scheduler_.post_deferred_completions(ops);
  429. service = service->next_;
  430. }
  431. }
  432. void signal_set_service::add_service(signal_set_service* service)
  433. {
  434. signal_state* state = get_signal_state();
  435. static_mutex::scoped_lock lock(state->mutex_);
  436. #if !defined(BOOST_ASIO_WINDOWS) && !defined(__CYGWIN__)
  437. // If this is the first service to be created, open a new pipe.
  438. if (state->service_list_ == 0)
  439. open_descriptors();
  440. #endif // !defined(BOOST_ASIO_WINDOWS) && !defined(__CYGWIN__)
  441. // If a scheduler_ object is thread-unsafe then it must be the only
  442. // scheduler used to create signal_set objects.
  443. if (state->service_list_ != 0)
  444. {
  445. if (!BOOST_ASIO_CONCURRENCY_HINT_IS_LOCKING(SCHEDULER,
  446. service->scheduler_.concurrency_hint())
  447. || !BOOST_ASIO_CONCURRENCY_HINT_IS_LOCKING(SCHEDULER,
  448. state->service_list_->scheduler_.concurrency_hint()))
  449. {
  450. std::logic_error ex(
  451. "Thread-unsafe execution context objects require "
  452. "exclusive access to signal handling.");
  453. boost::asio::detail::throw_exception(ex);
  454. }
  455. }
  456. // Insert service into linked list of all services.
  457. service->next_ = state->service_list_;
  458. service->prev_ = 0;
  459. if (state->service_list_)
  460. state->service_list_->prev_ = service;
  461. state->service_list_ = service;
  462. #if !defined(BOOST_ASIO_WINDOWS) \
  463. && !defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  464. && !defined(__CYGWIN__)
  465. // Register for pipe readiness notifications.
  466. int read_descriptor = state->read_descriptor_;
  467. lock.unlock();
  468. service->reactor_.register_internal_descriptor(reactor::read_op,
  469. read_descriptor, service->reactor_data_, new pipe_read_op);
  470. #endif // !defined(BOOST_ASIO_WINDOWS)
  471. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  472. // && !defined(__CYGWIN__)
  473. }
  474. void signal_set_service::remove_service(signal_set_service* service)
  475. {
  476. signal_state* state = get_signal_state();
  477. static_mutex::scoped_lock lock(state->mutex_);
  478. if (service->next_ || service->prev_ || state->service_list_ == service)
  479. {
  480. #if !defined(BOOST_ASIO_WINDOWS) \
  481. && !defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  482. && !defined(__CYGWIN__)
  483. // Disable the pipe readiness notifications.
  484. int read_descriptor = state->read_descriptor_;
  485. lock.unlock();
  486. service->reactor_.deregister_internal_descriptor(
  487. read_descriptor, service->reactor_data_);
  488. service->reactor_.cleanup_descriptor_data(service->reactor_data_);
  489. lock.lock();
  490. #endif // !defined(BOOST_ASIO_WINDOWS)
  491. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  492. // && !defined(__CYGWIN__)
  493. // Remove service from linked list of all services.
  494. if (state->service_list_ == service)
  495. state->service_list_ = service->next_;
  496. if (service->prev_)
  497. service->prev_->next_ = service->next_;
  498. if (service->next_)
  499. service->next_->prev_= service->prev_;
  500. service->next_ = 0;
  501. service->prev_ = 0;
  502. #if !defined(BOOST_ASIO_WINDOWS) && !defined(__CYGWIN__)
  503. // If this is the last service to be removed, close the pipe.
  504. if (state->service_list_ == 0)
  505. close_descriptors();
  506. #endif // !defined(BOOST_ASIO_WINDOWS) && !defined(__CYGWIN__)
  507. }
  508. }
  509. void signal_set_service::open_descriptors()
  510. {
  511. #if !defined(BOOST_ASIO_WINDOWS) \
  512. && !defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  513. && !defined(__CYGWIN__)
  514. signal_state* state = get_signal_state();
  515. int pipe_fds[2];
  516. if (::pipe(pipe_fds) == 0)
  517. {
  518. state->read_descriptor_ = pipe_fds[0];
  519. ::fcntl(state->read_descriptor_, F_SETFL, O_NONBLOCK);
  520. state->write_descriptor_ = pipe_fds[1];
  521. ::fcntl(state->write_descriptor_, F_SETFL, O_NONBLOCK);
  522. #if defined(FD_CLOEXEC)
  523. ::fcntl(state->read_descriptor_, F_SETFD, FD_CLOEXEC);
  524. ::fcntl(state->write_descriptor_, F_SETFD, FD_CLOEXEC);
  525. #endif // defined(FD_CLOEXEC)
  526. }
  527. else
  528. {
  529. boost::system::error_code ec(errno,
  530. boost::asio::error::get_system_category());
  531. boost::asio::detail::throw_error(ec, "signal_set_service pipe");
  532. }
  533. #endif // !defined(BOOST_ASIO_WINDOWS)
  534. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  535. // && !defined(__CYGWIN__)
  536. }
  537. void signal_set_service::close_descriptors()
  538. {
  539. #if !defined(BOOST_ASIO_WINDOWS) \
  540. && !defined(BOOST_ASIO_WINDOWS_RUNTIME) \
  541. && !defined(__CYGWIN__)
  542. signal_state* state = get_signal_state();
  543. if (state->read_descriptor_ != -1)
  544. ::close(state->read_descriptor_);
  545. state->read_descriptor_ = -1;
  546. if (state->write_descriptor_ != -1)
  547. ::close(state->write_descriptor_);
  548. state->write_descriptor_ = -1;
  549. #endif // !defined(BOOST_ASIO_WINDOWS)
  550. // && !defined(BOOST_ASIO_WINDOWS_RUNTIME)
  551. // && !defined(__CYGWIN__)
  552. }
  553. void signal_set_service::start_wait_op(
  554. signal_set_service::implementation_type& impl, signal_op* op)
  555. {
  556. scheduler_.work_started();
  557. signal_state* state = get_signal_state();
  558. static_mutex::scoped_lock lock(state->mutex_);
  559. registration* reg = impl.signals_;
  560. while (reg)
  561. {
  562. if (reg->undelivered_ > 0)
  563. {
  564. --reg->undelivered_;
  565. op->signal_number_ = reg->signal_number_;
  566. scheduler_.post_deferred_completion(op);
  567. return;
  568. }
  569. reg = reg->next_in_set_;
  570. }
  571. impl.queue_.push(op);
  572. }
  573. } // namespace detail
  574. } // namespace asio
  575. } // namespace boost
  576. #include <boost/asio/detail/pop_options.hpp>
  577. #endif // BOOST_ASIO_DETAIL_IMPL_SIGNAL_SET_SERVICE_IPP