include/boost/corosio/io/io_object.hpp

91.0% Lines (61/67) 95.6% List of functions (43/46) 53.3% Branches (16/30)
io_object.hpp
f(x) Functions (46)
Function Calls Lines Branches Blocks
boost::corosio::io_object::implementation::implementation() :53 20145x 100.0% 100.0% boost::corosio::io_object::implementation::~implementation() :55 20145x 100.0% 100.0% boost::corosio::io_object::io_service::io_service() :64 12012x 100.0% 100.0% boost::corosio::io_object::io_service::~io_service() :66 12012x 100.0% 100.0% boost::corosio::io_object::io_service::close(boost::corosio::io_object::handle&) :75 6274x 100.0% 100.0% <unknown function 85> :85 boost::corosio::io_object::handle::~handle() :91 96526x 100.0% 66.7% 100.0% boost::corosio::io_object::handle::handle() :101 0 0.0% 0.0% boost::corosio::io_object::handle::handle(boost::capy::execution_context&, boost::corosio::io_object::io_service&) :104 39938x 100.0% 100.0% boost::corosio::io_object::handle::handle(boost::corosio::io_object::handle&&) :112 56592x 100.0% 100.0% boost::corosio::io_object::handle::operator=(boost::corosio::io_object::handle&&) :120 71x 100.0% 50.0% 87.0% boost::corosio::io_object::handle::operator bool() const :140 57005x 100.0% 100.0% boost::corosio::io_object::handle::service() const :146 23331x 100.0% 100.0% boost::corosio::io_object::handle::get() const :152 1716357x 100.0% 100.0% boost::corosio::io_object::handle::reset(boost::corosio::io_object::implementation*) :161 5934x 100.0% 50.0% 83.0% boost::corosio::io_object::handle::context() const :172 12x 100.0% 100.0% boost::corosio::io_object::context() const :179 12x 100.0% 100.0% boost::corosio::io_object::~io_object() :185 21954x 100.0% 100.0% boost::corosio::io_object::io_object() :188 0 0.0% 0.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::file_service>(boost::capy::execution_context&) :200 33x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::kqueue_local_datagram_service>(boost::capy::execution_context&) :200 13x 100.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::kqueue_local_stream_acceptor_service>(boost::capy::execution_context&) :200 6x 100.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::kqueue_local_stream_service>(boost::capy::execution_context&) :200 14x 100.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::kqueue_tcp_acceptor_service>(boost::capy::execution_context&) :200 11x 0.0% 0.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::kqueue_tcp_service>(boost::capy::execution_context&) :200 18x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::kqueue_udp_service>(boost::capy::execution_context&) :200 18x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::local_datagram_service>(boost::capy::execution_context&) :200 100x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::local_stream_acceptor_service>(boost::capy::execution_context&) :200 32x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::local_stream_service>(boost::capy::execution_context&) :200 92x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::posix_random_access_file_service>(boost::capy::execution_context&) :200 10x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::posix_resolver_service>(boost::capy::execution_context&) :200 42x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::posix_signal_service>(boost::capy::execution_context&) :200 104x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::posix_stream_file_service>(boost::capy::execution_context&) :200 10x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::random_access_file_service>(boost::capy::execution_context&) :200 33x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::select_local_datagram_service>(boost::capy::execution_context&) :200 13x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::select_local_stream_acceptor_service>(boost::capy::execution_context&) :200 6x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::select_local_stream_service>(boost::capy::execution_context&) :200 14x 100.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::select_tcp_acceptor_service>(boost::capy::execution_context&) :200 11x 0.0% 0.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::select_tcp_service>(boost::capy::execution_context&) :200 18x 0.0% 0.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::select_udp_service>(boost::capy::execution_context&) :200 18x 100.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::tcp_acceptor_service>(boost::capy::execution_context&) :200 1084x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::tcp_service>(boost::capy::execution_context&) :200 11993x 80.0% 50.0% 66.0% boost::corosio::io_object::handle boost::corosio::io_object::create_handle<boost::corosio::detail::udp_service>(boost::capy::execution_context&) :200 146x 80.0% 50.0% 66.0% boost::corosio::io_object::io_object(boost::corosio::io_object::handle) :210 19969x 100.0% 100.0% boost::corosio::io_object::io_object(boost::corosio::io_object&&) :213 1987x 100.0% 100.0% boost::corosio::io_object::operator=(boost::corosio::io_object&&) :216 4x 100.0% 50.0% 100.0%
Line Branch TLA Hits Source Code
1 //
2 // Copyright (c) 2025 Vinnie Falco ([email protected])
3 // Copyright (c) 2026 Steve Gerbino
4 //
5 // Distributed under the Boost Software License, Version 1.0. (See accompanying
6 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
7 //
8 // Official repository: https://github.com/cppalliance/corosio
9 //
10
11 #ifndef BOOST_COROSIO_IO_IO_OBJECT_HPP
12 #define BOOST_COROSIO_IO_IO_OBJECT_HPP
13
14 #include <boost/corosio/detail/config.hpp>
15 #include <boost/corosio/detail/except.hpp>
16 #include <boost/capy/ex/execution_context.hpp>
17
18 #include <utility>
19
20 namespace boost::corosio {
21
22 /** Base class for platform I/O objects.
23
24 Provides common infrastructure for I/O objects that wrap kernel
25 resources (sockets, timers, signal handlers, acceptors). Derived
26 classes dispatch operations through a platform-specific vtable
27 (IOCP, epoll, kqueue, io_uring).
28
29 @par Semantics
30 Only concrete platform I/O types should inherit from `io_object`.
31 Test mocks, decorators, and stream adapters must not inherit from
32 this class. Use concepts or templates for generic I/O algorithms.
33
34 @par Thread Safety
35 Distinct objects: Safe.
36 Shared objects: Unsafe. All operations on a single I/O object
37 must be serialized.
38
39 @note Intended as a protected base class. The handle member
40 `h_` is accessible to derived classes.
41
42 @see io_stream, tcp_socket, tcp_acceptor
43 */
44 class BOOST_COROSIO_DECL io_object
45 {
46 public:
47 class handle;
48
49 /** Base interface for platform I/O implementations.
50
51 Derived classes provide platform-specific operation dispatch.
52 */
53 struct implementation
54 {
55 20145x virtual ~implementation() = default;
56 };
57
58 /** Service interface for I/O object lifecycle management.
59
60 Platform backends implement this interface to manage the
61 creation, closing, and destruction of I/O object
62 implementations.
63 */
64 struct io_service
65 {
66 12012x virtual ~io_service() = default;
67
68 /// Construct a new implementation instance.
69 virtual implementation* construct() = 0;
70
71 /// Destroy the implementation, closing kernel resources and freeing memory.
72 virtual void destroy(implementation*) = 0;
73
74 /// Close the I/O object, releasing kernel resources without deallocating.
75 6274x virtual void close(handle&) {}
76 };
77
78 /** RAII wrapper for I/O object implementation lifetime.
79
80 Manages ownership of the platform-specific implementation,
81 automatically destroying it when the handle goes out of scope.
82 */
83 class handle
84 {
85 capy::execution_context* ctx_ = nullptr;
86 io_service* svc_ = nullptr;
87 implementation* impl_ = nullptr;
88
89 public:
90 /// Destroy the handle and its implementation.
91 96526x ~handle()
92 48263x {
93
2/2
✓ Branch 0 taken 28367 times.
✓ Branch 1 taken 19896 times.
48263x if (impl_)
94 {
95
1/2
✓ Branch 0 taken 19896 times.
✗ Branch 1 not taken.
19896x svc_->close(*this);
96
1/2
✓ Branch 0 taken 19896 times.
✗ Branch 1 not taken.
19896x svc_->destroy(impl_);
97 19896x }
98 96526x }
99
100 /// Construct an empty handle.
101 handle() = default;
102
103 /// Construct a handle bound to a context and service.
104 39938x handle(capy::execution_context& ctx, io_service& svc)
105 19969x : ctx_(&ctx)
106 19969x , svc_(&svc)
107 19969x , impl_(svc_->construct())
108 19969x {
109 39938x }
110
111 /// Move construct from another handle.
112 56592x handle(handle&& other) noexcept
113 28296x : ctx_(std::exchange(other.ctx_, nullptr))
114 28296x , svc_(std::exchange(other.svc_, nullptr))
115 28296x , impl_(std::exchange(other.impl_, nullptr))
116 28296x {
117 56592x }
118
119 /// Move assign from another handle.
120 71x handle& operator=(handle&& other) noexcept
121 {
122
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 71 times.
71x if (this != &other)
123 {
124
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 71 times.
71x if (impl_)
125 {
126
1/2
✓ Branch 0 taken 71 times.
✗ Branch 1 not taken.
71x svc_->close(*this);
127
1/2
✓ Branch 0 taken 71 times.
✗ Branch 1 not taken.
71x svc_->destroy(impl_);
128 71x }
129 71x ctx_ = std::exchange(other.ctx_, nullptr);
130 71x svc_ = std::exchange(other.svc_, nullptr);
131 71x impl_ = std::exchange(other.impl_, nullptr);
132 71x }
133 71x return *this;
134 }
135
136 handle(handle const&) = delete;
137 handle& operator=(handle const&) = delete;
138
139 /// Return true if the handle owns an implementation.
140 57005x explicit operator bool() const noexcept
141 {
142 57005x return impl_ != nullptr;
143 }
144
145 /// Return the associated I/O service.
146 23331x io_service& service() const noexcept
147 {
148 23331x return *svc_;
149 }
150
151 /// Return the platform implementation.
152 1716357x implementation* get() const noexcept
153 {
154 1716357x return impl_;
155 }
156
157 /** Replace the implementation, destroying the old one.
158
159 @param p The new implementation to own. May be nullptr.
160 */
161 5934x void reset(implementation* p) noexcept
162 {
163
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 5934 times.
5934x if (impl_)
164 {
165
1/2
✓ Branch 0 taken 5934 times.
✗ Branch 1 not taken.
5934x svc_->close(*this);
166
1/2
✓ Branch 0 taken 5934 times.
✗ Branch 1 not taken.
5934x svc_->destroy(impl_);
167 5934x }
168 5934x impl_ = p;
169 5934x }
170
171 /// Return the execution context.
172 12x capy::execution_context& context() const noexcept
173 {
174 12x return *ctx_;
175 }
176 };
177
178 /// Return the execution context.
179 12x capy::execution_context& context() const noexcept
180 {
181 12x return h_.context();
182 }
183
184 protected:
185 21954x virtual ~io_object() = default;
186
187 /// Default construct for virtual base initialization.
188 io_object() noexcept = default;
189
190 /** Create a handle bound to a service found in the context.
191
192 @tparam Service The service type whose key_type is used for lookup.
193 @param ctx The execution context to search for the service.
194
195 @return A handle owning a freshly constructed implementation.
196
197 @throws std::logic_error if the service is not installed.
198 */
199 template<class Service>
200 13839x static handle create_handle(capy::execution_context& ctx)
201 {
202 13839x auto* svc = ctx.find_service<Service>();
203
4/8
✓ Branch 0 taken 13735 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 55 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 25 times.
✗ Branch 5 not taken.
✓ Branch 6 taken 24 times.
✗ Branch 7 not taken.
13839x if (!svc)
204 detail::throw_logic_error(
205 "io_object::create_handle: service not installed");
206 13839x return handle(ctx, *svc);
207 }
208
209 /// Construct an I/O object from a handle.
210 19969x explicit io_object(handle h) noexcept : h_(std::move(h)) {}
211
212 /// Move construct from another I/O object.
213 1987x io_object(io_object&& other) noexcept : h_(std::move(other.h_)) {}
214
215 /// Move assign from another I/O object.
216 4x io_object& operator=(io_object&& other) noexcept
217 {
218
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 4 times.
4x if (this != &other)
219 4x h_ = std::move(other.h_);
220 4x return *this;
221 }
222
223 io_object(io_object const&) = delete;
224 io_object& operator=(io_object const&) = delete;
225
226 /// The platform I/O handle owned by this object.
227 handle h_;
228 };
229
230 } // namespace boost::corosio
231
232 #endif
233