为什么我无法将IPv6套接字绑定到本地链路地址?

11
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
#include <stdio.h>

void error(char *msg)
{
  perror(msg);
  exit(0);
}
int main(int argc, char *argv[])
{
   int sock, length, fromlen, n;
   struct sockaddr_in6 server;
   struct sockaddr_in6  from;

   int portNr = 5555;
   char buf[1024];

   length = sizeof (struct sockaddr_in6);

   sock=socket(AF_INET6, SOCK_DGRAM, 0);
   if (sock < 0) error("Opening socket");

   bzero((char *)&server, length);
   server.sin6_family=AF_INET6;
   server.sin6_addr=in6addr_any;
   server.sin6_port=htons(portNr);

   inet_pton( AF_INET6, "fe80::21f:29ff:feed:2f7e", (void *)&server.sin6_addr.s6_addr);
   //inet_pton( AF_INET6, "::1", (void *)&server.sin6_addr.s6_addr);

   if (bind(sock,(struct sockaddr *)&server,length)<0)
       error("binding");
   fromlen = sizeof(struct sockaddr_in6);
   while (1) {
       n = recvfrom(sock,buf,1024,0,(struct sockaddr *)&from,&fromlen);
       if (n < 0) error("recvfrom");
       write(1,"Received a datagram: ",21);
       write(1,buf,n);
       n = sendto(sock,"Got your message\n",17,
                  0,(struct sockaddr *)&from,fromlen);
       if (n  < 0) error("sendto");
   }
}

当我编译和运行上述代码时,我得到了:
binding: Invalid argument

如果将绑定更改为::1,并在源代码中保持其他内容不变,则代码可以正常工作!所以你能告诉我我的代码有什么问题吗?提前感谢。

1个回答

17

对于链接本地地址,您还需要指定与该地址关联的网络接口的范围ID...类似这样:

server.sin6_scope_id = 5;   /* or whatever the scope ID is for the network interface you want to communicate over */
你可以使用getifaddrs()函数来查找系统上可用的各种作用域ID及其相应的网络接口。
(是的,这有点麻烦...或者你可以尝试将"%en0"附加到传递给inet_pton()的字符串末尾,inet_pton()可能会为你完成工作...我不确定inet_pton()是否处理该语法)

3
inet_pton不处理附加%if的字符串。你可以使用getaddrinfo来处理带有%符号的IPv6地址字符串表示。请参阅手册页面http://linux.die.net/man/3/getaddrinfo - Benjamin Maurer
3
对于阅读此答案的人,请注意 scope_id 是您想要通信的接口的索引。由于每个接口默认都有 fe80,因此我们需要指定该链路本地地址所指的接口。因此,值 5 被用作示例。要获取该索引,您需要使用 if_nametoindex (const char *ifname) 函数从接口名称中获取。接口名称是通过遍历由 getifaddrs(struct ifaddrs **ifap) 填充的数据结构返回的。 - wget

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接