/* */ #ifndef _D_UTIL_H_ #define _D_UTIL_H_ #include "common.h" #include #include #include #include #include #include #include #include #include #include #include #include "SharedHandle.h" #include "IntSequence.h" #include "a2time.h" #include "a2netcompat.h" #include "a2functional.h" namespace aria2 { class Randomizer; class BitfieldMan; class BinaryStream; class FileEntry; #define STRTOLL(X) strtoll(X, reinterpret_cast(0), 10) #define STRTOULL(X) strtoull(X, reinterpret_cast(0), 10) #define START_INDEX(OFFSET, PIECE_LENGTH) ((OFFSET)/(PIECE_LENGTH)) #define END_INDEX(OFFSET, LENGTH, PIECE_LENGTH) (((OFFSET)+(LENGTH)-1)/(PIECE_LENGTH)) #define DIV_FLOOR(X,Y) ((X)/(Y)+((X)%(Y)? 1:0)) #ifdef WORDS_BIGENDIAN inline uint64_t ntoh64(uint64_t x) { return x; } inline uint64_t hton64(uint64_t x) { return x; } #else // !WORDS_BIGENDIAN inline uint64_t byteswap64(uint64_t x) { uint64_t v1 = ntohl(x & 0x00000000ffffffff); uint64_t v2 = ntohl(x >> 32); return (v1 << 32)|v2; } inline uint64_t ntoh64(uint64_t x) { return byteswap64(x); } inline uint64_t hton64(uint64_t x) { return byteswap64(x); } #endif // !WORDS_BIGENDIAN namespace util { void split(std::pair& hp, const std::string& src, char delim); std::pair split(const std::string& src, const std::string& delims); template std::string uitos(T value, bool comma = false) { std::string str; if(value == 0) { str = "0"; return str; } unsigned int count = 0; while(value) { ++count; char digit = value%10+'0'; if(comma && count > 3 && count%3 == 1) { str += ','; } str += digit; value /= 10; } std::reverse(str.begin(), str.end()); return str; } template std::string itos(T value, bool comma = false) { bool flag = false; if(value < 0) { flag = true; value = -value; } std::string str = uitos(value, comma); if(flag) { str.insert(str.begin(), '-'); } return str; } /** * Computes difference in micro-seconds between tv1 and tv2, * assuming tv1 is newer than tv2. * If tv1 is older than tv2, then this method returns 0. */ int64_t difftv(struct timeval tv1, struct timeval tv2); int32_t difftvsec(struct timeval tv1, struct timeval tv2); extern const std::string DEFAULT_TRIM_CHARSET; std::string trim(const std::string& src, const std::string& trimCharset = DEFAULT_TRIM_CHARSET); void trimSelf(std::string& str, const std::string& trimCharset = DEFAULT_TRIM_CHARSET); bool startsWith(const std::string& target, const std::string& part); bool endsWith(const std::string& target, const std::string& part); std::string replace(const std::string& target, const std::string& oldstr, const std::string& newstr); std::string urlencode(const unsigned char* target, size_t len); std::string urlencode(const std::string& target); bool inRFC3986ReservedChars(const char c); bool inRFC3986UnreservedChars(const char c); std::string urldecode(const std::string& target); std::string torrentUrlencode(const unsigned char* target, size_t len); std::string torrentUrlencode(const std::string& target); std::string toHex(const unsigned char* src, size_t len); std::string toHex(const char* src, size_t len); std::string toHex(const std::string& src); // Converts hexadecimal ascii string 'src' into packed binary form and // return the result. If src is not well formed, then empty string is // returned. std::string fromHex(const std::string& src); FILE* openFile(const std::string& filename, const std::string& mode); bool isPowerOf(int num, int base); std::string secfmt(time_t sec); int32_t parseInt(const std::string& s, int32_t base = 10); uint32_t parseUInt(const std::string& s, int base = 10); bool parseUIntNoThrow(uint32_t& result, const std::string& s, int base = 10); int64_t parseLLInt(const std::string& s, int32_t base = 10); uint64_t parseULLInt(const std::string& s, int base = 10); IntSequence parseIntRange(const std::string& src); // Parses string which specifies the range of piece index for higher // priority and appends those indexes into result. The input string // src can contain 2 keywords "head" and "tail". To include both // keywords, they must be separated by comma. "head" means the pieces // where the first byte of each file sits. "tail" means the pieces // where the last byte of each file sits. These keywords can take one // parameter, SIZE. For example, if "head=SIZE" is specified, pieces // in the range of first SIZE bytes of each file get higher // priority. SIZE can include K or M(1K = 1024, 1M = 1024K). // If SIZE is omitted, SIZE=defaultSize is used. // // sample: head=512K,tail=512K void parsePrioritizePieceRange (std::vector& result, const std::string& src, const std::vector >& fileEntries, size_t pieceLength, uint64_t defaultSize = 1048576 /* 1MiB */); // this function temporarily put here std::string getContentDispositionFilename(const std::string& header); std::string randomAlpha(size_t length, const SharedHandle& randomizer); std::string toUpper(const std::string& src); std::string toLower(const std::string& src); bool isNumbersAndDotsNotation(const std::string& name); void setGlobalSignalHandler(int signal, void (*handler)(int), int flags); std::string getHomeDir(); int64_t getRealSize(const std::string& sizeWithUnit); std::string abbrevSize(int64_t size); template void toStream (InputIterator first, InputIterator last, std::ostream& os) { os << _("Files:") << "\n"; os << "idx|path/length" << "\n"; os << "===+===========================================================================" << "\n"; int32_t count = 1; for(; first != last; ++first, ++count) { os << std::setw(3) << count << "|" << (*first)->getPath() << "\n"; os << " |" << util::abbrevSize((*first)->getLength()) << "B (" << util::uitos((*first)->getLength(), true) << ")\n"; os << "---+---------------------------------------------------------------------------" << "\n"; } } void sleep(long seconds); void usleep(long microseconds); bool isNumber(const std::string& what); bool isLowercase(const std::string& what); bool isUppercase(const std::string& what); /** * Converts alphabets to unsigned int, assuming alphabets as a base 26 * integer and 'a' or 'A' is 0. * This function assumes alphabets includes only a-z. * Upper case are allowed but all letters must be upper case. * If overflow occurs, returns 0. */ unsigned int alphaToNum(const std::string& alphabets); void mkdirs(const std::string& dirpath); void convertBitfield(BitfieldMan* dest, const BitfieldMan* src); // binaryStream has to be opened before calling this function. std::string toString(const SharedHandle& binaryStream); #ifdef HAVE_POSIX_MEMALIGN void* allocateAlignedMemory(size_t alignment, size_t size); #endif // HAVE_POSIX_MEMALIGN std::pair getNumericNameInfo(const struct sockaddr* sockaddr, socklen_t len); std::string htmlEscape(const std::string& src); // Joins path element specified in [first, last). If ".." is found, // it eats the previous element if it exists. If "." is found, it // is just ignored and it is not appeared in the result. template std::string joinPath(InputIterator first, InputIterator last) { std::deque elements; for(;first != last; ++first) { if(*first == "..") { if(!elements.empty()) { elements.pop_back(); } } else if(*first == ".") { // do nothing } else { elements.push_back(*first); } } return strjoin(elements.begin(), elements.end(), "/"); } // Parses INDEX=PATH format string. INDEX must be an unsigned // integer. std::map::value_type parseIndexPath(const std::string& line); std::map createIndexPathMap(std::istream& i); /** * Take a string src which is a delimited list and add its elements * into result. result is stored in out. */ template OutputIterator split(const std::string& src, OutputIterator out, const std::string& delims, bool doTrim = false) { std::string::size_type p = 0; while(1) { std::string::size_type np = src.find_first_of(delims, p); if(np == std::string::npos) { std::string term = src.substr(p); if(doTrim) { term = util::trim(term); } if(!term.empty()) { *out = term; ++out; } break; } std::string term = src.substr(p, np-p); if(doTrim) { term = util::trim(term); } p = np+1; if(!term.empty()) { *out = term; ++out; } } return out; } void generateRandomData(unsigned char* data, size_t length); // Saves data to file whose name is filename. If overwrite is true, // existing file is overwritten. Otherwise, this function doesn't do // nothing. If data is saved successfully, return true. Otherwise // returns false. bool saveAs (const std::string& filename, const std::string& data, bool overwrite=false); } // namespace util } // namespace aria2 #endif // _D_UTIL_H_