include/boost/corosio/native/detail/posix/posix_resolver_service.hpp

81.2% Lines (237/292) 93.5% List of functions (29/31)
posix_resolver_service.hpp
f(x) Functions (31)
Function Calls Lines Blocks
boost::corosio::detail::posix_resolver_service::posix_resolver_service(boost::capy::execution_context&, boost::corosio::detail::scheduler&) :37 515x 100.0% 73.0% boost::corosio::detail::posix_resolver_service::~posix_resolver_service() :45 1030x 100.0% 100.0% boost::corosio::detail::posix_resolver_service::destroy(boost::corosio::io_object::implementation*) :52 29x 100.0% 100.0% boost::corosio::detail::posix_resolver_service::pool() :67 26x 100.0% 100.0% boost::corosio::detail::posix_resolver_service::single_threaded() const :73 26x 100.0% 100.0% boost::corosio::detail::posix_resolver_detail::flags_to_hints(boost::corosio::resolve_flags) :107 16x 73.3% 80.0% boost::corosio::detail::posix_resolver_detail::flags_to_ni_flags(boost::corosio::reverse_flags) :128 10x 90.9% 93.0% boost::corosio::detail::posix_resolver_detail::convert_results(addrinfo*, std::basic_string_view<char, std::char_traits<char> >, std::basic_string_view<char, std::char_traits<char> >) :145 13x 100.0% 74.0% boost::corosio::detail::posix_resolver_detail::make_gai_error(int) :171 4x 16.1% 17.0% boost::corosio::detail::posix_resolver::posix_resolver(boost::corosio::detail::posix_resolver_service&) :236 29x 100.0% 100.0% boost::corosio::detail::posix_resolver::resolve_op::reset() :244 16x 100.0% 100.0% boost::corosio::detail::posix_resolver::resolve_op::operator()() :258 16x 93.3% 90.0% boost::corosio::detail::posix_resolver::resolve_op::destroy() :283 0 0.0% 0.0% boost::corosio::detail::posix_resolver::resolve_op::request_cancel() :289 33x 100.0% 100.0% boost::corosio::detail::posix_resolver::resolve_op::start(std::stop_token const&) :295 16x 83.3% 71.0% boost::corosio::detail::posix_resolver::reverse_resolve_op::reset() :307 10x 100.0% 100.0% boost::corosio::detail::posix_resolver::reverse_resolve_op::operator()() :321 10x 93.8% 93.0% boost::corosio::detail::posix_resolver::reverse_resolve_op::destroy() :349 0 0.0% 0.0% boost::corosio::detail::posix_resolver::reverse_resolve_op::request_cancel() :355 33x 100.0% 100.0% boost::corosio::detail::posix_resolver::reverse_resolve_op::start(std::stop_token const&) :361 10x 83.3% 71.0% boost::corosio::detail::posix_resolver::resolve(std::__n4861::coroutine_handle<void>, boost::capy::executor_ref, std::basic_string_view<char, std::char_traits<char> >, std::basic_string_view<char, std::char_traits<char> >, boost::corosio::resolve_flags, std::stop_token, std::error_code*, boost::corosio::resolver_results*) :373 16x 76.0% 78.0% boost::corosio::detail::posix_resolver::reverse_resolve(std::__n4861::coroutine_handle<void>, boost::capy::executor_ref, boost::corosio::endpoint const&, boost::corosio::reverse_flags, std::stop_token, std::error_code*, boost::corosio::reverse_resolver_result*) :420 10x 75.0% 76.0% boost::corosio::detail::posix_resolver::cancel() :465 33x 100.0% 100.0% boost::corosio::detail::posix_resolver::do_resolve_work(boost::corosio::detail::pool_work_item*) :472 16x 100.0% 93.0% boost::corosio::detail::posix_resolver::do_reverse_resolve_work(boost::corosio::detail::pool_work_item*) :512 10x 100.0% 100.0% boost::corosio::detail::posix_resolver_service::shutdown() :564 515x 71.4% 78.0% boost::corosio::detail::posix_resolver_service::construct() :582 29x 100.0% 71.0% boost::corosio::detail::posix_resolver_service::destroy_impl(boost::corosio::detail::posix_resolver&) :597 29x 100.0% 67.0% boost::corosio::detail::posix_resolver_service::post(boost::corosio::detail::scheduler_op*) :605 26x 100.0% 100.0% boost::corosio::detail::posix_resolver_service::work_finished() :617 26x 100.0% 100.0% boost::corosio::detail::get_resolver_service(boost::capy::execution_context&, boost::corosio::detail::scheduler&) :625 515x 100.0% 100.0%
Line TLA Hits Source Code
1 //
2 // Copyright (c) 2026 Steve Gerbino
3 //
4 // Distributed under the Boost Software License, Version 1.0. (See accompanying
5 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6 //
7 // Official repository: https://github.com/cppalliance/corosio
8 //
9
10 #ifndef BOOST_COROSIO_NATIVE_DETAIL_POSIX_POSIX_RESOLVER_SERVICE_HPP
11 #define BOOST_COROSIO_NATIVE_DETAIL_POSIX_POSIX_RESOLVER_SERVICE_HPP
12
13 #include <boost/corosio/detail/platform.hpp>
14
15 #if BOOST_COROSIO_POSIX
16
17 #include <boost/corosio/native/detail/posix/posix_resolver.hpp>
18 #include <boost/corosio/native/native_scheduler.hpp>
19 #include <boost/corosio/detail/thread_pool.hpp>
20
21 #include <unordered_map>
22
23 namespace boost::corosio::detail {
24
25 /** Resolver service for POSIX backends.
26
27 Owns all posix_resolver instances. Thread lifecycle is managed
28 by the thread_pool service.
29 */
30 class BOOST_COROSIO_DECL posix_resolver_service final
31 : public capy::execution_context::service
32 , public io_object::io_service
33 {
34 public:
35 using key_type = posix_resolver_service;
36
37 515x posix_resolver_service(capy::execution_context& ctx, scheduler& sched)
38 1030x : sched_(&sched)
39 1030x , pool_(ctx.make_service<thread_pool>())
40 515x , single_threaded_(
41 515x static_cast<native_scheduler&>(sched).single_threaded_)
42 {
43 515x }
44
45 1030x ~posix_resolver_service() override = default;
46
47 posix_resolver_service(posix_resolver_service const&) = delete;
48 posix_resolver_service& operator=(posix_resolver_service const&) = delete;
49
50 io_object::implementation* construct() override;
51
52 29x void destroy(io_object::implementation* p) override
53 {
54 29x auto& impl = static_cast<posix_resolver&>(*p);
55 29x impl.cancel();
56 29x destroy_impl(impl);
57 29x }
58
59 void shutdown() override;
60 void destroy_impl(posix_resolver& impl);
61
62 void post(scheduler_op* op);
63 void work_started() noexcept;
64 void work_finished() noexcept;
65
66 /** Return the resolver thread pool. */
67 26x thread_pool& pool() noexcept
68 {
69 26x return pool_;
70 }
71
72 /** Return true if single-threaded mode is active. */
73 26x bool single_threaded() const noexcept
74 {
75 26x return single_threaded_;
76 }
77
78 private:
79 scheduler* sched_;
80 thread_pool& pool_;
81 bool single_threaded_;
82 std::mutex mutex_;
83 intrusive_list<posix_resolver> resolver_list_;
84 std::unordered_map<posix_resolver*, std::shared_ptr<posix_resolver>>
85 resolver_ptrs_;
86 };
87
88 /** Get or create the resolver service for the given context.
89
90 This function is called by the concrete scheduler during initialization
91 to create the resolver service with a reference to itself.
92
93 @param ctx Reference to the owning execution_context.
94 @param sched Reference to the scheduler for posting completions.
95 @return Reference to the resolver service.
96 */
97 posix_resolver_service&
98 get_resolver_service(capy::execution_context& ctx, scheduler& sched);
99
100 // ---------------------------------------------------------------------------
101 // Inline implementation
102 // ---------------------------------------------------------------------------
103
104 // posix_resolver_detail helpers
105
106 inline int
107 16x posix_resolver_detail::flags_to_hints(resolve_flags flags)
108 {
109 16x int hints = 0;
110
111 16x if ((flags & resolve_flags::passive) != resolve_flags::none)
112 hints |= AI_PASSIVE;
113 16x if ((flags & resolve_flags::numeric_host) != resolve_flags::none)
114 11x hints |= AI_NUMERICHOST;
115 16x if ((flags & resolve_flags::numeric_service) != resolve_flags::none)
116 8x hints |= AI_NUMERICSERV;
117 16x if ((flags & resolve_flags::address_configured) != resolve_flags::none)
118 hints |= AI_ADDRCONFIG;
119 16x if ((flags & resolve_flags::v4_mapped) != resolve_flags::none)
120 hints |= AI_V4MAPPED;
121 16x if ((flags & resolve_flags::all_matching) != resolve_flags::none)
122 hints |= AI_ALL;
123
124 16x return hints;
125 }
126
127 inline int
128 10x posix_resolver_detail::flags_to_ni_flags(reverse_flags flags)
129 {
130 10x int ni_flags = 0;
131
132 10x if ((flags & reverse_flags::numeric_host) != reverse_flags::none)
133 5x ni_flags |= NI_NUMERICHOST;
134 10x if ((flags & reverse_flags::numeric_service) != reverse_flags::none)
135 5x ni_flags |= NI_NUMERICSERV;
136 10x if ((flags & reverse_flags::name_required) != reverse_flags::none)
137 1x ni_flags |= NI_NAMEREQD;
138 10x if ((flags & reverse_flags::datagram_service) != reverse_flags::none)
139 ni_flags |= NI_DGRAM;
140
141 10x return ni_flags;
142 }
143
144 inline resolver_results
145 13x posix_resolver_detail::convert_results(
146 struct addrinfo* ai, std::string_view host, std::string_view service)
147 {
148 13x std::vector<resolver_entry> entries;
149 13x entries.reserve(4); // Most lookups return 1-4 addresses
150
151 26x for (auto* p = ai; p != nullptr; p = p->ai_next)
152 {
153 13x if (p->ai_family == AF_INET)
154 {
155 11x auto* addr = reinterpret_cast<sockaddr_in*>(p->ai_addr);
156 11x auto ep = from_sockaddr_in(*addr);
157 11x entries.emplace_back(ep, host, service);
158 }
159 2x else if (p->ai_family == AF_INET6)
160 {
161 2x auto* addr = reinterpret_cast<sockaddr_in6*>(p->ai_addr);
162 2x auto ep = from_sockaddr_in6(*addr);
163 2x entries.emplace_back(ep, host, service);
164 }
165 }
166
167 26x return resolver_results(std::move(entries));
168 13x }
169
170 inline std::error_code
171 4x posix_resolver_detail::make_gai_error(int gai_err)
172 {
173 // Map GAI errors to appropriate generic error codes
174 4x switch (gai_err)
175 {
176 case EAI_AGAIN:
177 // Temporary failure - try again later
178 return std::error_code(
179 static_cast<int>(std::errc::resource_unavailable_try_again),
180 std::generic_category());
181
182 case EAI_BADFLAGS:
183 // Invalid flags
184 return std::error_code(
185 static_cast<int>(std::errc::invalid_argument),
186 std::generic_category());
187
188 case EAI_FAIL:
189 // Non-recoverable failure
190 return std::error_code(
191 static_cast<int>(std::errc::io_error), std::generic_category());
192
193 case EAI_FAMILY:
194 // Address family not supported
195 return std::error_code(
196 static_cast<int>(std::errc::address_family_not_supported),
197 std::generic_category());
198
199 case EAI_MEMORY:
200 // Memory allocation failure
201 return std::error_code(
202 static_cast<int>(std::errc::not_enough_memory),
203 std::generic_category());
204
205 4x case EAI_NONAME:
206 // Host or service not found
207 4x return std::error_code(
208 static_cast<int>(std::errc::no_such_device_or_address),
209 4x std::generic_category());
210
211 case EAI_SERVICE:
212 // Service not supported for socket type
213 return std::error_code(
214 static_cast<int>(std::errc::invalid_argument),
215 std::generic_category());
216
217 case EAI_SOCKTYPE:
218 // Socket type not supported
219 return std::error_code(
220 static_cast<int>(std::errc::not_supported),
221 std::generic_category());
222
223 case EAI_SYSTEM:
224 // System error - use errno
225 return std::error_code(errno, std::generic_category());
226
227 default:
228 // Unknown error
229 return std::error_code(
230 static_cast<int>(std::errc::io_error), std::generic_category());
231 }
232 }
233
234 // posix_resolver
235
236 29x inline posix_resolver::posix_resolver(posix_resolver_service& svc) noexcept
237 29x : svc_(svc)
238 {
239 29x }
240
241 // posix_resolver::resolve_op implementation
242
243 inline void
244 16x posix_resolver::resolve_op::reset() noexcept
245 {
246 16x host.clear();
247 16x service.clear();
248 16x flags = resolve_flags::none;
249 16x stored_results = resolver_results{};
250 16x gai_error = 0;
251 16x cancelled.store(false, std::memory_order_relaxed);
252 16x stop_cb.reset();
253 16x ec_out = nullptr;
254 16x out = nullptr;
255 16x }
256
257 inline void
258 16x posix_resolver::resolve_op::operator()()
259 {
260 16x stop_cb.reset(); // Disconnect stop callback
261
262 16x bool const was_cancelled = cancelled.load(std::memory_order_acquire);
263
264 16x if (ec_out)
265 {
266 16x if (was_cancelled)
267 *ec_out = capy::error::canceled;
268 16x else if (gai_error != 0)
269 3x *ec_out = posix_resolver_detail::make_gai_error(gai_error);
270 else
271 13x *ec_out = {}; // Clear on success
272 }
273
274 16x if (out && !was_cancelled && gai_error == 0)
275 13x *out = std::move(stored_results);
276
277 16x impl->svc_.work_finished();
278 16x cont_op.cont.h = h;
279 16x dispatch_coro(ex, cont_op.cont).resume();
280 16x }
281
282 inline void
283 posix_resolver::resolve_op::destroy()
284 {
285 stop_cb.reset();
286 }
287
288 inline void
289 33x posix_resolver::resolve_op::request_cancel() noexcept
290 {
291 33x cancelled.store(true, std::memory_order_release);
292 33x }
293
294 inline void
295 16x posix_resolver::resolve_op::start(std::stop_token const& token)
296 {
297 16x cancelled.store(false, std::memory_order_release);
298 16x stop_cb.reset();
299
300 16x if (token.stop_possible())
301 stop_cb.emplace(token, canceller{this});
302 16x }
303
304 // posix_resolver::reverse_resolve_op implementation
305
306 inline void
307 10x posix_resolver::reverse_resolve_op::reset() noexcept
308 {
309 10x ep = endpoint{};
310 10x flags = reverse_flags::none;
311 10x stored_host.clear();
312 10x stored_service.clear();
313 10x gai_error = 0;
314 10x cancelled.store(false, std::memory_order_relaxed);
315 10x stop_cb.reset();
316 10x ec_out = nullptr;
317 10x result_out = nullptr;
318 10x }
319
320 inline void
321 10x posix_resolver::reverse_resolve_op::operator()()
322 {
323 10x stop_cb.reset(); // Disconnect stop callback
324
325 10x bool const was_cancelled = cancelled.load(std::memory_order_acquire);
326
327 10x if (ec_out)
328 {
329 10x if (was_cancelled)
330 *ec_out = capy::error::canceled;
331 10x else if (gai_error != 0)
332 1x *ec_out = posix_resolver_detail::make_gai_error(gai_error);
333 else
334 9x *ec_out = {}; // Clear on success
335 }
336
337 10x if (result_out && !was_cancelled && gai_error == 0)
338 {
339 27x *result_out = reverse_resolver_result(
340 27x ep, std::move(stored_host), std::move(stored_service));
341 }
342
343 10x impl->svc_.work_finished();
344 10x cont_op.cont.h = h;
345 10x dispatch_coro(ex, cont_op.cont).resume();
346 10x }
347
348 inline void
349 posix_resolver::reverse_resolve_op::destroy()
350 {
351 stop_cb.reset();
352 }
353
354 inline void
355 33x posix_resolver::reverse_resolve_op::request_cancel() noexcept
356 {
357 33x cancelled.store(true, std::memory_order_release);
358 33x }
359
360 inline void
361 10x posix_resolver::reverse_resolve_op::start(std::stop_token const& token)
362 {
363 10x cancelled.store(false, std::memory_order_release);
364 10x stop_cb.reset();
365
366 10x if (token.stop_possible())
367 stop_cb.emplace(token, canceller{this});
368 10x }
369
370 // posix_resolver implementation
371
372 inline std::coroutine_handle<>
373 16x posix_resolver::resolve(
374 std::coroutine_handle<> h,
375 capy::executor_ref ex,
376 std::string_view host,
377 std::string_view service,
378 resolve_flags flags,
379 std::stop_token token,
380 std::error_code* ec,
381 resolver_results* out)
382 {
383 16x if (svc_.single_threaded())
384 {
385 *ec = std::make_error_code(std::errc::operation_not_supported);
386 op_.cont_op.cont.h = h;
387 return dispatch_coro(ex, op_.cont_op.cont);
388 }
389
390 16x auto& op = op_;
391 16x op.reset();
392 16x op.h = h;
393 16x op.ex = ex;
394 16x op.impl = this;
395 16x op.ec_out = ec;
396 16x op.out = out;
397 16x op.host = host;
398 16x op.service = service;
399 16x op.flags = flags;
400 16x op.start(token);
401
402 // Keep io_context alive while resolution is pending
403 16x op.ex.on_work_started();
404
405 // Prevent impl destruction while work is in flight
406 16x resolve_pool_op_.resolver_ = this;
407 16x resolve_pool_op_.ref_ = this->shared_from_this();
408 16x resolve_pool_op_.func_ = &posix_resolver::do_resolve_work;
409 16x if (!svc_.pool().post(&resolve_pool_op_))
410 {
411 // Pool shut down — complete with cancellation
412 resolve_pool_op_.ref_.reset();
413 op.cancelled.store(true, std::memory_order_release);
414 svc_.post(&op_);
415 }
416 16x return std::noop_coroutine();
417 }
418
419 inline std::coroutine_handle<>
420 10x posix_resolver::reverse_resolve(
421 std::coroutine_handle<> h,
422 capy::executor_ref ex,
423 endpoint const& ep,
424 reverse_flags flags,
425 std::stop_token token,
426 std::error_code* ec,
427 reverse_resolver_result* result_out)
428 {
429 10x if (svc_.single_threaded())
430 {
431 *ec = std::make_error_code(std::errc::operation_not_supported);
432 reverse_op_.cont_op.cont.h = h;
433 return dispatch_coro(ex, reverse_op_.cont_op.cont);
434 }
435
436 10x auto& op = reverse_op_;
437 10x op.reset();
438 10x op.h = h;
439 10x op.ex = ex;
440 10x op.impl = this;
441 10x op.ec_out = ec;
442 10x op.result_out = result_out;
443 10x op.ep = ep;
444 10x op.flags = flags;
445 10x op.start(token);
446
447 // Keep io_context alive while resolution is pending
448 10x op.ex.on_work_started();
449
450 // Prevent impl destruction while work is in flight
451 10x reverse_pool_op_.resolver_ = this;
452 10x reverse_pool_op_.ref_ = this->shared_from_this();
453 10x reverse_pool_op_.func_ = &posix_resolver::do_reverse_resolve_work;
454 10x if (!svc_.pool().post(&reverse_pool_op_))
455 {
456 // Pool shut down — complete with cancellation
457 reverse_pool_op_.ref_.reset();
458 op.cancelled.store(true, std::memory_order_release);
459 svc_.post(&reverse_op_);
460 }
461 10x return std::noop_coroutine();
462 }
463
464 inline void
465 33x posix_resolver::cancel() noexcept
466 {
467 33x op_.request_cancel();
468 33x reverse_op_.request_cancel();
469 33x }
470
471 inline void
472 16x posix_resolver::do_resolve_work(pool_work_item* w) noexcept
473 {
474 16x auto* pw = static_cast<pool_op*>(w);
475 16x auto* self = pw->resolver_;
476
477 16x struct addrinfo hints{};
478 16x hints.ai_family = AF_UNSPEC;
479 16x hints.ai_socktype = SOCK_STREAM;
480 16x hints.ai_flags = posix_resolver_detail::flags_to_hints(self->op_.flags);
481
482 16x struct addrinfo* ai = nullptr;
483 48x int result = ::getaddrinfo(
484 32x self->op_.host.empty() ? nullptr : self->op_.host.c_str(),
485 32x self->op_.service.empty() ? nullptr : self->op_.service.c_str(), &hints,
486 &ai);
487
488 16x if (!self->op_.cancelled.load(std::memory_order_acquire))
489 {
490 16x if (result == 0 && ai)
491 {
492 26x self->op_.stored_results = posix_resolver_detail::convert_results(
493 13x ai, self->op_.host, self->op_.service);
494 13x self->op_.gai_error = 0;
495 }
496 else
497 {
498 3x self->op_.gai_error = result;
499 }
500 }
501
502 16x if (ai)
503 13x ::freeaddrinfo(ai);
504
505 // Move ref to stack before post — post may trigger destroy_impl
506 // which erases the last shared_ptr, destroying *self (and *pw)
507 16x auto ref = std::move(pw->ref_);
508 16x self->svc_.post(&self->op_);
509 16x }
510
511 inline void
512 10x posix_resolver::do_reverse_resolve_work(pool_work_item* w) noexcept
513 {
514 10x auto* pw = static_cast<pool_op*>(w);
515 10x auto* self = pw->resolver_;
516
517 10x sockaddr_storage ss{};
518 socklen_t ss_len;
519
520 10x if (self->reverse_op_.ep.is_v4())
521 {
522 8x auto sa = to_sockaddr_in(self->reverse_op_.ep);
523 8x std::memcpy(&ss, &sa, sizeof(sa));
524 8x ss_len = sizeof(sockaddr_in);
525 }
526 else
527 {
528 2x auto sa = to_sockaddr_in6(self->reverse_op_.ep);
529 2x std::memcpy(&ss, &sa, sizeof(sa));
530 2x ss_len = sizeof(sockaddr_in6);
531 }
532
533 char host[NI_MAXHOST];
534 char service[NI_MAXSERV];
535
536 10x int result = ::getnameinfo(
537 reinterpret_cast<sockaddr*>(&ss), ss_len, host, sizeof(host), service,
538 sizeof(service),
539 posix_resolver_detail::flags_to_ni_flags(self->reverse_op_.flags));
540
541 10x if (!self->reverse_op_.cancelled.load(std::memory_order_acquire))
542 {
543 10x if (result == 0)
544 {
545 9x self->reverse_op_.stored_host = host;
546 9x self->reverse_op_.stored_service = service;
547 9x self->reverse_op_.gai_error = 0;
548 }
549 else
550 {
551 1x self->reverse_op_.gai_error = result;
552 }
553 }
554
555 // Move ref to stack before post — post may trigger destroy_impl
556 // which erases the last shared_ptr, destroying *self (and *pw)
557 10x auto ref = std::move(pw->ref_);
558 10x self->svc_.post(&self->reverse_op_);
559 10x }
560
561 // posix_resolver_service implementation
562
563 inline void
564 515x posix_resolver_service::shutdown()
565 {
566 515x std::lock_guard<std::mutex> lock(mutex_);
567
568 // Cancel all resolvers (sets cancelled flag checked by pool threads)
569 515x for (auto* impl = resolver_list_.pop_front(); impl != nullptr;
570 impl = resolver_list_.pop_front())
571 {
572 impl->cancel();
573 }
574
575 // Clear the map which releases shared_ptrs.
576 // The thread pool service shuts down separately via
577 // execution_context service ordering.
578 515x resolver_ptrs_.clear();
579 515x }
580
581 inline io_object::implementation*
582 29x posix_resolver_service::construct()
583 {
584 29x auto ptr = std::make_shared<posix_resolver>(*this);
585 29x auto* impl = ptr.get();
586
587 {
588 29x std::lock_guard<std::mutex> lock(mutex_);
589 29x resolver_list_.push_back(impl);
590 29x resolver_ptrs_[impl] = std::move(ptr);
591 29x }
592
593 29x return impl;
594 29x }
595
596 inline void
597 29x posix_resolver_service::destroy_impl(posix_resolver& impl)
598 {
599 29x std::lock_guard<std::mutex> lock(mutex_);
600 29x resolver_list_.remove(&impl);
601 29x resolver_ptrs_.erase(&impl);
602 29x }
603
604 inline void
605 26x posix_resolver_service::post(scheduler_op* op)
606 {
607 26x sched_->post(op);
608 26x }
609
610 inline void
611 posix_resolver_service::work_started() noexcept
612 {
613 sched_->work_started();
614 }
615
616 inline void
617 26x posix_resolver_service::work_finished() noexcept
618 {
619 26x sched_->work_finished();
620 26x }
621
622 // Free function to get/create the resolver service
623
624 inline posix_resolver_service&
625 515x get_resolver_service(capy::execution_context& ctx, scheduler& sched)
626 {
627 515x return ctx.make_service<posix_resolver_service>(sched);
628 }
629
630 } // namespace boost::corosio::detail
631
632 #endif // BOOST_COROSIO_POSIX
633
634 #endif // BOOST_COROSIO_NATIVE_DETAIL_POSIX_POSIX_RESOLVER_SERVICE_HPP
635