`
budairenqin
  • 浏览: 201481 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Netty源码细节3--accept(Linux os层 + Netty层代码细节)

阅读更多
转自己的在公司发的文章:

前言

本菜鸟有过几年的网络IO相关经验, java层面netty也一直关注, 最近想对自己所了解的netty做一个系列的笔记, 不光技术水平有限, 写作水平更有限, 难免有错误之处欢迎指正, 共同学习.

上一篇讲了bind, 这篇分析一下accept的细节, 我觉得网络IO相关开发很多时候不能仅仅局限于java层, 尤其从accept开始一个连接诞生了, 什么拥塞控制啊, 滑动窗口啊等等一系列底层的问题可能就开始会渐渐困扰到你了, 这一章尝试先从linux内核的tcp实现开始分析accept

源码来自linux-2.6.11.12, 还参考了[TCP_IP.Architecture,.Design.and.Implementation.in.Linux]一书

linux的代码就不往这里贴了, 一个是太多, 篇幅控制不了(主要代码都在tcp_ipv4.c以及tcp_input.c中), 再一个是本屌丝只会java, os层我解释越多错误就会越多,最终会误导读者.


accept概述

accept属于tcp被动打开环节(被动打开请参考tcp状态变迁图), 被动打开的前提是你这一端listen, listen时创建了一个监听套接字, 专门负责监听, 不负责传输数据.
当一个SYN到达服务器时, linux内核并不会创建sock结构体, 只是创建一个轻量级的request_sock结构体,里面能唯一确定是哪一个客户端发来的SYN的信息.
接着服务端发送SYN + ACK给客户端, 总结下来是两个步骤:
    1.建立request_sock
    2.回复SYN + ACK
接着客户端需要回复ACK, 此时服务端从ACK这个包中取出相应信息去查找之前相同客户端发过来的SYN时候创建的request_sock结构体, 到这里内核才会为这条连接创建真正的重量级sock结构体.
但是sock还只是socket的子集(socket结构体中有一个指针sock * sk), 此时一条连接至少还需要绑定一个fd(文件描述符)才能传输数据, accept系统调用后将绑定一个fd.

accept流程图:



1.tcp_v4_rcv()是传输层报文处理入口, 主要做以下事情:
    a)从报文中获取TCP头部数据, 验证TCP首部检验和
    b)根据TCP首部中的信息来设置TCP控制块中的值,这里要进行字节序的转换
    c)接着会调用__tcp_v4_lookup()

2.__tcp_v4_lookup()用来查找传输控制块sk, 如果未找到则直接给对端发RST(我们java层常看到的connection reset by peer就是RST导致, 很多情况下会给对端发送RST,找不到sk只是RST众多导火索中的一个).

3.接着检查第二步中找到的传输控制块sk, 如果进程没有访问sk, 会接着调用tcp_v4_do_rcv()来正常接收, tcp_v4_do_rcv()是传输层处理TCP段的主入口

4.如果sk->sk_state == TCP_LISTEN, 代表是监听套接字, 则应该处理被动连接(注意下accept的连接就是被动连接)

5.sock *nsk = tcp_v4_hnd_req(sk, skb);
    tcp_v4_hnd_req()处理半连接状态的ACK消息, 这里分两种情况:
        1)tcp_v4_hnd_req()直接返回了nsk并且nsk == sk(这代表现在是第一次握手), 此时沿着上图左边虚线框里的路径继续往下执行.
        2)tcp_v4_hnd_req()里面调用tcp_v4_search_req()根据TCP四元组(源端口、源地址、目的地址)在父传输控制块的散列表中查找相应的连接请求块, 那说明两次握手已完成, 直接调用tcp_check_req()进行三次握手确认.
        此时沿着右边虚线框执行.

**一.先分析左边第一条链路, 也就是处理SYN**
    a)首先是tcp_rcv_state_process(), 除了ESTABLISHED和TIME_WAIT状态外,其他状态下的TCP段处理都由这个函数实现. 如果是处理SYN, 它会调用tcp_v4_conn_request()来处理.
    b)tcp_v4_conn_request()函数里会做如下检查:
        1)SYN queue是不是满了? 如果满了并且没有启用syncookie功能, 则直接丢弃连接
        2)accept queue是不是满了?如果满了并且SYN请求队列中至少有一个握手过程中没有重传,则丢弃
    c)通过了b)中的检查, 说明可以接收并处理请求, 调用tcp_openreq_alloc()先分配一个请求块.
    d)接着调用tcp_parse_options()解析TCP段中的选项
    e)然后初始化好连接请求块后就可以调用tcp_v4_send_synack()像客户端发送SYN + ACK了
    f)最后调用tcp_v4_synq_add()将这个sk加入SYN queue中.
最后注意下其实sk只是一个轻量级的request_sock, 毕竟sock结构体比request_sock大的多, 犯不着三次握手都没建立起来我就建立一个大的结构体.
现在一个sock已经进入SYN queue, 目前的阶段是握手的第二步, 收到SYN并且回复对端SYN + ACK(希望你记得上一章我们讲backlog时提到过的两个队列, SYN queue就是其中一个)

**二.接下来第二条链路, 右边的虚线框**
    a) tcp_v4_search_req()通过TCP四元组查到了对应的请求块, 说明两次握手已经完成, 进行第三次握手确认, 也就是处理ACK.
    b)如果检查第三次握手的ACK段是有效的, 则调用tcp_v4_syn_recv_sock()创建子传输控制块.
    c)tcp_v4_syn_recv_sock()方法里有很多初始化操作
        1)创建子传输控制块,并进行初始化(这里是真正的重量级sock了)
        2)设置路由缓存,确定输出网络接口的特性
        3)根据路由中的路径MTU信息,设置控制块MSS
        4)与本地端口进行绑定
        最后会返回一个真正的重量级sock(注意区别前边提到的sk == request_sock)
    d)接着调用tcp_synq_unlink()将sk从SYN queue中移除, 告别半连接身份
    e)现在通过tcp_acceptq_queue()把这个重量级的sock加入的accept queue, 到此这个TCP连接已经可以被我们的应用层netty拿去玩了.

好吧, 我知道上面的文字中很多东西没有详细展开, 只关注java层的同学可能看着稍微吃力, 下面贴上两个图, 一个tcp三路握手, 一个tcp状态变迁图








第一个图来自耗子叔的[TCP那些事]http://coolshell.cn/articles/11564.html, 三次握手过程

第二个图在网上随便搜的, 来源不清楚了, 不过最终的源头肯定是[TCP/IP详解]一书了, 这是一个TCP状态变迁图, 我们上面分析的accept过程属于被动打开, 可以仔细对照图看一下.图中所有的TCP状态这里不解释了, 篇幅控制不住了, 大家参照[TCP/IP详解]一书.

现在铺垫完了, 开始分析netty的accept过程. 又要拿出第一章(EventLoop)的代码了, 多路复用IO的dispatch:

private static void processSelectedKey(SelectionKey k, AbstractNioChannel ch) {
    final AbstractNioChannel.NioUnsafe unsafe = ch.unsafe();
    // ......
    try {
        int readyOps = k.readyOps();
        // Also check for readOps of 0 to workaround possible JDK bug which may otherwise lead
        // to a spin loop
        if ((readyOps & (SelectionKey.OP_READ | SelectionKey.OP_ACCEPT)) != 0 || readyOps == 0) {
            unsafe.read();
            if (!ch.isOpen()) {
                // Connection already closed - no need to handle write.
                return;
            }
        }
        if ((readyOps & SelectionKey.OP_WRITE) != 0) {
            // Call forceFlush which will also take care of clear the OP_WRITE once there is nothing left to write
            ch.unsafe().forceFlush();
        }
        if ((readyOps & SelectionKey.OP_CONNECT) != 0) {
            // remove OP_CONNECT as otherwise Selector.select(..) will always return without blocking
            // See https://github.com/netty/netty/issues/924
            int ops = k.interestOps();
            ops &= ~SelectionKey.OP_CONNECT;
            k.interestOps(ops);

            unsafe.finishConnect();
        }
    } catch (CancelledKeyException ignored) {
        unsafe.close(unsafe.voidPromise());
    }
}


以后分析read, write等逻辑是都要从这个代码开始, 现在我们只关心OP_ACCEPT, 由前两章的分析我们知道, 这里调用的是NioMessageUnsafe#read()

    public void read() {
        // ......
        final int maxMessagesPerRead = config.getMaxMessagesPerRead();
        final ChannelPipeline pipeline = pipeline();
        boolean closed = false;
        Throwable exception = null;
        try {
            try {
                for (;;) {
                    int localRead = doReadMessages(readBuf);
                    if (localRead == 0) {
                        break;
                    }
                    if (localRead < 0) {
                        closed = true;
                        break;
                    }

                    // stop reading and remove op
                    if (!config.isAutoRead()) {
                        break;
                    }

                    if (readBuf.size() >= maxMessagesPerRead) {
                        break;
                    }
                }
            } catch (Throwable t) {
                exception = t;
            }
            setReadPending(false);
            int size = readBuf.size();
            for (int i = 0; i < size; i ++) {
                pipeline.fireChannelRead(readBuf.get(i));
            }

            readBuf.clear();
            pipeline.fireChannelReadComplete();

            if (exception != null) {
                if (exception instanceof IOException && !(exception instanceof PortUnreachableException)) {
                    closed = !(AbstractNioMessageChannel.this instanceof ServerChannel);
                }
                pipeline.fireExceptionCaught(exception);
            }
            if (closed) {
                if (isOpen()) {
                    close(voidPromise());
                }
            }
        } finally {
            if (!config.isAutoRead() && !isReadPending()) {
                removeReadOp();
            }
        }
    }
}


1. maxMessagesPerRead的默认值在NioMessageUnsafe中为16, 尽可能的一次多accept些连接, 在os层我们提到了accept queue会满, 所以应用层越早拿走accept queue中的连接越好.

2. 接下来重头戏是doReadMessages


protected int doReadMessages(List<Object> buf) throws Exception {
    SocketChannel ch = javaChannel().accept();
    try {
        if (ch != null) {
            buf.add(new NioSocketChannel(this, ch));
            return 1;
        }
    } catch (Throwable ignored) {}
    return 0;
}


    a)javaChannel().accept()会通过accept系统调用从os的accept queue中拿出一个连接并包装成SocketChannel

    b)接着又包装一层netty的NioSocketChannel之后放进buf中.

    c)NioSocketChannel构造方法将SocketChannel感兴趣的事件设置成OP_READ, 并设置成非阻塞模式.


3. 我们回到unsafe#read()方法, 如果每次调用doReadMessages都能拿到一个channel, 那么一直拿到16个以上的channel再跳出循环, 原因在第一点中已经说了.
    如果localRead == 0, 表示此时os 的 accept queue中可能已经没有就绪连接了, 所以也跳出循环.

4. 接下来触发channelRead event:
    pipeline.fireChannelRead(readBuf.get(i));
    channelRead是inbound event, 回想之前pipeline中的顺序(head--> ServerBootstrapAcceptor-->tail), 会调用ServerBootstrapAcceptor的channelRead()


public void channelRead(ChannelHandlerContext ctx, Object msg) {
        final Channel child = (Channel) msg;

        child.pipeline().addLast(childHandler);

        // 设置child options, attrs

        try {
            childGroup.register(child).addListener(new ChannelFutureListener() {
                @Override
                public void operationComplete(ChannelFuture future) throws Exception {
                    if (!future.isSuccess()) {
                        forceClose(child, future.cause());
                    }
                }
            });
        } catch (Throwable t) {
            forceClose(child, t);
        }
    }


前两篇开篇实例有如下代码:

b.childHandler(new ChannelInitializer<SocketChannel>() {
         @Override
         public void initChannel(SocketChannel ch) throws Exception {
             ChannelPipeline p = ch.pipeline();
             p.addLast(new EchoServerHandler());
         }
     });


     1.child.pipeline().addLast(childHandler)就是将这里我们自己用户逻辑相关的handler加入到 channel 的pipeline里面(注意这是worker的pipeline, 前面提到的都是boss 的 pipeline)

     2.设置child options, attrs

     3.接下里从workerGroup中拿出一个workerEventLoop并将channel注册到其中, register()的逻辑和第二篇讲bind时bossEventLoop的注册基本是一样的, 这里我们不再重复讲了.


到这里, 一次accept流程, 就完成了, 现在这个channel就有workerEventLoop来处理读写等事件了.
  • 大小: 55.7 KB
  • 大小: 96.3 KB
  • 大小: 179.8 KB
分享到:
评论
1 楼 dengchang1 2016-11-01  
好文章。 详细看了《Netty源码细节1--IO线程(EventLoop)》《Netty源码细节3--accept》,分析浅显易懂,受益匪浅。大神能把bind部分的分析也分享下不,盼复 :-)

相关推荐

    Netty权威指南-Netty源码

    在深入探讨 Netty 源码之前,我们先了解一下 Netty 的核心概念和架构。 Netty 的核心组件包括:ByteBuf(字节缓冲区)、Channel(通道)、EventLoop(事件循环)、Pipeline(处理链)以及Handler(处理器)。...

    Netty大纲-同步netty专栏

    3. **Netty入门** - **概述**:Netty由JBOSS创始人Jochen Meskel开发,它具有高性能、低延迟、易用性等优点,广泛应用于分布式系统、游戏服务器、RPC框架等领域。 - **HelloWorld**:Netty的基本使用包括服务器端...

    Netty5.0架构剖析和源码解读

    Netty源码分析 ##### 3.1. 服务端创建 Netty是一个高性能的网络应用程序框架,其核心优势在于异步事件驱动的I/O处理机制。接下来我们将深入分析Netty服务端的创建过程。 ###### 3.1.1. 服务端启动辅助类...

    Java版水果管理系统源码-awesome-netty:netty最佳实践

    Java版水果管理系统源码 前言 Netty作为一款高性能的网络通信框架,广泛应用在RPC框架,MQ组件和游戏行业的基础通信, 每款框架都有不同的参数和配置来满足不同的使用场景,也有相应的最佳实践方式,这里整理了近些年在...

    抓到 Netty 一个 Bug,顺带来透彻地聊一下 Netty 是如何高效接收网络连接的.doc

    Netty 是一个高性能、异步事件驱动的网络应用程序框架,常用于开发服务器和客户端的通信应用。在本文中,我们将深入探讨 Netty 如何高效地处理客户端连接,以及作者在研究过程中发现的一个影响 Netty 接收连接吞吐量...

    netty-starter:Netty权威指南学习之旅

    Netty学习day01:同步阻塞式I/O源码分析服务端TimerServerTimerServer根据传入的参数设置监听端口,如果没有入参,使用默认值8080。启动后会发现主线程阻塞在ServerSocket的accept()方法上:可通过JConsole查看线程...

    java一个简单的即时通讯工具的设计与开发(源代码+LW).zip

    在Java中,通常使用ServerSocket类来创建服务器,然后用accept()方法等待客户端连接。一旦连接建立,就可以通过Socket对象进行通信。 2. **客户端**:用户界面展示聊天信息,允许用户发送和接收消息。在Java中,...

    安卓Android源码——(Socket协议).zip

    当客户端连接请求到达时,调用`accept()`方法获取新的Socket实例,然后通过该Socket进行数据交互。 - **客户端**:创建Socket,指定服务器的IP地址和端口号,调用`connect()`方法建立连接。之后,可以使用Socket的...

    websocket客户端和服务端

    2. **握手过程**:服务器收到请求后,会返回一个101 Switching Protocols响应,同样包含`Upgrade`和`Connection`字段,并用`Sec-WebSocket-Accept`头确认接收到的key。 3. **数据帧传输**:连接建立后,客户端和...

    Android应用源码之Android应用源码安卓与PC的Socket通信项目C#版+Java版_应用.zip

    - 对于性能优化,可以考虑使用NIO(非阻塞I/O)或Netty框架来提高并发处理能力。 - 错误处理和断线重连机制是必不可少的,确保在网络不稳定时仍能保持通信。 5. 实战应用: 这个项目可以应用于多种场景,如智能...

    多线程,网络套接字课件源码

    3. **套接字API**:如在Java中的`java.net.Socket`和`ServerSocket`类,C中的`socket()`、`bind()`、`listen()`、`accept()`、`connect()`、`send()`和`recv()`函数。 4. **异步套接字**:非阻塞IO或异步事件驱动...

    Java 两台服务器之间传递文件

    下面将详细讲解这个过程,以及如何利用源码和工具来实现这一目标。 首先,我们需要了解Java中的网络编程基础。在Java中,Socket是进行网络通信的基础,它提供了客户端与服务器之间的连接。对于文件传输,我们通常会...

    Java监听器学习 统计当前在线人数

    - 为简化开发,可以使用第三方库如Netty、Grizzly或Jetty,它们提供了高级的网络编程框架,可以更方便地实现监听器和会话管理。 在实际项目中,可能还需要考虑到网络延迟、超时处理、安全问题(如SSL/TLS加密)...

    \"java 网络编程\"简单总结

    同时,阅读和理解开源项目如Netty的源码,能帮助我们更深入地了解Java网络编程的底层实现。 总结,Java网络编程是开发者必备的技能之一,涵盖了TCP/IP原理、Socket编程、NIO以及网络协议的使用。通过学习和实践,...

    java socket 客户端和服务端例子

    在TCP/IP模型中,Socket是位于传输层的应用编程接口(API),它为应用层提供了进程间通信的能力。Java Socket库提供了一种标准的方式来创建和使用这些套接字。 服务端(server)程序通常会先启动,它会监听特定的IP...

    一个合适于JAVA初学者的JAVA通信

    博文可能提供了简单的Java通信示例代码,通过分析这些源码,初学者可以更好地理解如何在实际项目中应用上述概念。 总结来说,这个主题对初学者来说是一个很好的起点,它会介绍Java网络通信的基本原理和实践技巧,...

    java网络通信编程基础知识介绍

    例如,Tomcat、Jetty这样的Web服务器,或者Netty这样的高性能网络库,它们的源码可以加深对网络编程原理的理解。同时,IDE如IntelliJ IDEA或Eclipse,以及调试工具如Wireshark,可以帮助开发者在实践中调试和分析...

    Java网络编程总结

    另外,标签中提到的“源码”和“工具”,暗示我们可以深入研究Java网络库的源代码,理解其内部实现,这对于优化网络程序性能和解决疑难问题非常有帮助。例如,可以学习Apache MINA、Netty等高性能网络框架的源码,...

Global site tag (gtag.js) - Google Analytics