104e5c324SShuo Chen// excerpts from http://code.google.com/p/muduo/
204e5c324SShuo Chen//
304e5c324SShuo Chen// Use of this source code is governed by a BSD-style license
404e5c324SShuo Chen// that can be found in the License file.
504e5c324SShuo Chen//
604e5c324SShuo Chen// Author: Shuo Chen (chenshuo at chenshuo dot com)
704e5c324SShuo Chen
804e5c324SShuo Chen#include "Connector.h"
904e5c324SShuo Chen
1004e5c324SShuo Chen#include "Channel.h"
1104e5c324SShuo Chen#include "EventLoop.h"
1204e5c324SShuo Chen#include "SocketsOps.h"
1304e5c324SShuo Chen
1404e5c324SShuo Chen#include "logging/Logging.h"
1504e5c324SShuo Chen
1604e5c324SShuo Chen#include <boost/bind.hpp>
1704e5c324SShuo Chen
1804e5c324SShuo Chen#include <errno.h>
1904e5c324SShuo Chen
2004e5c324SShuo Chenusing namespace muduo;
2104e5c324SShuo Chen
2204e5c324SShuo Chenconst int Connector::kMaxRetryDelayMs;
2304e5c324SShuo Chen
2404e5c324SShuo ChenConnector::Connector(EventLoop* loop, const InetAddress& serverAddr)
2504e5c324SShuo Chen  : loop_(loop),
2604e5c324SShuo Chen    serverAddr_(serverAddr),
2704e5c324SShuo Chen    connect_(false),
2804e5c324SShuo Chen    state_(kDisconnected),
2904e5c324SShuo Chen    retryDelayMs_(kInitRetryDelayMs)
3004e5c324SShuo Chen{
3104e5c324SShuo Chen  LOG_DEBUG << "ctor[" << this << "]";
3204e5c324SShuo Chen}
3304e5c324SShuo Chen
3404e5c324SShuo ChenConnector::~Connector()
3504e5c324SShuo Chen{
3604e5c324SShuo Chen  LOG_DEBUG << "dtor[" << this << "]";
37f4e8e3d3SShuo Chen  loop_->cancel(timerId_);
3804e5c324SShuo Chen  assert(!channel_);
3904e5c324SShuo Chen}
4004e5c324SShuo Chen
4104e5c324SShuo Chenvoid Connector::start()
4204e5c324SShuo Chen{
4304e5c324SShuo Chen  connect_ = true;
4404e5c324SShuo Chen  loop_->runInLoop(boost::bind(&Connector::startInLoop, this)); // FIXME: unsafe
4504e5c324SShuo Chen}
4604e5c324SShuo Chen
4704e5c324SShuo Chenvoid Connector::startInLoop()
4804e5c324SShuo Chen{
4904e5c324SShuo Chen  loop_->assertInLoopThread();
5004e5c324SShuo Chen  assert(state_ == kDisconnected);
5104e5c324SShuo Chen  if (connect_)
5204e5c324SShuo Chen  {
5304e5c324SShuo Chen    connect();
5404e5c324SShuo Chen  }
5504e5c324SShuo Chen  else
5604e5c324SShuo Chen  {
5704e5c324SShuo Chen    LOG_DEBUG << "do not connect";
5804e5c324SShuo Chen  }
5904e5c324SShuo Chen}
6004e5c324SShuo Chen
6104e5c324SShuo Chenvoid Connector::connect()
6204e5c324SShuo Chen{
6304e5c324SShuo Chen  int sockfd = sockets::createNonblockingOrDie();
6404e5c324SShuo Chen  int ret = sockets::connect(sockfd, serverAddr_.getSockAddrInet());
6504e5c324SShuo Chen  int savedErrno = (ret == 0) ? 0 : errno;
6604e5c324SShuo Chen  switch (savedErrno)
6704e5c324SShuo Chen  {
6804e5c324SShuo Chen    case 0:
6904e5c324SShuo Chen    case EINPROGRESS:
7004e5c324SShuo Chen    case EINTR:
7104e5c324SShuo Chen    case EISCONN:
7204e5c324SShuo Chen      connecting(sockfd);
7304e5c324SShuo Chen      break;
7404e5c324SShuo Chen
7504e5c324SShuo Chen    case EAGAIN:
7604e5c324SShuo Chen    case EADDRINUSE:
7704e5c324SShuo Chen    case EADDRNOTAVAIL:
7804e5c324SShuo Chen    case ECONNREFUSED:
7904e5c324SShuo Chen    case ENETUNREACH:
8004e5c324SShuo Chen      retry(sockfd);
8104e5c324SShuo Chen      break;
8204e5c324SShuo Chen
8304e5c324SShuo Chen    case EACCES:
8404e5c324SShuo Chen    case EPERM:
8504e5c324SShuo Chen    case EAFNOSUPPORT:
8604e5c324SShuo Chen    case EALREADY:
8704e5c324SShuo Chen    case EBADF:
8804e5c324SShuo Chen    case EFAULT:
8904e5c324SShuo Chen    case ENOTSOCK:
9004e5c324SShuo Chen      LOG_SYSERR << "connect error in Connector::startInLoop " << savedErrno;
9104e5c324SShuo Chen      sockets::close(sockfd);
9204e5c324SShuo Chen      break;
9304e5c324SShuo Chen
9404e5c324SShuo Chen    default:
9504e5c324SShuo Chen      LOG_SYSERR << "Unexpected error in Connector::startInLoop " << savedErrno;
9604e5c324SShuo Chen      sockets::close(sockfd);
9704e5c324SShuo Chen      // connectErrorCallback_();
9804e5c324SShuo Chen      break;
9904e5c324SShuo Chen  }
10004e5c324SShuo Chen}
10104e5c324SShuo Chen
10204e5c324SShuo Chenvoid Connector::restart()
10304e5c324SShuo Chen{
10404e5c324SShuo Chen  loop_->assertInLoopThread();
10504e5c324SShuo Chen  setState(kDisconnected);
10604e5c324SShuo Chen  retryDelayMs_ = kInitRetryDelayMs;
10704e5c324SShuo Chen  connect_ = true;
10804e5c324SShuo Chen  startInLoop();
10904e5c324SShuo Chen}
11004e5c324SShuo Chen
11104e5c324SShuo Chenvoid Connector::stop()
11204e5c324SShuo Chen{
11304e5c324SShuo Chen  connect_ = false;
114fdb5c17cSShuo Chen  loop_->cancel(timerId_);
11504e5c324SShuo Chen}
11604e5c324SShuo Chen
11704e5c324SShuo Chenvoid Connector::connecting(int sockfd)
11804e5c324SShuo Chen{
11904e5c324SShuo Chen  setState(kConnecting);
12004e5c324SShuo Chen  assert(!channel_);
12104e5c324SShuo Chen  channel_.reset(new Channel(loop_, sockfd));
12204e5c324SShuo Chen  channel_->setWriteCallback(
12304e5c324SShuo Chen      boost::bind(&Connector::handleWrite, this)); // FIXME: unsafe
12404e5c324SShuo Chen  channel_->setErrorCallback(
12504e5c324SShuo Chen      boost::bind(&Connector::handleError, this)); // FIXME: unsafe
12604e5c324SShuo Chen
12704e5c324SShuo Chen  // channel_->tie(shared_from_this()); is not working,
12804e5c324SShuo Chen  // as channel_ is not managed by shared_ptr
12904e5c324SShuo Chen  channel_->enableWriting();
13004e5c324SShuo Chen}
13104e5c324SShuo Chen
13204e5c324SShuo Chenint Connector::removeAndResetChannel()
13304e5c324SShuo Chen{
13404e5c324SShuo Chen  channel_->disableAll();
13504e5c324SShuo Chen  loop_->removeChannel(get_pointer(channel_));
13604e5c324SShuo Chen  int sockfd = channel_->fd();
13704e5c324SShuo Chen  // Can't reset channel_ here, because we are inside Channel::handleEvent
13804e5c324SShuo Chen  loop_->queueInLoop(boost::bind(&Connector::resetChannel, this)); // FIXME: unsafe
13904e5c324SShuo Chen  return sockfd;
14004e5c324SShuo Chen}
14104e5c324SShuo Chen
14204e5c324SShuo Chenvoid Connector::resetChannel()
14304e5c324SShuo Chen{
14404e5c324SShuo Chen  channel_.reset();
14504e5c324SShuo Chen}
14604e5c324SShuo Chen
14704e5c324SShuo Chenvoid Connector::handleWrite()
14804e5c324SShuo Chen{
14904e5c324SShuo Chen  LOG_TRACE << "Connector::handleWrite " << state_;
15004e5c324SShuo Chen
15104e5c324SShuo Chen  if (state_ == kConnecting)
15204e5c324SShuo Chen  {
15304e5c324SShuo Chen    int sockfd = removeAndResetChannel();
15404e5c324SShuo Chen    int err = sockets::getSocketError(sockfd);
15504e5c324SShuo Chen    if (err)
15604e5c324SShuo Chen    {
15704e5c324SShuo Chen      LOG_WARN << "Connector::handleWrite - SO_ERROR = "
15804e5c324SShuo Chen               << err << " " << strerror_tl(err);
15904e5c324SShuo Chen      retry(sockfd);
16004e5c324SShuo Chen    }
16104e5c324SShuo Chen    else if (sockets::isSelfConnect(sockfd))
16204e5c324SShuo Chen    {
16304e5c324SShuo Chen      LOG_WARN << "Connector::handleWrite - Self connect";
16404e5c324SShuo Chen      retry(sockfd);
16504e5c324SShuo Chen    }
16604e5c324SShuo Chen    else
16704e5c324SShuo Chen    {
16804e5c324SShuo Chen      setState(kConnected);
16904e5c324SShuo Chen      if (connect_)
17004e5c324SShuo Chen      {
17104e5c324SShuo Chen        newConnectionCallback_(sockfd);
17204e5c324SShuo Chen      }
17304e5c324SShuo Chen      else
17404e5c324SShuo Chen      {
17504e5c324SShuo Chen        sockets::close(sockfd);
17604e5c324SShuo Chen      }
17704e5c324SShuo Chen    }
17804e5c324SShuo Chen  }
17904e5c324SShuo Chen  else
18004e5c324SShuo Chen  {
18104e5c324SShuo Chen    // what happened?
18204e5c324SShuo Chen    assert(state_ == kDisconnected);
18304e5c324SShuo Chen  }
18404e5c324SShuo Chen}
18504e5c324SShuo Chen
18604e5c324SShuo Chenvoid Connector::handleError()
18704e5c324SShuo Chen{
18804e5c324SShuo Chen  LOG_ERROR << "Connector::handleError";
18904e5c324SShuo Chen  assert(state_ == kConnecting);
19004e5c324SShuo Chen
19104e5c324SShuo Chen  int sockfd = removeAndResetChannel();
19204e5c324SShuo Chen  int err = sockets::getSocketError(sockfd);
19304e5c324SShuo Chen  LOG_TRACE << "SO_ERROR = " << err << " " << strerror_tl(err);
19404e5c324SShuo Chen  retry(sockfd);
19504e5c324SShuo Chen}
19604e5c324SShuo Chen
19704e5c324SShuo Chenvoid Connector::retry(int sockfd)
19804e5c324SShuo Chen{
19904e5c324SShuo Chen  sockets::close(sockfd);
20004e5c324SShuo Chen  setState(kDisconnected);
20104e5c324SShuo Chen  if (connect_)
20204e5c324SShuo Chen  {
203f4e8e3d3SShuo Chen    LOG_INFO << "Connector::retry - Retry connecting to "
204f4e8e3d3SShuo Chen             << serverAddr_.toHostPort() << " in "
205f4e8e3d3SShuo Chen             << retryDelayMs_ << " milliseconds. ";
206f4e8e3d3SShuo Chen    timerId_ = loop_->runAfter(retryDelayMs_/1000.0,  // FIXME: unsafe
207f4e8e3d3SShuo Chen                               boost::bind(&Connector::startInLoop, this));
20804e5c324SShuo Chen    retryDelayMs_ = std::min(retryDelayMs_ * 2, kMaxRetryDelayMs);
20904e5c324SShuo Chen  }
21004e5c324SShuo Chen  else
21104e5c324SShuo Chen  {
21204e5c324SShuo Chen    LOG_DEBUG << "do not connect";
21304e5c324SShuo Chen  }
21404e5c324SShuo Chen}
21504e5c324SShuo Chen
216