#pragma once #include "../state.hpp" #include "../ft_sha1_info.hpp" #include "../command_line.hpp" #include #include #include #include #include #include namespace States { // we are either sending or receiving // we have full info struct SHA1 final : public StateI { public: // general interface SHA1( ToxClient& tcl, const CommandLine& cl, mio::mmap_sink&& file_map, const FTInfoSHA1&& sha1_info, const std::vector&& sha1_info_data, //const std::vector&& sha1_info_hash, const SHA1Digest&& sha1_info_hash, std::vector&& have_chunk ); ~SHA1(void) override = default; bool iterate(float delta) override; std::unique_ptr nextState(void) override; public: // callbacks // sha1_info void onFT1ReceiveRequestSHA1Info(uint32_t group_number, uint32_t peer_number, const uint8_t* file_id, size_t file_id_size) override; bool onFT1ReceiveInitSHA1Info(uint32_t group_number, uint32_t peer_number, const uint8_t* file_id, size_t file_id_size, const uint8_t transfer_id, const size_t file_size) override; void onFT1ReceiveDataSHA1Info(uint32_t group_number, uint32_t peer_number, uint8_t transfer_id, size_t data_offset, const uint8_t* data, size_t data_size) override; void onFT1SendDataSHA1Info(uint32_t group_number, uint32_t peer_number, uint8_t transfer_id, size_t data_offset, uint8_t* data, size_t data_size) override; // sha1_chunk void onFT1ReceiveRequestSHA1Chunk(uint32_t group_number, uint32_t peer_number, const uint8_t* file_id, size_t file_id_size) override; bool onFT1ReceiveInitSHA1Chunk(uint32_t group_number, uint32_t peer_number, const uint8_t* file_id, size_t file_id_size, const uint8_t transfer_id, const size_t file_size) override; void onFT1ReceiveDataSHA1Chunk(uint32_t group_number, uint32_t peer_number, uint8_t transfer_id, size_t data_offset, const uint8_t* data, size_t data_size) override; void onFT1SendDataSHA1Chunk(uint32_t group_number, uint32_t peer_number, uint8_t transfer_id, size_t data_offset, uint8_t* data, size_t data_size) override; private: // avoids duplicates // clears timer if exists void queueUpRequestInfo(uint32_t group_number, uint32_t peer_number); void queueUpRequestChunk(uint32_t group_number, uint32_t peer_number, const SHA1Digest& hash); std::optional chunkIndex(const SHA1Digest& hash) const; size_t chunkSize(size_t chunk_index) const; bool haveChunk(const SHA1Digest& hash) const; private: mio::mmap_sink _file_map; // writable if not all const FTInfoSHA1 _sha1_info; const std::vector _sha1_info_data; const SHA1Digest _sha1_info_hash; float _io_log_timer {0.f}; size_t _bytes_up {0}; size_t _bytes_up_last_log {0}; size_t _bytes_down {0}; size_t _bytes_down_last_log {0}; // index is the same as for info std::vector _have_chunk; bool _have_all {false}; size_t _have_count {0}; std::deque _chunk_want_queue; // chunk_index -> time since request std::map _chunks_requested; size_t _max_concurrent_in {32}; size_t _max_concurrent_out {16}; std::minstd_rand _rng {1337}; std::uniform_int_distribution _distrib; std::unordered_map _chunk_hash_to_index; // group_number, peer_number std::deque> _queue_requested_info; // group_number, peer_number, transfer_id, seconds since (remote) activity std::vector> _transfers_requested_info; // group_number, peer_number, chunk_hash std::deque> _queue_requested_chunk; // group_number, peer_number, transfer_id(i/o), seconds since (remote) activity, chunk index std::vector> _transfers_sending_chunk; std::vector> _transfers_receiving_chunk; }; } // States