1354280cfSShuo Chen// excerpts from http://code.google.com/p/muduo/
2354280cfSShuo Chen//
3354280cfSShuo Chen// Use of this source code is governed by a BSD-style license
4354280cfSShuo Chen// that can be found in the License file.
5354280cfSShuo Chen//
6354280cfSShuo Chen// Author: Shuo Chen (chenshuo at chenshuo dot com)
7354280cfSShuo Chen
8354280cfSShuo Chen#include "TcpServer.h"
9354280cfSShuo Chen
10354280cfSShuo Chen#include "logging/Logging.h"
11354280cfSShuo Chen#include "Acceptor.h"
12354280cfSShuo Chen#include "EventLoop.h"
13354280cfSShuo Chen#include "EventLoopThreadPool.h"
14354280cfSShuo Chen#include "SocketsOps.h"
15354280cfSShuo Chen
16354280cfSShuo Chen#include <boost/bind.hpp>
17354280cfSShuo Chen
18354280cfSShuo Chen#include <stdio.h>  // snprintf
19354280cfSShuo Chen
20354280cfSShuo Chenusing namespace muduo;
21354280cfSShuo Chen
22354280cfSShuo ChenTcpServer::TcpServer(EventLoop* loop, const InetAddress& listenAddr)
23354280cfSShuo Chen  : loop_(CHECK_NOTNULL(loop)),
24354280cfSShuo Chen    name_(listenAddr.toHostPort()),
25354280cfSShuo Chen    acceptor_(new Acceptor(loop, listenAddr)),
26354280cfSShuo Chen    threadPool_(new EventLoopThreadPool(loop)),
27354280cfSShuo Chen    started_(false),
28354280cfSShuo Chen    nextConnId_(1)
29354280cfSShuo Chen{
30354280cfSShuo Chen  acceptor_->setNewConnectionCallback(
31354280cfSShuo Chen      boost::bind(&TcpServer::newConnection, this, _1, _2));
32354280cfSShuo Chen}
33354280cfSShuo Chen
34354280cfSShuo ChenTcpServer::~TcpServer()
35354280cfSShuo Chen{
36354280cfSShuo Chen}
37354280cfSShuo Chen
38354280cfSShuo Chenvoid TcpServer::setThreadNum(int numThreads)
39354280cfSShuo Chen{
40354280cfSShuo Chen  assert(0 <= numThreads);
41354280cfSShuo Chen  threadPool_->setThreadNum(numThreads);
42354280cfSShuo Chen}
43354280cfSShuo Chen
44354280cfSShuo Chenvoid TcpServer::start()
45354280cfSShuo Chen{
46354280cfSShuo Chen  if (!started_)
47354280cfSShuo Chen  {
48354280cfSShuo Chen    started_ = true;
49354280cfSShuo Chen    threadPool_->start();
50354280cfSShuo Chen  }
51354280cfSShuo Chen
52354280cfSShuo Chen  if (!acceptor_->listenning())
53354280cfSShuo Chen  {
54354280cfSShuo Chen    loop_->runInLoop(
55354280cfSShuo Chen        boost::bind(&Acceptor::listen, get_pointer(acceptor_)));
56354280cfSShuo Chen  }
57354280cfSShuo Chen}
58354280cfSShuo Chen
59354280cfSShuo Chenvoid TcpServer::newConnection(int sockfd, const InetAddress& peerAddr)
60354280cfSShuo Chen{
61354280cfSShuo Chen  loop_->assertInLoopThread();
62354280cfSShuo Chen  char buf[32];
63354280cfSShuo Chen  snprintf(buf, sizeof buf, "#%d", nextConnId_);
64354280cfSShuo Chen  ++nextConnId_;
65354280cfSShuo Chen  std::string connName = name_ + buf;
66354280cfSShuo Chen
67354280cfSShuo Chen  LOG_INFO << "TcpServer::newConnection [" << name_
68354280cfSShuo Chen           << "] - new connection [" << connName
69354280cfSShuo Chen           << "] from " << peerAddr.toHostPort();
70354280cfSShuo Chen  InetAddress localAddr(sockets::getLocalAddr(sockfd));
71354280cfSShuo Chen  // FIXME poll with zero timeout to double confirm the new connection
72354280cfSShuo Chen  EventLoop* ioLoop = threadPool_->getNextLoop();
73354280cfSShuo Chen  TcpConnectionPtr conn(
74354280cfSShuo Chen      new TcpConnection(ioLoop, connName, sockfd, localAddr, peerAddr));
75354280cfSShuo Chen  connections_[connName] = conn;
76354280cfSShuo Chen  conn->setConnectionCallback(connectionCallback_);
77354280cfSShuo Chen  conn->setMessageCallback(messageCallback_);
78354280cfSShuo Chen  conn->setWriteCompleteCallback(writeCompleteCallback_);
79354280cfSShuo Chen  conn->setCloseCallback(
80354280cfSShuo Chen      boost::bind(&TcpServer::removeConnection, this, _1)); // FIXME: unsafe
81354280cfSShuo Chen  ioLoop->runInLoop(boost::bind(&TcpConnection::connectEstablished, conn));
82354280cfSShuo Chen}
83354280cfSShuo Chen
84354280cfSShuo Chenvoid TcpServer::removeConnection(const TcpConnectionPtr& conn)
85354280cfSShuo Chen{
86354280cfSShuo Chen  // FIXME: unsafe
87354280cfSShuo Chen  loop_->runInLoop(boost::bind(&TcpServer::removeConnectionInLoop, this, conn));
88354280cfSShuo Chen}
89354280cfSShuo Chen
90354280cfSShuo Chenvoid TcpServer::removeConnectionInLoop(const TcpConnectionPtr& conn)
91354280cfSShuo Chen{
92354280cfSShuo Chen  loop_->assertInLoopThread();
93354280cfSShuo Chen  LOG_INFO << "TcpServer::removeConnectionInLoop [" << name_
94354280cfSShuo Chen           << "] - connection " << conn->name();
95354280cfSShuo Chen  size_t n = connections_.erase(conn->name());
96354280cfSShuo Chen  assert(n == 1); (void)n;
97354280cfSShuo Chen  EventLoop* ioLoop = conn->getLoop();
98354280cfSShuo Chen  ioLoop->queueInLoop(
99354280cfSShuo Chen      boost::bind(&TcpConnection::connectDestroyed, conn));
100354280cfSShuo Chen}
101354280cfSShuo Chen
102