• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //
2 // transmit_file.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 <ctime>
12 #include <iostream>
13 #include <string>
14 #include <boost/bind/bind.hpp>
15 #include <boost/shared_ptr.hpp>
16 #include <boost/enable_shared_from_this.hpp>
17 #include <boost/asio.hpp>
18 
19 #if defined(BOOST_ASIO_HAS_WINDOWS_OVERLAPPED_PTR)
20 
21 using boost::asio::ip::tcp;
22 using boost::asio::windows::overlapped_ptr;
23 using boost::asio::windows::random_access_handle;
24 
25 typedef boost::asio::basic_stream_socket<tcp,
26     boost::asio::io_context::executor_type> tcp_socket;
27 
28 typedef boost::asio::basic_socket_acceptor<tcp,
29     boost::asio::io_context::executor_type> tcp_acceptor;
30 
31 // A wrapper for the TransmitFile overlapped I/O operation.
32 template <typename Handler>
transmit_file(tcp_socket & socket,random_access_handle & file,Handler handler)33 void transmit_file(tcp_socket& socket,
34     random_access_handle& file, Handler handler)
35 {
36   // Construct an OVERLAPPED-derived object to contain the handler.
37   overlapped_ptr overlapped(socket.get_executor().context(), handler);
38 
39   // Initiate the TransmitFile operation.
40   BOOL ok = ::TransmitFile(socket.native_handle(),
41       file.native_handle(), 0, 0, overlapped.get(), 0, 0);
42   DWORD last_error = ::GetLastError();
43 
44   // Check if the operation completed immediately.
45   if (!ok && last_error != ERROR_IO_PENDING)
46   {
47     // The operation completed immediately, so a completion notification needs
48     // to be posted. When complete() is called, ownership of the OVERLAPPED-
49     // derived object passes to the io_context.
50     boost::system::error_code ec(last_error,
51         boost::asio::error::get_system_category());
52     overlapped.complete(ec, 0);
53   }
54   else
55   {
56     // The operation was successfully initiated, so ownership of the
57     // OVERLAPPED-derived object has passed to the io_context.
58     overlapped.release();
59   }
60 }
61 
62 class connection
63   : public boost::enable_shared_from_this<connection>
64 {
65 public:
66   typedef boost::shared_ptr<connection> pointer;
67 
create(boost::asio::io_context & io_context,const std::string & filename)68   static pointer create(boost::asio::io_context& io_context,
69       const std::string& filename)
70   {
71     return pointer(new connection(io_context, filename));
72   }
73 
socket()74   tcp_socket& socket()
75   {
76     return socket_;
77   }
78 
start()79   void start()
80   {
81     boost::system::error_code ec;
82     file_.assign(::CreateFile(filename_.c_str(), GENERIC_READ, 0, 0,
83           OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, 0), ec);
84     if (file_.is_open())
85     {
86       transmit_file(socket_, file_,
87           boost::bind(&connection::handle_write, shared_from_this(),
88             boost::asio::placeholders::error,
89             boost::asio::placeholders::bytes_transferred));
90     }
91   }
92 
93 private:
connection(boost::asio::io_context & io_context,const std::string & filename)94   connection(boost::asio::io_context& io_context, const std::string& filename)
95     : socket_(io_context),
96       filename_(filename),
97       file_(io_context)
98   {
99   }
100 
handle_write(const boost::system::error_code &,size_t)101   void handle_write(const boost::system::error_code& /*error*/,
102       size_t /*bytes_transferred*/)
103   {
104     boost::system::error_code ignored_ec;
105     socket_.shutdown(tcp_socket::shutdown_both, ignored_ec);
106   }
107 
108   tcp_socket socket_;
109   std::string filename_;
110   random_access_handle file_;
111 };
112 
113 class server
114 {
115 public:
server(boost::asio::io_context & io_context,unsigned short port,const std::string & filename)116   server(boost::asio::io_context& io_context,
117       unsigned short port, const std::string& filename)
118     : acceptor_(io_context, tcp::endpoint(tcp::v4(), port)),
119       filename_(filename)
120   {
121     start_accept();
122   }
123 
124 private:
start_accept()125   void start_accept()
126   {
127     connection::pointer new_connection =
128       connection::create(acceptor_.get_executor().context(), filename_);
129 
130     acceptor_.async_accept(new_connection->socket(),
131         boost::bind(&server::handle_accept, this, new_connection,
132           boost::asio::placeholders::error));
133   }
134 
handle_accept(connection::pointer new_connection,const boost::system::error_code & error)135   void handle_accept(connection::pointer new_connection,
136       const boost::system::error_code& error)
137   {
138     if (!error)
139     {
140       new_connection->start();
141     }
142 
143     start_accept();
144   }
145 
146   tcp_acceptor acceptor_;
147   std::string filename_;
148 };
149 
main(int argc,char * argv[])150 int main(int argc, char* argv[])
151 {
152   try
153   {
154     if (argc != 3)
155     {
156       std::cerr << "Usage: transmit_file <port> <filename>\n";
157       return 1;
158     }
159 
160     boost::asio::io_context io_context;
161 
162     using namespace std; // For atoi.
163     server s(io_context, atoi(argv[1]), argv[2]);
164 
165     io_context.run();
166   }
167   catch (std::exception& e)
168   {
169     std::cerr << e.what() << std::endl;
170   }
171 
172   return 0;
173 }
174 
175 #else // defined(BOOST_ASIO_HAS_WINDOWS_OVERLAPPED_PTR)
176 # error Overlapped I/O not available on this platform
177 #endif // defined(BOOST_ASIO_HAS_WINDOWS_OVERLAPPED_PTR)
178