1 //
2 // udp_server.cpp
3 // ~~~~~~~~~~~~~~
4 //
5 // Copyright (c) 2003-2021 Christopher M. Kohlhoff (chris at kohlhoff dot com)
6 //
7 // Distributed under the Boost Software License, Version 1.0. (See accompanying
8 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
9 //
10
11 #include <boost/asio/io_context.hpp>
12 #include <boost/asio/ip/udp.hpp>
13 #include <boost/shared_ptr.hpp>
14 #include <cstdio>
15 #include <cstdlib>
16 #include <cstring>
17 #include <vector>
18 #include "allocator.hpp"
19
20 using boost::asio::ip::udp;
21
22 #include <boost/asio/yield.hpp>
23
24 class udp_server : boost::asio::coroutine
25 {
26 public:
udp_server(boost::asio::io_context & io_context,unsigned short port,std::size_t buf_size)27 udp_server(boost::asio::io_context& io_context,
28 unsigned short port, std::size_t buf_size) :
29 socket_(io_context, udp::endpoint(udp::v4(), port)),
30 buffer_(buf_size)
31 {
32 }
33
operator ()(boost::system::error_code ec,std::size_t n=0)34 void operator()(boost::system::error_code ec, std::size_t n = 0)
35 {
36 reenter (this) for (;;)
37 {
38 yield socket_.async_receive_from(
39 boost::asio::buffer(buffer_),
40 sender_, ref(this));
41
42 if (!ec)
43 {
44 for (std::size_t i = 0; i < n; ++i) buffer_[i] = ~buffer_[i];
45 socket_.send_to(boost::asio::buffer(buffer_, n), sender_, 0, ec);
46 }
47 }
48 }
49
asio_handler_allocate(std::size_t n,udp_server * s)50 friend void* asio_handler_allocate(std::size_t n, udp_server* s)
51 {
52 return s->allocator_.allocate(n);
53 }
54
asio_handler_deallocate(void * p,std::size_t,udp_server * s)55 friend void asio_handler_deallocate(void* p, std::size_t, udp_server* s)
56 {
57 s->allocator_.deallocate(p);
58 }
59
60 struct ref
61 {
refudp_server::ref62 explicit ref(udp_server* p)
63 : p_(p)
64 {
65 }
66
operator ()udp_server::ref67 void operator()(boost::system::error_code ec, std::size_t n = 0)
68 {
69 (*p_)(ec, n);
70 }
71
72 private:
73 udp_server* p_;
74
asio_handler_allocate(std::size_t n,ref * r)75 friend void* asio_handler_allocate(std::size_t n, ref* r)
76 {
77 return asio_handler_allocate(n, r->p_);
78 }
79
asio_handler_deallocate(void * p,std::size_t n,ref * r)80 friend void asio_handler_deallocate(void* p, std::size_t n, ref* r)
81 {
82 asio_handler_deallocate(p, n, r->p_);
83 }
84 };
85
86 private:
87 udp::socket socket_;
88 std::vector<unsigned char> buffer_;
89 udp::endpoint sender_;
90 allocator allocator_;
91 };
92
93 #include <boost/asio/unyield.hpp>
94
main(int argc,char * argv[])95 int main(int argc, char* argv[])
96 {
97 if (argc != 5)
98 {
99 std::fprintf(stderr,
100 "Usage: udp_server <port1> <nports> "
101 "<bufsize> {spin|block}\n");
102 return 1;
103 }
104
105 unsigned short first_port = static_cast<unsigned short>(std::atoi(argv[1]));
106 unsigned short num_ports = static_cast<unsigned short>(std::atoi(argv[2]));
107 std::size_t buf_size = std::atoi(argv[3]);
108 bool spin = (std::strcmp(argv[4], "spin") == 0);
109
110 boost::asio::io_context io_context(1);
111 std::vector<boost::shared_ptr<udp_server> > servers;
112
113 for (unsigned short i = 0; i < num_ports; ++i)
114 {
115 unsigned short port = first_port + i;
116 boost::shared_ptr<udp_server> s(new udp_server(io_context, port, buf_size));
117 servers.push_back(s);
118 (*s)(boost::system::error_code());
119 }
120
121 if (spin)
122 for (;;) io_context.poll();
123 else
124 io_context.run();
125 }
126