MongoDB爱好者
垂直技术交流平台

MongoDB transport_layer网络传输层模块源码实现三

 

1. 说明

在之前的<<MongoDB网络传输处理源码实现及性能调优-体验内核性能极致设计>>和<<MongoDB transport_layer网络传输层模块源码实现二>>一文中分析了如何阅读百万级大工程源码、Asio网络库实现、线程模型、transport_layer套接字处理及传输层管理子模块、session会话子模块、Ticket数据收发子模块、service_entry_point服务入口点子模块。

本文将继续分析网络传输层模块中service_state_machine服务状态机子模块,即状态机调度子模块的源码实现。

 

2. service_state_machine状态机调度子模块

service_state_machine状态机处理模块主要复杂一次完整请求的状态转换,确保请求可以按照指定的流程顺利进行,最终实现客户端的正常MongoDB访问。该模块核心代码实现主要由以下三个源码文件实现(test为测试相关,可以忽略):

2.1 核心代码实现


在service_entry_point服务入口点子模块分析中,当接收到一个新的链接后,在ServiceEntryPointImpl::startSession(…)回调函数中会构造一个ServiceStateMachine ssm类,从而实现了新链接、session、ssm的一一映射关系。其中,ServiceStateMachine 类实现对ThreadGuard(线程守护)有较多的依赖,因此本文从这两个类核心代码实现来分析整个状态机调度模块的内部设计细节。

2.1.1 ThreadGuard类核心代码实现

ThreadGuard也就是”线程守护”类,该类主要用于工作线程名的管理维护、ssm归属管理、该ssm对应session链接的回收处理等。该类核心成员及接口实现如下:

class ServiceStateMachine::ThreadGuard {  
  ......  
public:  
  // create a ThreadGuard which will take ownership of the SSM in this thread.  
  //ThreadGuard初始化,标记ssm所有权属于本线程  
  explicit ThreadGuard(ServiceStateMachine* ssm) : _ssm{ssm} {  
      //获取ssm状态机所有权  
      auto owned = _ssm->_owned.compareAndSwap(Ownership::kUnowned, Ownership::kOwned);  
      //如果是一个链接一个线程模型,则条件满足  
      if (owned == Ownership::kStatic) {   
      //一个链接一个线程模型,由于链接始终由同一个线程处理,因此该链接对应的ssm始终归属于同一个线程  
          dassert(haveClient());  
          dassert(Client::getCurrent() == _ssm->_dbClientPtr);  
          //标识归属权已确定  
          _haveTakenOwnership = true;  
          return;  
      }  
        ......
      //adaptive 动态线程模式走下面的模式  
 
      //把当前线程的线程名零时保存起来  
      auto oldThreadName = getThreadName();   
      //ssm保存的线程名和当前线程名不同  
      if (oldThreadName != _ssm->_threadName) {  
          //即将修改线程名了,把修改前的线程名保存到_oldThreadName  
          _ssm->_oldThreadName = getThreadName().toString();  
          //把运行本ssm状态机的线程名改为conn-x线程  
          setThreadName(_ssm->_threadName); //把当前线程改名为_threadName  
      }  
 
      //设置本线程对应client信息,一个链接对应一个client,标识本client当前归属于本线程处理  
      Client::setCurrent(std::move(_ssm->_dbClient));  
        //本状态机ssm所有权有了,归属于运行本ssm的线程  
      _haveTakenOwnership = true;  
  }  
  ......  
  //重新赋值  
  ThreadGuard& operator=(ThreadGuard&& other) {  
      if (this != &other) {  
          _ssm = other._ssm;  
          _haveTakenOwnership = other._haveTakenOwnership;  
          //原来的other所有权失效  
          other._haveTakenOwnership = false;  
      }  
  //返回  
      return *this;  
  };  
 
  //析构函数  
  ~ThreadGuard() {  
  //ssm所有权已确定,则析构的时候,调用release处理,恢复线程原有线程名  
      if (_haveTakenOwnership)  
          release();  
  }  
 
  //一个链接一个线程模型,标记_owned为kStatic,也就是线程名永远不会改变  
  void markStaticOwnership() {  
      dassert(static_cast<bool>(*this));  
      _ssm->_owned.store(Ownership::kStatic);  
  }  
 
  //恢复原有线程名,同时把client信息从调度线程归还给状态机  
  void release() {  
        auto owned = _ssm->_owned.load();  
        //adaptive异步线程池模式满足if条件,表示SSM固定归属于某个线程  
      if (owned != Ownership::kStatic) {  
      //本线程拥有currentClient信息,于是把它归还给SSM状态机  
          if (haveClient()) {  
              _ssm->_dbClient = Client::releaseCurrent();  
          }  
            //恢复到以前的线程名  
          if (!_ssm->_oldThreadName.empty()) {  
              //恢复到老线程名  
              setThreadName(_ssm->_oldThreadName);   
          }  
      }  
      //状态机状态进入end,则调用对应回收hook处理  
      if (_ssm->state() == State::Ended) {  
          //链接关闭的回收处理 ServiceStateMachine::setCleanupHook  
          auto cleanupHook = std::move(_ssm->_cleanupHook);  
          if (cleanupHook)  
              cleanupHook();  
          return;  
      }  
 
      //归属权失效  
      _haveTakenOwnership = false;  
      //归属状态变为未知  
      if (owned == Ownership::kOwned) {  
          _ssm->_owned.store(Ownership::kUnowned);  
      }  
  }  
 
private:  
  //本线程守护当前对应的ssm  
  ServiceStateMachine* _ssm;  
  //默认false,标识该状态机ssm不归属于任何线程  
  bool _haveTakenOwnership = false;  
}

从上面的代码分析可以看出线程守护类作用比较明确,就是守护当前线程的归属状态,并记录状态机ssm不同状态变化前后的线程名。此外,状态机ssm对应的session链接如果进入end状态,则该链接的资源回收释放也由该类完成。

查看mongod或者mongos实例,如果启动实例的时候配置了”serviceExecutor: adaptive”会发现这些进程下面有很多线程名为”conn-x”和”worker-x”线程,同时同一个线程线程名可能发生改变,这个过程就是由ThreadGuard线程守护类来实现。synchronous一个链接一个线程模型只有”conn-x”线程,adaptive线程模型将同时存在有线程名为”conn-x”和”worker-x”的线程,具体原理讲在后面继续分析,如下图:

说明:synchronous线程模式对应worker初始线程名为”conn-x”,adaptive线程模型对应worker初始线程名为”worker-x”。

ThreadGuard线程守护类和状态机ssm(service_state_machine)关联紧密,客户端请求处理的内部ssm状态转换也和该类密切关联,请看后续分析。

 

2.1.2 ServiceStateMachine 类核心代码实现


service_state_machine状态机处理模块核心代码实现通过ServiceStateMachine类完成,该类的核心结构成员和函数接口如下:

类核心成员功能说明如下表:

成员名 功能说明
_state SSM状态机当前所处状态
_sep 服务入口,mongod对应ServiceEntryPointMongod;mongos对应 ServiceEntryPointMongos
_transportMode synchronous及adaptive模式,也就是线程模型是一个链接一个线程还是动态线程池
_serviceContext 服务上下文,mongod和mongos分别对应:ServiceContextMongoD、ServiceContextNoop
_sessionHandle 本ssm对应的session信息,默认对应ASIOSession
_dbClient 也就是本次请求对应的客户端信息
_dbClientPtr 指向上面的_dbClient
_threadName SSM所属线程的线程名,初始化为”conn-n”,adaptive线程模型一次请求中的不同状态会修改线程名
_oldThreadName 修改线程名之前的线程名称
_cleanupHook session链接回收处理
_inMessage 接收处理的message信息 一个完整的报文就记录在该msg中
_owned 默认初始化kUnowned,标识本SSM状态机处于非活跃状态,主要用来判断是否需要在状态转换中跟新线程名,只对动态线程模型生效

我们知道,链接、session、SSM状态机一一对应,他们也拥有对应的归属权,这里的归属权指的就是当前SSM归属于那个线程,也就是当前SSM状态机调度模块由那个线程实现。归属权通过Ownership类标记,该类保护如下几种状态,每种状态功能说明如下:

归属码 功能说明
kUnowned 未知状态,也就是SSM当前不归属与具体的线程
kOwned 针对adaptive线程模型,表示当前SSM状态转换处理由具体的线程负责,也就是归属权明确
kStatic 针对synchronous线程模型,也就是一个链接一个线程模型。由于一个链接始终由同一个线程处理,因此该链接对应SSM也就始终归属于同一个线程,整个运行状态不会改变。

Mongodb服务端接收到客户端请求后的数据接收、协议解析、从db层获取数据、发送数据给客户端都是通过SSM状态机进行有序的状态转换处理,SSM调度处理过程中保护多个状态,每个状态对应一个状态码,具体状态码及其功能说明如下表所示:

状态码 功能说明
Created ServiceStateMachine::ServiceStateMachine()构造函数初始状态
Source 下面两种情况会进入该状态:1. 新连接到来第一次进入SSM处理;2. 客户端请求已完成(已经发送DB获取的数据给客户端),等待调度进行该链接的下一轮请求
SourceWait 等待获取客户端的数据
Process 接收到一个完整MongoDB报文后进入该状态,开始进行协议解析、db层数据访问
SinkWait 等待数据发送成功,发送数据给客户端过程中
EndSession 接收或者发送数据异常、链接关闭,session对应客户端,同时进入Ended状态
Ended session回收处理,进入EndSession后立马通过_cleanupSession进入该状态

以上是SSM处理请求过程的状态码信息,状态转换的具体实现过程请参考后面的核心代码分析。listerner线程接收到新的客户端链接后会调用通过service_entry_point服务入口点子模块的ssm->start()接口进入SSM状态机调度模块,该接口相关的源码实现如下:

ServiceStateMachine::start()接口调用ServiceStateMachine::scheduleNextWithGuard()来启动状态机运行。scheduleNextWithGuard()接口最核心的作用就是调用service_executor服务运行子模块(线程模型子模块)的schedule()接口来把状态机调度任务入队到ASIO网络库的一个全局队列(adaptive动态线程模型),如果是一个链接一个线程模型,则任务入队到线程级私有队列。

adaptive线程模型,任务入队以及工作线程调度任务执行的流程将在后续的线程模型子模块中分析,也可以参考:<<MongoDB网络传输处理源码实现及性能调优-体验内核性能极致设计>>

此外,scheduleNextWithGuard()入队到全局队列的任务就是本模块后面需要分析的SSM状态机任务,这些task任务通过本函数接口的func (…)进行封装,然后通过线程模型子模块入队到一个全局队列。Func(…)这个task任务中会直接调用runNextInGuard()接口来进行状态转换处理,该接口也就是入队到ASIO全局队列的任务,核心代码功能如下:

从上面的代码实现可以看出,真正入队到全局队列中的任务类型只有三个,分别是:

1)    接收MongoDB数据的task任务,简称为readTask。

2)    接收到一个完整MongoDB数据后的后续处理(包括协议解析、命令处理、DB获取数据、发送数据给客户端等),简称为dealTask。

3)    接收或者发送数据异常、链接关闭等引起的后续资源释放,简称为cleanTask。

下面针对这三种task任务核心代码实现进行分析:

readTask任务核心代码实现

readTask任务核心代码实现由_sourceMessage()接口实现,具体代码如下:

SSM调度的第一个任务就是readTask任务,从上面的源码分析可以看出,该任务就是通过ticket数据分发模块从ASIO网络库读取一个完整长度的MongoDB报文,然后执行sourceCallback回调。进入该回调函数后,即刻设置SSM状态为State::Process状态,然后调用scheduleNextWithGuard(…)把dealTask任务入队到ASIO的全局队列(adaptive线程模型),或者入队到线程级私有队列(synchronous线程模型)等待worker线程调度执行。

这里有个细节,在把dealTask入队的时候,携带了kMayRecurse标记,该标记标识该任务可以递归调用,也就是该任务可以由当前线程继续执行,这样也就可以保证同一个请求的taskRead任务和dealTask任务由同一个线程处理。任务递归调度,可以参考后面的线程模型子模块源码实现。

dealTask任务核心代码实现

当读取到一个完整长度的MongoDB报文后,就会把dealTask任务入队到全局队列,然后由worker线程调度执行该task任务。dealTask任务的核心代码实现如下:

readTask通过ticket数据分发子模块读取一个完整长度的MongoDB报文后,开始dealTask任务逻辑,该任务也就是processMessage(…)。该接口中核心实现就是调用mongod和mongos实例对应的服务入口类的handleRequest(…)接口来完成后续的command命令处理、DB层数据访问等,访问到的数据存储到DbResponse中,最终在通过sinkMessage(…)把数据发送出去。

真正的MongoDB内部处理流程实际上就是通过该dealTask任务完成,该任务也是处理整个请求中资源耗费最重的一个环节。在该task任务中,当数据成功发送给客户端后,该session链接对应的SSM状态机进入State::Source状态,继续等待worker线程调度来完成后续该链接的新请求。

cleanTask任务

在数据读写过程、客户端链接关闭、访问DB数据层等任何一个环节异常,则会进入State::EndSession状态。该状态对应得任务实现相对比较简单,具体代码实现如下:

进入该状态后直接由本线程进行session资源回收和client资源释放处理,而无需状态机调度worker线程来回收。

 

2.2 关于worker线程名和guardthread线程守护类


前面得分析我们知道,当线程模型为adaptive动态线程模型的时候,mongod和mongos实例对应的子线程中有很多名为“conn-xx”和”worker-xx”的线程,而且同一个线程可能一会儿线程名为“conn-xx”,下一次又变为了”worker-xx”。这个线程名的初始命名和线程名更改与ServiceStateMachine状态机调度类、guardthread线程守护类、worker线程模型等都有关系。

Worker线程由ServiceExecutor线程模型子模块创建,请参考后续线程模型子模块相关章节。默认初始化线程名为”conn-x”,初始化代码实现如下:

//ServiceStateMachine::create调用,ServiceStateMachine类初始化构造  
ServiceStateMachine::ServiceStateMachine(...)  
  ......  
  //线程名初始化:conn-xx,xx代码session id  
  _threadName{str::stream() << "conn-" << _session()->id()} {}   
}  
 
class Session {  
  ......  
  //sessionID,自增生成  
  const Id _id;  
}  
 
//全局sessionIdCounter计数器,初始化为0  
AtomicUInt64 sessionIdCounter(0);  
 
//session id自增  
Session::Session() : _id(sessionIdCounter.addAndFetch(1)) {}

SSM状态处理过程中,会把一个完整的请求过程 = readTask任务 + dealTask任务,这两个任务都是通过SSM状态机和ServiceExecutor线程模型子模块的worker线程配合调度完成,在任务处理过程中处理同一个任务的线程可能会有多次线程名更改,这个就是结合guardthread线程守护类来完成,以一个线程名切换更改伪代码实现为例:

worker_thread_run_task(...)  
{  
  //如果是adaptive线程模型,当前worker线程名为"worker-xx"  
  print(threadName)  
  //业务逻辑处理1  
  ......  
     
  //初始化构造ThreadGuard,这里面修改线程名为_ssm->_threadName,也就是"conn-xx",  
  //同时保存原来的线程名"worker-xx"到_ssm->_oldThreadName中  
  ThreadGuard guard(this);  
  //如果是adaptive线程模型,线程名打印内容为"conn-xx"  
  print(threadName)  
  //业务逻辑处理2  
  ......  
  //恢复_ssm->_oldThreadName保存的线程名"worker-xx"  
  guard.release();  
     
  //如果是adaptive线程模型,线程名恢复为"worker-xx"  
  print(threadName)  
}

从上面的伪代码可以看出,adaptive线程模型对应worker线程名为”worker”,在进入ThreadGuard guard(this)流程后,线程名更改为”conn-xx”线程,当guard.release()释放后恢复原有”worker-xx”线程名。

结合前面的SSM状态处理流程,adaptive线程模型可以得到如下总结:底层网络IO数据读写过程,worker线程名会改为”worker-xx”,其他非网络IO的mongodb内部逻辑处理线程名为”conn-xx”。所以,如果查看mongod或者mongos进程所有线程名的时候,如果发现线程名为”worker-xx”,说明当前线程在处理网络IO;如果发现线程名为”conn-xx”,则说明当前线程在处理内部逻辑处理,对于mongod实例可以理解为主要处理磁盘IO。

由于synchronous同步线程模型,同一链接对应的所有客户端请求至始至终都有同一线程处理,所以整个处理线程名不会改变,也没必要修改线程名,整个过程都是”conn-xx”线程名。

2.3 该模块函数接口总结大全


前面分析了主要核心接口源码实现,很多其他接口没有一一列举详细分析,该模块u所有接口功能总结如下,更多接口代码实现详见Mongodb内核源码详细注释分析

类名 函数接口 功能说明
ThreadGuard ThreadGuard(…) 线程守护初始化,把worker线程名改为”conn-xx”
operator=(…) 类赋值
~ThreadGuard() 类析构,析构函数中调用release()接口
operator  bool() 获取所有权标识
markStaticOwnership() synchronous线程模型默认归属权为kStatic
release() 1. 如果ssm状态为Ended,则链接回收处理  2. 恢复worker原来的线程名(adaptive:”worker-xx”)
ServiceStateMachine ServiceStateMachine(…) ServiceStateMachine类初始化构造赋值
_session() 获取当前ssm对应的session信息
_sourceMessage(…) 等待通过ASIO库接收网络IO数据
_sinkMessage(…) 等待通过ASIO库发送网络IO数据
_sourceCallback(…) 接收到一个完整长度mongodb报文的回调处理
_sinkCallback(…) 发送一个完整应答报文后的回调处理
_processMessage(…) 开始内部处理,如协议解析、命令处理、DB层数据访问等
_runNextInGuard(…) SSM状态机调度运行
start(…) 接收到一个新链接通过start()接口启用SSM状态机
_scheduleNextWithGuard(…) readTask和dealTask交由worker线程处理
terminate(…) 调用TransportLayerASIO::end()进行套接字回收处理
setCleanupHook(…) 设置clean hook,也就是回收处理
state() 获取SSM当前状态
_terminateAndLogIfError(…) 打印回收日志并进行terminate回收处理
_cleanupSession(…) session会话回收处理

3. 总结

本文主要分析了service_state_machine状态机子模块,该模块把session对应的客户端请求转换为readTask任务、dealTask任务和cleanTask任务,前两个任务通过worker线程完成调度处理,cleanTask任务在内部处理异常或者链接关闭的时候由本线程直接执行,而不是通过worker线程调度执行。

这三个任务处理过程会分别对应到Created、Source、SourceWait、Process、SinkWait、EndSession、Ended七种状态的一种或者多种,具体详见前面的状态码分析。一个正常的客户端请求状态转换过程如下:

1)    链接刚建立的第一次请求状态转换过程:

Created->Source -> SourceWait -> Process -> SinkWait -> Source

2)    该链接后续的请求状态转换过程:

Source -> SourceWait -> Process -> SinkWait -> Source

此外,SSM状态机调度模块通ServiceStateMachine::_scheduleNextWithGuard(…)接口和线程模型子模块关联起来。SSM通过该接口完成worker线程初始创建、task任务入队处理,下期将分析<<网络线程模型子模块>>详细源码实现。

说明:

该模块更多接口实现细节详见MongoDB内核源码注释:

MongoDB内核源码详细注释分析

更多文章:

常用高并发网络线程模型设计及MongoDB线程模型优化实践

MongoDB网络传输处理源码实现及性能调优-体验内核性能极致设计

OPPO百万级高并发MongoDB集群性能数十倍提升优化实践

MongoDB网络传输层模块源码实现二

作者:杨亚洲

前滴滴出行技术专家,现任OPPO文档数据库MongoDB负责人,负责oppo千万级峰值TPS/十万亿级数据量文档数据库MongoDB内核研发及运维工作,一直专注于分布式缓存、高性能服务端、数据库、中间件等相关研发。Github账号地址:

https://github.com/y123456yz

 

 

 

 

 

 

 

 

 

MongoDB中文手册翻译正在进行中,欢迎更多朋友在自己的空闲时间学习并进行文档翻译,您的翻译将由社区专家进行审阅,并拥有署名权更新到中文用户手册和发布到社区微信内容平台。点击下方图片即可领取翻译任务——

 

更多问题可以添加社区助理小芒果微信(mongoingcom)咨询,进入社区微信交流群请备注“mongo”。

赞(14)
未经允许不得转载:MongoDB中文社区 » MongoDB transport_layer网络传输层模块源码实现三

评论 抢沙发

评论前必须登录!