/* Copyright <2018-2022> Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #ifndef INCLUDE_DOWNLOAD_RING_DOWNLOAD_HPP_ #define INCLUDE_DOWNLOAD_RING_DOWNLOAD_HPP_ #include "common.hpp" #include "download/i_download.hpp" #include "utils/native_file.hpp" #include "types/repertory.hpp" namespace repertory { class buffered_reader; class app_config; class ring_download final : public virtual i_download { private: struct io_action { io_action(std::mutex &mtx, std::condition_variable &cv, std::function action) : mutex(mtx), notify(cv), action(action) {} std::mutex &mutex; std::condition_variable ¬ify; std::function action; }; public: ring_download(const app_config &config, filesystem_item fsi, const api_reader_callback &api_reader, const std::uint64_t &handle, const std::size_t &chunk_size, const std::size_t &ring_buffer_size); ~ring_download() override; private: const app_config &config_; const filesystem_item fsi_; const api_reader_callback &api_reader_; const std::uint64_t handle_; const std::size_t chunk_size_; boost::dynamic_bitset<> ring_state_; const std::size_t total_chunks_ = 0u; api_error error_ = api_error::success; native_file_ptr buffer_file_; std::string buffer_file_path_; buffered_reader *buffered_reader_ = nullptr; bool disable_download_end_ = false; std::size_t head_chunk_index_ = 0ull; std::mutex io_mutex_; std::condition_variable io_notify_; std::deque> io_queue_; std::unique_ptr io_thread_; std::size_t read_chunk_ = 0ull; std::unique_ptr buffer_thread_; std::mutex read_mutex_; std::condition_variable read_notify_; bool stop_requested_ = false; std::size_t write_chunk_ = 0ull; std::mutex write_mutex_; private: void buffer_thread(std::size_t start_chunk_index); void io_thread(); bool is_active() const { return not stop_requested_ && (error_ == api_error::success); } bool queue_io_item(std::mutex &m, std::condition_variable &cv, const bool &is_read, std::function action); void read(std::size_t read_chunk_index, std::size_t read_size, std::size_t read_offset, std::vector &data); void set_api_error(const api_error &error); void start(const std::size_t &start_chunk_index); void stop(); public: api_error allocate(const std::uint64_t & /*handle*/, const std::uint64_t & /*size*/, const allocator_callback & /*allocator*/, const completer_callback & /*completer*/) override { return api_error::not_implemented; } api_error download_all() override { return api_error::not_implemented; } api_error get_result() const override { return error_; } std::string get_source_path() const override { return buffer_file_path_; } void get_state_information(filesystem_item &, std::size_t &, std::size_t &, boost::dynamic_bitset<> &, boost::dynamic_bitset<> &) override {} bool get_write_supported() const override { return false; } void notify_stop_requested() override; bool pause() override { return false; } api_error read_bytes(const std::uint64_t &, std::size_t read_size, const std::uint64_t &read_offset, std::vector &data) override; void reset_timeout(const bool &) override {} void resume() override {} void set_api_path(const std::string &) override {} void set_disable_download_end(const bool &disable) override { disable_download_end_ = disable; } api_error write_bytes(const std::uint64_t &, const std::uint64_t &, std::vector, std::size_t &bytes_written, const completer_callback & /*completer*/) override { bytes_written = 0u; return api_error::not_implemented; } }; } // namespace repertory #endif // INCLUDE_DOWNLOAD_RING_DOWNLOAD_HPP_