我需要为现有的仅支持IPv4的基于套接字的应用程序添加IPv6支持.为了迎接即将到来的IPv6时代,有人命令我为应用程序添加IPv6接口,并在尝试与应用程序通信时让外界选择IPv4接口或IPv6接口.
我的问题是:对于Linux中的套接字处理API级别,处理基于IPv4的套接字和基于IPv6的套接字之间没有区别吗?
更多的是,是否可以让套接字侦听具有相同端口的两个IP地址?如果这是真的,那么实现这个要求是一项微不足道的工作,我想.
解决方法
用1个TCP套接字监听2个不同的IP地址是不可能的,但是如果你使用
in6addr_any地址监听所有接口,那么也将包括所有的IPv4地址(虽然我相信例如linux有一个内核选项来禁用该映射).
对于您是使用IPv4还是IPv6,(较新版本)套接字API非常透明,但必须非常谨慎地处理IPv4应用程序的编码方式.
例如此IPv4代码接受连接并打印出远程主机的地址:
struct sockaddr_in client_addr; socklen_t addr_len = sizeof(client_addr); client_data->fd = accept(server_fd,(struct sockaddr*)&client_addr,&addr_len); log_printf("New client from %s\n",inet_ntoa(client_addr.sin_addr.s_addr));
必须转换为以下内容,它处理IPv4和IPv6
struct sockaddr_storage client_addr; char numeric_addr[INET6_ADDRSTRLEN]; socklen_t addr_len = sizeof(client_addr); client_data->fd = accept(server_fd,&addr_len); if(client_addr.ss_family == AF_INET) log_printf("New client from %s\n",inet_ntop(client_addr.ss_family,((struct sockaddr_in*)&client_addr)->sin_addr.s_addr,numeric_addr,sizeof numeric_addr)); else if(client_addr.ss_family == AF_INET6) log_printf("New client from %s\n",((struct sockaddr_in6*)&client_addr)->sin6_addr,sizeof numeric_addr));
虽然我相信你可以用getaddrinfo()更优雅和透明地做到这一点
以下是有关IP层独立性的附加说明:
http://uw714doc.sco.com/en/SDK_netapi/sockC.PortIPv4appIPv6.html
http://www.kame.net/newsletter/19980604/