为什么 `accept(2)` 需要将 `sockaddr` 的长度作为一个单独的指针?

4

我很难理解accept(2)的原型。

我有一个简单的服务器,可以接受连接并向客户端返回"Hello world!\n"。系统调用accept(2)需要一个指向struct sockaddr的指针和一个指向socklen_t的指针来存储结构的长度。但是sockaddr已经有了一个名为sa_len的字段,似乎就是为此而设计的。

此外,我还有一个简单的服务器(在OSX下编译,希望在Linux下也能编译),它会打印出传递给accept的socklen_t以及sa_len的值:在OSX上它们是相同的,都是28。

编辑:经过更多测试,似乎sa_len并不一定与指针中存储的长度相同。

为什么accept(2)需要一个单独的长度指针?


以下是示例服务器的参考代码。您可以使用以下命令进行编译:

gcc -Wall -Wextra main.c

然后运行:

./a.out

在另一个终端上连接它:
telnet 127.0.0.1 3000

#include <errno.h>
#include <sys/socket.h>
#include <unistd.h>
#include <netdb.h>
#include <stdio.h>
#include <netinet/in.h>
#include <strings.h>

// default port if none if passed in the program arguments
#define FTP_PORT_DEFAULT ("3000")

// number of connections allow to wait for a call to accept
#define FTP_BACKLOG (5)

int main(int ac, char **av)
{
    struct addrinfo     hints; // hints used to get address information
    struct addrinfo     *sai; // server address information
    int                 ss; // server socket
    char            *port;

    port = (ac >= 2) ? av[1] : FTP_PORT_DEFAULT;
    bzero(&hints, sizeof(hints));

    // using AF_INIT6 instead of PF_INET6
    // https://dev59.com/sGw15IYBdhLWcg3wQ5di
    hints.ai_family = AF_INET6;
    hints.ai_socktype = SOCK_STREAM;

    // intended for use with `bind`
    hints.ai_flags = AI_PASSIVE;

    // passing `NULL` as the hostname tells `getaddrinfo` to use
    // any available local IP address
    if (getaddrinfo(NULL, port, &hints, &sai) != 0)
        return (-1);

    if ((ss =
         socket(sai->ai_family, sai->ai_socktype, sai->ai_protocol))== -1)
        return (-1);

    if (bind(ss, sai->ai_addr, sai->ai_addrlen) == -1 ||
        listen(ss, FTP_BACKLOG) == -1)
    {
        close(ss);
        return (-1);
    }

    while (1)
    {
        int cs;
        struct sockaddr csockaddr;
        unsigned csockaddr_len;

        bzero(&csockaddr, sizeof(csockaddr));
        cs = accept(ss, &csockaddr, &csockaddr_len);
        // handle "fatal" errors that I don't think I can recover from
        // other than by relaunching the server
        if (cs == EBADF || cs == ECONNABORTED || cs == ENOTSOCK ||
                cs == EOPNOTSUPP)
        {
            break ;
        }
        printf("my len %u / internal len %u\n",
                   csockaddr_len, csockaddr.sa_len);

        if (fork() == 0)
        {
            write(cs, "Hello world!\n", 13);
            close(cs);
        }
        break;
    }
    close(ss);

    return 0;
}

2
accept 实际上需要一个 socklen_t*,而不是一个 unsigned int*。在 4.3BSD 中它需要一个 int*,然后在早期的 POSIX 版本中需要一个 size_t*,然后 POSIX 改变了主意,让实现定义 socklen_t 为这两种类型之一。 - Fred Foo
你说得对。我用的是OSX,它似乎使用了“unsigned int”。已在问题中进行了更正。 - conradkleinespel
在调用accept之前,您需要初始化csockaddr_len。有关详细信息,请参阅accept(2)手册页面。 - sigjuice
1个回答

5

sa_len不具备可移植性。它只在1988年被BSD添加,比4.3BSD晚几年,但它不在POSIX和Linux中。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接