LCOV - code coverage report
Current view: top level - corosio/native/detail/reactor - reactor_op_complete.hpp (source / functions) Coverage Total Hit Missed
Test: coverage_remapped.info Lines: 90.0 % 100 90 10
Test Date: 2026-03-17 15:52:49 Functions: 100.0 % 16 16

           TLA  Line data    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_REACTOR_REACTOR_OP_COMPLETE_HPP
      11                 : #define BOOST_COROSIO_NATIVE_DETAIL_REACTOR_REACTOR_OP_COMPLETE_HPP
      12                 : 
      13                 : #include <boost/corosio/detail/dispatch_coro.hpp>
      14                 : #include <boost/corosio/native/detail/endpoint_convert.hpp>
      15                 : #include <boost/corosio/native/detail/make_err.hpp>
      16                 : #include <boost/corosio/io/io_object.hpp>
      17                 : 
      18                 : #include <coroutine>
      19                 : #include <mutex>
      20                 : #include <utility>
      21                 : 
      22                 : #include <netinet/in.h>
      23                 : #include <sys/socket.h>
      24                 : #include <unistd.h>
      25                 : 
      26                 : namespace boost::corosio::detail {
      27                 : 
      28                 : /** Complete a base read/write operation.
      29                 : 
      30                 :     Translates the recorded errno and cancellation state into
      31                 :     an error_code, stores the byte count, then resumes the
      32                 :     caller via symmetric transfer.
      33                 : 
      34                 :     @tparam Op The concrete operation type.
      35                 :     @param op The operation to complete.
      36                 : */
      37                 : template<typename Op>
      38                 : void
      39 HIT       88866 : complete_io_op(Op& op)
      40                 : {
      41           88866 :     op.stop_cb.reset();
      42           88866 :     op.socket_impl_->desc_state_.scheduler_->reset_inline_budget();
      43                 : 
      44           88866 :     if (op.cancelled.load(std::memory_order_acquire))
      45             308 :         *op.ec_out = capy::error::canceled;
      46           88558 :     else if (op.errn != 0)
      47 MIS           0 :         *op.ec_out = make_err(op.errn);
      48 HIT       88558 :     else if (op.is_read_operation() && op.bytes_transferred == 0)
      49 MIS           0 :         *op.ec_out = capy::error::eof;
      50                 :     else
      51 HIT       88558 :         *op.ec_out = {};
      52                 : 
      53           88866 :     *op.bytes_out = op.bytes_transferred;
      54                 : 
      55           88866 :     capy::executor_ref saved_ex(op.ex);
      56           88866 :     std::coroutine_handle<> saved_h(op.h);
      57           88866 :     auto prevent = std::move(op.impl_ptr);
      58           88866 :     dispatch_coro(saved_ex, saved_h).resume();
      59           88866 : }
      60                 : 
      61                 : /** Complete a connect operation with endpoint caching.
      62                 : 
      63                 :     On success, queries the local endpoint via getsockname and
      64                 :     caches both endpoints in the socket impl. Then resumes the
      65                 :     caller via symmetric transfer.
      66                 : 
      67                 :     @tparam Op The concrete connect operation type.
      68                 :     @param op The operation to complete.
      69                 : */
      70                 : template<typename Op>
      71                 : void
      72            4500 : complete_connect_op(Op& op)
      73                 : {
      74            4500 :     op.stop_cb.reset();
      75            4500 :     op.socket_impl_->desc_state_.scheduler_->reset_inline_budget();
      76                 : 
      77            4500 :     bool success =
      78            4500 :         (op.errn == 0 && !op.cancelled.load(std::memory_order_acquire));
      79                 : 
      80            4500 :     if (success && op.socket_impl_)
      81                 :     {
      82            4494 :         endpoint local_ep;
      83            4494 :         sockaddr_storage local_storage{};
      84            4494 :         socklen_t local_len = sizeof(local_storage);
      85            4494 :         if (::getsockname(
      86                 :                 op.fd, reinterpret_cast<sockaddr*>(&local_storage),
      87            4494 :                 &local_len) == 0)
      88            4494 :             local_ep = from_sockaddr(local_storage);
      89            4494 :         op.socket_impl_->set_endpoints(local_ep, op.target_endpoint);
      90                 :     }
      91                 : 
      92            4500 :     if (op.cancelled.load(std::memory_order_acquire))
      93 MIS           0 :         *op.ec_out = capy::error::canceled;
      94 HIT        4500 :     else if (op.errn != 0)
      95               6 :         *op.ec_out = make_err(op.errn);
      96                 :     else
      97            4494 :         *op.ec_out = {};
      98                 : 
      99            4500 :     capy::executor_ref saved_ex(op.ex);
     100            4500 :     std::coroutine_handle<> saved_h(op.h);
     101            4500 :     auto prevent = std::move(op.impl_ptr);
     102            4500 :     dispatch_coro(saved_ex, saved_h).resume();
     103            4500 : }
     104                 : 
     105                 : /** Construct and register a peer socket from an accepted fd.
     106                 : 
     107                 :     Creates a new socket impl via the acceptor's associated
     108                 :     socket service, registers it with the scheduler, and caches
     109                 :     the local and remote endpoints.
     110                 : 
     111                 :     @tparam SocketImpl The concrete socket implementation type.
     112                 :     @tparam AcceptorImpl The concrete acceptor implementation type.
     113                 :     @param acceptor_impl The acceptor that accepted the connection.
     114                 :     @param accepted_fd The accepted file descriptor (set to -1 on success).
     115                 :     @param peer_storage The peer address from accept().
     116                 :     @param impl_out Output pointer for the new socket impl.
     117                 :     @param ec_out Output pointer for any error.
     118                 :     @return True on success, false on failure.
     119                 : */
     120                 : template<typename SocketImpl, typename AcceptorImpl>
     121                 : bool
     122            4484 : setup_accepted_socket(
     123                 :     AcceptorImpl* acceptor_impl,
     124                 :     int& accepted_fd,
     125                 :     sockaddr_storage const& peer_storage,
     126                 :     io_object::implementation** impl_out,
     127                 :     std::error_code* ec_out)
     128                 : {
     129            4484 :     auto* socket_svc = acceptor_impl->service().tcp_service();
     130            4484 :     if (!socket_svc)
     131                 :     {
     132 MIS           0 :         *ec_out = make_err(ENOENT);
     133               0 :         return false;
     134                 :     }
     135                 : 
     136 HIT        4484 :     auto& impl = static_cast<SocketImpl&>(*socket_svc->construct());
     137            4484 :     impl.set_socket(accepted_fd);
     138                 : 
     139            4484 :     impl.desc_state_.fd = accepted_fd;
     140                 :     {
     141            4484 :         std::lock_guard lock(impl.desc_state_.mutex);
     142            4484 :         impl.desc_state_.read_op    = nullptr;
     143            4484 :         impl.desc_state_.write_op   = nullptr;
     144            4484 :         impl.desc_state_.connect_op = nullptr;
     145            4484 :     }
     146            4484 :     socket_svc->scheduler().register_descriptor(accepted_fd, &impl.desc_state_);
     147                 : 
     148            4484 :     impl.set_endpoints(
     149                 :         acceptor_impl->local_endpoint(), from_sockaddr(peer_storage));
     150                 : 
     151            4484 :     if (impl_out)
     152            4484 :         *impl_out = &impl;
     153            4484 :     accepted_fd = -1;
     154            4484 :     return true;
     155                 : }
     156                 : 
     157                 : /** Complete an accept operation.
     158                 : 
     159                 :     Sets up the peer socket on success, or closes the accepted
     160                 :     fd on failure. Then resumes the caller via symmetric transfer.
     161                 : 
     162                 :     @tparam SocketImpl The concrete socket implementation type.
     163                 :     @tparam Op The concrete accept operation type.
     164                 :     @param op The operation to complete.
     165                 : */
     166                 : template<typename SocketImpl, typename Op>
     167                 : void
     168            4496 : complete_accept_op(Op& op)
     169                 : {
     170            4496 :     op.stop_cb.reset();
     171            4496 :     op.acceptor_impl_->desc_state_.scheduler_->reset_inline_budget();
     172                 : 
     173            4496 :     bool success =
     174            4496 :         (op.errn == 0 && !op.cancelled.load(std::memory_order_acquire));
     175                 : 
     176            4496 :     if (op.cancelled.load(std::memory_order_acquire))
     177              12 :         *op.ec_out = capy::error::canceled;
     178            4484 :     else if (op.errn != 0)
     179 MIS           0 :         *op.ec_out = make_err(op.errn);
     180                 :     else
     181 HIT        4484 :         *op.ec_out = {};
     182                 : 
     183            4496 :     if (success && op.accepted_fd >= 0 && op.acceptor_impl_)
     184                 :     {
     185            4484 :         if (!setup_accepted_socket<SocketImpl>(
     186            4484 :                 op.acceptor_impl_, op.accepted_fd, op.peer_storage, op.impl_out,
     187                 :                 op.ec_out))
     188 MIS           0 :             success = false;
     189                 :     }
     190                 : 
     191 HIT        4496 :     if (!success || !op.acceptor_impl_)
     192                 :     {
     193              12 :         if (op.accepted_fd >= 0)
     194                 :         {
     195 MIS           0 :             ::close(op.accepted_fd);
     196               0 :             op.accepted_fd = -1;
     197                 :         }
     198 HIT          12 :         if (op.impl_out)
     199              12 :             *op.impl_out = nullptr;
     200                 :     }
     201                 : 
     202            4496 :     capy::executor_ref saved_ex(op.ex);
     203            4496 :     std::coroutine_handle<> saved_h(op.h);
     204            4496 :     auto prevent = std::move(op.impl_ptr);
     205            4496 :     dispatch_coro(saved_ex, saved_h).resume();
     206            4496 : }
     207                 : 
     208                 : /** Complete a datagram operation (send_to or recv_from).
     209                 : 
     210                 :     For recv_from operations, writes the source endpoint from the
     211                 :     recorded sockaddr_storage into the caller's endpoint pointer.
     212                 :     Then resumes the caller via symmetric transfer.
     213                 : 
     214                 :     @tparam Op The concrete datagram operation type.
     215                 :     @param op The operation to complete.
     216                 :     @param source_out Optional pointer to store source endpoint
     217                 :         (non-null for recv_from, null for send_to).
     218                 : */
     219                 : template<typename Op>
     220                 : void
     221              14 : complete_datagram_op(Op& op, endpoint* source_out)
     222                 : {
     223              14 :     op.stop_cb.reset();
     224              14 :     op.socket_impl_->desc_state_.scheduler_->reset_inline_budget();
     225                 : 
     226              14 :     if (op.cancelled.load(std::memory_order_acquire))
     227               6 :         *op.ec_out = capy::error::canceled;
     228               8 :     else if (op.errn != 0)
     229 MIS           0 :         *op.ec_out = make_err(op.errn);
     230                 :     else
     231 HIT           8 :         *op.ec_out = {};
     232                 : 
     233              14 :     *op.bytes_out = op.bytes_transferred;
     234                 : 
     235              20 :     if (source_out && !op.cancelled.load(std::memory_order_acquire) &&
     236               6 :         op.errn == 0)
     237               6 :         *source_out = from_sockaddr(op.source_storage);
     238                 : 
     239              14 :     capy::executor_ref saved_ex(op.ex);
     240              14 :     std::coroutine_handle<> saved_h(op.h);
     241              14 :     auto prevent = std::move(op.impl_ptr);
     242              14 :     dispatch_coro(saved_ex, saved_h).resume();
     243              14 : }
     244                 : 
     245                 : } // namespace boost::corosio::detail
     246                 : 
     247                 : #endif // BOOST_COROSIO_NATIVE_DETAIL_REACTOR_REACTOR_OP_COMPLETE_HPP
        

Generated by: LCOV version 2.3