listen()函数仅在TCP服务器端调用,它做两个事情:将套接字转换到LISTEN状态和设置套接上的最大连接队列。listen()对应的内核实现为sys_listen(),下面开始对其实现作具体的分析。
一、sys_listen()函数
sys_listen()的源码实现及分析如下所示:
/*
* Perform a listen. Basically, we allow the protocol to do anything
* necessary for a listen, and if that works, we mark the socket as
* ready for listening.
*/
SYSCALL_DEFINE2(listen, int, fd, int, backlog)
{
struct socket *sock;
int err, fput_needed;
int somaxconn;
sock = sockfd_lookup_light(fd, &err, &fput_needed);
if (sock) {
/*
* sysctl_somaxconn存储的是服务器监听时,允许每个套接字连接队列长度
* 的最大值,默认值是SOMAXCONN,即128,在sysctl_core_net_init()函数中初始化。
* 在proc文件系统中可以通过修改/proc/sys/net/core/somaxconn文件来修改这个值。
*/
somaxconn = sock_net(sock->sk)->core.sysctl_somaxconn;
/*
* 如果指定的最大连接数超过系统限制,则使用系统当前允许的连接队列
* 中连接的最大数。
*/
if ((unsigned)backlog > somaxconn)
backlog = somaxconn;
err = security_socket_listen(sock, backlog);
if (!err)
/*
* 如果是TCP套接字,sock->ops指向的是inet_stream_ops,
* sock->ops是在inet_create()函数中初始化,所以listen接口
* 调用的是inet_listen()函数。
*/
err = sock->ops->listen(sock, backlog);
fput_light(sock->file, fput_needed);
}
return err;
}
sys_listen()的代码流程图如下所示:
sys_listen()的代码流程和sys_bind()很像,都是先调用sockfd_lookup_light()获取描述符对应的socket实例,然后通过调用sock->ops中的操作接口来完成真正的操作。接下来看这段代码:
if ((unsigned)backlog > somaxconn)
backlog = somaxconn;
这里可以看出,如果指定的最大连接队列数超过系统限制,会使用系统中设置的最大连接队列数。所以,如果想扩大套接字的连接队列,只调整listen()的backlog参数是没用的,还要修改系统的设置才行。