/* */ #include "InitiatorMSEHandshakeCommand.h" #include "PeerInitiateConnectionCommand.h" #include "PeerInteractionCommand.h" #include "DownloadEngine.h" #include "DlAbortEx.h" #include "message.h" #include "prefs.h" #include "SocketCore.h" #include "Logger.h" #include "LogFactory.h" #include "Peer.h" #include "PeerConnection.h" #include "BtRuntime.h" #include "PeerStorage.h" #include "PieceStorage.h" #include "Option.h" #include "MSEHandshake.h" #include "ARC4Encryptor.h" #include "RequestGroup.h" #include "DownloadContext.h" #include "bittorrent_helper.h" #include "util.h" #include "fmt.h" #include "array_fun.h" namespace aria2 { InitiatorMSEHandshakeCommand::InitiatorMSEHandshakeCommand (cuid_t cuid, RequestGroup* requestGroup, const SharedHandle& p, DownloadEngine* e, const SharedHandle& btRuntime, const SharedHandle& s) : PeerAbstractCommand(cuid, p, e, s), requestGroup_(requestGroup), btRuntime_(btRuntime), sequence_(INITIATOR_SEND_KEY), mseHandshake_(new MSEHandshake(cuid, s, getOption().get())) { disableReadCheckSocket(); setWriteCheckSocket(getSocket()); setTimeout(getOption()->getAsInt(PREF_PEER_CONNECTION_TIMEOUT)); btRuntime_->increaseConnections(); requestGroup_->increaseNumCommand(); } InitiatorMSEHandshakeCommand::~InitiatorMSEHandshakeCommand() { requestGroup_->decreaseNumCommand(); btRuntime_->decreaseConnections(); delete mseHandshake_; } bool InitiatorMSEHandshakeCommand::executeInternal() { if(mseHandshake_->getWantRead()) { mseHandshake_->read(); } bool done = false; while(!done) { switch(sequence_) { case INITIATOR_SEND_KEY: { if(!getSocket()->isWritable(0)) { getDownloadEngine()->addCommand(this); return false; } setTimeout(getOption()->getAsInt(PREF_BT_TIMEOUT)); mseHandshake_->initEncryptionFacility(true); mseHandshake_->sendPublicKey(); sequence_ = INITIATOR_SEND_KEY_PENDING; break; } case INITIATOR_SEND_KEY_PENDING: if(mseHandshake_->send()) { sequence_ = INITIATOR_WAIT_KEY; } else { done = true; } break; case INITIATOR_WAIT_KEY: { if(mseHandshake_->receivePublicKey()) { mseHandshake_->initCipher (bittorrent::getInfoHash(requestGroup_->getDownloadContext()));; mseHandshake_->sendInitiatorStep2(); sequence_ = INITIATOR_SEND_STEP2_PENDING; } else { done = true; } break; } case INITIATOR_SEND_STEP2_PENDING: if(mseHandshake_->send()) { sequence_ = INITIATOR_FIND_VC_MARKER; } else { done = true; } break; case INITIATOR_FIND_VC_MARKER: { if(mseHandshake_->findInitiatorVCMarker()) { sequence_ = INITIATOR_RECEIVE_PAD_D_LENGTH; } else { done = true; } break; } case INITIATOR_RECEIVE_PAD_D_LENGTH: { if(mseHandshake_->receiveInitiatorCryptoSelectAndPadDLength()) { sequence_ = INITIATOR_RECEIVE_PAD_D; } else { done = true; } break; } case INITIATOR_RECEIVE_PAD_D: { if(mseHandshake_->receivePad()) { SharedHandle peerConnection (new PeerConnection(getCuid(), getPeer(), getSocket())); if(mseHandshake_->getNegotiatedCryptoType() == MSEHandshake::CRYPTO_ARC4){ peerConnection->enableEncryption(mseHandshake_->getEncryptor(), mseHandshake_->getDecryptor()); size_t buflen = mseHandshake_->getBufferLength(); mseHandshake_->getDecryptor()->encrypt(buflen, mseHandshake_->getBuffer(), mseHandshake_->getBuffer()); peerConnection->presetBuffer(mseHandshake_->getBuffer(), buflen); } else { peerConnection->presetBuffer(mseHandshake_->getBuffer(), mseHandshake_->getBufferLength()); } PeerInteractionCommand* c = new PeerInteractionCommand (getCuid(), requestGroup_, getPeer(), getDownloadEngine(), btRuntime_, pieceStorage_, peerStorage_, getSocket(), PeerInteractionCommand::INITIATOR_SEND_HANDSHAKE, peerConnection); getDownloadEngine()->addCommand(c); return true; } else { done = true; } break; } } } if(mseHandshake_->getWantRead()) { setReadCheckSocket(getSocket()); } else { disableReadCheckSocket(); } if(mseHandshake_->getWantWrite()) { setWriteCheckSocket(getSocket()); } else { disableWriteCheckSocket(); } getDownloadEngine()->addCommand(this); return false; } void InitiatorMSEHandshakeCommand::tryNewPeer() { if(peerStorage_->isPeerAvailable() && btRuntime_->lessThanEqMinPeers()) { SharedHandle peer = peerStorage_->getUnusedPeer(); peer->usedBy(getDownloadEngine()->newCUID()); PeerInitiateConnectionCommand* command = new PeerInitiateConnectionCommand(peer->usedBy(), requestGroup_, peer, getDownloadEngine(), btRuntime_); command->setPeerStorage(peerStorage_); command->setPieceStorage(pieceStorage_); getDownloadEngine()->addCommand(command); } } bool InitiatorMSEHandshakeCommand::prepareForNextPeer(time_t wait) { if(sequence_ == INITIATOR_SEND_KEY) { // We don't try legacy handshake when connection did not // established. tryNewPeer(); return true; } else if(getOption()->getAsBool(PREF_BT_REQUIRE_CRYPTO)) { A2_LOG_INFO(fmt("CUID#%" PRId64 " - Establishing connection using legacy" " BitTorrent handshake is disabled by preference.", getCuid())); tryNewPeer(); return true; } else { // try legacy BitTorrent handshake A2_LOG_INFO(fmt("CUID#%" PRId64 " - Retry using legacy BitTorrent handshake.", getCuid())); PeerInitiateConnectionCommand* command = new PeerInitiateConnectionCommand(getCuid(), requestGroup_, getPeer(), getDownloadEngine(), btRuntime_, false); command->setPeerStorage(peerStorage_); command->setPieceStorage(pieceStorage_); getDownloadEngine()->addCommand(command); return true; } } void InitiatorMSEHandshakeCommand::onAbort() { if(getOption()->getAsBool(PREF_BT_REQUIRE_CRYPTO)) { peerStorage_->returnPeer(getPeer()); } } bool InitiatorMSEHandshakeCommand::exitBeforeExecute() { return btRuntime_->isHalt(); } void InitiatorMSEHandshakeCommand::setPeerStorage (const SharedHandle& peerStorage) { peerStorage_ = peerStorage; } void InitiatorMSEHandshakeCommand::setPieceStorage (const SharedHandle& pieceStorage) { pieceStorage_ = pieceStorage; } const SharedHandle