linux TCP发送过程源码分析

本文深入分析Linux内核3.15.2版本中TCP发送过程,涉及Socket层的send()、sendto()等函数的流程,包括Socket数据结构、应用层使用、系统调用及Socket层操作。核心是通过inet_sendmsg()调用TCP层的tcp_sendmsg()进行数据发送。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

linux TCP发送过程源码分析——socket层

内核版本:3.15.2 

Socket数据结构关系

发送流程图

    以下是send()、sendto()、sendmsg()和sendmmsg()的发送流程图,这四个函数除了在系统调用层面上有些差别,在Socket层和TCP层的实现都是相同的。

 

应用层 

    应用层可以使用以下Socket函数来发送数据:

ssize_t write(int fd, const void *buf, size_t count);
ssize_t send(int s, const void *buf, size_t len, int flags);
ssize_t sendto(int s, const void *buf, size_t len, int flags, const struct sockaddr *to, socklen_t tolen);
ssize_t sendmsg(int s, const struct msghdr *msg, int flags);
int sendmmsg(int s, struct mmsghdr *msgvec, unsigned int vlen, unsigned int flags);

     这些发送函数有什么区别呢?当flags为0时,send()和write()功能相同。send(s, buf, len, flags)和sendto(s, buf, len, flags, NULL, 0)功能相同。write()和send()在套接字处于连接状态时可以使用,而sendto()、sendmsg()和sendmmsg()在任何时候都可用。用户层的数据最终都是以消息头来描述的。

struct msghdr

struct msghdr { 
    void *msg_name; /* optional address,目的地址 */ 
    socklen_t msg_namelen; /* size of address,目的地址的长度 */ 
    struct iovec *msg_iov; /* scatter/gather array,分散的数据块数组 */ 
    size_t msg_iovlen; /* #elements in msg_iov,分散的数据块个数 */ 
    void *msg_control; /* ancillary data, 控制数据 */ 
    socklen_t msg_controllen; /* ancillary data buffer len,控制数据的长度 */ 
    int msg_flags; /* flags on received message */ 
};

 

struct iovec

/* Structure for scatter/gather I/O. */ 
struct iovec { 
    void *iov_base; /* Pointer to data. */ 
    size_t iov_len; /* Length of data. */ 
};

发送默认为阻塞发送,也可以设置为非阻塞发送。非阻塞标志:O_NONBLOCK、MSG_DONTWAIT。

系统调用 

    发送函数是由glibc提供的,声明位于include/sys/socket.h中,实现位于sysdeps/mach/hurd/connect.c中,主要是用来从用户空间进入名为sys_socketcall的系统调用,并传递参数。sys_socketcall()实际上是所有

socket函数进入内核空间的共同入口。

SYSCALL_
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值