我正在用 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() 调用的操作稍微不正确。
您设置了指针*p:
p = sockadr;
您将 &sockadr 作为最后一个参数传递给 getaddrinfo():
status = getaddrinfo(NULL, "8080", &hints, &sockadr)
您在调用 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/