今日学习内容~
- socket() 生成一个连接套接字文件描述符sockfd
可作为稍后bind()函数绑定对象 - bind() 将连接套接字绑定到要监听的地址和端口上
- 服务程序解析配置文件得到监听地址和端口,使用bind()函数将sockfd绑定到监听地址和端口上。
可作为listen()函数监听对象
- 绑定了地址和端口的套接字,再加上配置文件指定的协议类型,构成三元组(protocal,src_addr,src_port)
- 服务程序可以监听多个地址、端口,实现多实例。本质是多次调用socket()+bind()生成并绑定多个套接字实现
- 服务程序解析配置文件得到监听地址和端口,使用bind()函数将sockfd绑定到监听地址和端口上。
- connect() 向某个被服务器监听的套接字发起连接请求
- 只有连接请求方才会调用connect函数
- 执行connect前,发起方生成一个绑定随机端口的sockfd
- 使用connec函数时带上连接的目的地址和目的端口,即服务端监听套接字绑定的地址和端口。
- 同时connect还需要带上本地客户端的地址和端口。
- 三次握手建立连接之后,连接两端的套接字都构成五元组(protocal,src_addr,src_port,dest_addr,dest_port)
- listen() 监听bind()绑定的addr+port套接字
套接字从CLOSE状态变为LISTEN状态,listenfd- 初次监控
- 进程/线程(监听者)被阻塞在select()或poll()上
- 直到有SYN信息写入被监听的sockfd的recv_buffer
DMA方式拷贝
- 内核被唤醒,将SYN数据拷贝到kernel buffer并处理
TCP连接建立与关闭由内核完成;处理包括判断SYN合法等
- 内核准备SYN+ACK,并从kernel buffer->send buffer->网卡(DMA)
与网卡通信是DMA方式
- 在半连接队列中创建一个SYN_RECV状态的新项
- 使用select()/poll()方式继续监听listenfd
- 再次监控-ACK
对应连接ESTABLISHED- ACK数据写入recv buffer再次唤醒内核,表示有某个客户端响应服务端内核的SYN(SYN+ACK)请求。
- 内核将数据拷入kernel buffer处理,将半连接队列中对应项移入全连接队列,状态ESTABLISHED
- ESTABLISHED连接可由内核通过accept()进行消费
用户进程发起accept系统调用,内核完成消费
- accept()处理过的连接会从全连接队列中移除,表明TCP建立连接完成,两端用户进程可以通信。
- 直到有一端使用close()/shutdown()再次唤醒内核关闭连接,之间的通信无需内核参与。
- 再次监控-SYN
新建SYN_RCVD半连接项- 若新写入数据是新的SYN请求而非ACK,则与初次监控处理一致,将其拷入kernel buffer处理,并新建SYN_RCVD半连接项,继续监听listenfd
- listen维护队列
半连接syn queue和全连接accept queue(连接已完成队列)- 监听者收到某客户端的SYN请求并恢复SYN+ACK后,在半连接队列尾部创建该连接的SYN_RECV项
包括客户端地址和客户端端口
- 监听者收到某客户端的ACK信息后,通过分析数据匹配半连接项,并将该项移至全连接项,状态ESTABLISHED,等待内核通过accept()消费
accept之后两端用户进程可以直接通信,无需内核
- 全连接队列项被accept()后,该项对应连接已经建立完毕,两端的用户进程使用自己的send buffer进行通信,无需内核介入。
- 监听套接字listenfd的send buffer只接受TCP SYN或ACK请求
- TCP已建立连接的socket buffer主要存通信数据,如HTTP请求/响应等
- 监听者收到某客户端的SYN请求并恢复SYN+ACK后,在半连接队列尾部创建该连接的SYN_RECV项
- backlog参数
- linux 2.2之前是listen()参数,为半连接+全连接队列长度和的MAX值
实际只有一个队列,用SYN_RCVD和ESTABLISHED区分
- Linux 2.2开始
- backlog参数表示全连接的最大长度
- /proc/sys/net/ipv4/tcp_max_syn_backlog 半连接队列的最大长度
syn queue/syn backlog 半连接队列
- /proc/sys/net/core/somaxconn 硬限制已完成队列最大长度,默认128
backlog的最大值的上限somaxconn
- 全连接队列的最大长度为min(backlog,somaxconn)。
- linux 2.2之前是listen()参数,为半连接+全连接队列长度和的MAX值
- 初次监控