Asio Extensions
Additional functionality built on top of (Boost.)Asio
thread_pool_file_service.hpp
1 /// @file
2 /// Declares the thread_pool_file_service class.
3 ///
4 /// @copyright Copyright (c) 2015 Tim Niederhausen (tim@rnc-ag.de)
5 /// Distributed under the Boost Software License, Version 1.0.
6 /// (See accompanying file LICENSE_1_0.txt or copy at
7 /// http://www.boost.org/LICENSE_1_0.txt)
8 
9 #ifndef ASIOEXT_THREADPOOLFILESERVICE_HPP
10 #define ASIOEXT_THREADPOOLFILESERVICE_HPP
11 
12 #include "asioext/detail/config.hpp"
13 
14 #if ASIOEXT_HAS_PRAGMA_ONCE
15 #pragma once
16 #endif
17 
18 #include "asioext/file_handle.hpp"
19 #include "asioext/file_perms.hpp"
20 #include "asioext/file_attrs.hpp"
21 #include "asioext/seek_origin.hpp"
22 #include "asioext/cancellation_token.hpp"
23 #include "asioext/async_result.hpp"
24 
25 #include "asioext/detail/move_support.hpp"
26 #include "asioext/detail/cstdint.hpp"
27 #include "asioext/detail/service_base.hpp"
28 #include "asioext/detail/mutex.hpp"
29 #include "asioext/detail/work.hpp"
30 #include "asioext/detail/thread_group.hpp"
31 
32 #if defined(ASIOEXT_HAS_BOOST_FILESYSTEM) || defined(ASIOEXT_IS_DOCUMENTATION)
33 # include <boost/filesystem/path.hpp>
34 #endif
35 
36 ASIOEXT_NS_BEGIN
37 
38 /// @ingroup files_handle
39 /// @brief A FileService utilizing a thread-pool for async operations.
40 ///
41 /// This FileService class uses a thread-pool to emulate asynchronous file I/O.
43 #if !defined(ASIOEXT_IS_DOCUMENTATION)
44  : public asioext::detail::service_base<thread_pool_file_service>
45 #else
46  : public asio::io_service::service
47 #endif
48 {
49 public:
50 #if defined(ASIOEXT_IS_DOCUMENTATION)
51  /// The unique service identifier.
52  static asio::io_service::id id;
53 #endif
54 
55 #if defined(ASIOEXT_IS_DOCUMENTATION)
56  /// The native handle type.
57  typedef implementation_defined native_handle_type;
58 #else
60 #endif
61 
62 #if defined(ASIOEXT_IS_DOCUMENTATION)
63  /// The type of a file implementation.
64  typedef implementation_defined implementation_type;
65 #else
67  {
68  public:
70  : next_(0)
71  , prev_(0)
72  {
73  // ctor
74  }
75 
76  private:
77  // Only this service will have access to the internal values.
78  friend class thread_pool_file_service;
79 
80  file_handle handle_;
81  cancellation_token_source cancel_token_;
82 
83  // Pointers to adjacent handle implementations in linked list.
84  implementation_type* next_;
85  implementation_type* prev_;
86  };
87 #endif
88 
89  /// Construct a new file service for the specified io_service.
90  ///
91  /// @param io_service The io_service that will own this service object.
92  ///
93  /// @param num_threads The number of threads that shall be spawned to
94  /// execute file I/O operations. Defaults to 1.
95  ASIOEXT_DECL explicit thread_pool_file_service(asio::io_service& io_service,
96  std::size_t num_threads = 1);
97 
98  /// Destroy all user-defined handler objects owned by the service.
99  ASIOEXT_DECL void shutdown_service();
100 
101  /// Construct a new file implementation.
102  ASIOEXT_DECL void construct(implementation_type& impl);
103 
104 #ifdef ASIOEXT_HAS_MOVE
105  /// Move-construct a new file implementation.
106  ASIOEXT_DECL void move_construct(implementation_type& impl,
107  implementation_type& other_impl)
108  ASIOEXT_NOEXCEPT;
109 
110  /// Move-assign from another file implementation.
111  ASIOEXT_DECL void move_assign(implementation_type& impl,
112  thread_pool_file_service& other_service,
113  implementation_type& other_impl);
114 #endif
115 
116  /// Destroy a file implementation.
117  ASIOEXT_DECL void destroy(implementation_type& impl);
118 
119  /// Open a handle to the given file.
120  ASIOEXT_DECL void open(implementation_type& impl,
121  const char* filename,
122  open_flags flags,
123  file_perms perms, file_attrs attrs,
124  error_code& ec) ASIOEXT_NOEXCEPT;
125 
126 #if defined(ASIOEXT_WINDOWS) || defined(ASIOEXT_IS_DOCUMENTATION)
127  /// Open a handle to the given file.
128  ASIOEXT_DECL void open(implementation_type& impl,
129  const wchar_t* filename,
130  open_flags flags,
131  file_perms perms, file_attrs attrs,
132  error_code& ec) ASIOEXT_NOEXCEPT;
133 #endif
134 
135 #if defined(ASIOEXT_HAS_BOOST_FILESYSTEM) || defined(ASIOEXT_IS_DOCUMENTATION)
136  /// Open a handle to the given file.
137  ASIOEXT_DECL void open(implementation_type& impl,
138  const boost::filesystem::path& filename,
139  open_flags flags,
140  file_perms perms, file_attrs attrs,
141  error_code& ec) ASIOEXT_NOEXCEPT;
142 #endif
143 
144  /// Assign a native handle to a file implementation.
145  ASIOEXT_DECL void assign(implementation_type& impl,
146  const native_handle_type& handle,
147  error_code& ec) ASIOEXT_NOEXCEPT;
148 
149  /// Determine whether the file handle is open.
150  bool is_open(const implementation_type& impl) const ASIOEXT_NOEXCEPT
151  {
152  return impl.handle_.is_open();
153  }
154 
155  /// Destroy a file implementation.
156  ASIOEXT_DECL void close(implementation_type& impl, error_code& ec)
157  ASIOEXT_NOEXCEPT;
158 
159  /// Get the native file handle representation.
161  {
162  return impl.handle_.native_handle();
163  }
164 
165  /// Get the current file pointer position.
166  ASIOEXT_DECL uint64_t position(implementation_type& impl,
167  error_code& ec) ASIOEXT_NOEXCEPT;
168 
169  /// Change the current file pointer position.
170  ASIOEXT_DECL uint64_t seek(implementation_type& impl,
171  seek_origin origin,
172  int64_t offset,
173  error_code& ec) ASIOEXT_NOEXCEPT;
174 
175  /// Get the file size.
176  ASIOEXT_DECL uint64_t size(implementation_type& impl,
177  error_code& ec) ASIOEXT_NOEXCEPT;
178 
179  /// Set the file size.
180  ASIOEXT_DECL void size(implementation_type& impl, uint64_t new_size,
181  error_code& ec) ASIOEXT_NOEXCEPT;
182 
183  /// Get the file permissions.
184  ASIOEXT_DECL file_perms permissions(implementation_type& impl,
185  error_code& ec) ASIOEXT_NOEXCEPT;
186 
187  /// Set the file permissions.
188  ASIOEXT_DECL void permissions(implementation_type& impl,
189  file_perms new_perms,
190  error_code& ec) ASIOEXT_NOEXCEPT;
191 
192  /// Set the file permissions.
193  ASIOEXT_DECL void permissions(implementation_type& impl,
194  file_perms new_perms, file_perm_options opts,
195  error_code& ec) ASIOEXT_NOEXCEPT;
196 
197  /// Get the file attributes.
198  ASIOEXT_DECL file_attrs attributes(implementation_type& impl,
199  error_code& ec) ASIOEXT_NOEXCEPT;
200 
201  /// Set the file attributes.
202  ASIOEXT_DECL void attributes(implementation_type& impl,
203  file_attrs new_attrs,
204  error_code& ec) ASIOEXT_NOEXCEPT;
205 
206  /// Set the file attributes.
207  ASIOEXT_DECL void attributes(implementation_type& impl,
208  file_attrs new_attrs, file_attr_options opts,
209  error_code& ec) ASIOEXT_NOEXCEPT;
210 
211  /// Get the file times.
212  ASIOEXT_DECL file_times times(implementation_type& impl,
213  error_code& ec) ASIOEXT_NOEXCEPT;
214 
215  /// Set the file times.
216  ASIOEXT_DECL void times(implementation_type& impl,
217  const file_times& new_times,
218  error_code& ec) ASIOEXT_NOEXCEPT;
219 
220  /// Cancel all operations associated with the handle.
221  ASIOEXT_DECL void cancel(implementation_type& impl,
222  error_code& ec) ASIOEXT_NOEXCEPT;
223 
224  /// Read some data. Returns the number of bytes received.
225  template <typename MutableBufferSequence>
226  size_t read_some(implementation_type& impl,
227  const MutableBufferSequence& buffers,
228  error_code& ec) ASIOEXT_NOEXCEPT;
229 
230  /// Write the given data. Returns the number of bytes written.
231  template <typename ConstBufferSequence>
232  size_t write_some(implementation_type& impl,
233  const ConstBufferSequence& buffers,
234  error_code& ec) ASIOEXT_NOEXCEPT;
235 
236  /// Read some data at a specified offset. Returns the number of bytes received.
237  template <typename MutableBufferSequence>
238  size_t read_some_at(implementation_type& impl, uint64_t offset,
239  const MutableBufferSequence& buffers,
240  error_code& ec) ASIOEXT_NOEXCEPT;
241 
242  /// Write the given data at the specified offset. Returns the number of bytes
243  /// written.
244  template <typename ConstBufferSequence>
245  size_t write_some_at(implementation_type& impl, uint64_t offset,
246  const ConstBufferSequence& buffers,
247  error_code& ec) ASIOEXT_NOEXCEPT;
248 
249  /// Start an asynchronous read. The buffer for the data being received must be
250  /// valid for the lifetime of the asynchronous operation.
251  template <typename MutableBufferSequence, typename Handler>
252  ASIOEXT_INITFN_RESULT_TYPE(Handler, void(error_code, std::size_t))
253  async_read_some(implementation_type& impl,
254  const MutableBufferSequence& buffers,
255  ASIOEXT_MOVE_ARG(Handler) handler);
256 
257  /// Start an asynchronous write. The data being written must be valid for the
258  /// lifetime of the asynchronous operation.
259  template <typename ConstBufferSequence, typename Handler>
260  ASIOEXT_INITFN_RESULT_TYPE(Handler, void(error_code, std::size_t))
261  async_write_some(implementation_type& impl,
262  const ConstBufferSequence& buffers,
263  ASIOEXT_MOVE_ARG(Handler) handler);
264 
265  /// Start an asynchronous read at a specified offset. The buffer for the data
266  /// being received must be valid for the lifetime of the asynchronous
267  /// operation.
268  template <typename MutableBufferSequence, typename Handler>
269  ASIOEXT_INITFN_RESULT_TYPE(Handler, void(error_code, std::size_t))
270  async_read_some_at(implementation_type& impl, uint64_t offset,
271  const MutableBufferSequence& buffers,
272  ASIOEXT_MOVE_ARG(Handler) handler);
273 
274  /// Start an asynchronous write at a specified offset. The data being written
275  /// must be valid for the lifetime of the asynchronous operation.
276  template <typename ConstBufferSequence, typename Handler>
277  ASIOEXT_INITFN_RESULT_TYPE(Handler, void(error_code, std::size_t))
278  async_write_some_at(implementation_type& impl, uint64_t offset,
279  const ConstBufferSequence& buffers,
280  ASIOEXT_MOVE_ARG(Handler) handler);
281 
282  /// @private
283  // This is needed for tests.
284  asio::io_service& get_pool_io_service()
285  {
286  return pool_;
287  }
288 
289 private:
290  struct thread_function
291  {
292  asio::io_service* service_;
293  void operator()();
294  };
295 
296  // Helper function to close a handle when the associated object is being
297  // destroyed.
298  ASIOEXT_DECL void close_for_destruction(implementation_type& impl);
299 
300  // The io_service that runs on the thread pool.
301  asio::io_service pool_;
302 
303  // A work helper to keep |pool_| running.
304  detail::work work_;
305 
306  // The thread pool.
307  detail::thread_group pool_threads_;
308 
309  // Mutex to protect access to the linked list of implementations.
310  detail::mutex mutex_;
311 
312  // The head of a linked list of all implementations.
313  implementation_type* impl_list_;
314 };
315 
316 ASIOEXT_NS_END
317 
318 #include "asioext/impl/thread_pool_file_service.hpp"
319 
320 #if defined(ASIOEXT_HEADER_ONLY)
321 # include "asioext/impl/thread_pool_file_service.cpp"
322 #endif
323 
324 #endif
open_flags
Specifies semantics for opening files.
Definition: open_flags.hpp:45
Container for various times associated with a file.
Definition: file_handle.hpp:34
file_perms
Names for permissions.
Definition: file_perms.hpp:31
implementation_defined native_handle_type
The operating system&#39;s native file handle type.
Definition: file_handle.hpp:78
file_perm_options
Options to control the behavior of asioext::file_handle::permissions(file_perms,file_perm_options) ...
Definition: file_perms.hpp:119
implementation_defined implementation_type
The type of a file implementation.
Definition: thread_pool_file_service.hpp:64
file_attrs
Names for file attributes.
Definition: file_attrs.hpp:32
implementation_defined native_handle_type
The native handle type.
Definition: thread_pool_file_service.hpp:57
Manager for cancellation_tokens.
Definition: cancellation_token.hpp:26
static asio::io_service::id id
The unique service identifier.
Definition: thread_pool_file_service.hpp:52
automatically_chosen error_code
Typedef for the error_code class used by this library.
Definition: error_code.hpp:37
bool is_open(const implementation_type &impl) const noexcept
Determine whether the file handle is open.
Definition: thread_pool_file_service.hpp:150
unique_file_handle open(const char *filename, open_flags flags, file_perms perms=file_perms::create_default, file_attrs attrs=file_attrs::none)
Open a file and return its handle.
seek_origin
Specifies the various position offset origins.
Definition: seek_origin.hpp:25
A FileService utilizing a thread-pool for async operations.
Definition: thread_pool_file_service.hpp:42
file_attr_options
Options to control the behavior of asioext::file_handle::attributes(file_attrs,file_attr_options) ...
Definition: file_attrs.hpp:99
native_handle_type native_handle(implementation_type &impl) noexcept
Get the native file handle representation.
Definition: thread_pool_file_service.hpp:160
A thin and lightweight wrapper around a native file handle.
Definition: file_handle.hpp:73