TLA Line data Source code
1 : //
2 : // Copyright (c) 2025 Vinnie Falco (vinnie.falco@gmail.com)
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_TEST_SOCKET_PAIR_HPP
12 : #define BOOST_COROSIO_TEST_SOCKET_PAIR_HPP
13 :
14 : #include <boost/corosio/io_context.hpp>
15 : #include <boost/corosio/tcp_acceptor.hpp>
16 : #include <boost/corosio/tcp_socket.hpp>
17 : #include <boost/corosio/socket_option.hpp>
18 : #include <boost/capy/ex/run_async.hpp>
19 : #include <boost/capy/task.hpp>
20 :
21 : #include <cstdio>
22 : #include <stdexcept>
23 : #include <system_error>
24 : #include <utility>
25 :
26 : namespace boost::corosio::test {
27 :
28 : /** Create a connected pair of sockets.
29 :
30 : Creates two sockets connected via loopback TCP sockets.
31 : Data written to one socket can be read from the other.
32 :
33 : @tparam Socket The socket type (default `tcp_socket`).
34 : @tparam Acceptor The acceptor type (default `tcp_acceptor`).
35 :
36 : @param ctx The I/O context for the sockets.
37 :
38 : @return A pair of connected sockets.
39 : */
40 : template<
41 : class Socket = tcp_socket,
42 : class Acceptor = tcp_acceptor,
43 : bool Linger = true>
44 : std::pair<Socket, Socket>
45 HIT 58 : make_socket_pair(io_context& ctx)
46 : {
47 58 : auto ex = ctx.get_executor();
48 :
49 58 : std::error_code accept_ec;
50 58 : std::error_code connect_ec;
51 58 : bool accept_done = false;
52 58 : bool connect_done = false;
53 :
54 58 : Acceptor acc(ctx);
55 58 : acc.open();
56 58 : acc.set_option(socket_option::reuse_address(true));
57 58 : if (auto ec = acc.bind(endpoint(ipv4_address::loopback(), 0)))
58 MIS 0 : throw std::runtime_error("socket_pair bind failed: " + ec.message());
59 HIT 58 : if (auto ec = acc.listen())
60 MIS 0 : throw std::runtime_error("socket_pair listen failed: " + ec.message());
61 HIT 58 : auto port = acc.local_endpoint().port();
62 :
63 58 : Socket s1(ctx);
64 58 : Socket s2(ctx);
65 58 : s2.open();
66 :
67 58 : capy::run_async(ex)(
68 116 : [](Acceptor& a, Socket& s, std::error_code& ec_out,
69 : bool& done_out) -> capy::task<> {
70 : auto [ec] = co_await a.accept(s);
71 : ec_out = ec;
72 : done_out = true;
73 : }(acc, s1, accept_ec, accept_done));
74 :
75 58 : capy::run_async(ex)(
76 116 : [](Socket& s, endpoint ep, std::error_code& ec_out,
77 : bool& done_out) -> capy::task<> {
78 : auto [ec] = co_await s.connect(ep);
79 : ec_out = ec;
80 : done_out = true;
81 : }(s2, endpoint(ipv4_address::loopback(), port), connect_ec,
82 : connect_done));
83 :
84 58 : ctx.run();
85 58 : ctx.restart();
86 :
87 58 : if (!accept_done || accept_ec)
88 : {
89 MIS 0 : std::fprintf(
90 : stderr, "socket_pair: accept failed (done=%d, ec=%s)\n",
91 : accept_done, accept_ec.message().c_str());
92 0 : acc.close();
93 0 : throw std::runtime_error("socket_pair accept failed");
94 : }
95 :
96 HIT 58 : if (!connect_done || connect_ec)
97 : {
98 MIS 0 : std::fprintf(
99 : stderr, "socket_pair: connect failed (done=%d, ec=%s)\n",
100 : connect_done, connect_ec.message().c_str());
101 0 : acc.close();
102 0 : s1.close();
103 0 : throw std::runtime_error("socket_pair connect failed");
104 : }
105 :
106 HIT 58 : acc.close();
107 :
108 : if constexpr (Linger)
109 : {
110 4 : s1.set_option(socket_option::linger(true, 0));
111 4 : s2.set_option(socket_option::linger(true, 0));
112 : }
113 :
114 116 : return {std::move(s1), std::move(s2)};
115 58 : }
116 :
117 : } // namespace boost::corosio::test
118 :
119 : #endif
|