TCP源码浅读(6)三次握手-被动方接收ACK

本文深入解析Linux内核如何处理TCP三次握手的最后阶段。当服务端接收到第三次握手的ACK包后,从tcp_v4_rcv入口开始,检查包的合法性,创建新的tcp_sock表示全连接,并将其状态设为TCP_SYN_RECV,随后添加到相关数据结构中。最终,通过tcp_child_process将连接状态改为TCP_ESTABLISHED,准备进行数据交互。整个过程涉及到了socket、请求队列和连接状态的管理。

版本:linux 4.18.1

作为学习笔记,本篇只讨论常规的交互过程,旨在理清 linux 内核对 TCP 相关的信息管理。

第四节 我们讨论了被动方接受 SYN 后的逻辑,简单回顾一下:

收到 SYN 请求包后,服务端创建一个 request_sock 对象来表示这个 半连接,并添加到 ehash 中管理,此时该 sock 的状态为 TCP_NEW_SYN_RECV。然后就向客户端发送了第二次握手包。

现在我们看下服务端接收到第三次握手后的处理。

入口函数 tcp_v4_rcv 老相识了:

1、首先查找 sock,会在 ehash 中找到代表半连接的 request_sock,它处于 TCP_NEW_SYN_RECV 状态;

2、根据 request_sock 关联到 inet_listen_hashbucket 监听该端口的 tcp_sock (TCP_LISTEN);

3、调用子流程 tcp_check_req 对 ACK 包做合法性检查,检查通过后,创建一个新的 tcp_sock ,用来表示和客户端的全连接;

4、对新的 tcp_sock 进一步处理。

流程相比之前稍微复杂一些,我们分步骤阅读,首先,进入 tcp_check_req 看下:

1、对客户端发来的第二次握手包做合法性检查,比如序列号是否正确,控制位检查等...;

2、检查通过后,创建一个全新的 tcp_sock 表示这个连接,状态为 TCP_SYN_RECV

3、把这个新的 sock 添加到 inet_bind_bucketowners 链表 和 ehash 中管理,将代表半连接的 request_sock 从 ehash 中删除;

4、最后将新建的 sock 添加到全连接队列,同时更新半连接数和全连接队列值。

看下执行到这时内核管理信息:

看着眼花缭乱,对着源码看其实很清晰的。

 最后我们进入 tcp_child_process 看下: 

这里会将表示这个全连接的 sock 状态进一步修改为 TCP_ESTABLISHED,然后通知应用程序可以进行数据的交互。

此时状态图和上图一样,只是状态有变更,此处就省略了。

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值