/* */ #include "PeerListenCommand.h" #include #include #include "DownloadEngine.h" #include "Peer.h" #include "RequestGroupMan.h" #include "RecoverableException.h" #include "message.h" #include "ReceiverMSEHandshakeCommand.h" #include "Logger.h" #include "LogFactory.h" #include "SocketCore.h" #include "SimpleRandomizer.h" #include "util.h" #include "fmt.h" namespace aria2 { PeerListenCommand::PeerListenCommand (cuid_t cuid, DownloadEngine* e, int family) : Command(cuid), e_(e), family_(family) {} PeerListenCommand::~PeerListenCommand() {} bool PeerListenCommand::bindPort(uint16_t& port, SegList& sgl) { socket_.reset(new SocketCore()); std::vector ports; while(sgl.hasNext()) { ports.push_back(sgl.next()); } std::random_shuffle(ports.begin(), ports.end(), *SimpleRandomizer::getInstance().get()); const int ipv = (family_ == AF_INET) ? 4 : 6; for(std::vector::const_iterator i = ports.begin(), eoi = ports.end(); i != eoi; ++i) { port = *i; try { socket_->bind(0, port, family_); socket_->beginListen(); socket_->setNonBlockingMode(); A2_LOG_NOTICE(fmt(_("IPv%d BitTorrent: listening to port %u"), ipv, port)); return true; } catch(RecoverableException& ex) { A2_LOG_ERROR_EX(fmt("IPv%d BitTorrent: failed to bind port %u", ipv, port), ex); socket_->closeConnection(); } } return false; } uint16_t PeerListenCommand::getPort() const { if(!socket_) { return 0; } else { std::pair addr; socket_->getAddrInfo(addr); return addr.second; } } bool PeerListenCommand::execute() { if(e_->isHaltRequested() || e_->getRequestGroupMan()->downloadFinished()) { return true; } for(int i = 0; i < 3 && socket_->isReadable(0); ++i) { SharedHandle peerSocket; try { peerSocket.reset(socket_->acceptConnection()); std::pair peerInfo; peerSocket->getPeerInfo(peerInfo); peerSocket->setNonBlockingMode(); SharedHandle peer(new Peer(peerInfo.first, peerInfo.second, true)); cuid_t cuid = e_->newCUID(); Command* command = new ReceiverMSEHandshakeCommand(cuid, peer, e_, peerSocket); e_->addCommand(command); A2_LOG_DEBUG(fmt("Accepted the connection from %s:%u.", peer->getIPAddress().c_str(), peer->getPort())); A2_LOG_DEBUG(fmt("Added CUID#%" PRId64 " to receive BitTorrent/MSE handshake.", cuid)); } catch(RecoverableException& ex) { A2_LOG_DEBUG_EX(fmt(MSG_ACCEPT_FAILURE, getCuid()), ex); } } e_->addCommand(this); return false; } } // namespace aria2