LCOV - code coverage report
Current view: top level - corosio/native - native_tcp_acceptor.hpp (source / functions) Coverage Total Hit Missed
Test: coverage_remapped.info Lines: 92.0 % 25 23 2
Test Date: 2026-02-18 17:16:27 Functions: 100.0 % 14 14

           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_NATIVE_TCP_ACCEPTOR_HPP
      11                 : #define BOOST_COROSIO_NATIVE_NATIVE_TCP_ACCEPTOR_HPP
      12                 : 
      13                 : #include <boost/corosio/tcp_acceptor.hpp>
      14                 : #include <boost/corosio/backend.hpp>
      15                 : 
      16                 : #if BOOST_COROSIO_HAS_EPOLL
      17                 : #include <boost/corosio/native/detail/epoll/epoll_acceptor_service.hpp>
      18                 : #endif
      19                 : 
      20                 : #if BOOST_COROSIO_HAS_SELECT
      21                 : #include <boost/corosio/native/detail/select/select_acceptor_service.hpp>
      22                 : #endif
      23                 : 
      24                 : #if BOOST_COROSIO_HAS_KQUEUE
      25                 : #include <boost/corosio/native/detail/kqueue/kqueue_acceptor_service.hpp>
      26                 : #endif
      27                 : 
      28                 : #if BOOST_COROSIO_HAS_IOCP
      29                 : #include <boost/corosio/native/detail/iocp/win_acceptor_service.hpp>
      30                 : #endif
      31                 : 
      32                 : namespace boost::corosio {
      33                 : 
      34                 : /** An asynchronous TCP acceptor with devirtualized accept operations.
      35                 : 
      36                 :     This class template inherits from @ref tcp_acceptor and shadows
      37                 :     the `accept` operation with a version that calls the backend
      38                 :     implementation directly, allowing the compiler to inline through
      39                 :     the entire call chain.
      40                 : 
      41                 :     Non-async operations (`listen`, `close`, `cancel`) remain
      42                 :     unchanged and dispatch through the compiled library.
      43                 : 
      44                 :     A `native_tcp_acceptor` IS-A `tcp_acceptor` and can be passed
      45                 :     to any function expecting `tcp_acceptor&`.
      46                 : 
      47                 :     @tparam Backend A backend tag value (e.g., `epoll`).
      48                 : 
      49                 :     @par Thread Safety
      50                 :     Same as @ref tcp_acceptor.
      51                 : 
      52                 :     @see tcp_acceptor, epoll_t, iocp_t
      53                 : */
      54                 : template<auto Backend>
      55                 : class native_tcp_acceptor : public tcp_acceptor
      56                 : {
      57                 :     using backend_type = decltype(Backend);
      58                 :     using impl_type    = typename backend_type::acceptor_type;
      59                 :     using service_type = typename backend_type::acceptor_service_type;
      60                 : 
      61 HIT           4 :     impl_type& get_impl() noexcept
      62                 :     {
      63               4 :         return *static_cast<impl_type*>(h_.get());
      64                 :     }
      65                 : 
      66                 :     struct native_accept_awaitable
      67                 :     {
      68                 :         native_tcp_acceptor& acc_;
      69                 :         tcp_socket& peer_;
      70                 :         std::stop_token token_;
      71                 :         mutable std::error_code ec_;
      72                 :         mutable io_object::implementation* peer_impl_ = nullptr;
      73                 : 
      74               4 :         native_accept_awaitable(
      75                 :             native_tcp_acceptor& acc, tcp_socket& peer) noexcept
      76               4 :             : acc_(acc)
      77               4 :             , peer_(peer)
      78                 :         {
      79               4 :         }
      80                 : 
      81               4 :         bool await_ready() const noexcept
      82                 :         {
      83               4 :             return token_.stop_requested();
      84                 :         }
      85                 : 
      86               4 :         capy::io_result<> await_resume() const noexcept
      87                 :         {
      88               4 :             if (token_.stop_requested())
      89 MIS           0 :                 return {make_error_code(std::errc::operation_canceled)};
      90 HIT           4 :             if (!ec_)
      91               4 :                 acc_.reset_peer_impl(peer_, peer_impl_);
      92               4 :             return {ec_};
      93                 :         }
      94                 : 
      95               4 :         auto await_suspend(std::coroutine_handle<> h, capy::io_env const* env)
      96                 :             -> std::coroutine_handle<>
      97                 :         {
      98               4 :             token_ = env->stop_token;
      99              12 :             return acc_.get_impl().accept(
     100              12 :                 h, env->executor, token_, &ec_, &peer_impl_);
     101                 :         }
     102                 :     };
     103                 : 
     104                 : public:
     105                 :     /** Construct a native acceptor from an execution context.
     106                 : 
     107                 :         @param ctx The execution context that will own this acceptor.
     108                 :     */
     109               4 :     explicit native_tcp_acceptor(capy::execution_context& ctx)
     110               4 :         : tcp_acceptor(create_handle<service_type>(ctx))
     111                 :     {
     112               4 :     }
     113                 : 
     114                 :     /** Construct a native acceptor from an executor.
     115                 : 
     116                 :         @param ex The executor whose context will own the acceptor.
     117                 :     */
     118                 :     template<class Ex>
     119                 :         requires(!std::same_as<std::remove_cvref_t<Ex>, native_tcp_acceptor>) &&
     120                 :         capy::Executor<Ex>
     121                 :     explicit native_tcp_acceptor(Ex const& ex)
     122                 :         : native_tcp_acceptor(ex.context())
     123                 :     {
     124                 :     }
     125                 : 
     126                 :     /// Move construct.
     127                 :     native_tcp_acceptor(native_tcp_acceptor&&) noexcept = default;
     128                 : 
     129                 :     /// Move assign.
     130                 :     native_tcp_acceptor& operator=(native_tcp_acceptor&&) noexcept = default;
     131                 : 
     132                 :     native_tcp_acceptor(native_tcp_acceptor const&)            = delete;
     133                 :     native_tcp_acceptor& operator=(native_tcp_acceptor const&) = delete;
     134                 : 
     135                 :     /** Asynchronously accept an incoming connection.
     136                 : 
     137                 :         Calls the backend implementation directly, bypassing virtual
     138                 :         dispatch. Otherwise identical to @ref tcp_acceptor::accept.
     139                 : 
     140                 :         @param peer The socket to receive the accepted connection.
     141                 : 
     142                 :         @return An awaitable yielding `io_result<>`.
     143                 : 
     144                 :         @throws std::logic_error if the acceptor is not listening.
     145                 :     */
     146               4 :     auto accept(tcp_socket& peer)
     147                 :     {
     148               4 :         if (!is_open())
     149 MIS           0 :             detail::throw_logic_error("accept: acceptor not listening");
     150 HIT           4 :         return native_accept_awaitable(*this, peer);
     151                 :     }
     152                 : };
     153                 : 
     154                 : } // namespace boost::corosio
     155                 : 
     156                 : #endif
        

Generated by: LCOV version 2.3