Linux服务器套接字-错误的文件描述符

tre*_*nki 5 c sockets linux

我在Linux下的服务器套接字有问题。由于某种我不知道的原因,服务器套接字消失了,我Bad file descriptor在等待传入连接的select调用中遇到错误。当我在其他线程中关闭不相关的套接字连接时,总是会出现此问题。这在带有2.6.36内核的嵌入式Linux上发生。

有谁知道为什么会这样?服务器套接字会完全消失,这是否正常Bad file descriptor

编辑: 其他套接字代码实现了VNC服务器,并在完全不同的线程中运行。其他代码中唯一的特殊之处是使用,setjmp/longjmp但这不应该成为问题。

创建服务器套接字的代码如下:

int server_socket = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);

struct sockaddr_in saddr;
memset(&saddr, 0, sizeof(saddr));
saddr.sin_family = AF_INET;
saddr.sin_addr.s_addr = htonl(INADDR_ANY);
saddr.sin_port = htons(1234);

const int optionval = 1;
setsockopt(server_socket, SOL_SOCKET, SO_REUSEADDR, &optionval, sizeof(optionval));

if (bind(server_socket, (struct sockaddr *) &saddr, sizeof(saddr)) < 0) {
    perror("bind");
    return 0;
}

if (listen(server_socket, 1) < 0) {
    perror("listen");
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

我等待使用以下代码的传入连接:

static int WaitForConnection(int server_socket, struct timeval *timeout)
{
    fd_set read_fds;

    FD_ZERO(&read_fds);
    int max_sd = server_socket;
    FD_SET(server_socket, &read_fds);

    // This select will result in 'EBADFD' in the error case.
    // Even though the server socket was not closed with 'close'.
    int res = select(max_sd + 1, &read_fds, NULL, NULL, timeout);
    if (res > 0) {
        struct sockaddr_in caddr;
        socklen_t clen = sizeof(caddr);
        return accept(server_socket, (struct sockaddr *) &caddr, &clen);
    }

    return -1;
}
Run Code Online (Sandbox Code Playgroud)

编辑: 发生问题时,我目前只是重启服务器,但我不明白为什么服务器套接字ID突然会变成无效的文件描述符:

int error = 0;
socklen_t len = sizeof (error);
int retval = getsockopt (server_socket, SOL_SOCKET, SO_ERROR, &error, &len );
if (retval < 0) {
    close(server_socket);
    goto server_start;
}
Run Code Online (Sandbox Code Playgroud)

Gre*_*ape 6

套接字(文件描述符)通常会遇到与C. 每当您关闭套接字时,不要忘记分配-1给保持描述符值的变量:

close(socket);
socket = -1;
Run Code Online (Sandbox Code Playgroud)

就像你对C指针所做的那样

free(buffer);
buffer = NULL;
Run Code Online (Sandbox Code Playgroud)

如果你忘记这样做,你可以稍后关闭套接字两次,因为free()如果它是一个指针,你会记忆两次。

另一个问题可能与人们通常忘记的事实有关:UNIX 环境中的文件描述符从0. 如果代码中的某个地方你有

struct FooData {
    int foo;
    int socket;
    ...
}

// Either
FooData my_data_1 = {0};
// Or
FooData my_data_2;
memset(&my_data_2, 0, sizeof(my_data_2));
Run Code Online (Sandbox Code Playgroud)

在这两种情况下my_data_1my_data_2都有一个有效的描述符 ( socket) 值。后来,一些负责释放FooData结构的代码可能会盲目地使用close()这个描述符,这恰好是您服务器的侦听套接字(0)。


Rah*_*ali -3

在 Linux 中,一旦创建连接并关闭,则必须等待一段时间才能建立新连接。与 Linux 中一样,socket 不会释放端口号。一旦你关闭套接字。

或者

您重复使用套接字,然后就会出现错误的文件描述符。