0

我正在开发一个后效插件,我正在尝试集成 raknet,它是一个 c++ 网络库。当 raknet 库试图通过调用获取 ipv4 地址时

gethostbyname

然后它抛出错误访问冲突读取位置 0xFFFFFFFFFFFFFFFF

int idx=0;
char ac[ 80 ];
int err = gethostname( ac, sizeof( ac ) );
(void) err;
RakAssert(err != -1);

struct hostent *phe = gethostbyname( ac );

if ( phe == 0 )
{
RakAssert(phe!=0);
return ;
}
for ( idx = 0; idx < MAXIMUM_NUMBER_OF_INTERNAL_IDS; ++idx )
{
if (phe->h_addr_list[ idx ] == 0)
    break;

memcpy(&addresses[idx].address.addr4.sin_addr,phe->h_addr_list[ idx ], sizeof(struct in_addr));
}

while (idx < MAXIMUM_NUMBER_OF_INTERNAL_IDS)
{
addresses[idx]=UNASSIGNED_SYSTEM_ADDRESS;
idx++;
}

这是我看到的一些图片。

http://jacobsgriffith.com/stackoverflow/noaccesserror.png

我读过这个,看起来图书馆没有错误地实现它。 关于 gethostbyname 的 Microsoft 文档

当我将鼠标悬停在 h_addr_list 和 h_aliases 上时,我得到 .

http://jacobsgriffith.com/stackoverflow/noaccess.jpg

有人有什么想法吗?为什么会失败我很确定这是一个常见的功能。

另一件事,winsock 和 winsock2 的 gethostbyname 函数的实现有什么区别吗?

4

2 回答 2

1

我很惊讶 Windows 实现不是线程安全的,并且对每个主机都使用线程本地存储。但无论如何...

只需使用getaddrinfo来解析主机名。它是线程安全的,旨在替代 gethostname。

但您的最终目标是枚举盒子上的本地 IP 地址。在这种情况下,只需在 UNIX 上使用getifaddrs并在 Windows 上使用 GetAdaptersInfo 和 GetAdatperAddresses的组合来枚举本地 IP 地址。您还可以在 Windows上将 SIO_ADDRESS_LIST_QUERY ioctl与虚拟套接字一起使用。

于 2013-03-07T08:05:13.103 回答
0

这是我想出的解决方案。显然 gethostname 不是线程安全的。After Effects 是多线程的。

我换了这个

int idx=0;
char ac[ 80 ];
int err = gethostname( ac, sizeof( ac ) );
(void) err;
RakAssert(err != -1);

struct hostent *phe = gethostbyname( ac );

if ( phe == 0 )
{
    RakAssert(phe!=0);
    return ;
}
for ( idx = 0; idx < MAXIMUM_NUMBER_OF_INTERNAL_IDS; ++idx )
{
    if (phe->h_addr_list[ idx ] == 0)
        break;

    memcpy(&addresses[idx].address.addr4.sin_addr,phe->h_addr_list[ idx ],sizeof(struct in_addr));
}

while (idx < MAXIMUM_NUMBER_OF_INTERNAL_IDS)
{
    addresses[idx]=UNASSIGNED_SYSTEM_ADDRESS;
    idx++;
}

有了这个

int idx=0;
struct addrinfo* feed_server = NULL;
struct addrinfo hints;
memset(&hints, 0, sizeof(struct addrinfo));
hints.ai_family = AF_INET;
getaddrinfo("localhost", NULL, &hints, &feed_server);
struct addrinfo *res;
for(res = feed_server; res != NULL; res = res->ai_next) {
    struct sockaddr_in* saddr = (struct sockaddr_in*)res->ai_addr;
    //char* ipv4Str = inet_ntoa(saddr->sin_addr);
    memcpy(&addresses[idx].address.addr4.sin_addr, &saddr->sin_addr, sizeof(struct in_addr));
    idx++;
} 
while (idx < MAXIMUM_NUMBER_OF_INTERNAL_IDS) {
    addresses[idx]=UNASSIGNED_SYSTEM_ADDRESS;
    idx++;
}

这就是我走那条路的原因。有人反对吗?

gethostname 不是线程安全的

于 2013-03-07T07:39:24.630 回答