TCP层sendmsg系统调用的实现分析
概述
sendmsg系统调用在tcp层的实现是tcp_sendmsg函数,该函数完成以下任务:从用户空间读取数据,拷贝到内核skb,将skb加入到发送队列的任务,调用发送函数;函数在执行过程中会锁定控制块,避免软中断在tcp层的影响;函数核心流程为,在发送数据时,查看是否能够将数据合并到发送队列中最后一个skb中,如果不能合并,则新申请一个skb;拷贝过程中,如果skb的线性区域有空间,则优先使用线性区域,线性区域空间不足,则使用分页区域;拷贝完成后,调用发送函数发送数据;
代码分析
int tcp_sendmsg(struct sock *sk, struct msghdr *msg, size_t size)
{
struct tcp_sock *tp = tcp_sk(sk);
struct sk_buff *skb;
struct sockcm_cookie sockc;
int flags, err, copied = ;
int mss_now = , size_goal, copied_syn = ;
bool process_backlog = false;
bool sg;
long timeo; /* 加锁,避免与软中断的冲突 */
lock_sock(sk); /* 获取标记 */
flags = msg->msg_flags; /* fastopen和defer */
if (unlikely(flags & MSG_FASTOPEN || inet_sk(sk)->defer_connect)) {
err = tcp_sendmsg_fastopen(sk, msg, &copied_syn, size);
if (err == -EINPROGRESS && copied_syn > )
goto out;
else if (err)
goto out_err;
} /* 获取阻塞时间,非阻塞为0 */
timeo = sock_sndtimeo(sk, flags & MSG_DONTWAIT); /* 限速检查 */
tcp_rate_check_app_limited(sk); /* is sending application-limited? */ /* Wait for a connection to finish. One exception is TCP Fast Open
* (passive side) where data is allowed to be sent before a connection
* is fully established.
*/
/* 等待连接完成状态,fastopen的被动打开方例外 */
if ((( << sk->sk_state) & ~(TCPF_ESTABLISHED | TCPF_CLOSE_WAIT)) &&
!tcp_passive_fastopen(sk)) {
/* 等待连接完成 */
err = sk_stream_wait_connect(sk, &timeo);
if (err != )
goto do_error;
} if (unlikely(tp->repair)) {
if (tp->repair_queue == TCP_RECV_QUEUE) {
copied = tcp_send_rcvq(sk, msg, size);
goto out_nopush;
} err = -EINVAL;
if (tp->repair_queue == TCP_NO_QUEUE)
goto out_err; /* 'common' sending to sendq */
} sockc.tsflags = sk->sk_tsflags;
if (msg->msg_controllen) {
err = sock_cmsg_send(sk, msg, &sockc);
if (unlikely(err)) {
err = -EINVAL;
goto out_err;
}
} /* This should be in poll */
/* 清除异步队列已满标记 */
sk_clear_bit(SOCKWQ_ASYNC_NOSPACE, sk); /* Ok commence sending. */
copied = ; restart:
/* 获取mss,gso情况下size_goal记录总mss=页数*mss */
mss_now = tcp_send_mss(sk, &size_goal, flags); err = -EPIPE;
/* 已经关闭了发送端 */
if (sk->sk_err || (sk->sk_shutdown & SEND_SHUTDOWN))
goto do_error; /* 网卡分散聚合标记 */
sg = !!(sk->sk_route_caps & NETIF_F_SG); /* 遍历发送缓存 */
while (msg_data_left(msg)) {
int copy = ;
int max = size_goal; /* 拿到发送队列的尾部skb */
skb = tcp_write_queue_tail(sk);
/* 有skb未发送 */
if (tcp_send_head(sk)) {
/* 网卡不支持校验和,调整最大长度 */
if (skb->ip_summed == CHECKSUM_NONE)
max = mss_now;
/* 该skb还能够容纳的数据长度 */
copy = max - skb->len;
} /* 剩余空间为0,或者不能合并,分配一个新的skb */
if (copy <= || !tcp_skb_can_collapse_to(skb)) {
bool first_skb; new_segment:
/* Allocate new segment. If the interface is SG,
* allocate skb fitting to single page.
*/
/* 空闲内存不足,进入等待 */
if (!sk_stream_memory_free(sk))
goto wait_for_sndbuf; if (process_backlog && sk_flush_backlog(sk)) {
process_backlog = false;
goto restart;
}
first_skb = skb_queue_empty(&sk->sk_write_queue); /* 分配skb */
skb = sk_stream_alloc_skb(sk,
select_size(sk, sg, first_skb),
sk->sk_allocation,
first_skb);
/* 分配失败,等待 */
if (!skb)
goto wait_for_memory; process_backlog = true;
/*
* Check whether we can use HW checksum.
*/
/* 网卡允许计算校验和 */
if (sk_check_csum_caps(sk))
skb->ip_summed = CHECKSUM_PARTIAL; /* 添加到发送队列 */
skb_entail(sk, skb);
copy = size_goal;
max = size_goal; /* All packets are restored as if they have
* already been sent. skb_mstamp isn't set to
* avoid wrong rtt estimation.
*/
if (tp->repair)
TCP_SKB_CB(skb)->sacked |= TCPCB_REPAIRED;
} /* Try to append data to the end of skb. */
/* 拷贝的数据不能超过实际数据块的长度 */
if (copy > msg_data_left(msg))
copy = msg_data_left(msg); /* Where to copy to? */
/* 线性区域还有空间 */
if (skb_availroom(skb) > ) {
/* We have some space in skb head. Superb! */
/* 取要拷贝的数量和线性区域的较小值 */
copy = min_t(int, copy, skb_availroom(skb));
/* 从用户空间拷贝到内核 */
err = skb_add_data_nocache(sk, skb, &msg->msg_iter, copy);
if (err)
goto do_fault;
}
/* 线性区域没有空间,则使用分页区 */
else {
bool merge = true;
/* 获取页数量 */
int i = skb_shinfo(skb)->nr_frags;
/* 获取缓存的页 */
struct page_frag *pfrag = sk_page_frag(sk); /* 检查是否有足够的空间,空间不足则申请新页,失败则等待 */
if (!sk_page_frag_refill(sk, pfrag))
goto wait_for_memory; /* 不能合并 */
if (!skb_can_coalesce(skb, i, pfrag->page,
pfrag->offset)) {
/* 页数量超过限制 || 网卡不支持分散聚合*/
if (i >= sysctl_max_skb_frags || !sg) {
/* 增加push标记,尽快处理数据 */
tcp_mark_push(tp, skb);
/* 新申请一个skb */
goto new_segment;
}
/* 不合并 */
merge = false;
} /* 获取能够合并的空间 */
copy = min_t(int, copy, pfrag->size - pfrag->offset); /* 发送合法性检查 */
if (!sk_wmem_schedule(sk, copy))
goto wait_for_memory; /* 拷贝数据到分页 */
err = skb_copy_to_page_nocache(sk, &msg->msg_iter, skb,
pfrag->page,
pfrag->offset,
copy);
if (err)
goto do_error; /* Update the skb. */
if (merge) {
/* 合并的,需要增加数据量 */
skb_frag_size_add(&skb_shinfo(skb)->frags[i - ], copy);
} else {
/* 非合并的,对新页进行初始化 */
skb_fill_page_desc(skb, i, pfrag->page,
pfrag->offset, copy);
page_ref_inc(pfrag->page);
} /* 记录新的偏移 */
pfrag->offset += copy;
} /* 第一次拷贝,清除push标记 */
if (!copied)
TCP_SKB_CB(skb)->tcp_flags &= ~TCPHDR_PSH; /* 更新发送队列的最后一个序号 */
tp->write_seq += copy;
/* 更新skb结束序号 */
TCP_SKB_CB(skb)->end_seq += copy;
tcp_skb_pcount_set(skb, ); /* 更新拷贝数量 */
copied += copy; /* 数据都拷贝完成 */
if (!msg_data_left(msg)) {
if (unlikely(flags & MSG_EOR))
TCP_SKB_CB(skb)->eor = ;
goto out;
} /* 还能继续拷贝数据 带外数据 修复模式,继续拷贝 */
if (skb->len < max || (flags & MSG_OOB) || unlikely(tp->repair))
continue; /* 需要使用push标记 */
if (forced_push(tp)) {
/* 打psh标记 */
tcp_mark_push(tp, skb);
/* 发送队列中的多个skb */
__tcp_push_pending_frames(sk, mss_now, TCP_NAGLE_PUSH);
}
/* 否则,有数据要发送,则发送一个skb */
else if (skb == tcp_send_head(sk))
tcp_push_one(sk, mss_now);
continue; wait_for_sndbuf:
/* 设置空间不足标记 */
set_bit(SOCK_NOSPACE, &sk->sk_socket->flags);
wait_for_memory:
/* 已经有拷贝数据到发送队列,则发送之 */
if (copied)
tcp_push(sk, flags & ~MSG_MORE, mss_now,
TCP_NAGLE_PUSH, size_goal); /* 等待内存足够 */
err = sk_stream_wait_memory(sk, &timeo);
if (err != )
goto do_error; /* 重新计算mss */
mss_now = tcp_send_mss(sk, &size_goal, flags);
} out:
/* 已经拷贝数据到发送队列,则发送之 */
if (copied) {
tcp_tx_timestamp(sk, sockc.tsflags, tcp_write_queue_tail(sk));
tcp_push(sk, flags, mss_now, tp->nonagle, size_goal);
}
out_nopush:
release_sock(sk);
return copied + copied_syn; do_fault:
/* skb中没有数据,从发送队列中删除skb */
if (!skb->len) {
tcp_unlink_write_queue(skb, sk);
/* It is the one place in all of TCP, except connection
* reset, where we can be unlinking the send_head.
*/
tcp_check_send_head(sk, skb);
sk_wmem_free_skb(sk, skb);
} do_error:
if (copied + copied_syn)
goto out;
out_err:
err = sk_stream_error(sk, flags, err);
/* make sure we wake any epoll edge trigger waiter */
if (unlikely(skb_queue_len(&sk->sk_write_queue) == &&
err == -EAGAIN)) {
sk->sk_write_space(sk);
tcp_chrono_stop(sk, TCP_CHRONO_SNDBUF_LIMITED);
}
release_sock(sk);
return err;
}
TCP层sendmsg系统调用的实现分析的更多相关文章
- TCP层shutdown系统调用的实现分析
概述 shutdown系统调用在tcp层会调用两个函数,对于ESTABLISHED状态需要调用tcp_shutdown关闭连接,对于LISTEN和SYN_SENT状态则需要以非阻塞模式调用tcp_di ...
- TCP层recvmsg系统调用的实现分析
概述 recvmsg系统调用在tcp层的实现是tcp_recvmsg函数,该函数完成从接收队列中读取数据复制到用户空间的任务:函数在执行过程中会锁定控制块,避免软中断在tcp层的影响:函数会涉及从接收 ...
- TCP层close系统调用的实现分析
在调用close系统调用关闭套接字时,如果套接字引用计数已经归零,则需继续向上层调用其close实现,tcp为tcp_close:本文仅介绍tcp部分,前置部分请参考本博关于close系统调用的文章: ...
- TCP层accept系统调用的实现分析
inet_csk_accept函数实现了tcp协议accept操作,其主要完成的功能是,从已经完成三次握手的队列中取控制块,如果没有已经完成的连接,则需要根据阻塞标记来来区分对待,若非阻塞则直接返回, ...
- TCP层bind系统调用的实现分析
说明:该文章中部分代码未能完全理解透彻,可能对您造成误解,请慎读: 并建议您先阅读本博另外一篇文章:<Linux TCP套接字选项 之 SO_REUSEADDR && SO_RE ...
- 套接字之sendmsg系统调用
sendmsg系统调用允许在用户空间构造消息头和控制信息,用此函数可以发送多个数据缓冲区的数据,并支持控制信息:当调用进入内核后,会将用户端的user_msghdr对应拷贝到内核的msghdr中,然后 ...
- Python Tornado框架(TCP层)
Tornado在TCP层里的工作机制 上一节是关于应用层的协议 HTTP,它依赖于传输层协议 TCP,例如服务器是如何绑定端口的?HTTP 服务器的 handle_stream 是在什么时候被调用的呢 ...
- TCP拥塞控制算法 优缺点 适用环境 性能分析
[摘要]对多种TCP拥塞控制算法进行简要说明,指出它们的优缺点.以及它们的适用环境. [关键字]TCP拥塞控制算法 优点 缺点 适用环境公平性 公平性 公平性是在发生拥塞时各源端(或同一源端 ...
- Android系统进程间通信Binder机制在应用程序框架层的Java接口源代码分析
文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6642463 在前面几篇文章中,我们详细介绍了A ...
随机推荐
- java基础4(线程)
1.请简单描述什么是并行,什么是并发? 并行:指两个或多个事件在同一时刻发生(同时发生). 并发:指两个或多个事件在同一个时间段内发生. 通俗易懂版: 你吃饭吃到一半,电话来了,你一直到吃完了以后才去 ...
- 如何使用koa搭建一个简单服务
1.首先检测是否已经有node环境? 把Windows的黑窗体的命令行工具调用出来 敲击命令行node -v , 然后,就可以看到这个打印出了一个版本号,这就证明我们的node.js已经是安装 ...
- frp 路由穿透(github开源穿透软件)
server配置(windows):下载: https://github.com/fatedier/frp/releases [common] # 服务器端端口 bind_port = # 客户端连接 ...
- Windows问题
常用工具 DisplayFusion 官网 电脑分屏,V9.4 Pro 破解版 问题解决 Win64位注册表导入方法 64位Windows操作系统注册表不同于32位Windows操作系统,Win64 ...
- 解决remix在线编译器连接本地私有链环境不成功的问题
一.部署合约到私有链环境 选择"environment"里的"Web3 Provider" 弹出RPC连接地址输入框 输入我们Geth客户端安装服务器的IP:9 ...
- Java语言基础(12)
1 构造方法重载 在一个类内部,编写多个构造方法,创建对象的时候,根据需求的不同,调用不同的构造方法创建对象,实现不同的初始化. 案例:Demo1 public class Demo1 { publi ...
- 本地phpmyadmin 访问远程数据库服务器
第一步:打开/phpmyadmin/libraries 目录 第二步:修改config.default.php 文件(linux下可用vim编辑,FreeBSD下可用vi或是ee编辑)找到“All ...
- 浅入深出Vue:登录
上一篇我们实现了注册功能,现在我们来实现一下登录功能. 准备工作 新建登录组件 添加登录组件的路由对象 新建登录组件 Login.vue: <template> <div> & ...
- git报错remote: error: cannot run hooks/post-receive: No such file or directory
错误情况如下图所示: 如果你也显示这个错误但是其实在该路径上有上有这个文件,那么显然你遇到和我一样的情况,即你是Windows下创建的文件,但是试图在Lunix系统去打开它.这是在Windows下调用 ...
- MySQL 关于自定义函数的操作
-- 函数 --> 模块化,封装,代码复用 create function 函数名([参数列表]) returns 数据类型 begin SQL语句: return 值: end; 示例: -- ...