/* */ #ifndef D_PIECE_H #define D_PIECE_H #include "common.h" #include #include #include #include #include "Command.h" #include "a2functional.h" namespace aria2 { class BitfieldMan; class WrDiskCache; class WrDiskCacheEntry; class DiskAdaptor; class MessageDigest; class Piece { private: std::unique_ptr bitfield_; std::unique_ptr wrCache_; std::unique_ptr mdctx_; std::vector users_; std::string hashType_; size_t index_; int64_t length_; int64_t nextBegin_; bool usedBySegment_; Piece(const Piece& piece) = delete; Piece& operator=(const Piece& piece) = delete; public: static const int32_t BLOCK_LENGTH = 16_k; Piece(); Piece(size_t index, int64_t length, int32_t blockLength = BLOCK_LENGTH); ~Piece(); bool operator==(const Piece& piece) const { return index_ == piece.index_; } bool operator<(const Piece& piece) const { return index_ < piece.index_; } // TODO This function only used by unit tests bool getMissingUnusedBlockIndex(size_t& index) const; // Appends at most n missing unused block index to indexes. For all // i in retrieved indexes, call bitfield->setUseBit(i). This // function just append index to indexes and it doesn't remove // anything from it. Returns the number of indexes to retrieved. size_t getMissingUnusedBlockIndex(std::vector& indexes, size_t n) const; bool getFirstMissingBlockIndexWithoutLock(size_t& index) const; bool getAllMissingBlockIndexes(unsigned char* misbitfield, size_t mislen) const; void completeBlock(size_t blockIndex); void cancelBlock(size_t blockIndex); size_t countCompleteBlock() const; size_t countMissingBlock() const; bool hasBlock(size_t blockIndex) const; /** * Returns true if all blocks of this piece have been downloaded, otherwise * returns false. */ bool pieceComplete() const; size_t countBlock() const; int32_t getBlockLength(size_t index) const; int32_t getBlockLength() const; size_t getIndex() const { return index_; } void setIndex(size_t index) { index_ = index; } int64_t getLength() const { return length_; } void setLength(int64_t length) { length_ = length; } const unsigned char* getBitfield() const; void setBitfield(const unsigned char* bitfield, size_t len); size_t getBitfieldLength() const; void clearAllBlock(WrDiskCache* diskCache); void setAllBlock(); std::string toString() const; bool isBlockUsed(size_t index) const; // Calculates completed length int64_t getCompletedLength(); void setHashType(const std::string& hashType); // Updates hash value. This function compares begin and private variable // nextBegin_ and only when they are equal, hash is updated eating data and // returns true. Otherwise returns false. bool updateHash(int64_t begin, const unsigned char* data, size_t dataLength); bool isHashCalculated() const; // Returns raw hash value, not hex digest, which is calculated // by updateHash(). Please note that this function returns hash // value only once. Second invocation without updateHash() returns // empty string. std::string getDigest(); void destroyHashContext(); // Returns raw hash value, not hex digest, which is calculated using // cached data and data on disk. std::string getDigestWithWrCache(size_t pieceLength, const std::shared_ptr& adaptor); /** * Loses current bitfield state. */ void reconfigure(int64_t length); void addUser(cuid_t cuid); void removeUser(cuid_t cuid); bool getUsed() const { return !users_.empty(); } bool usedBy(cuid_t cuid) const; bool getUsedBySegment() const { return usedBySegment_; } void setUsedBySegment(bool f) { usedBySegment_ = f; } void initWrCache(WrDiskCache* diskCache, const std::shared_ptr& diskAdaptor); void flushWrCache(WrDiskCache* diskCache); void clearWrCache(WrDiskCache* diskCache); void updateWrCache(WrDiskCache* diskCache, unsigned char* data, size_t offset, size_t len, size_t capacity, int64_t goff); void updateWrCache(WrDiskCache* diskCache, unsigned char* data, size_t offset, size_t len, int64_t goff) { updateWrCache(diskCache, data, offset, len, len, goff); } size_t appendWrCache(WrDiskCache* diskCache, int64_t goff, const unsigned char* data, size_t len); void releaseWrCache(WrDiskCache* diskCache); WrDiskCacheEntry* getWrDiskCacheEntry() const { return wrCache_.get(); } }; } // namespace aria2 #endif // D_PIECE_H