NAT概念
网络地址转换。出口网关,将内网IP地址和端口,映射成出口网关公网IP地址和端口。
是为了解决IP地址不够用而产生的路由器解决方案。
NAT类型
https://www.cnblogs.com/colin-vio/p/13323228.html
- 锥形NAT,在内网某个客户端向外网主机ip:port发送过数据包的前提下,当网关收到外网不同ip,不同port发过来的数据包时,网关根据内部缓存的映射表找到目的IP地址和端口,然后向内网转发。
- IP限制锥形NAT,在内网某个客户端向外网主机ip:port发送过数据包的前提下,当网关收到外网相同ip,不同port发过来的数据包时,网关根据内部缓存的映射表找到目的IP地址和端口,然后向内网转发。
- 端口限制锥形NAT,在内网某个客户端向外网主机ip:port发送过数据包的前提下,只有该外网相同ip,相同port的数据包才能进入内网。
- 对称NAT,在安全策略上,和端口限制锥形NAT时一样的。
锥形NAT和对称NAT的区别是,客户端用同一个本地端口同时请求不同的外网服务器,锥形NAT分配相同的端口号,对称NAT会分配不同的端口号。
1,2,3的区别在于安全策略上。
测试
测试环境的网络拓扑结构
p2p server在树莓派
p2p client在vm虚拟机和Hyper V虚拟机
代码
server.c
#include <stdio.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/select.h>
int main(int argc,char **argv){
if(argc < 2){
return 0;
}
int port = atoi(argv[1]);
int server_fd;
struct sockaddr_in server_addr;
socklen_t server_addr_len;
server_fd = socket(AF_INET,SOCK_DGRAM,0);
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
bind(server_fd,(struct sockaddr*)&server_addr,sizeof(server_addr));
char tmp[1024] = {
0},tmp2[1024];
int len;
struct sockaddr_in client_addr,client2_addr;
socklen_t client_addr_len=sizeof(client_addr),client2_addr_len=sizeof(client2_addr);
for(;;){
len = recvfrom(server_fd,tmp,sizeof(tmp)