/* */ #include "PeerMessageUtil.h" #include "DlAbortEx.h" #include "a2netcompat.h" #include #include namespace aria2 { uint8_t PeerMessageUtil::getId(const unsigned char* msg) { return msg[0]; } uint32_t PeerMessageUtil::getIntParam(const unsigned char* msg, size_t pos) { uint32_t nParam; memcpy(&nParam, msg+pos, sizeof(nParam)); return ntohl(nParam); } uint16_t PeerMessageUtil::getShortIntParam(const unsigned char* msg, size_t pos) { uint16_t nParam; memcpy(&nParam, msg+pos, sizeof(nParam)); return ntohs(nParam); } void PeerMessageUtil::checkIndex(size_t index, size_t pieces) { if(!(index < pieces)) { throw new DlAbortEx("Invalid index: %zu", index); } } void PeerMessageUtil::checkBegin(uint32_t begin, size_t pieceLength) { if(!(begin < pieceLength)) { throw new DlAbortEx("Invalid begin: %u", begin); } } void PeerMessageUtil::checkLength(size_t length) { if(length > MAX_BLOCK_LENGTH) { throw new DlAbortEx("Length too long: %zu > %uKB", length, MAX_BLOCK_LENGTH/1024); } if(length == 0) { throw new DlAbortEx("Invalid length: %zu", length); } } void PeerMessageUtil::checkRange(uint32_t begin, size_t length, size_t pieceLength) { if(!(0 < length)) { throw new DlAbortEx("Invalid range: begin=%u, length=%zu", begin, length); } uint32_t end = begin+length; if(!(end <= pieceLength)) { throw new DlAbortEx("Invalid range: begin=%u, length=%zu", begin, length); } } void PeerMessageUtil::checkBitfield(const unsigned char* bitfield, size_t bitfieldLength, size_t pieces) { if(!(bitfieldLength == (pieces+7)/8)) { throw new DlAbortEx("Invalid bitfield length: %zu", bitfieldLength); } char lastbyte = bitfield[bitfieldLength-1]; for(size_t i = 0; i < 8-pieces%8 && pieces%8 != 0; ++i) { if(!(((lastbyte >> i) & 1) == 0)) { throw new DlAbortEx("Invalid bitfield"); } } } void PeerMessageUtil::setIntParam(unsigned char* dest, uint32_t param) { uint32_t nParam = htonl(param); memcpy(dest, &nParam, sizeof(nParam)); } void PeerMessageUtil::setShortIntParam(unsigned char* dest, uint16_t param) { uint16_t nParam = htons(param); memcpy(dest, &nParam, sizeof(nParam)); } void PeerMessageUtil::createPeerMessageString(unsigned char* msg, size_t msgLength, size_t payloadLength, uint8_t messageId) { assert(msgLength >= 5); memset(msg, 0, msgLength); setIntParam(msg, payloadLength); msg[4] = messageId; } bool PeerMessageUtil::createcompact(unsigned char* compact, const std::string& addr, uint16_t port) { struct addrinfo hints; struct addrinfo* res; memset(&hints, 0, sizeof(hints)); hints.ai_family = AF_INET; // since compact peer format is ipv4 only. hints.ai_flags = AI_NUMERICHOST; if(getaddrinfo(addr.c_str(), 0, &hints, &res)) { return false; } struct sockaddr_in* in = reinterpret_cast(res->ai_addr); uint32_t* addrp = (uint32_t*)compact; *addrp = in->sin_addr.s_addr; uint16_t* portp = (uint16_t*)(compact+4); *portp = htons(port); freeaddrinfo(res); return true; } std::pair PeerMessageUtil::unpackcompact(const unsigned char* compact) { struct sockaddr_in in; memset(&in, 0, sizeof(in)); in.sin_family = AF_INET; in.sin_addr.s_addr = *reinterpret_cast(compact); in.sin_port = 0; char host[NI_MAXHOST]; int s; s = getnameinfo(reinterpret_cast(&in), sizeof(in), host, NI_MAXHOST, 0, NI_MAXSERV, NI_NUMERICHOST); if(s) { return std::pair(); } uint16_t port = ntohs(*(uint16_t*)(compact+sizeof(uint32_t))); return std::pair(host, port); } } // namespace aria2