2

在我的程序中,我在 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() 如何获取字节?以及如何触发事件?

谢谢!

4

1 回答 1

2

正如@Celada 指出的那样,在您的情况下,getpeername()适用于 a 的大小,它的大小不足以容纳有关 IPv6 地址的信息。struct sockaddr *struct sockaddr_in

struct sockaddr将被实现为 astruct sockaddr_in或 a ,struct sockaddr_in6具体取决于套接字是处理 IPv4 还是 IPv6 地址。

您正在分配struct sockaddr_in16 个字节的 a(仅用于 IPv4 地址的空间)。IPv6 需要struct sockaddr_in628 个字节的 a 并在第 16 个字节之后存储其 IP 地址。因此将 a 读struct sockaddr_in作 astruct 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 的网络编程指南struct sockaddr 和 pals页面

因此,当您尝试读取一段struct sockaddr_in时间getpeername()内填充struct sockaddr_in6信息的 IP 地址时,您实际上是在sin6_flowinfo读取struct sockaddr_in6. 该流信息通常为零,除非使用流 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 的网络编程指南的getpeername()页面

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

于 2014-09-03T09:28:20.800 回答