sockets - 在什么情况下 getpeername 返回 IP :PORT 0. 0.0.0:0

标签 sockets networking network-programming libevent

在我的程序中,我在 libevent 事件循环中为 connfd 注册了一个 EV_READ 事件。 触发此事件时,我使用 getpeername 获取对等方的 IP/PORT 地址

socklen_t socklen;
struct sockaddr_in client_addr;
socklen = sizeof(client_addr);
retval = getpeername(connfd, (struct sockaddr *)&client_addr, &socklen);
if(retval == -1) perror("getpeername error!\n");

但有时,它会返回 0.0.0.0:0

然后我注意到错误是Transport endpoint is not connected

但 recv(connfd,buf,..) 返回 1273 ,这意味着它接收到 1273 个字节。 如果连接结束,recv() 如何获取字节? 以及如何触发事件?

谢谢!

最佳答案

正如@Celada 指出的那样,getpeername()struct sockaddr * 上工作,在您的情况下,它的大小是 struct sockaddr_in,不足以保存有关 IPv6 地址的信息。

这个 struct sockaddr 将被实现为 struct sockaddr_instruct sockaddr_in6 取决于套接字是处理 IPv4 还是一个 IPv6 地址。

您正在分配一个 16 字节的 struct sockaddr_in(仅用于 IPv4 地址的空间)。 IPv6 需要一个 struct sockaddr_in6,它是 28 个字节,在前 16 个字节之后存储它的 IP 地址。因此将 struct sockaddr_in 读取为 struct sockaddr_in6

下一部分解释了为什么您可能会阅读 0.0.0.0:0。如果您对解决方案更感兴趣,请直接转到最后一节。


为什么要用当前代码读取一些归零的 IP 地址?

下面是这些结构的定义:

struct sockaddr_in {
    short            sin_family;   // e.g. AF_INET, AF_INET6 <-- 2 bytes
    unsigned short   sin_port;     // e.g. htons(3490) <-- 2 bytes
    struct in_addr   sin_addr;     // (only contains IPv4 address) <-- 4 bytes
    char             sin_zero[8];  // zero this if you want to <-- 8 bytes
};

struct sockaddr_in6 {
    u_int16_t       sin6_family;   // address family, AF_INET6 <-- 2 bytes
    u_int16_t       sin6_port;     // port number, Network Byte Order <-- 2 bytes
    u_int32_t       sin6_flowinfo; // IPv6 flow information <-- 4 bytes
    struct in6_addr sin6_addr;     // IPv6 address <-- 16 bytes
    u_int32_t       sin6_scope_id; // Scope ID <-- 4 bytes
};

来源:Beej's Guide to Network Programming's struct sockaddr and pals page

因此,当您尝试从 struct sockaddr_in 读取 IP 地址时,getpeername()struct sockaddr_in6 信息填充它,< strong>您实际上正在阅读 struct sockaddr_in6sin6_flowinfo 部分。该流信息通常为零,除非连接明确标记有流 ID。 这就是为什么您会读取归零的 IP 地址。


如何提供通用存储以便 getpeername() 在任何一种情况下都能正常工作,而不可能(确实)在对等方的连接类型上有超前的情报?

这就是 struct sockaddr_storage 发挥作用的地方!

这是您应该用来初始化空间以提供 getpeername() 的类型,将其转换为 struct sockaddr *。它旨在保存任何连接类型的信息。

您的代码应该是:

socklen_t socklen;
struct sockaddr_storage client_addr;
socklen = sizeof(client_addr);
retval = getpeername(connfd, (struct sockaddr *)&client_addr, &socklen);
if(retval == -1) perror("getpeername error!\n");

要使用该结构,您应该将其转换回 struct sockaddr_instruct sockaddr_in6,具体取决于 ss_family 的值struct sockaddr_storage 的字段。

if (addr.ss_family == AF_INET) {
    struct sockaddr_in *s = (struct sockaddr_in *)&addr;
    port = ntohs(s->sin_port);
    inet_ntop(AF_INET, &s->sin_addr, ipstr, sizeof ipstr);
} else if(addr.ss_family == AF_INET6) {
    struct sockaddr_in6 *s = (struct sockaddr_in6 *)&addr;
    port = ntohs(s->sin6_port);
    inet_ntop(AF_INET6, &s->sin6_addr, ipstr, sizeof ipstr);
}

来源(已修改):Beej's Guide to Network Programming's getpeername() page

是的,由于 IP 地址长度从 IPv4 到 IPv6 不同,因此没有通用函数为您执行此操作,返回 IP 地址...因为您仍然需要知道您收到的是 32 位还是 128 位,因此再次检查地址族!

关于sockets - 在什么情况下 getpeername 返回 IP :PORT 0. 0.0.0:0,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17220006/

相关文章:

c++ - c/c++ 中的网络拓扑

javascript - 连接到表单提交 Node js上的套接字

java - 网络连接断开时如何使Java Client Socket保持 Activity 状态?

Azure 私有(private) DNS 配置不适用于 P2S VPN

sockets - sslscan.c :94:25: fatal error: openssl/err. h : No such file or directory compilation terminated. #163

c - net-snmp解析代码,如何解析MIB?

linux - 从内核空间中给定的结构套接字获取结构文件

java - 在java中使用套接字的本地客户端的ID

c - 在 c 中通过 TCP 发送 MQTT 连接数据包

java - 区分同一台计算机上的多个客户端