本文主要分析listen()的内核实现,包括它的系统调用、Socket层实现、半连接队列,以及监听哈希表。

内核版本:3.6

Author:zhangskd @ csdn blog

应用层

int listen(int sockfd, int backlog);

Accept incoming connections and a queue limit for incoming connections.

backlog的定义

Now it specifies the queue length for completely established sockets waiting to be accepted,

instead of the number of incomplete connection requests. The maximum length of the queue

for incomplete sockets can be set using the tcp_max_syn_backlog sysctl. When syncookies

are enabled there is no logical maximum length and this sysctl setting is ignored.

If the socket is of type AF_INET, and the backlog argument is greater than the constant SOMAXCONN

(128 default), it is silently truncated to SOMAXCONN.

全连接队列的最大长度:

backlog保存的是完成三次握手、等待accept的全连接,而不是半连接。

负载不高时,backlog不用太大。(For complete connections)

系统最大的、未处理的全连接数量为:min(backlog, somaxconn),net.core.somaxconn默认为128。

这个值最终存储于sk->sk_max_ack_backlog。

半连接队列的最大长度:

tcp_max_syn_backlog默认值为256。(For incomplete connections)

当使用SYN Cookie时,这个参数变为无效。

半连接队列的最大长度为backlog、somaxconn、tcp_max_syn_backlog的最小值。

系统调用

listen()是由glibc提供的,声明位于include/sys/socket.h中,实现位于sysdeps/mach/hurd/listen.c中,

主要是用来从用户空间进入名为sys_socketcall的系统调用,并传递参数。sys_socketcall()实际上是所

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

在sys_socketcall()中会调用sys_listen()。

SYSCALL_DEFINE2(socketcall, int, call, unsigned long __user *, args)
{
...
switch(call) {
...
case SYS_LISTEN:
err = sys_listen(a0, a1);
break;
...
}
return err;
}

经过了socket层的总入口sys_socketcall(),现在进入sys_listen()。

SYSCALL_DEFINE2(listen, int, fd, int, backlog)
{
struct socket *sock;
int err, fput_needed;
int somaxconn; /* 通过文件描述符fd,找到对应的socket。
* 以fd为索引从当前进程的文件描述符表files_struct中找到对应的file实例,
* 然后从file实例的private_data成员中获取socket实例。
*/
sock = sockfd_lookup_light(fd, &err, &fput_needed); if (sock) {
/* backlog不能超过系统参数somaxconn */
somaxconn = sock_net(sock->sk)->core.sysctl_somaxconn;
if ((unsigned int)backlog > somaxconn)
backlog = somaxconn; err = security_socket_listen(sock, backlog); if (! err)
/* socket层的操作函数,如果是SOCK_STREAM的话,proto_ops是inet_stream_ops,
* 接下来调用的是inet_listen()。
*/
err = sock->ops->listen(sock, backlog); fput_light(sock->file, fput_needed);
}
}

socket层

SOCK_STREAM套接口的socket层操作函数集实例为inet_stream_ops,其中监听函数为inet_listen()。

const struct proto_ops inet_stream_ops = {
.family = PF_INET,
.owner = THIS_MODULE,
...
.listen = inet_listen, /* socket层的实现 */
...
};
/*
* @sk_ack_backlog: current listen backlog
* @sk_max_ack_backlog: listen backlog set in listen()
*/
struct sock {
...
unsigned short sk_ack_backlog; /* 当前的backlog,当前全连接队列长度 */
unsigned short sk_max_ack_backlog; /* 最大的backlog,最大全连接队列长度 */
...
};

检查套接口的状态、当前连接的状态是否合法,然后调用inet_csk_listen_start()启动监听。

/* Move a socket into listening state. */
int inet_listen(struct socket *sock, int backlog)
{
struct sock *sk = sock->sk;
unsigned char old_state;
int err; lock_sock(sk);
err = -EINVAL; /* 此时套接口状态需为SS_UNCONNECTED,套接口类型需为SOCK_STREAM */
if (sock->state != SS_UNCONNECTED || sock->type != SOCK_STREAM)
goto out; old_state = sk->sk_state; /* 当前连接状态 */ /* 当前的连接需为CLOSED或LISTEN状态 */
if (! ((1 << old_state) & (TCPF_CLOSE | TCPF_LISTEN)))
goto out; /* Really, if the socket is already in listen state
* we can only allow the backlog to be adjusted.
*/
if (old_state != TCP_LISTEN) {
/* 启动监听 */
err = inet_csk_listen_start(sk, backlog);
if (err)
goto out;
} sk->sk_max_ack_backlog = backlog; /* 最大全连接队列长度 */
err = 0 out:
release_sock(sk);
return err;
}

启动监听时,做的工作主要包括:

1. 创建半连接队列的实例,初始化全连接队列。

2. 初始化sock的一些变量,把它的状态设为TCP_LISTEN。

3. 检查端口是否可用,防止bind()后其它进程修改了端口信息。

4. 把sock链接进入监听哈希表listening_hash中。

int inet_csk_listen_start(struct sock *sk, const in nr_table_entries)
{
struct inet_sock *inet = inet_sk(sk);
struct inet_connection_sock *icsk = inet_csk(sk); /* 初始化全连接队列,创建半连接队列的实例 */
int rc = reqsk_queue_alloc(&icsk->icsk_accept_queue, nr_table_entries); if (rc != 0)
return rc; sk->sk_max_ack_backlog = 0; /* 在返回inet_listen()时赋值 */
sk->sk_ack_backlog = 0;
inet_csk_delack_init(sk); /* icsk->icsk_ack结构清零 */ sk->sk_state = TCP_LISTEN; /* 把sock的状态置为LISTEN */ /* 检查端口是否仍然可用,防止bind()后其它进程修改了端口信息 */
if (! sk->sk_prot->get_port(sk, inet->inet_num)) {
inet->inet_sport = htons(inet->inet_num);
sk_dst_reset(sk);
sk->sk_prot->hash(sk); /* 把sock链接入监听哈希表中 */
return 0;
} sk->sk_state = TCP_CLOSE;
/* 如果端口不可用,则释放半连接队列 */
__reqsk_queue_destroy(&icsk->icsk_accept_queue);
return -EADDRINUSE;
}

半连接队列

listen_sock结构用于保存SYN_RECV状态的连接请求块,所以也叫半连接队列。

(1)创建

queue是连接请求控制块,nr_table_entries是半连接的最大个数,即backlog。

int reqsk_queue_alloc(struct request_sock_queue *queue, unsigned int nr_table_entries)
{
size_t lopt_size = sizeof(struct listen_sock);
struct listen_sock *lopt; /* nr_table_entries必需在[8, sysctl_max_syn_backlog]之间,默认是[8, 256]
* 但实际上在sys_listen()中要求backlog <= sysctl_somaxconn(默认为128)
* 所以此时默认区间为[8, 128]
*/
nr_table_entries = min_t(u32, nr_table_entries, sysctl_max_syn_backlog);
nr_table_entries = max_t(u32, nr_table_entries, 8); /* 使nr_table_entries = 2^n,向上取整 */
nr_table_entries = roundup_pow_of_two(nr_table_entries + 1); /* 增加一个指针数组的长度 */
lopt_size += nr_table_entries * sizeof(struct request_sock *); if (lopt_size > PAGE_SIZE)
lopt = vzalloc(lopt_size); /* 如果申请内存大于1页,则申请虚拟地址连续的空间 */
else
lopt = kzalloc(lopt_size, GFP_KERNEL); /* 申请内存在1页内,则申请物理地址连续的空间 */ /* 相当于把max_qlen_log设置为nr_table_entries以2为底的对数 */
for (lopt->max_qlen_log = 3, (1 << lopt->max_qlen_log) < nr_table_entries; lopt->max_qlen_log++); get_random_bytes(&lopt->hash_rnd, sizeof(lopt->hash_rnd); /* 获取一个随机数 */ rwlock_init(&queue->syn_wait_lock);
queue->rskq_accept_head = NULL; /* 全连接队列置为空 */
lopt->nr_table_entries = nr_table_entries; /* 半连接队列的最大长度 */ write_lock_bh(&queue->syn_wait_lock);
queue->listen_opt = lopt; /* 初始化半连接队列 */
write_unlock_bh(&queue->syn_wait_lock); return 0;
}
/*
* Maximum number of SYN_RECV sockets in queue per LISTEN socket.
* One SYN_RECV socket costs about 80bytes on a 32bit machine.
* It would be better to replace it with a global counter for all sockets but then
* some measure against one socket starving all other sockets would be needed.
*
* The minimum value of it is 128. Experiments with real servers show that it is
* absolutely not enough even at 100conn/sec. 256 cures most of problems.
* This value is adjusted to 128 for low memory machines, and it will increase
* in proportion to the memory of machine.
* Note: Dont forget somaxconn that may limit backlog too.
*/
int sysctl_max_syn_backlog = 256;

(2)销毁

销毁连接请求块中的listen_sock实例,释放半连接队列。

void __reqsk_queue_destroy(struct request_sock_queue *queue)
{
struct listen_sock *lopt;
size_t lopt_size; lopt = queue->listen_opt;
lopt_size = sizeof(struct listen_sock) + lopt->nr_table_entries *
sizeof(struct request_sock *); if (lopt_size > PAGE_SIZE)
vfree(lopt);
else
kfree(lopt);
}

监听哈希表

inet_hash()用于把sock链入监听哈希表listening_hash,或者已建立连接的哈希表ehash。

struct proto tcp_prot = {
...
.hash = inet_hash,
.unhash = inet_unhash,
.get_port = inet_csk_get_port,
...
};

当sock状态为TCP_LISTEN时,inet_hash()把sock链接入tcp_hashinfo->listening_hash哈希表。

void inet_hash(struct sock *sk)
{
if (sk->sk_state != TCP_CLOSE) {
local_bh_disable();
__inet_hash(sk);
local_bh_enable();
}
} static void __inet_hash(struct sock *sk)
{
struct inet_hashinfo *hashinfo = sk->sk_prot->h.hashinfo;
struct inet_listen_hashbucket *ilb; /* sock不处于listen状态时 */
if (sk->sk_state != TCP_LISTEN) {
__inet_hash_nolisten(sk, NULL); /* 这里对应的是已建立的连接 */
return;
} /* 要求sk->sk_node没被使用? */
WARN_ON(! sk_unhashed(sk)); /* 根据端口号,找到对应的监听哈希桶 */
lib = &hashinfo->listening_hash[inet_sk_listen_hashfn(sk)]; spin_lock(&lib->lock);
/* 把sock放入监听哈希桶的头,链接节点为sk->sk_nulls_node */
__sk_nulls_add_node_rcu(sk, &ilb->head); sock_prot_inuse_add(sock_net(sk), sk->sk_prot, 1); /* 此CPU上该协议的使用数加一 */
spin_lock(&lib->lock);
} static inline bool sk_unhashed(const struct sock *sk)
{
return hlist_unhashed(&sk->sk_node);
} static inline int hlist_unhashed(const struct hlist_node *h)
{
return ! h->pprev;
}

Socket层实现系列 — listen()的实现的更多相关文章

  1. Socket层实现系列 — send()类发送函数的实现

    主要内容:socket发送函数的系统调用.Socket层实现. 内核版本:3.15.2 我的博客:http://blog.csdn.net/zhangskd 发送流程图 以下是send().sendt ...

  2. Socket层实现系列 — connect()的实现

    主要内容:connect()的Socket层实现.期间进程的睡眠和唤醒. 内核版本:3.15.2 我的博客:http://blog.csdn.net/zhangskd 应用层 int connect( ...

  3. Socket层实现系列 — 信号驱动的异步等待

    主要内容:Socket的异步通知机制. 内核版本:3.15.2 我的博客:http://blog.csdn.net/zhangskd 概述 socket上定义了几个IO事件:状态改变事件.有数据可读事 ...

  4. Socket层实现系列 — 睡眠驱动的同步等待

    主要内容:Socket的同步等待机制,connect和accept等待的实现. 内核版本:3.15.2 我的博客:http://blog.csdn.net/zhangskd 概述 socket上定义了 ...

  5. Socket层实现系列 — accept()的实现(一)

    本文主要介绍了accept()的系统调用.Socket层实现,以及TCP层实现. 内核版本:3.6 Author:zhangskd @ csdn blog 应用层 int accept(int soc ...

  6. Socket层实现系列 — getsockname()和getpeername()的实现

    本文主要介绍了getsockname()和getpeername()的内核实现. 内核版本:3.6 Author:zhangskd @ csdn blog 应用层 int getsockname(in ...

  7. Socket层实现系列 — bind()的实现(一)

    bind()函数的使用方法很简单,但是它是怎么实现的呢? 笔者从应用层出发,沿着网络协议栈,分析了bind()的系统调用.Socket层实现,以及它的TCP层实现. 本文主要内容:bind()的系统调 ...

  8. Socket层实现系列 — bind()的实现(二)

    本文主要内容:bind()的TCP层实现.端口的冲突处理,以及不同内核版本的实现差异. 内核版本:3.6 Author:zhangskd @ csdn blog TCP层实现 SOCK_STREAM套 ...

  9. Socket层实现系列 — I/O事件及其处理函数

    主要内容:Socket I/O事件的定义.I/O处理函数的实现. 内核版本:3.15.2 我的博客:http://blog.csdn.net/zhangskd I/O事件定义 sock中定义了几个I/ ...

随机推荐

  1. Dynamics CRM 非声明验证方式下连接组织服务的两种方式的性能测试

    今天看了勇哥的博文"http://luoyong0201.blog.163.com/blog/static/1129305201510153391392/",又认识到了一种新的连接 ...

  2. 没想到你是这样的UDP

    UDP是国际标准化组织为互联网设定的标准中的传输层中的一个协议.TCP/IP协议簇是一个很庞大的家族,但是今天我们就来看一看这个面向无连接的传输层在Java中是怎样通过编程实现的. 原理性知识 在Ja ...

  3. Android开发学习之路--Notification之初体验

    一般当我们收到短信啊,微信啊,或者有些app的提醒,我们都会在通知栏收到一天简单的消息,然后点击消息进入到app里面,其实android中有专门的Notification的类可以完成这个工作,这里就实 ...

  4. 学习笔记3-开发与运行(卸载)第一个ANDROID应用

    新建Android项目 1.      配置好Android坏境以后,新建项目选择Android Project. 2.      选择针对哪个平台开发的应用(Android2/Android4等) ...

  5. Android 文件操作心得体会

    android 的文件操作说白了就是Java的文件操作的处理.所以如果对Java的io文件操作比较熟悉的话,android的文件操作就是小菜一碟了.好了,话不多说,开始今天的正题吧. 先从一个小项目入 ...

  6. Win7 Eclipse Hadoop2.4插件配置

    准备工作: 1.下载hadoop2x-eclipse-plugin-master.zip Github地址:https://github.com/winghc/hadoop2x-eclipse-plu ...

  7. 精通CSS+DIV网页样式与布局--CSS文字效果

    上篇文章,小编简单的介绍了一下CSS的一些基本语法,学习内容不是很复杂,都是CSS的一些入门知识,但是万丈高楼平地起,搭好地基,高楼大厦不在话下,学习任何课程,都必须从基础开始,一步一个脚印,踏实坚定 ...

  8. SQL Sever2005卸载问题解决措施

      在安装SQLServer 2005时,曾遇到过SQL database service不能安装类似问题,曾经花费3个小时时间,最终将其安装成功.将其大概纠错过程记录如下,以作为前车之鉴.      ...

  9. Android开发技巧——使用PopupWindow实现弹出菜单

    在本文当中,我将会与大家分享一个封装了PopupWindow实现弹出菜单的类,并说明它的实现与使用. 因对界面的需求,android原生的弹出菜单已不能满足我们的需求,自定义菜单成了我们的唯一选择,在 ...

  10. Mybatis执行ReuseExecutor(五)

    ReuseExecutor顾名思义就是重复使用执行,其定义了一个Map<String, Statement>,将执行的sql作为key,将执行的Statement作为value保存,这样执 ...