在我的程序中,我在libevent事件循环中为connfd注册了一个EV_READ事件。当触发此事件时,我使用 getpeername
获取对等方的 IP/端口地址
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_in
或struct sockaddr_in6
实现,具体取决于套接字是处理 IPv4 还是 IPv6 地址。
您正在分配 16 个字节(仅 IPv4 地址的空间(的struct sockaddr_in
。IPv6 需要一个 28 字节的struct sockaddr_in6
,并将其 IP 地址存储在前 16 个字节之后。因此将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
信息填充它时,您实际上是在读取struct sockaddr_in6
的sin6_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_in
或 struct sockaddr_in6
,具体取决于struct sockaddr_storage
的 ss_family
字段的值。
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 位,因此再次检查地址系列!