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<class Socket = tcp_socket, class Acceptor = tcp_acceptor>
41 : std::pair<Socket, Socket>
42 HIT 4 : make_socket_pair(io_context& ctx)
43 : {
44 4 : auto ex = ctx.get_executor();
45 :
46 4 : std::error_code accept_ec;
47 4 : std::error_code connect_ec;
48 4 : bool accept_done = false;
49 4 : bool connect_done = false;
50 :
51 4 : Acceptor acc(ctx);
52 4 : acc.open();
53 4 : acc.set_option(socket_option::reuse_address(true));
54 4 : if (auto ec = acc.bind(endpoint(ipv4_address::loopback(), 0)))
55 MIS 0 : throw std::runtime_error("socket_pair bind failed: " + ec.message());
56 HIT 4 : if (auto ec = acc.listen())
57 MIS 0 : throw std::runtime_error("socket_pair listen failed: " + ec.message());
58 HIT 4 : auto port = acc.local_endpoint().port();
59 :
60 4 : Socket s1(ctx);
61 4 : Socket s2(ctx);
62 4 : s2.open();
63 :
64 4 : capy::run_async(ex)(
65 8 : [](Acceptor& a, Socket& s, std::error_code& ec_out,
66 : bool& done_out) -> capy::task<> {
67 : auto [ec] = co_await a.accept(s);
68 : ec_out = ec;
69 : done_out = true;
70 : }(acc, s1, accept_ec, accept_done));
71 :
72 4 : capy::run_async(ex)(
73 8 : [](Socket& s, endpoint ep, std::error_code& ec_out,
74 : bool& done_out) -> capy::task<> {
75 : auto [ec] = co_await s.connect(ep);
76 : ec_out = ec;
77 : done_out = true;
78 : }(s2, endpoint(ipv4_address::loopback(), port), connect_ec,
79 : connect_done));
80 :
81 4 : ctx.run();
82 4 : ctx.restart();
83 :
84 4 : if (!accept_done || accept_ec)
85 : {
86 MIS 0 : std::fprintf(
87 : stderr, "socket_pair: accept failed (done=%d, ec=%s)\n",
88 : accept_done, accept_ec.message().c_str());
89 0 : acc.close();
90 0 : throw std::runtime_error("socket_pair accept failed");
91 : }
92 :
93 HIT 4 : if (!connect_done || connect_ec)
94 : {
95 MIS 0 : std::fprintf(
96 : stderr, "socket_pair: connect failed (done=%d, ec=%s)\n",
97 : connect_done, connect_ec.message().c_str());
98 0 : acc.close();
99 0 : s1.close();
100 0 : throw std::runtime_error("socket_pair connect failed");
101 : }
102 :
103 HIT 4 : acc.close();
104 :
105 4 : s1.set_option(socket_option::linger(true, 0));
106 4 : s2.set_option(socket_option::linger(true, 0));
107 :
108 8 : return {std::move(s1), std::move(s2)};
109 4 : }
110 :
111 : } // namespace boost::corosio::test
112 :
113 : #endif
|