LCOV - code coverage report
Current view: top level - corosio/native/detail/epoll - epoll_traits.hpp (source / functions) Coverage Total Hit Missed
Test: coverage_remapped.info Lines: 79.5 % 39 31 8
Test Date: 2026-05-06 00:07:59 Functions: 90.9 % 11 10 1

           TLA  Line data    Source code
       1                 : //
       2                 : // Copyright (c) 2026 Michael Vandeberg
       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_EPOLL_EPOLL_TRAITS_HPP
      11                 : #define BOOST_COROSIO_NATIVE_DETAIL_EPOLL_EPOLL_TRAITS_HPP
      12                 : 
      13                 : #include <boost/corosio/detail/platform.hpp>
      14                 : 
      15                 : #if BOOST_COROSIO_HAS_EPOLL
      16                 : 
      17                 : #include <boost/corosio/native/detail/make_err.hpp>
      18                 : #include <boost/corosio/native/detail/reactor/reactor_descriptor_state.hpp>
      19                 : 
      20                 : #include <system_error>
      21                 : 
      22                 : #include <errno.h>
      23                 : #include <netinet/in.h>
      24                 : #include <sys/socket.h>
      25                 : 
      26                 : /* epoll backend traits.
      27                 : 
      28                 :    Captures the platform-specific behavior of the Linux epoll backend:
      29                 :    atomic SOCK_NONBLOCK|SOCK_CLOEXEC on socket(), accept4() for
      30                 :    accepted connections, and sendmsg(MSG_NOSIGNAL) for writes.
      31                 : */
      32                 : 
      33                 : namespace boost::corosio::detail {
      34                 : 
      35                 : class epoll_scheduler;
      36                 : 
      37                 : struct epoll_traits
      38                 : {
      39                 :     using scheduler_type    = epoll_scheduler;
      40                 :     using desc_state_type   = reactor_descriptor_state;
      41                 : 
      42                 :     static constexpr bool needs_write_notification = false;
      43                 : 
      44                 :     // No extra per-socket state or lifecycle hooks needed for epoll.
      45                 :     struct stream_socket_hook
      46                 :     {
      47 HIT          32 :         std::error_code on_set_option(
      48                 :             int fd, int level, int optname,
      49                 :             void const* data, std::size_t size) noexcept
      50                 :         {
      51              32 :             if (::setsockopt(
      52                 :                     fd, level, optname, data,
      53              32 :                     static_cast<socklen_t>(size)) != 0)
      54 MIS           0 :                 return make_err(errno);
      55 HIT          32 :             return {};
      56                 :         }
      57           18753 :         static void pre_shutdown(int) noexcept {}
      58            6237 :         static void pre_destroy(int) noexcept {}
      59                 :     };
      60                 : 
      61                 :     struct write_policy
      62                 :     {
      63 MIS           0 :         static ssize_t write(int fd, iovec* iovecs, int count) noexcept
      64                 :         {
      65               0 :             msghdr msg{};
      66               0 :             msg.msg_iov    = iovecs;
      67               0 :             msg.msg_iovlen = static_cast<std::size_t>(count);
      68                 : 
      69                 :             ssize_t n;
      70                 :             do
      71                 :             {
      72               0 :                 n = ::sendmsg(fd, &msg, MSG_NOSIGNAL);
      73                 :             }
      74               0 :             while (n < 0 && errno == EINTR);
      75               0 :             return n;
      76                 :         }
      77                 : 
      78 HIT      126459 :         static ssize_t write_one(
      79                 :             int fd, void const* data, std::size_t size) noexcept
      80                 :         {
      81                 :             ssize_t n;
      82                 :             do
      83                 :             {
      84          126459 :                 n = ::send(fd, data, size, MSG_NOSIGNAL);
      85                 :             }
      86          126459 :             while (n < 0 && errno == EINTR);
      87          126459 :             return n;
      88                 :         }
      89                 :     };
      90                 : 
      91                 :     struct accept_policy
      92                 :     {
      93            4118 :         static int do_accept(
      94                 :             int fd, sockaddr_storage& peer, socklen_t& addrlen) noexcept
      95                 :         {
      96            4118 :             addrlen = sizeof(peer);
      97                 :             int new_fd;
      98                 :             do
      99                 :             {
     100            4118 :                 new_fd = ::accept4(
     101                 :                     fd, reinterpret_cast<sockaddr*>(&peer), &addrlen,
     102                 :                     SOCK_NONBLOCK | SOCK_CLOEXEC);
     103                 :             }
     104            4118 :             while (new_fd < 0 && errno == EINTR);
     105            4118 :             return new_fd;
     106                 :         }
     107                 :     };
     108                 : 
     109                 :     // Create a nonblocking, close-on-exec socket using Linux's atomic flags.
     110            2237 :     static int create_socket(int family, int type, int protocol) noexcept
     111                 :     {
     112            2237 :         return ::socket(family, type | SOCK_NONBLOCK | SOCK_CLOEXEC, protocol);
     113                 :     }
     114                 : 
     115                 :     // Apply protocol-specific options after socket creation.
     116                 :     // For IP sockets, sets IPV6_V6ONLY on AF_INET6 (best-effort).
     117                 :     static std::error_code
     118            2123 :     configure_ip_socket(int fd, int family) noexcept
     119                 :     {
     120            2123 :         if (family == AF_INET6)
     121                 :         {
     122              14 :             int one = 1;
     123              14 :             (void)::setsockopt(
     124                 :                 fd, IPPROTO_IPV6, IPV6_V6ONLY, &one, sizeof(one));
     125                 :         }
     126            2123 :         return {};
     127                 :     }
     128                 : 
     129                 :     // Apply protocol-specific options for acceptor sockets.
     130                 :     // For IP acceptors, sets IPV6_V6ONLY=0 (dual-stack, best-effort).
     131                 :     static std::error_code
     132              94 :     configure_ip_acceptor(int fd, int family) noexcept
     133                 :     {
     134              94 :         if (family == AF_INET6)
     135                 :         {
     136               9 :             int val = 0;
     137               9 :             (void)::setsockopt(
     138                 :                 fd, IPPROTO_IPV6, IPV6_V6ONLY, &val, sizeof(val));
     139                 :         }
     140              94 :         return {};
     141                 :     }
     142                 : 
     143                 :     // No extra configuration needed for local (unix) sockets on epoll.
     144                 :     static std::error_code
     145              20 :     configure_local_socket(int /*fd*/) noexcept
     146                 :     {
     147              20 :         return {};
     148                 :     }
     149                 : 
     150                 :     // Non-mutating validation for fds adopted via assign(). Used when
     151                 :     // the caller retains fd ownership responsibility.
     152                 :     static std::error_code
     153              14 :     validate_assigned_fd(int /*fd*/) noexcept
     154                 :     {
     155              14 :         return {};
     156                 :     }
     157                 : };
     158                 : 
     159                 : } // namespace boost::corosio::detail
     160                 : 
     161                 : #endif // BOOST_COROSIO_HAS_EPOLL
     162                 : 
     163                 : #endif // BOOST_COROSIO_NATIVE_DETAIL_EPOLL_EPOLL_TRAITS_HPP
        

Generated by: LCOV version 2.3