/* */ #include "PeerMessageUtil.h" #include "DlAbortEx.h" #include "a2netcompat.h" #include #include namespace aria2 { int8_t PeerMessageUtil::getId(const unsigned char* msg) { return msg[0]; } int32_t PeerMessageUtil::getIntParam(const unsigned char* msg, int32_t offset) { int32_t nParam; memcpy(&nParam, msg+offset, sizeof(int32_t)); return ntohl(nParam); } int16_t PeerMessageUtil::getShortIntParam(const unsigned char* msg, int32_t offset) { int16_t nParam; memcpy(&nParam, msg+offset, sizeof(int16_t)); return ntohs(nParam); } void PeerMessageUtil::checkIndex(int32_t index, int32_t pieces) { if(!(0 <= index && index < (int32_t)pieces)) { throw new DlAbortEx("Invalid index: %d", index); } } void PeerMessageUtil::checkBegin(int32_t begin, int32_t pieceLength) { if(!(0 <= begin && begin < (int32_t)pieceLength)) { throw new DlAbortEx("Invalid begin: %d", begin); } } void PeerMessageUtil::checkLength(int32_t length) { if(length > MAX_BLOCK_LENGTH) { throw new DlAbortEx("Length too long: %d > %dKB", length, MAX_BLOCK_LENGTH/1024); } if(length <= 0) { throw new DlAbortEx("Invalid length: %d", length); } } void PeerMessageUtil::checkRange(int32_t begin, int32_t length, int32_t pieceLength) { if(!(0 <= begin && 0 < length)) { throw new DlAbortEx("Invalid range: begin=%d, length=%d", begin, length); } int32_t end = begin+length; if(!(0 < end && end <= pieceLength)) { throw new DlAbortEx("Invalid range: begin=%d, length=%d", begin, length); } } void PeerMessageUtil::checkBitfield(const unsigned char* bitfield, int32_t bitfieldLength, int32_t pieces) { if(!(bitfieldLength == (pieces+7)/8)) { throw new DlAbortEx("Invalid bitfield length: %d", bitfieldLength); } char lastbyte = bitfield[bitfieldLength-1]; for(int32_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, int32_t param) { int32_t nParam = htonl(param); memcpy(dest, &nParam, sizeof(int32_t)); } void PeerMessageUtil::setShortIntParam(unsigned char* dest, int16_t param) { int16_t nParam = htons(param); memcpy(dest, &nParam, sizeof(int16_t)); } void PeerMessageUtil::createPeerMessageString(unsigned char* msg, int32_t msgLength, int32_t payloadLength, int8_t messageId) { assert(msgLength >= 5); memset(msg, 0, msgLength); setIntParam(msg, payloadLength); msg[4] = messageId; } bool PeerMessageUtil::createcompact(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 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