OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
udp_socket.hpp
1#pragma once
3#include <ossia/network/sockets/configuration.hpp>
4
5#include <boost/asio/io_context.hpp>
6#include <boost/asio/ip/udp.hpp>
7#include <boost/asio/local/datagram_protocol.hpp>
8#include <boost/asio/placeholders.hpp>
9#include <boost/asio/strand.hpp>
10#include <boost/asio/write.hpp>
11
12#include <nano_signal_slot.hpp>
13
14namespace ossia::net
15{
16
17class udp_receive_socket
18{
19 using proto = boost::asio::ip::udp;
20
21public:
22 udp_receive_socket(boost::asio::io_context& ctx)
23 : m_context{ctx}
24 , m_socket{boost::asio::make_strand(ctx)}
25 {
26 }
27
28 udp_receive_socket(const inbound_socket_configuration& conf, boost::asio::io_context& ctx)
29 : m_context{ctx}
30 , m_endpoint{boost::asio::ip::make_address(conf.bind), conf.port}
31 , m_socket{boost::asio::make_strand(ctx)}
32 {
33 }
34
35 ~udp_receive_socket() = default;
36
37 void assign(int sock) { m_socket.assign(boost::asio::ip::udp::v4(), sock); }
38 void open()
39 {
40 m_socket.open(boost::asio::ip::udp::v4());
41 m_socket.bind(m_endpoint);
42 }
43
44 void close()
45 {
46 if(m_socket.is_open())
47 {
48 boost::asio::post(m_context, [this] {
49 m_socket.close();
50 on_close();
51 });
52 }
53 }
54
55 template <typename F>
56 void receive(F f)
57 {
58 m_socket.async_receive_from(
59 boost::asio::mutable_buffer(&m_data[0], std::size(m_data)), m_endpoint,
60 [this, f](auto ec, std::size_t sz) {
61 if(ec == boost::asio::error::operation_aborted)
62 return;
63
64 if(!ec && sz > 0)
65 {
66 try
67 {
68 f(m_data, sz);
69 }
70 catch(const std::exception& e)
71 {
72 ossia::logger().error("[udp_socket::receive]: {}", e.what());
73 }
74 catch(...)
75 {
76 ossia::logger().error("[udp_socket::receive]: unknown error");
77 }
78 }
79
80 this->receive(f);
81 });
82 }
83
84 Nano::Signal<void()> on_close;
85
86 boost::asio::io_context& m_context;
87 proto::endpoint m_endpoint;
88 proto::socket m_socket;
89 alignas(16) char m_data[65535];
90};
91
92class udp_send_socket
93{
94 using proto = boost::asio::ip::udp;
95
96public:
97 udp_send_socket(const outbound_socket_configuration& conf, boost::asio::io_context& ctx)
98 : m_context{ctx}
99 , m_endpoint{boost::asio::ip::make_address(conf.host), conf.port}
100 , m_socket{boost::asio::make_strand(ctx)}
101 , m_broadcast{conf.broadcast}
102 {
103 }
104
105 udp_send_socket(
106 const boost::asio::ip::address& host, const uint16_t port,
107 boost::asio::io_context& ctx)
108 : m_context{ctx}
109 , m_endpoint{host, port}
110 , m_socket{boost::asio::make_strand(ctx)}
111 {
112 }
113
114 void connect()
115 {
116 m_socket.open(boost::asio::ip::udp::v4());
117
118 m_socket.set_option(boost::asio::ip::udp::socket::reuse_address(true));
119
120 if(m_broadcast)
121 m_socket.set_option(boost::asio::socket_base::broadcast(true));
122 }
123
124 void close()
125 {
126 if(m_socket.is_open())
127 {
128 boost::asio::post(m_context, [this] {
129 m_socket.close();
130 on_close();
131 });
132 }
133 }
134
135 void write(const proto::endpoint& ep, const char* data, std::size_t sz)
136 {
137 boost::system::error_code ec;
138 m_socket.send_to(boost::asio::const_buffer(data, sz), ep, 0, ec);
139 }
140
141 void write(const char* data, std::size_t sz) { write(m_endpoint, data, sz); }
142
143 Nano::Signal<void()> on_close;
144
145 boost::asio::io_context& m_context;
146 proto::endpoint m_endpoint;
147 proto::socket m_socket;
148 bool m_broadcast{};
149};
150
151}
spdlog::logger & logger() noexcept
Where the errors will be logged. Default is stderr.
Definition context.cpp:118