Unix网络编程学习笔记之第12章 IPv4与IPv6的互操作性

一、 简介

假设我们本章讨论的主机都是支持双栈的,即支持IPv4地址,也支持Ipv6地址。

我们本次讨论的点:客户端与服务器端使用的是不同类型的地址。因为相同类型的地址没什么可讲的。

二、 IPv4客户端与IPv6服务器

即,客户端使用IPv4地址套接字来通信,服务器端使用IPv6地址套接字通信。

原理:

0. 首先IPv6服务器主机保证既有IPv4地址,又有IPv6地址。

1. IPv4客户端通过getaddrinfo函数,找到服务器端的IPv4地址,然后进行连接。

2. 来自客户端的IPv4的SYN到达服务器端,服务器端内核就把这个IPv4的SYN,映射为IPv6的SYN(就是把IPv4的IP地址映射为IPv6的IP地址),然后交给进程。所以服务器端accept、recvfrom的套接字地址都是IPv6的。

3. 然后服务器端进程进行响应ACK,进程发送一个IPv6的ACK到内核,内核查看目的地址,就知道这个IPv6的地址是经过映射过来的。所以内核就是这个IPv6地址映射为IPv4地址发送出去。

4. 这样就正常通信了,即实际上还是通过IPv4包进行通信的。然而服务器和客户端进程都不知情的,这些工作由服务器内核完成。

注意:如果IPv6服务器主机没有IPv4地址,则这样的通信无法完成。

三、 IPv6客户端与IPv4服务器

即,客户端使用IPv6地址套接字来通信,服务器端使用IPv4地址套接字通信。

原理:

0. 首先IPv4服务器主机中没有IPv6地址。

1. IPv6客户端通过getaddrinfo函数,且hints的结构中的标志为AI_V4MAPPED。则我们就可以通过getaddrinfo函数得到服务器主机的IPv4地址映射的IPv6地址。

2. 然后客户端使用这个IPv6地址调用connect,而内核检测出这个IPv6的地址是映射的,所以就会把这个IPv6地址转成IPv4地址,然后发送出去。

3. 这样IPv4服务器就得到了IPv4的SYN,然后响应IPv4的ACK。

4. 客户端接收到这个IPv4的ACK后,内核就会把这个IPv4的ACK转为IPv6的ACK,从而进行通信。

5. 实际上还是通过IPv4的包进行通信的。服务器和客户端进程不知情。由客户端内核完成。

注意:如果此时IPv4服务器主机有IPv6地址,则尽管hints的结构中的标志为AI_V4MAPPED,但是getaddrinfo也返回原有的IPv6地址,这样的话,双方无法完成通信。这时我们可以通过在主机名上加-4,来规定只查询A记录。这样就可以通信了。

四、 错误的组合

上述的两种错误情况,都是因为内核无法将IPv6地址转为为IPv4地址,因为IPv6有128位,而IPv4地址只有32位。

总结一下:即

即:

1. 显然如果双方都是单栈主机,也双方的协议必须相同。

2. IPv4双栈客户,无法与IPv6的单栈服务器通信。即上面的第1种错误情况。

3. Ipv6单栈客户,无法与Ipv4的双栈服务器通信。

4. IPv6双栈客户是否可以与IPv4双栈客户通信取决于实现,即如果getaddrinfo获取的是IPv4映射的IPv6地址,则可以通信。如果获取的是真正的IPv6地址,则无法通信。

5. 其实问题就是在于实现了IPv6的主机上,尽量也要实现IPv4,这样的话,我们可以看到,去除表的第二行和第而列,则表中的(无)就没有了。只剩下(无*)。

五、 IPv6地址测试宏

有一些IPv6应用程序想要知道,某个IPv6地址到底是IPv4映射过来的,还是本身就是IPv6地址。我们可以使用宏:

#include     <netinet/in.h>
int IN6_IS_ADDR_V4MAPPED(const structin6_addr* apt);

来进行检测。

六、 建议

尽量写一些与具体协议无关的函数。

尽量使用与具体协议无关的函数。

还有从上述的表中可以看出,我们在编写服务器时,在服务器主机支持双栈协议的情况下,把服务器的地址写成IPv6地址,这样可以接收任何协议的客户端。

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-30 21:09:42

Unix网络编程学习笔记之第12章 IPv4与IPv6的互操作性的相关文章

Unix网络编程学习笔记之第11章 名字与地址转换

一. 域名系统(DNS) 1. 简介 DNS主要用于主机名和IP地址之间的映射. 主机名可以是简单的名字ljm,也可以是全限定域名ljm.localdomainbaidu.com等. 2.资源记录 DNS中的条目称为资源记录(RR).我们感兴趣的RR类型只有几个: A             A记录把一个主机名映射为一个32位的IPv4地址. AAAA    4A记录把一个主机名映射为一个128位的IPv6地址. 例如: ljm               IN      A    127.0.

Unix网络编程学习笔记之第7章 套接字选项

一.获取/设置套接字选项的方法 一个套接字描述符相关联的套接字选项很多.获取/设置套接字选项的方法: 1.  getsockopt和setsockopt函数 2. fcntl函数 3. ioctl函数 二. getsockopt和setsockopt函数 int getsockopt(int sockfd, int level, int optname, void* optval, socklen_t* optlen); int setsockopt(int sockfd, int level,

Unix网络编程学习笔记之第2章 TCP和UDP

TCP 1. TCP面向连接的协议,是一个字节流协议,没有任何记录边界.发送的是数据分组. 2. TCP提供了可靠性:确认重传和重组 (1) TCP每发送一份数据都会要求对端进行确认.如果超时,就会重传.TCP会估计往返时间RTT,以确定等待多长时间重传. (2) 如果多次发送数据分组,TCP可以保证分组的按序达到.即会根据序列号进行重组. 3. TCP提供流量控制 TCP在任何时刻通知对端,它此时一次能够接受多少字节的数据,即通告窗口.该窗口指出接受缓冲区当前可用的空间. 4. 为何说TCP是

Unix网络编程学习笔记之第6章 I/O复用:select和poll函数

一.I/O复用应用场合 1. 当客户处理多个描述符(既有标准输入,又有网络套接字)时,必须使用IO复用. 2. 一个客户同时处理多个套接字是可能的. 3. 如果一个服务器既要处理监听套接字,又要处理已连接套接字,一般就要使用I/O复用. 4. 如果一个服务器既要处理TCP,又要处理UDP,一般就要I/O复用. 5. 如果一个服务器要处理多个服务或协议,就要用到I/O复用. 其实IO复用就是一个进程/线程处理多个套接字描述符. 二. I/O模型 Unix提供了5种I/O模型: 1. 阻塞式I/O模

Unix网络编程学习笔记之第8章 基于UDP套接字编程

一. UDP C/S的典型函数调用 UDP没有像TCP那样的连接,客户端直接sendto向某服务器发送数据,服务器端一直recvfrom阻塞,以接收任何客户端发送的数据. 二. sendto和recvfrom函数 int sendto(int sockfd, const void* buff, size_t nbytes, int flag, const struct sockaddr* to, socklen_taddrlen); int recvfrom(int sockfd, void*

Unix网络编程学习笔记之第1章 简介

一.一个简单的时间获取客户端 #include <sys/socket.h> #define MAXCON 50 #define MAXLINE 1024 #define PORT 13 void err_sys(const char* s) { fprintf(stderr, "%s\n",s); exit(1); } int main(int argc, char** argv) { int sockfd; structsockaddr_in servaddr; cha

Unix网络编程学习笔记之第5章 TCP客户端/服务器程序示例

一. 一个简单TCP回射服务端程序 #include "unp.h" #define MAXLINE 1024 #define PORT 13 #define CONMAX 5 void err_sys(const char* s) { fprintf(stderr, "%s\n",s); exit(1); } void str_echo(int connfd) { int nbyte; char buff[MAXLINE+1]; again: while(nbyt

Unix网络编程学习笔记之第4章 基于TCP套接字编程

1. socket函数 int socket(int family, int type,int protocol) 成返回一个套接字描述符.错误返回-1 其中family指定协议族,一般IPv4为AF_INET, IPv6为AF_INET6. 其中type指定套接字类型,字节流:SOCK_STREAM.   数据报:SOCK_DGRAM. 一般情况下通过family和type的组合都可以唯一确定一个套接字类型.所以一般我们就把protocol设为0就可以了. 有时在某些特殊情况下,family和

UNIX网络编程学习笔记2 需要用到的一些字节操纵和格式转换函数

当然这些东西是炒鸡无聊的,但是真当自己开始撸代码时才发现熟悉这些枯燥的函数能够节约大量的时间.于是总结一下: 字节序:低序字节存储在起始地址,这称为小端(little-endian),高序字节存储在起始地址,这称为大端(big-endian) 例:存放0x0A0B0C0D LE: 0D 0C 0B 0A BE: 0A 0B 0C 0D 小端的存放方式更加接近于人类思维 网际协议使用大端字节序来传送多字节整数(为何要规定一个字节序来传输ip和port? 呸 这样子协议才能正确“看懂”(解释)这些地