1a1bde736SShuo Chen// excerpts from http://code.google.com/p/muduo/ 2a1bde736SShuo Chen// 3a1bde736SShuo Chen// Use of this source code is governed by a BSD-style license 4a1bde736SShuo Chen// that can be found in the License file. 5a1bde736SShuo Chen// 6a1bde736SShuo Chen// Author: Shuo Chen (chenshuo at chenshuo dot com) 7a1bde736SShuo Chen 8a1bde736SShuo Chen#include "Connector.h" 9a1bde736SShuo Chen 10a1bde736SShuo Chen#include "Channel.h" 11a1bde736SShuo Chen#include "EventLoop.h" 12a1bde736SShuo Chen#include "SocketsOps.h" 13a1bde736SShuo Chen 14a1bde736SShuo Chen#include "logging/Logging.h" 15a1bde736SShuo Chen 16a1bde736SShuo Chen#include <boost/bind.hpp> 17a1bde736SShuo Chen 18a1bde736SShuo Chen#include <errno.h> 19a1bde736SShuo Chen 20a1bde736SShuo Chenusing namespace muduo; 21a1bde736SShuo Chen 22a1bde736SShuo Chenconst int Connector::kMaxRetryDelayMs; 23a1bde736SShuo Chen 24a1bde736SShuo ChenConnector::Connector(EventLoop* loop, const InetAddress& serverAddr) 25a1bde736SShuo Chen : loop_(loop), 26a1bde736SShuo Chen serverAddr_(serverAddr), 27a1bde736SShuo Chen connect_(false), 28a1bde736SShuo Chen state_(kDisconnected), 29a1bde736SShuo Chen retryDelayMs_(kInitRetryDelayMs) 30a1bde736SShuo Chen{ 31a1bde736SShuo Chen LOG_DEBUG << "ctor[" << this << "]"; 32a1bde736SShuo Chen} 33a1bde736SShuo Chen 34a1bde736SShuo ChenConnector::~Connector() 35a1bde736SShuo Chen{ 36a1bde736SShuo Chen LOG_DEBUG << "dtor[" << this << "]"; 37a1bde736SShuo Chen loop_->cancel(timerId_); 38a1bde736SShuo Chen assert(!channel_); 39a1bde736SShuo Chen} 40a1bde736SShuo Chen 41a1bde736SShuo Chenvoid Connector::start() 42a1bde736SShuo Chen{ 43a1bde736SShuo Chen connect_ = true; 44a1bde736SShuo Chen loop_->runInLoop(boost::bind(&Connector::startInLoop, this)); // FIXME: unsafe 45a1bde736SShuo Chen} 46a1bde736SShuo Chen 47a1bde736SShuo Chenvoid Connector::startInLoop() 48a1bde736SShuo Chen{ 49a1bde736SShuo Chen loop_->assertInLoopThread(); 50a1bde736SShuo Chen assert(state_ == kDisconnected); 51a1bde736SShuo Chen if (connect_) 52a1bde736SShuo Chen { 53a1bde736SShuo Chen connect(); 54a1bde736SShuo Chen } 55a1bde736SShuo Chen else 56a1bde736SShuo Chen { 57a1bde736SShuo Chen LOG_DEBUG << "do not connect"; 58a1bde736SShuo Chen } 59a1bde736SShuo Chen} 60a1bde736SShuo Chen 61a1bde736SShuo Chenvoid Connector::connect() 62a1bde736SShuo Chen{ 63a1bde736SShuo Chen int sockfd = sockets::createNonblockingOrDie(); 64a1bde736SShuo Chen int ret = sockets::connect(sockfd, serverAddr_.getSockAddrInet()); 65a1bde736SShuo Chen int savedErrno = (ret == 0) ? 0 : errno; 66a1bde736SShuo Chen switch (savedErrno) 67a1bde736SShuo Chen { 68a1bde736SShuo Chen case 0: 69a1bde736SShuo Chen case EINPROGRESS: 70a1bde736SShuo Chen case EINTR: 71a1bde736SShuo Chen case EISCONN: 72a1bde736SShuo Chen connecting(sockfd); 73a1bde736SShuo Chen break; 74a1bde736SShuo Chen 75a1bde736SShuo Chen case EAGAIN: 76a1bde736SShuo Chen case EADDRINUSE: 77a1bde736SShuo Chen case EADDRNOTAVAIL: 78a1bde736SShuo Chen case ECONNREFUSED: 79a1bde736SShuo Chen case ENETUNREACH: 80a1bde736SShuo Chen retry(sockfd); 81a1bde736SShuo Chen break; 82a1bde736SShuo Chen 83a1bde736SShuo Chen case EACCES: 84a1bde736SShuo Chen case EPERM: 85a1bde736SShuo Chen case EAFNOSUPPORT: 86a1bde736SShuo Chen case EALREADY: 87a1bde736SShuo Chen case EBADF: 88a1bde736SShuo Chen case EFAULT: 89a1bde736SShuo Chen case ENOTSOCK: 90a1bde736SShuo Chen LOG_SYSERR << "connect error in Connector::startInLoop " << savedErrno; 91a1bde736SShuo Chen sockets::close(sockfd); 92a1bde736SShuo Chen break; 93a1bde736SShuo Chen 94a1bde736SShuo Chen default: 95a1bde736SShuo Chen LOG_SYSERR << "Unexpected error in Connector::startInLoop " << savedErrno; 96a1bde736SShuo Chen sockets::close(sockfd); 97a1bde736SShuo Chen // connectErrorCallback_(); 98a1bde736SShuo Chen break; 99a1bde736SShuo Chen } 100a1bde736SShuo Chen} 101a1bde736SShuo Chen 102a1bde736SShuo Chenvoid Connector::restart() 103a1bde736SShuo Chen{ 104a1bde736SShuo Chen loop_->assertInLoopThread(); 105a1bde736SShuo Chen setState(kDisconnected); 106a1bde736SShuo Chen retryDelayMs_ = kInitRetryDelayMs; 107a1bde736SShuo Chen connect_ = true; 108a1bde736SShuo Chen startInLoop(); 109a1bde736SShuo Chen} 110a1bde736SShuo Chen 111a1bde736SShuo Chenvoid Connector::stop() 112a1bde736SShuo Chen{ 113a1bde736SShuo Chen connect_ = false; 114fdb5c17cSShuo Chen loop_->cancel(timerId_); 115a1bde736SShuo Chen} 116a1bde736SShuo Chen 117a1bde736SShuo Chenvoid Connector::connecting(int sockfd) 118a1bde736SShuo Chen{ 119a1bde736SShuo Chen setState(kConnecting); 120a1bde736SShuo Chen assert(!channel_); 121a1bde736SShuo Chen channel_.reset(new Channel(loop_, sockfd)); 122a1bde736SShuo Chen channel_->setWriteCallback( 123a1bde736SShuo Chen boost::bind(&Connector::handleWrite, this)); // FIXME: unsafe 124a1bde736SShuo Chen channel_->setErrorCallback( 125a1bde736SShuo Chen boost::bind(&Connector::handleError, this)); // FIXME: unsafe 126a1bde736SShuo Chen 127a1bde736SShuo Chen // channel_->tie(shared_from_this()); is not working, 128a1bde736SShuo Chen // as channel_ is not managed by shared_ptr 129a1bde736SShuo Chen channel_->enableWriting(); 130a1bde736SShuo Chen} 131a1bde736SShuo Chen 132a1bde736SShuo Chenint Connector::removeAndResetChannel() 133a1bde736SShuo Chen{ 134a1bde736SShuo Chen channel_->disableAll(); 135a1bde736SShuo Chen loop_->removeChannel(get_pointer(channel_)); 136a1bde736SShuo Chen int sockfd = channel_->fd(); 137a1bde736SShuo Chen // Can't reset channel_ here, because we are inside Channel::handleEvent 138a1bde736SShuo Chen loop_->queueInLoop(boost::bind(&Connector::resetChannel, this)); // FIXME: unsafe 139a1bde736SShuo Chen return sockfd; 140a1bde736SShuo Chen} 141a1bde736SShuo Chen 142a1bde736SShuo Chenvoid Connector::resetChannel() 143a1bde736SShuo Chen{ 144a1bde736SShuo Chen channel_.reset(); 145a1bde736SShuo Chen} 146a1bde736SShuo Chen 147a1bde736SShuo Chenvoid Connector::handleWrite() 148a1bde736SShuo Chen{ 149a1bde736SShuo Chen LOG_TRACE << "Connector::handleWrite " << state_; 150a1bde736SShuo Chen 151a1bde736SShuo Chen if (state_ == kConnecting) 152a1bde736SShuo Chen { 153a1bde736SShuo Chen int sockfd = removeAndResetChannel(); 154a1bde736SShuo Chen int err = sockets::getSocketError(sockfd); 155a1bde736SShuo Chen if (err) 156a1bde736SShuo Chen { 157a1bde736SShuo Chen LOG_WARN << "Connector::handleWrite - SO_ERROR = " 158a1bde736SShuo Chen << err << " " << strerror_tl(err); 159a1bde736SShuo Chen retry(sockfd); 160a1bde736SShuo Chen } 161a1bde736SShuo Chen else if (sockets::isSelfConnect(sockfd)) 162a1bde736SShuo Chen { 163a1bde736SShuo Chen LOG_WARN << "Connector::handleWrite - Self connect"; 164a1bde736SShuo Chen retry(sockfd); 165a1bde736SShuo Chen } 166a1bde736SShuo Chen else 167a1bde736SShuo Chen { 168a1bde736SShuo Chen setState(kConnected); 169a1bde736SShuo Chen if (connect_) 170a1bde736SShuo Chen { 171a1bde736SShuo Chen newConnectionCallback_(sockfd); 172a1bde736SShuo Chen } 173a1bde736SShuo Chen else 174a1bde736SShuo Chen { 175a1bde736SShuo Chen sockets::close(sockfd); 176a1bde736SShuo Chen } 177a1bde736SShuo Chen } 178a1bde736SShuo Chen } 179a1bde736SShuo Chen else 180a1bde736SShuo Chen { 181a1bde736SShuo Chen // what happened? 182a1bde736SShuo Chen assert(state_ == kDisconnected); 183a1bde736SShuo Chen } 184a1bde736SShuo Chen} 185a1bde736SShuo Chen 186a1bde736SShuo Chenvoid Connector::handleError() 187a1bde736SShuo Chen{ 188a1bde736SShuo Chen LOG_ERROR << "Connector::handleError"; 189a1bde736SShuo Chen assert(state_ == kConnecting); 190a1bde736SShuo Chen 191a1bde736SShuo Chen int sockfd = removeAndResetChannel(); 192a1bde736SShuo Chen int err = sockets::getSocketError(sockfd); 193a1bde736SShuo Chen LOG_TRACE << "SO_ERROR = " << err << " " << strerror_tl(err); 194a1bde736SShuo Chen retry(sockfd); 195a1bde736SShuo Chen} 196a1bde736SShuo Chen 197a1bde736SShuo Chenvoid Connector::retry(int sockfd) 198a1bde736SShuo Chen{ 199a1bde736SShuo Chen sockets::close(sockfd); 200a1bde736SShuo Chen setState(kDisconnected); 201a1bde736SShuo Chen if (connect_) 202a1bde736SShuo Chen { 203a1bde736SShuo Chen LOG_INFO << "Connector::retry - Retry connecting to " 204a1bde736SShuo Chen << serverAddr_.toHostPort() << " in " 205a1bde736SShuo Chen << retryDelayMs_ << " milliseconds. "; 206a1bde736SShuo Chen timerId_ = loop_->runAfter(retryDelayMs_/1000.0, // FIXME: unsafe 207a1bde736SShuo Chen boost::bind(&Connector::startInLoop, this)); 208a1bde736SShuo Chen retryDelayMs_ = std::min(retryDelayMs_ * 2, kMaxRetryDelayMs); 209a1bde736SShuo Chen } 210a1bde736SShuo Chen else 211a1bde736SShuo Chen { 212a1bde736SShuo Chen LOG_DEBUG << "do not connect"; 213a1bde736SShuo Chen } 214a1bde736SShuo Chen} 215a1bde736SShuo Chen 216