c - 在 VirtualBox 内的 Debian 上调用 socket() 时出现错误 97

标签 c linux sockets tcp

我正在用 C 创建一个简单的服务器。现在我在创建套接字来监听传入连接时遇到问题。 我使用的是在 VirtualBox 内运行的 x86 Debian 发行版。 我的代码如下:

#define _POSIX_SOURCE

#include <stdio.h>
#include <pthread.h>    /* threads, yo */
#include <stdlib.h>
#include <memory.h>     /* memset */

#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <netdb.h>

#include <arpa/inet.h>  /* helpers like inet_ntop */ 
#include <netinet/in.h>

void listenInput();

void* handleConn(void* args) {
    printf("Handling connection!\n");
    listenInput();
    return NULL;
}

void acceptData(int socket) {


}

void listenInput() {
    struct addrinfo hints;
    struct addrinfo *sockadr;
    char ipstr[INET6_ADDRSTRLEN];
    int status;
    struct addrinfo *p = sockadr;
    int s;
    int yes = 1;

    memset(&hints, 0, sizeof(hints));
    hints.ai_family = AF_INET;
    hints.ai_socktype = SOCK_STREAM;
    hints.ai_flags = AI_PASSIVE;

    printf("Getting addr info. Preparing to listen on port 8080\n");
    if((status = getaddrinfo(NULL, "8080", &hints, &sockadr)) != 0) {
        printf("getaddrinfo: %s\n", gai_strerror(status));
        return;
    }

    printf("Woo, got addr info!\n");
    printf("Calling socket(...)\n");
    if( -1 == (s = socket(p->ai_family, p->ai_socktype, p->ai_protocol))) {
        printf("Error creating socket (%d): %s\n", errno, gai_strerror(errno));
        return;
    }

        printf("Calling bind\n");
        bind(s, p->ai_addr, p->ai_addrlen);

        printf("Calling listen...\n");
        listen(s, 5);
        {
            struct sockaddr_storage their_addr;
            socklen_t addr_size = sizeof their_addr;

            printf("accept()...\n");
            int newsocket = accept(s, (struct sockaddr *)&their_addr, &addr_size);

            printf("Got a connection! New socket: %d\n", newsocket);
        }

        close(s);


    freeaddrinfo(sockadr);
}

int main(int argc, char** argv) {
    pthread_t   thread;
    void*       status;

    pthread_create(&thread, NULL, handleConn, NULL);
    pthread_join(thread, &status);
    printf("Joined thread!\n");
    return 0;
}

运行此程序时,socket() 失败并显示错误代码 97 - 不支持地址系列。 目前我还没有找到任何关于这个问题的相关信息。

这是我的虚拟机中 ifconfig 的输出:

eth0      Link encap:Ethernet  HWaddr 08:00:27:11:84:ee  
          inet addr:192.168.56.101  Bcast:192.168.56.255  Mask:255.255.255.0
          inet6 addr: fe80::a00:27ff:fe11:84ee/64 Scope:Link
          UP BROADCAST RUNNING MULTICAST  MTU:1500  Metric:1
          RX packets:122 errors:0 dropped:0 overruns:0 frame:0
          TX packets:104 errors:0 dropped:0 overruns:0 carrier:0
          collisions:0 txqueuelen:1000 
          RX bytes:17778 (17.3 KiB)  TX bytes:14571 (14.2 KiB)

lo        Link encap:Local Loopback  
          inet addr:127.0.0.1  Mask:255.0.0.0
          inet6 addr: ::1/128 Scope:Host
          UP LOOPBACK RUNNING  MTU:16436  Metric:1
          RX packets:64 errors:0 dropped:0 overruns:0 frame:0
          TX packets:64 errors:0 dropped:0 overruns:0 carrier:0
          collisions:0 txqueuelen:0 
          RX bytes:4652 (4.5 KiB)  TX bytes:4652 (4.5 KiB)

此外,netstat --tcp 显示其他正在监听的流套接字,因此其他应用程序似乎已成功创建它们。

最佳答案

问题是您对 getaddrinfo() 调用的操作稍微不正确。

  1. 您设置了指针*p:

    p = sockadr;
    
  2. 您将 &sockadr 作为最后一个参数传递给 getaddrinfo():

    status = getaddrinfo(NULL, "8080", &hints, &sockadr)
    
  3. 您在调用 socket() 时使用 sockadr值(通过 p)和bind()

请注意:&sockadr是指向指针的指针,因此它可以更改sockadr指向的地址。这就是实际发生的情况。如果您在 getaddrinfo() 之前和之后打印 sockadr 作为指针,您会看到它:

sockadr before getaddrinfo(): 0x7fef07d8fee2

Getting addr info. Preparing to listen on port 8080

sockadr after getaddrinfo(): 0x7fef000008c0

因此,可能的解决方案是在 getaddrinfo() 调用之后添加此行:

p = sockadr;

有关使用示例,请参阅 man 3 getaddrinfo

PS。不过我喜欢你的代码。它很干净并且几乎正确。

关于c - 在 VirtualBox 内的 Debian 上调用 socket() 时出现错误 97,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20031116/

相关文章:

c - argv[argc] ==?

c - 如果您不释放函数中动态分配的内存,会发生什么情况?

c - 如何在c中使用sscanf在分号后分隔数字

linux - Linux 上的轻量级版本控制软件

Java:同步套接字输入

python - 显示来自通过套接字发送的numpy Array的图像

c++ - 目录大小

c++ - 2个线程如何共享同一个缓存行

mysql - 由于语法错误无法导入 MySQL 数据库

java - 使用哪些缓冲区从 ServerSocket 创建的 Socket 接收输入流?