/* */ #ifndef D_SOCKET_BUFFER_H #define D_SOCKET_BUFFER_H #include "common.h" #include #include #include "SharedHandle.h" namespace aria2 { class SocketCore; struct ProgressUpdate { virtual ~ProgressUpdate() {} virtual void update(size_t length, bool complete) = 0; }; class SocketBuffer { private: class BufEntry { public: BufEntry(ProgressUpdate* progressUpdate) : progressUpdate_(progressUpdate) {} virtual ~BufEntry() { delete progressUpdate_; } virtual ssize_t send (const SharedHandle& socket, size_t offset) = 0; virtual bool final(size_t offset) const = 0; virtual size_t getLength() const = 0; virtual const unsigned char* getData() const = 0; void progressUpdate(size_t length, bool complete) { if(progressUpdate_) { progressUpdate_->update(length, complete); } } private: ProgressUpdate* progressUpdate_; }; class ByteArrayBufEntry:public BufEntry { public: ByteArrayBufEntry(unsigned char* bytes, size_t length, ProgressUpdate* progressUpdate); virtual ~ByteArrayBufEntry(); virtual ssize_t send (const SharedHandle& socket, size_t offset); virtual bool final(size_t offset) const; virtual size_t getLength() const; virtual const unsigned char* getData() const; private: unsigned char* bytes_; size_t length_; }; class StringBufEntry:public BufEntry { public: StringBufEntry(const std::string& s, ProgressUpdate* progressUpdate); StringBufEntry(); virtual ssize_t send (const SharedHandle& socket, size_t offset); virtual bool final(size_t offset) const; virtual size_t getLength() const; virtual const unsigned char* getData() const; void swap(std::string& s); private: std::string str_; }; SharedHandle socket_; std::deque > bufq_; // Offset of data in bufq_[0]. SocketBuffer tries to send bufq_[0], // but it cannot always send whole data. In this case, offset points // to the data to be sent in the next send() call. size_t offset_; public: SocketBuffer(const SharedHandle& socket); ~SocketBuffer(); // Don't allow copying SocketBuffer(const SocketBuffer&); SocketBuffer& operator=(const SocketBuffer&); // Feeds data pointered by bytes with length len into queue. This // object gets ownership of bytes, so caller must not delete or // later bytes after this call. This function doesn't send data. If // progressUpdate is not null, its update() function will be called // each time the data is sent. It will be deleted by this object. It // can be null. void pushBytes(unsigned char* bytes, size_t len, ProgressUpdate* progressUpdate = 0); // Feeds data into queue. This function doesn't send data. If // progressUpdate is not null, its update() function will be called // each time the data is sent. It will be deleted by this object. It // can be null. void pushStr(const std::string& data, ProgressUpdate* progressUpdate = 0); // Sends data in queue. Returns the number of bytes sent. ssize_t send(); // Returns true if queue is empty. bool sendBufferIsEmpty() const; size_t getBufferEntrySize() const { return bufq_.size(); } }; } // namespace aria2 #endif // D_SOCKET_BUFFER_H