2010-01-30 13:25:52
TCP连接的建立一般是服务器端进入监听状态,然后客户端调用connect来连接。本文分析connect调用的过程,也即第一个syn报文的发送。
connect对应的系统调用是sys_connect
上面函数中的sock是在前面的socket调用的时候就初始化的,对应于TCP的BSD socket 的操作符集是inet_stream_ops
注:网络子模块在内核初始化的时候就注册了TCP,UDP和RAW3中协议,具体在af_inet.c中的inet_init函数中。
从上面的结构中可以看到connect对应的函数是inet_stream_connect。下面分析该函数
对于INET socket中的tcp连接,协议特有操作符集为tcp_prot
可以看出这2个操作符集中好多函数是相同的,不过它们是出于不同层次中的,从系统调用的角度来看,BSD socket的操作符集是先
被调用的,然后再调用对应的INET socket的操作符集。
对于TCP,流程进入tcp_v4_connect函数
该函数的主要功能是准备好路由,如果源端口没有指定,还要选择一个端口,然后再次更新路由信息;代表该连接的sk结构加入
到对应的hash表中(已连接ehash)。获取一个write_seq,以及当 sysctl_tw_recycle设置时,读取上次连接(如果存在)
进入time-wait时保存的时戳,赋给当前连接的rx_opt结构中。
最后该函数调用tcp_connect来完成连接
该函数的流程是
- 初始化该次连接的sk结构
- 分配一个skb数据包
- 初始化skb
- skb加入发送队列后,调用tcp_transmit_skb发送该数据包
- 更新snd_nxt,并启动超时定时器。
首先来看sk结构的初始化,也就是tcp_connect_init函数
接下来发送数据包,调用tcp_transimit_skb函数
skb发送后,connect并没有返回,因为此时连接还没有建立,tcp进入等待状态,此时回到前面的inet_stream_connect函数
在发送syn后进入等待状态
下文中将分析第二步:服务器端返回syn_ack包