3#include <ossia/network/sockets/configuration.hpp>
5#include <boost/asio/io_context.hpp>
6#include <boost/asio/ip/multicast.hpp>
7#include <boost/asio/ip/udp.hpp>
8#include <boost/asio/local/datagram_protocol.hpp>
9#include <boost/asio/placeholders.hpp>
10#include <boost/asio/strand.hpp>
11#include <boost/asio/write.hpp>
14#include <sys/socket.h>
17#include <nano_signal_slot.hpp>
22class udp_receive_socket
24 using proto = boost::asio::ip::udp;
27 udp_receive_socket(boost::asio::io_context& ctx)
29 , m_socket{boost::asio::make_strand(ctx)}
33 udp_receive_socket(
const inbound_socket_configuration& conf, boost::asio::io_context& ctx)
35 , m_endpoint{boost::asio::ip::make_address(conf.bind), conf.port}
36 , m_socket{boost::asio::make_strand(ctx)}
37 , m_multicast_group{conf.multicast_group}
38 , m_multicast_interface{conf.multicast_interface}
42 ~udp_receive_socket() =
default;
44 void assign(
int sock) { m_socket.assign(boost::asio::ip::udp::v4(), sock); }
47 m_socket.open(boost::asio::ip::udp::v4());
48 if(!m_multicast_group.empty())
50 m_socket.set_option(boost::asio::ip::udp::socket::reuse_address(
true));
51#if defined(SO_REUSEPORT)
56 = boost::asio::detail::socket_option::boolean<SOL_SOCKET, SO_REUSEPORT>;
57 boost::system::error_code ec;
58 m_socket.set_option(reuse_port(
true), ec);
61 m_socket.bind(m_endpoint);
62 if(!m_multicast_group.empty())
64 const auto group = boost::asio::ip::make_address_v4(m_multicast_group);
65 const auto iface = boost::asio::ip::make_address_v4(m_multicast_interface);
66 m_socket.set_option(boost::asio::ip::multicast::join_group(group, iface));
72 if(m_socket.is_open())
74 boost::asio::post(m_context, [
this] {
77 m_socket.shutdown(boost::asio::ip::udp::socket::shutdown_both);
92 m_socket.async_receive_from(
93 boost::asio::mutable_buffer(&m_data[0], std::size(m_data)), m_endpoint,
94 [
this, f](
auto ec, std::size_t sz) {
95 if(ec == boost::asio::error::operation_aborted)
104 catch(
const std::exception& e)
106 ossia::logger().error(
"[udp_socket::receive]: {}", e.what());
110 ossia::logger().error(
"[udp_socket::receive]: unknown error");
118 Nano::Signal<void()> on_close;
120 boost::asio::io_context& m_context;
121 proto::endpoint m_endpoint;
122 proto::socket m_socket;
123 std::string m_multicast_group;
124 std::string m_multicast_interface;
125 alignas(16)
char m_data[65535];
130 using proto = boost::asio::ip::udp;
133 udp_send_socket(
const outbound_socket_configuration& conf, boost::asio::io_context& ctx)
135 , m_endpoint{boost::asio::ip::make_address(conf.host), conf.port}
136 , m_socket{boost::asio::make_strand(ctx)}
137 , m_broadcast{conf.broadcast}
138 , m_multicast_ttl{conf.multicast_ttl}
139 , m_multicast_interface{conf.multicast_interface}
140 , m_multicast_loopback{conf.multicast_loopback}
145 const boost::asio::ip::address& host,
const uint16_t port,
146 boost::asio::io_context& ctx)
148 , m_endpoint{host, port}
149 , m_socket{boost::asio::make_strand(ctx)}
155 m_socket.open(boost::asio::ip::udp::v4());
157 m_socket.set_option(boost::asio::ip::udp::socket::reuse_address(
true));
160 m_socket.set_option(boost::asio::socket_base::broadcast(
true));
162 if(m_endpoint.address().is_multicast())
165 m_socket.set_option(boost::asio::ip::multicast::hops(*m_multicast_ttl));
166 if(!m_multicast_interface.empty())
168 m_socket.set_option(boost::asio::ip::multicast::outbound_interface(
169 boost::asio::ip::make_address_v4(m_multicast_interface)));
171 if(m_multicast_loopback)
174 boost::asio::ip::multicast::enable_loopback(*m_multicast_loopback));
181 if(m_socket.is_open())
183 boost::asio::post(m_context, [
this] {
186 m_socket.shutdown(boost::asio::ip::udp::socket::shutdown_both);
197 void write(
const proto::endpoint& ep,
const char* data, std::size_t sz)
199 boost::system::error_code ec;
200 m_socket.send_to(boost::asio::const_buffer(data, sz), ep, 0, ec);
203 void write(
const char* data, std::size_t sz) { write(m_endpoint, data, sz); }
205 Nano::Signal<void()> on_close;
207 boost::asio::io_context& m_context;
208 proto::endpoint m_endpoint;
209 proto::socket m_socket;
211 std::optional<int> m_multicast_ttl;
212 std::string m_multicast_interface;
213 std::optional<bool> m_multicast_loopback;
spdlog::logger & logger() noexcept
Where the errors will be logged. Default is stderr.
Definition context.cpp:118