Linux 原始套接字缓冲区大小的上限是否为 256 K?

Myc*_*ese 5 c sockets buffer embedded-linux setsockopt

我在 Centos 中使用以下代码将原始套接字缓冲区大小更改为 400 KB,但是在将缓冲区大小设置为 256 KB 时得到了相同的结果。哪里不对了?或者这是套接字层的限制?内核版本为 2.6.34。谢谢!

int       rawsock;
socklen_t socklen;
int       optval;
int       bufsize = 400 * 1024;

rawsock = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));
if (rawsock < 0) {
    my_log(LOG_ERR, "error creating raw socket");
    return rawsock;
}

optval = 0;
socklen = 4;
err = getsockopt(rawsock, SOL_SOCKET, SO_RCVBUF, &optval, &socklen);
bail_error(err);
my_log("socket RX original buffer size = %d", optval);

optval = 0;
socklen = 4;
err = getsockopt(rawsock, SOL_SOCKET, SO_SNDBUF, &optval, &socklen);
bail_error(err);
my_log("socket TX original buffer size = %d", optval);

err = setsockopt(rawsock, SOL_SOCKET, SO_RCVBUF, &bufsize, sizeof(bufsize));
bail_error(err);

err = setsockopt(rawsock, SOL_SOCKET, SO_SNDBUF, &bufsize, sizeof(bufsize));
bail_error(err);

optval = 0;
socklen = 4;
err = getsockopt(rawsock, SOL_SOCKET, SO_RCVBUF, &optval, &socklen);
bail_error(err);
my_log("socket RX new buffer size = %d", optval);

optval = 0;
socklen = 4;
err = getsockopt(rawsock, SOL_SOCKET, SO_SNDBUF, &optval, &socklen);
bail_error(err);
my_log("socket TX new buffer size = %d", optval);
Run Code Online (Sandbox Code Playgroud)

运行后,结果如下:

socket RX original buffer size = 110592
socket TX original buffer size = 110592
socket RX new buffer size = 524288
socket TX new buffer size = 524288
Run Code Online (Sandbox Code Playgroud)

Nom*_*mal 5

您刚刚达到系统当前的 sysctl 限制net.core.wmem_max并且net.core.rmem_max.

如果进程具有超级用户权限,它可以使用SO_SNDBUFFORCESO_RCVBUFFORCEioctl 来覆盖限制。如果您的服务确实需要更大的缓冲区有真正的原因——也就是说,除了糟糕的开发或设计选择之外的任何其他原因——那么我推荐这种方式。通常没有这样的原因,在这种情况下,我建议您修复应用程序/服务代码。

您可以在系统范围内修改限制,但它们会影响所有进程。通常,默认值工作正常,但在某些特殊情况下(可能是具有非常宽但延迟长的网络连接的嵌入式服务器?)您可能希望修改它们。

要暂时执行此操作(直到下次启动),请以 root 身份运行sysctl -w net.core.wmem_max=bytessysctl -w net.core.rmem_max=bytes(其中bytes是以十进制数表示的新限制,以字节为单位)。

要使更改永久生效,请添加

net.core.rmem_max=bytes
net.core.wmem_max=bytes
Run Code Online (Sandbox Code Playgroud)

到您的/etc/sysctl.conf文件,或者目录中的新文件/etc/sysctl.d/(如果您的 Linux 发行版提供了该文件)。后者是更好的方法,因为它不会停止对默认配置文件的更新。

如果您想更深入地研究这些和其他套接字 ioctl,您可以查看内核net/core/sock.c文件及其sock_setsockopt()中的函数。


dvh*_*vhh 2

根据文档

SO_RCVBUF/proc/sys/net/core/rmem_max:最大允许值由文件设置

SO_SNDBUF:最大允许值由/proc/sys/net/core/wmem_max文件设置。

因此限制可能取决于您的系统的配置方式。