gpt4 book ai didi

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

转载 作者:可可西里 更新时间:2023-11-01 02:41:06 24 4
gpt4 key购买 nike

我正致力于在 C 中创建一个简单的服务器。现在我在创建套接字以监听传入连接时遇到问题。我在 x86 Debian 发行版上运行 VirtualBox。我的代码如下:

#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

附言。不过我喜欢你的代码。它很干净,几乎是正确的。

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

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com