TUN/TAP设备浅析

https://blog.csdn.net/lishuhuakai/article/details/73136442
TUN/TAP设备浅析
TUN设备
TUN 设备是一种虚拟网络设备,通过此设备,程序可以方便地模拟网络行为。TUN 模拟的是一个三层设备,也就是说,通过它可以处理来自网络层的数据,更通俗一点的说,通过它,我们可以处理 IP 数据包。

先来看看物理设备是如何工作的:
在这里插入图片描述

上图中的 eth0 表示我们主机已有的真实的网卡接口 (interface)。

网卡接口 eth0 所代表的真实网卡通过网线(wire)和外部网络相连,该物理网卡收到的数据包会经由接口 eth0 传递给内核的网络协议栈(Network Stack)。然后协议栈对这些数据包进行进一步的处理。

对于一些错误的数据包,协议栈可以选择丢弃;对于不属于本机的数据包,协议栈可以选择转发;而对于确实是传递给本机的数据包,而且该数据包确实被上层的应用所需要,协议栈会通过 Socket API 告知上层正在等待的应用程序。

下面看看 TUN 的工作方式:

在这里插入图片描述
在这里插入图片描述

我们知道,普通的网卡是通过网线来收发数据包的话,而 TUN 设备比较特殊,它通过一个文件收发数据包。

如上图所示,tunX 和上面的 eth0 在逻辑上面是等价的, tunX 也代表了一个网络接口,虽然这个接口是系统通过软件所模拟出来的.

网卡接口 tunX 所代表的虚拟网卡通过文件 /dev/tunX 与我们的应用程序(App) 相连,应用程序每次使用 write 之类的系统调用将数据写入该文件,这些数据会以网络层数据包的形式,通过该虚拟网卡,经由网络接口 tunX 传递给网络协议栈,同时该应用程序也可以通过 read 之类的系统调用,经由文件 /dev/tunX 读取到协议栈向 tunX 传递的所有数据包。

此外,协议栈可以像操纵普通网卡一样来操纵 tunX 所代表的虚拟网卡。比如说,给 tunX 设定 IP 地址,设置路由,总之,在协议栈看来,tunX 所代表的网卡和其他普通的网卡区别不大,当然,硬要说区别,那还是有的,那就是 tunX 设备不存在 MAC 地址,这个很好理解,tunX 只模拟到了网络层,要 MAC地址没有任何意义。当然,如果是 tapX 的话,在协议栈的眼中,tapX 和真是网卡没有任何区别。

如果我们使用 TUN 设备搭建一个基于 UDP 的 VPN ,那么整个处理过程可能是这幅样子:
在这里插入图片描述

首先,我们的应用程序通过 eth0 和远程的 UDP 程序相连,对方传递过来的 UDP 数据包经由左边的协议栈传递给了应用程序,UDP 数据包的内容其实是一个网络层的数据包,比如说 IP 数据报,应用程序接收到该数据包的数据(剥除了各种头部之后的 UDP 数据)之后,然后进行一定的处理,处理完成后将处理后的数据写入文件 /dev/tunX,这样,数据会第二次到达协议栈。需要注意的是,上图中绘制的两个协议栈其实是同一个协议栈,之所以这么画是为了叙述的方便。

TAP设备
TAP 设备与 TUN 设备工作方式完全相同,区别在于:

TUN 设备是一个三层设备,它只模拟到了 IP 层,即网络层 我们可以通过 /dev/tunX 文件收发 IP 层数据包,它无法与物理网卡做 bridge,但是可以通过三层交换(如 ip_forward)与物理网卡连通。可以使用ifconfig之类的命令给该设备设定 IP 地址。
TAP 设备是一个二层设备,它比 TUN 更加深入,通过 /dev/tapX 文件可以收发 MAC 层数据包,即数据链路层,拥有 MAC 层功能,可以与物理网卡做 bridge,支持 MAC 层广播。同样的,我们也可以通过ifconfig之类的命令给该设备设定 IP 地址,你如果愿意,我们可以给它设定 MAC 地址。
最后,关于文章中出现的二层,三层,我这里说明一下,第一层是物理层,第二层是数据链路层,第三层是网络层,第四层是传输层。

其实关于这两种设备的编程,基本上属于八股文,大家一般都这么干。

启动设备之前
有的linux 并没有将tun 模块编译到内核之中,所以,我们要做的第一件事情就是检查我们的系统是否支持 TUN/TAP 。具体如何检查和解决,请查看这里http://blog.csdn.net/lishuhuakai/article/details/70305543,这篇文章就不再赘述。

光有tun 模块还不够,我们还要创建上篇文章中所提到的文件,运行命令:

% sudo mknod /dev/net/tun c 10 200 # c表示为字符设备,10和200分别是主设备号和次设备号

这样,你到 /dev/net/ 目录下就可以看到一个名称为 tun 的文件了。当然这里的 tun 可以改成任意的你喜欢的名称。

启动设备
对于TUN设备,我们一般这样来初始化:

int 
tun_alloc(char dev[IFNAMSIZ]) // dev数组用于存储设备的名称
{
  struct ifreq ifr;
  int fd, err;

  if ((fd = open("/dev/net/tun", O_RDWR)) < 0) { // 打开文件
    perror("open");
    return -1;
  }

  bzero(&ifr, sizeof(ifr));

  /* Flags : IFF_TUN   - TUN设备
   *         IFF_TAP   - TAP设备
   *         IFF_NO_PI - 不需要提供包的信息
   */

  ifr.ifr_flags = IFF_TUN | IFF_NO_PI; // tun设备不包含以太网头部,而tap包含,仅此而已

  if (*dev) {
    strncpy(ifr.ifr_name, dev, IFNAMSIZ); 
  }

  if ((err = ioctl(fd, TUNSETIFF, (void *) &ifr)) < 0) { // 打开设备
    perror("ioctl TUNSETIFF");
    close(fd);
    return err;
  }

  // 一旦设备开启成功,系统会给设备分配一个名称对于tun设备,一般为tunX,X为从0开始的编号,对于tap设备
  // 一般为tapX,X为从0开始的编号
  strcpy(dev, ifr.ifr_name); // 拷贝设备的名称至dev中
  return fd;
}

如果我们想启动一个TAP 设备的话,很简单,将上面的ifr.ifr_flags = IFF_TUN | IFF_NO_PI;改为ifr.ifr_flags = IFF_TAP | IFF_NO_PI;即可,那么我们就启动了一个 TAP 设备。

设定网络地址
上面的代码打开了文件,并且返回了文件的描述符,但是还不够,对于一张网卡来说,我们还要给其配置网络地址,有时候甚至是路由信息,网卡才能够正常地工作。

一旦虚拟的 TUN/TAP 设备启动成功,我们便可以通过命令来给其设定地址。

我来举个例子,以一个 TAP 设备为例:

% sudo ip link set dev tap0 up # 启动tap0网卡,虽然网卡已经启动,但是此时使用ipconfig命令并不能看到tap0这个设备,因为我们还没有给其配置ip地址

% sudo ip address add dev tap0 10.0.1.5/24 # 给tap0设置ip地址

% ifconfig  # 此时在ifconfig命令下已经可以看到tap0设备了
tap0: flags=4163<UP,BROADCAST,RUNNING,MULTICAST>  mtu 1500
        inet 10.0.1.5  netmask 255.255.255.0  broadcast 0.0.0.0
        inet6 fe80::1872:80ff:fe20:46e2  prefixlen 64  scopeid 0x20<link>
        ether 1a:72:80:20:46:e2  txqueuelen 1000  (Ethernet)
        RX packets 0  bytes 0 (0.0 B)
        RX errors 0  dropped 0  overruns 0  frame 0
        TX packets 0  bytes 0 (0.0 B)
        TX errors 0  dropped 0 overruns 0  carrier 0  collisions 0


% ip route show # 显示所有的路由信息
default via 192.168.140.2 dev ens33 proto static metric 100
10.0.0.0/24 dev ens39 proto kernel scope link src 10.0.0.130 metric 100
10.0.1.0/24 dev tap0 proto kernel scope link src 10.0.1.5 # 给网卡设定ip后,系统自动添加了路由
192.168.140.0/24 dev ens33 proto kernel scope link src 192.168.140.133 metric 100

通过手动敲命令的方式来配置 tap0 设备,略显麻烦,其实我们可以直接在程序中调用 system 函数:

int 
run_cmd(char *cmd, ...)
{
    va_list ap;
    char buf[CMDBUFLEN];
    va_start(ap, cmd);
    vsnprintf(buf, CMDBUFLEN, cmd, ap);
    va_end(ap);
    if (debug) { // DEBUG模式下输出信息
        printf("EXEC: %s\n", buf);
    }
    return system(buf);
}

将上面的命令直接传递给 run_cmd 函数即可.

当然,如果你不喜欢这种方式,我们自然还可以有其他的方法,比如说使用下面的函数:

int
set_stack_attribute(char *dev)
{
    struct ifreq ifr;
    struct sockaddr_in addr;
    int sockfd, err = -1;

    bzero(&addr, sizeof(addr));
    addr.sin_family = AF_INET;
    inet_pton(AF_INET, tapaddr, &addr.sin_addr);

    bzero(&ifr, sizeof(ifr));
    strcpy(ifr.ifr_name, dev);
    bcopy(&addr, &ifr.ifr_addr, sizeof(addr));
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        perror("socket");
        return -1;
    }

    // ifconfig tap0 10.0.1.5 #设定ip地址
    if ((err = ioctl(sockfd, SIOCSIFADDR, (void *)&ifr)) < 0) {
        perror("ioctl SIOSIFADDR");
        goto done;
    }

    /* 获得接口的标志 */
    if ((err = ioctl(sockfd, SIOCGIFFLAGS, (void *)&ifr)) < 0) {
        perror("ioctl SIOCGIFADDR");
        goto done;
    }

    /* 设置接口的标志 */
    ifr.ifr_flags |= IFF_UP;
    // ifup tap0 #启动设备
    if ((err = ioctl(sockfd, SIOCSIFFLAGS, (void *)&ifr)) < 0) {
        perror("ioctl SIOCSIFFLAGS");
        goto done;
    }

    inet_pton(AF_INET, "255.255.255.0", &addr.sin_addr);
    bcopy(&addr, &ifr.ifr_netmask, sizeof(addr));
    // ifconfig tap0 10.0.1.5/24 #设定子网掩码
    if ((err = ioctl(sockfd, SIOCSIFNETMASK, (void *) &ifr)) < 0) {
        perror("ioctl SIOCSIFNETMASK");
        goto done;
    }


done:
    close(sockfd);
    return err;
}

上面的函数主要干的事情和上面的命令大致相同。

收发数据
收发数据非常简单,每次读取返回的文件描述符即可接收数据,没有数据到来时,会一直阻塞在哪里,当然,你也可以玩一下非阻塞 IO,然后想要发送数据的话,只需要将数据写入到该文件描述符对应的文件中即可。

上一篇文章主要讲述了TUN/TAP设备的一些原理,你可能会好奇,TUN/TAP设备究竟有什么用处呢?所以这篇文章,我想用一些实际的例子来回答这个问题。

例子源自陈硕老师的博客,博文中关于TUN/TAP设备的使用非常典型,对原文感兴趣的同学可以查看这里:http://blog.csdn.net/solstice/article/details/6579232

背景:在一台 PC 机上模拟 TCP 客户端程序发起连接请求,同时在该 PC 上创建虚拟网卡 tun0,接收连接请求

并送至 faketcp 应用程序,用于模拟 TCP 服务器端进行响应。

网络的拓扑结构如下:

拓扑结构

具体做法是:在主机 atom 上通过打开 /dev/net/tun 设备来创建一个 tun0 虚拟网卡,然后把这个网卡的地址设为192.168.0.1/24,这样 faketcp 程序就扮演了192.168.0.0/24 这个网段上的所有机器。atom 发给192.168.0.2 ~ 192.168.0.254的 IP 数据包都会发给 faketcp 程序,faketcp 程序可以模拟其中任何一个IP 给atom 发 IP 数据包。

程序分成几步来实现。

第一步:实现 icmp echo 协议,这样就能 ping 通 faketcp 了:

faketcp.h:

#include <algorithm>  // std::swap

#include <assert.h>
#include <stdint.h>
#include <string.h>
#include <arpa/inet.h>  // inet_ntop
#include <net/if.h>

struct SocketAddr
{
  uint32_t saddr, daddr;  // 源地址和目的地址
  uint16_t sport, dport;  // 源端口和目的端口

  bool operator==(const SocketAddr& rhs) const
  {
    return saddr == rhs.saddr && daddr == rhs.daddr && 
        sport == rhs.sport && dport == rhs.dport;
  }

  bool operator<(const SocketAddr& rhs) const
  {
    return memcmp(this, &rhs, sizeof(rhs)) < 0;
  }
};

int tun_alloc(char dev[IFNAMSIZ]);
uint16_t in_checksum(const void* buf, int len);

void icmp_input(int fd, const void* input, const void* payload, int len);

faketcp.cc:

#include "faketcp.h"

#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <linux/if_tun.h>
#include <netinet/in.h>
#include <netinet/ip_icmp.h>
#include <sys/ioctl.h>

int sethostaddr(const char* dev)
{
  struct ifreq ifr;
  bzero(&ifr, sizeof(ifr));
  strcpy(ifr.ifr_name, dev);
  struct sockaddr_in addr;
  bzero(&addr, sizeof addr);
  addr.sin_family = AF_INET;
  inet_pton(AF_INET, "192.168.0.1", &addr.sin_addr);
  //addr.sin_addr.s_addr = htonl(0xc0a80001);
  bcopy(&addr, &ifr.ifr_addr, sizeof addr);
  int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
  if (sockfd < 0)
    return sockfd;
  int err = 0;
  // ifconfig tun0 192.168.0.1
  if ((err = ioctl(sockfd, SIOCSIFADDR, (void *) &ifr)) < 0)
  {
    perror("ioctl SIOCSIFADDR");
    goto done;
  }
  // ifup tun0 其实就是启动tun0
  if ((err = ioctl(sockfd, SIOCGIFFLAGS, (void *) &ifr)) < 0)
  {
    perror("ioctl SIOCGIFFLAGS");
    goto done;
  }
  ifr.ifr_flags |= IFF_UP;
  if ((err = ioctl(sockfd, SIOCSIFFLAGS, (void *) &ifr)) < 0)
  {
    perror("ioctl SIOCSIFFLAGS");
    goto done;
  }
  // ifconfig tun0 192.168.0.1/24 # 配置子网掩码
  inet_pton(AF_INET, "255.255.255.0", &addr.sin_addr);
  bcopy(&addr, &ifr.ifr_netmask, sizeof addr);
  if ((err = ioctl(sockfd, SIOCSIFNETMASK, (void *) &ifr)) < 0)
  {
    perror("ioctl SIOCSIFNETMASK");
    goto done;
  }
done:
  close(sockfd);
  return err;
}

int tun_alloc(char dev[IFNAMSIZ])
{
  struct ifreq ifr;
  int fd, err;

  if ((fd = open("/dev/net/tun", O_RDWR)) < 0)
  {
    perror("open");
    return -1;
  }

  bzero(&ifr, sizeof(ifr));
  ifr.ifr_flags = IFF_TUN | IFF_NO_PI; // tun设备不包含以太网头部,而tap包含,仅此而已

  if (*dev)
  {
    strncpy(ifr.ifr_name, dev, IFNAMSIZ); 
  }

  if ((err = ioctl(fd, TUNSETIFF, (void *) &ifr)) < 0)
  {
    perror("ioctl TUNSETIFF");
    close(fd);
    return err;
  }
  strcpy(dev, ifr.ifr_name);
  if ((err = sethostaddr(dev)) < 0) // 设定地址等信息
    return err;

  return fd;
}

uint16_t in_checksum(const void* buf, int len)
{
  assert(len % 2 == 0);
  const uint16_t* data = static_cast<const uint16_t*>(buf);
  int sum = 0;
  for (int i = 0; i < len; i+=2)
  {
    sum += *data++;
  }
  while (sum >> 16)
    sum = (sum & 0xFFFF) + (sum >> 16);
  assert(sum <= 0xFFFF);
  return ~sum;
}

void icmp_input(int fd, const void* input, const void* payload, int len)
{
  const struct iphdr* iphdr = static_cast<const struct iphdr*>(input); // ip头部
  const struct icmphdr* icmphdr = static_cast<const struct icmphdr*>(payload); // icmp头部
  // const int icmphdr_size = sizeof(*icmphdr);
  const int iphdr_len = iphdr->ihl*4;

  if (icmphdr->type == ICMP_ECHO)
  {
    char source[INET_ADDRSTRLEN];
    char dest[INET_ADDRSTRLEN];
    inet_ntop(AF_INET, &iphdr->saddr, source, INET_ADDRSTRLEN);
    inet_ntop(AF_INET, &iphdr->daddr, dest, INET_ADDRSTRLEN);
    printf("%s > %s: ", source, dest);
    printf("ICMP echo request, id %d, seq %d, length %d\n",
           ntohs(icmphdr->un.echo.id),
           ntohs(icmphdr->un.echo.sequence),
           len - iphdr_len);

    union
    {
      unsigned char output[ETH_FRAME_LEN]; // 以太网头部
      struct
      {
        struct iphdr iphdr;
        struct icmphdr icmphdr;
      } out;
    };

    memcpy(output, input, len);
    out.icmphdr.type = ICMP_ECHOREPLY;
    out.icmphdr.checksum += ICMP_ECHO; // FIXME: not portable
    std::swap(out.iphdr.saddr, out.iphdr.daddr);
    write(fd, output, len);
  }
}

icmpecho.cc:

#include "faketcp.h"

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <netinet/ip.h>
#include <linux/if_ether.h>

int main()
{
  char ifname[IFNAMSIZ] = "tun%d";
  int fd = tun_alloc(ifname); // tun_alloc函数主要用于开启

  if (fd < 0)
  {
    fprintf(stderr, "tunnel interface allocation failed\n");
    exit(1);
  }

  printf("allocted tunnel interface %s\n", ifname);
  sleep(1);

  for (;;)
  {
    union
    {
      unsigned char buf[ETH_FRAME_LEN]; // 以太网头部
      struct iphdr iphdr;   // ip头部
    };

    const int iphdr_size = sizeof iphdr; // ip头部默认是20字节

    int nread = read(fd, buf, sizeof(buf));
    if (nread < 0)
    {
      perror("read");
      close(fd);
      exit(1);
    }
    printf("read %d bytes from tunnel interface %s.\n", nread, ifname);

    const int iphdr_len = iphdr.ihl*4;
    if (nread >= iphdr_size
        && iphdr.version == 4
        && iphdr_len >= iphdr_size
        && iphdr_len <= nread
        && iphdr.tot_len == htons(nread)
        && in_checksum(buf, iphdr_len) == 0)
    {
      const void* payload = buf + iphdr_len;
      if (iphdr.protocol == IPPROTO_ICMP)  // icmp协议
      {
        icmp_input(fd, buf, payload, nread);
      }
    }
    else
    {
      printf("bad packet\n");
      for (int i = 0; i < nread; ++i)
      {
        if (i % 4 == 0) printf("\n");
        printf("%02x ", buf[i]);
      }
      printf("\n");
    }
  }
  return 0;
}

运行方法,打开3个命令行窗口:

在第1个窗口运行sudo ./icmpecho,程序显示:
allocted tunnel interface tun0

在第2个窗口运行:
$ sudo ifconfig tun0 192.168.0.1/24 # 设定ip地址

$ sudo tcpdump -i tun0 # 用tcpdump抓取通过接口tun0的数据包

在第3个窗口运行:
$ ping 192.168.0.2

$ ping 192.168.0.3

$ ping 192.168.0.234

发现每个192.168.0.X 的IP 都能 ping 通。

第二步:实现拒接 TCP 连接的功能,即在收到SYN TCP segment的时候发送RST segment。

rejectall.cc:

#include "faketcp.h"

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <netinet/ip.h>
#include <netinet/tcp.h>
#include <linux/if_ether.h>

void tcp_input(int fd, const void* input, const void* payload, int tot_len)
{
  const struct iphdr* iphdr = static_cast<const struct iphdr*>(input); // ip头部
  const struct tcphdr* tcphdr = static_cast<const struct tcphdr*>(payload); // tcp头部
  const int iphdr_len = iphdr->ihl*4; // ip头部的大小
  const int tcp_seg_len = tot_len - iphdr_len; // tcp报文的大小
  const int tcphdr_size = sizeof(*tcphdr);
  if (tcp_seg_len >= tcphdr_size
      && tcp_seg_len >= tcphdr->doff*4)
  {
    const int tcphdr_len = tcphdr->doff*4;

    if (tcphdr->syn) // 收到了SYN分节
    {
      char source[INET_ADDRSTRLEN];
      char dest[INET_ADDRSTRLEN];
      inet_ntop(AF_INET, &iphdr->saddr, source, INET_ADDRSTRLEN); // 将ip转化为可读的字符串
      inet_ntop(AF_INET, &iphdr->daddr, dest, INET_ADDRSTRLEN);
      printf("IP %s.%d > %s.%d: ",
             source, ntohs(tcphdr->source), dest, ntohs(tcphdr->dest));
      printf("Flags [S], seq %u, win %d, length %d\n",
             ntohl(tcphdr->seq), // 序列号
             ntohs(tcphdr->window), // 窗口大小
             tot_len - iphdr_len - tcphdr_len);

      union
      {
        unsigned char output[ETH_FRAME_LEN];
        struct
        {
          struct iphdr iphdr;
          struct tcphdr tcphdr;
        } out;
      };

      assert(sizeof(out) == sizeof(struct iphdr) + sizeof(struct tcphdr));
      int output_len = sizeof(out);
      bzero(&out, output_len + 4);
      memcpy(output, input, sizeof(struct iphdr));

      out.iphdr.tot_len = htons(output_len);
      std::swap(out.iphdr.saddr, out.iphdr.daddr);
      out.iphdr.check = 0;
      out.iphdr.check = in_checksum(output, sizeof(struct iphdr));
      out.tcphdr.source = tcphdr->dest;  // 源地址和目的地址对调
      out.tcphdr.dest = tcphdr->source;
      out.tcphdr.seq = 0;
      out.tcphdr.ack_seq = htonl(ntohl(tcphdr->seq)+1); // 确认序列号
      out.tcphdr.doff = sizeof(struct tcphdr) / 4;
      out.tcphdr.ack = 1;
      out.tcphdr.rst = 1; // 注意这里的RST分节
      out.tcphdr.window = 0;
      unsigned char* pseudo = output + output_len;
      pseudo[0] = 0;
      pseudo[1] = IPPROTO_TCP;
      pseudo[2] = 0;
      pseudo[3] = sizeof(struct tcphdr);
      out.tcphdr.check = in_checksum(&out.iphdr.saddr, sizeof(struct tcphdr)+12);
      write(fd, output, output_len);
    }
  }
}

int main()
{
  char ifname[IFNAMSIZ] = "tun%d";
  int fd = tun_alloc(ifname);

  if (fd < 0)
  {
    fprintf(stderr, "tunnel interface allocation failed\n");
    exit(1);
  }

  printf("allocted tunnel interface %s\n", ifname);
  sleep(1);

  for (;;)
  {
    union
    {
      unsigned char buf[ETH_FRAME_LEN];
      struct iphdr iphdr;
    };

    const int iphdr_size = sizeof iphdr;

    int nread = read(fd, buf, sizeof(buf));
    if (nread < 0)
    {
      perror("read");
      close(fd);
      exit(1);
    }
    printf("read %d bytes from tunnel interface %s.\n", nread, ifname);

    const int iphdr_len = iphdr.ihl*4;
    if (nread >= iphdr_size
        && iphdr.version == 4
        && iphdr_len >= iphdr_size
        && iphdr_len <= nread
        && iphdr.tot_len == htons(nread)
        && in_checksum(buf, iphdr_len) == 0)
    {
      const void* payload = buf + iphdr_len;
      if (iphdr.protocol == IPPROTO_ICMP)
      {
        icmp_input(fd, buf, payload, nread);
      }
      else if (iphdr.protocol == IPPROTO_TCP)
      {
        tcp_input(fd, buf, payload, nread);
      }
    }
    else
    {
      printf("bad packet\n");
      for (int i = 0; i < nread; ++i)
      {
        if (i % 4 == 0) printf("\n");
        printf("%02x ", buf[i]);
      }
      printf("\n");
    }
  }

  return 0;
}

运行方法,打开3个命令行窗口,头两个窗口的操作与前面相同,运行的faketcp 程序是 ./rejectall。

在第3个窗口运行

$ nc 192.168.0.2 2000

$ nc 192.168.0.2 3333

$ nc 192.168.0.7 5555

发现向其中任意一个 IP 发起的 TCP 连接都被拒接了。

第三步:实现接受 TCP 连接的功能,即在接收到SYN TCP segment的时候发回 SYN + ACK。这个程序同时处理了连接断开的情况,即在收到FIN segment的时候发回 FIN + ACK。

acceptall.cc:

#include "faketcp.h"

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <netinet/ip.h>
#include <netinet/tcp.h>
#include <linux/if_ether.h>

void tcp_input(int fd, const void* input, const void* payload, int tot_len)
{
  const struct iphdr* iphdr = static_cast<const struct iphdr*>(input);
  const struct tcphdr* tcphdr = static_cast<const struct tcphdr*>(payload);
  const int iphdr_len = iphdr->ihl*4;
  const int tcp_seg_len = tot_len - iphdr_len;
  const int tcphdr_size = sizeof(*tcphdr);
  if (tcp_seg_len >= tcphdr_size
      && tcp_seg_len >= tcphdr->doff*4)
  {
    const int tcphdr_len = tcphdr->doff*4;

    char source[INET_ADDRSTRLEN];
    char dest[INET_ADDRSTRLEN];
    inet_ntop(AF_INET, &iphdr->saddr, source, INET_ADDRSTRLEN);
    inet_ntop(AF_INET, &iphdr->daddr, dest, INET_ADDRSTRLEN);
    printf("IP %s.%d > %s.%d: ",
           source, ntohs(tcphdr->source), dest, ntohs(tcphdr->dest));
    printf("Flags [%c], seq %u, win %d, length %d\n",
           tcphdr->syn ? 'S' : (tcphdr->fin ? 'F' : '.'),
           ntohl(tcphdr->seq),
           ntohs(tcphdr->window),
           tot_len - iphdr_len - tcphdr_len);

    union
    {
      unsigned char output[ETH_FRAME_LEN];
      struct
      {
        struct iphdr iphdr;
        struct tcphdr tcphdr;
      } out;
    };

    assert(sizeof(out) == sizeof(struct iphdr) + sizeof(struct tcphdr));
    int output_len = sizeof(out);
    bzero(&out, output_len + 4);
    memcpy(output, input, sizeof(struct iphdr));

    out.iphdr.tot_len = htons(output_len);
    std::swap(out.iphdr.saddr, out.iphdr.daddr);
    out.iphdr.check = 0;
    out.iphdr.check = in_checksum(output, sizeof(struct iphdr));

    out.tcphdr.source = tcphdr->dest;
    out.tcphdr.dest = tcphdr->source;
    out.tcphdr.ack_seq = htonl(ntohl(tcphdr->seq)+1);
    out.tcphdr.doff = sizeof(struct tcphdr) / 4;
    out.tcphdr.window = htons(5000);

    bool response = false;
    if (tcphdr->syn)
    {
      out.tcphdr.seq = htonl(123456); // 序列号随机产生
      out.tcphdr.syn = 1; // SYN
      out.tcphdr.ack = 1; // ACK
      response = true;
    }
    else if (tcphdr->fin) // 对于对方发送的FIN也需要接收是吧!
    {
      out.tcphdr.seq = htonl(123457);
      out.tcphdr.fin = 1;
      out.tcphdr.ack = 1;
      response = true;
    }

    unsigned char* pseudo = output + output_len;
    pseudo[0] = 0;
    pseudo[1] = IPPROTO_TCP;
    pseudo[2] = 0;
    pseudo[3] = sizeof(struct tcphdr);
    out.tcphdr.check = in_checksum(&out.iphdr.saddr, sizeof(struct tcphdr)+12);
    if (response)
    {
      write(fd, output, output_len);
    }
  }
}

int main()
{
  char ifname[IFNAMSIZ] = "tun%d";
  int fd = tun_alloc(ifname);

  if (fd < 0)
  {
    fprintf(stderr, "tunnel interface allocation failed\n");
    exit(1);
  }

  printf("allocted tunnel interface %s\n", ifname);
  sleep(1);

  for (;;)
  {
    union
    {
      unsigned char buf[ETH_FRAME_LEN];
      struct iphdr iphdr;
    };

    const int iphdr_size = sizeof iphdr;

    int nread = read(fd, buf, sizeof(buf));
    if (nread < 0)
    {
      perror("read");
      close(fd);
      exit(1);
    }
    printf("read %d bytes from tunnel interface %s.\n", nread, ifname);

    const int iphdr_len = iphdr.ihl*4;
    if (nread >= iphdr_size
        && iphdr.version == 4
        && iphdr_len >= iphdr_size
        && iphdr_len <= nread
        && iphdr.tot_len == htons(nread)
        && in_checksum(buf, iphdr_len) == 0)
    {
      const void* payload = buf + iphdr_len;
      if (iphdr.protocol == IPPROTO_ICMP)
      {
        icmp_input(fd, buf, payload, nread);
      }
      else if (iphdr.protocol == IPPROTO_TCP)
      {
        tcp_input(fd, buf, payload, nread);
      }
    }
    else
    {
      printf("bad packet\n");
      for (int i = 0; i < nread; ++i)
      {
        if (i % 4 == 0) printf("\n");
        printf("%02x ", buf[i]);
      }
      printf("\n");
    }
  }

  return 0;
}

运行方法,打开3个命令行窗口,步骤与前面相同,运行的 faketcp 程序是 ./acceptall。

这次会发现 nc 能和192.168.0.X中的每一个 IP 每一个 PORT 都能连通。还可以在第4个窗口中运行 netstat -tpn,以确认连接确实建立起来了。

如果在 nc 中输入数据,数据会堆积在操作系统中,表现为netstat 显示的发送队列 (Send-Q)的长度增加。

第四步:在第三步接受TCP连接的基础上,实现接收数据,即在收到包含 payload 数据的 TCP segment时发回ACK。

discardall.cc:

#include "faketcp.h"

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <netinet/ip.h>
#include <netinet/tcp.h>
#include <linux/if_ether.h>

void tcp_input(int fd, const void* input, const void* payload, int tot_len)
{
    const struct iphdr* iphdr = static_cast<const struct iphdr*>(input);
    const struct tcphdr* tcphdr = static_cast<const struct tcphdr*>(payload);
    const int iphdr_len = iphdr->ihl*4;
    const int tcp_seg_len = tot_len - iphdr_len;
    const int tcphdr_size = sizeof(*tcphdr);
    if (tcp_seg_len >= tcphdr_size
            && tcp_seg_len >= tcphdr->doff*4)
    {
        const int tcphdr_len = tcphdr->doff*4;
        const int payload_len = tot_len - iphdr_len - tcphdr_len;

        char source[INET_ADDRSTRLEN];
        char dest[INET_ADDRSTRLEN];
        inet_ntop(AF_INET, &iphdr->saddr, source, INET_ADDRSTRLEN); // 将ip地址变得可读
        inet_ntop(AF_INET, &iphdr->daddr, dest, INET_ADDRSTRLEN);
        printf("IP %s.%d > %s.%d: ",
                source, ntohs(tcphdr->source), dest, ntohs(tcphdr->dest));
        printf("Flags [%c], seq %u, win %d, length %d\n",
                tcphdr->syn ? 'S' : (tcphdr->fin ? 'F' : '.'),
                ntohl(tcphdr->seq),
                ntohs(tcphdr->window),
                payload_len);

        union
        {
            unsigned char output[ETH_FRAME_LEN];
            struct
            {
                struct iphdr iphdr;
                struct tcphdr tcphdr;
            } out;
        };

        assert(sizeof(out) == sizeof(struct iphdr) + sizeof(struct tcphdr));
        int output_len = sizeof(out);
        bzero(&out, output_len + 4);
        memcpy(output, input, sizeof(struct iphdr));

        out.iphdr.tot_len = htons(output_len);
        std::swap(out.iphdr.saddr, out.iphdr.daddr);
        out.iphdr.check = 0;
        out.iphdr.check = in_checksum(output, sizeof(struct iphdr));

        out.tcphdr.source = tcphdr->dest; // 目的地址和源地址倒换
        out.tcphdr.dest = tcphdr->source;
        out.tcphdr.doff = sizeof(struct tcphdr) / 4;
        out.tcphdr.window = htons(5000);

        bool response = false;
        if (tcphdr->syn)
        {
            out.tcphdr.seq = htonl(123456);
            out.tcphdr.ack_seq = htonl(ntohl(tcphdr->seq)+1);
            out.tcphdr.syn = 1;
            out.tcphdr.ack = 1;
            response = true;
        }
        else if (tcphdr->fin)
        {
            out.tcphdr.seq = htonl(123457);
            out.tcphdr.ack_seq = htonl(ntohl(tcphdr->seq)+1);
            out.tcphdr.fin = 1;
            out.tcphdr.ack = 1;
            response = true;
        }
        else if (payload_len > 0)
        {
            out.tcphdr.seq = htonl(123457);
            out.tcphdr.ack_seq = htonl(ntohl(tcphdr->seq)+payload_len); // 确认的序列号
            out.tcphdr.ack = 1; // ack,不发送数据,仅发送确认号
            response = true;
        }

        unsigned char* pseudo = output + output_len;
        pseudo[0] = 0;
        pseudo[1] = IPPROTO_TCP;
        pseudo[2] = 0;
        pseudo[3] = sizeof(struct tcphdr);
        out.tcphdr.check = in_checksum(&out.iphdr.saddr, sizeof(struct tcphdr)+12);
        if (response)
        {
            write(fd, output, output_len);
        }
    }
}

int main()
{
    char ifname[IFNAMSIZ] = "tun%d";
    int fd = tun_alloc(ifname);

    if (fd < 0)
    {
        fprintf(stderr, "tunnel interface allocation failed\n");
        exit(1);
    }

    printf("allocted tunnel interface %s\n", ifname);
    sleep(1);

    for (;;)
    {
        union
        {
            unsigned char buf[ETH_FRAME_LEN];
            struct iphdr iphdr;
        };

        const int iphdr_size = sizeof iphdr;

        int nread = read(fd, buf, sizeof(buf));
        if (nread < 0)
        {
            perror("read");
            close(fd);
            exit(1);
        }
        printf("read %d bytes from tunnel interface %s.\n", nread, ifname);

        const int iphdr_len = iphdr.ihl*4;
        if (nread >= iphdr_size
                && iphdr.version == 4
                && iphdr_len >= iphdr_size
                && iphdr_len <= nread
                && iphdr.tot_len == htons(nread)
                && in_checksum(buf, iphdr_len) == 0)
        {
            const void* payload = buf + iphdr_len;
            if (iphdr.protocol == IPPROTO_ICMP)
            {
                icmp_input(fd, buf, payload, nread);
            }
            else if (iphdr.protocol == IPPROTO_TCP)
            {
                tcp_input(fd, buf, payload, nread);
            }
        }
        else
        {
            printf("bad packet\n");
            for (int i = 0; i < nread; ++i)
            {
                if (i % 4 == 0) printf("\n");
                printf("%02x ", buf[i]);
            }
            printf("\n");
        }
    }

    return 0;
}

运行方法,打开3个命令行窗口,步骤与前面相同,运行的faketcp程序是./acceptall。

这次会发现 nc 能和192.168.0.X中的每一个IP 每一个PORT 都能连通,数据也能发出去。还可以在第4个窗口中运行netstat -tpn,以确认连接确实建立起来了,并且发送队列的长度为0;

这一步已经解决了前面的问题2,扮演任意 TCP 服务端。

第五步:解决前面的问题1,扮演客户端向atom 发起任意多的连接。

connectmany.cc:

#include "faketcp.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <netinet/ip.h>
#include <netinet/tcp.h>
#include <linux/if_ether.h>

void tcp_input(int fd, const void* input, const void* payload, int tot_len, bool passive)
{
    const struct iphdr* iphdr = static_cast<const struct iphdr*>(input);
    const struct tcphdr* tcphdr = static_cast<const struct tcphdr*>(payload);
    const int iphdr_len = iphdr->ihl*4;
    const int tcp_seg_len = tot_len - iphdr_len;
    const int tcphdr_size = sizeof(*tcphdr);
    if (tcp_seg_len >= tcphdr_size
            && tcp_seg_len >= tcphdr->doff*4)
    {
        const int tcphdr_len = tcphdr->doff*4;
        const int payload_len = tot_len - iphdr_len - tcphdr_len;

        char source[INET_ADDRSTRLEN];
        char dest[INET_ADDRSTRLEN];
        inet_ntop(AF_INET, &iphdr->saddr, source, INET_ADDRSTRLEN);
        inet_ntop(AF_INET, &iphdr->daddr, dest, INET_ADDRSTRLEN);
        printf("IP %s.%d > %s.%d: ",
                source, ntohs(tcphdr->source), dest, ntohs(tcphdr->dest));
        printf("Flags [%c], seq %u, win %d, length %d\n",
                tcphdr->syn ? 'S' : (tcphdr->fin ? 'F' : '.'),
                ntohl(tcphdr->seq),
                ntohs(tcphdr->window),
                payload_len);

        union
        {
            unsigned char output[ETH_FRAME_LEN];
            struct
            {
                struct iphdr iphdr;
                struct tcphdr tcphdr;
            } out;
        };

        assert(sizeof(out) == sizeof(struct iphdr) + sizeof(struct tcphdr));
        int output_len = sizeof(out);
        bzero(&out, output_len + 4);
        memcpy(output, input, sizeof(struct iphdr));

        out.iphdr.tot_len = htons(output_len);
        std::swap(out.iphdr.saddr, out.iphdr.daddr);
        out.iphdr.check = 0;
        out.iphdr.check = in_checksum(output, sizeof(struct iphdr));

        out.tcphdr.source = tcphdr->dest;
        out.tcphdr.dest = tcphdr->source;
        out.tcphdr.doff = sizeof(struct tcphdr) / 4;
        out.tcphdr.window = htons(5000);

        bool response = false;
        if (tcphdr->syn) // 对方发起连接,或者对方发送了确认的syn和ack
        {
            out.tcphdr.seq = htonl(passive ? 123456 : 123457);
            out.tcphdr.ack_seq = htonl(ntohl(tcphdr->seq)+1);
            if (passive) // passive==true表示被动接收连接,表示对方连过来
            {
                out.tcphdr.syn = 1;
            }
            // 否则的话,表示自己主动发送的连接,接收到了对方的syn和ack,我们只需要发送一个ack即可
            out.tcphdr.ack = 1; 
            response = true;
        }
        else if (tcphdr->fin) // 对方关闭连接
        {
            out.tcphdr.seq = htonl(123457);
            out.tcphdr.ack_seq = htonl(ntohl(tcphdr->seq)+1);
            out.tcphdr.fin = 1;
            out.tcphdr.ack = 1;
            response = true;
        }
        else if (payload_len > 0)
        {
            out.tcphdr.seq = htonl(123457);
            out.tcphdr.ack_seq = htonl(ntohl(tcphdr->seq)+payload_len);
            out.tcphdr.ack = 1;
            response = true;
        }

        unsigned char* pseudo = output + output_len;
        pseudo[0] = 0;
        pseudo[1] = IPPROTO_TCP;
        pseudo[2] = 0;
        pseudo[3] = sizeof(struct tcphdr);
        out.tcphdr.check = in_checksum(&out.iphdr.saddr, sizeof(struct tcphdr)+12);
        if (response)
        {
            write(fd, output, output_len);
        }
    }
}

// connect_one 发起一个tcp连接?
bool connect_one(int fd, uint32_t daddr, int dport, uint32_t saddr, int sport)
{
    {
        union
        {
            unsigned char output[ETH_FRAME_LEN];
            struct
            {
                struct iphdr iphdr;
                struct tcphdr tcphdr;
            } out;
        };

        bzero(&out, (sizeof out)+4);

        out.iphdr.version = IPVERSION;
        out.iphdr.ihl = sizeof(out.iphdr)/4;
        out.iphdr.tos = 0;
        out.iphdr.tot_len = htons(sizeof(out));
        out.iphdr.id = 55564;
        out.iphdr.frag_off |= htons(IP_DF);
        out.iphdr.ttl = IPDEFTTL;
        out.iphdr.protocol = IPPROTO_TCP;
        out.iphdr.saddr = saddr;
        out.iphdr.daddr = daddr;
        out.iphdr.check = in_checksum(output, sizeof(struct iphdr));

        out.tcphdr.source = sport; // 端口号
        out.tcphdr.dest = dport;
        out.tcphdr.seq = htonl(123456);
        out.tcphdr.ack_seq = 0;
        out.tcphdr.doff = sizeof(out.tcphdr)/4;
        out.tcphdr.syn = 1;       // 主动发起连接
        out.tcphdr.window = htons(4096);

        unsigned char* pseudo = output + sizeof out;
        pseudo[0] = 0;
        pseudo[1] = IPPROTO_TCP;
        pseudo[2] = 0;
        pseudo[3] = sizeof(struct tcphdr);
        out.tcphdr.check = in_checksum(&out.iphdr.saddr, sizeof(struct tcphdr)+12);

        write(fd, output, sizeof out); // 发送连接
    }

    union
    {
        unsigned char buf[ETH_FRAME_LEN];
        struct iphdr iphdr;
    };

    const int iphdr_size = sizeof iphdr;

    int nread = read(fd, buf, sizeof(buf)); // 接收到回复之后
    if (nread < 0)
    {
        perror("read");
        close(fd);
        exit(1);
    }
    // printf("read %d bytes from tunnel interface %s.\n", nread, ifname);

    if (nread >= iphdr_size
            && iphdr.version == 4
            && iphdr.ihl*4 >= iphdr_size
            && iphdr.ihl*4 <= nread
            && iphdr.tot_len == htons(nread)
            && in_checksum(buf, iphdr.ihl*4) == 0)
    {
        const void* payload = buf + iphdr.ihl*4;
        if (iphdr.protocol == IPPROTO_ICMP)
        {
            icmp_input(fd, buf, payload, nread);
        }
        else if (iphdr.protocol == IPPROTO_TCP) // tcp 报文
        {
            tcp_input(fd, buf, payload, nread, false); // 注意到这里的false,表示是自己主动发起连接
        }
    }

    return true;
}

void connect_many(int fd, const char* ipstr, int port, int count)
{
    uint32_t destip;
    inet_pton(AF_INET, ipstr, &destip); // 连接到目的ip

    uint32_t srcip = ntohl(destip)+1;
    int srcport = 1024; // 端口从1024开始

    for (int i = 0; i < count; ++i)
    {
        connect_one(fd, destip, htons(port), htonl(srcip), htons(srcport));
        srcport++; // 源端口在不断加1
        if (srcport > 0xFFFF)
        {
            srcport = 1024;
            srcip++;
        }
    }
}

void usage()
{
}

int main(int argc, char* argv[])
{
    if (argc < 4)
    {
        usage();
        return 0;
    }

    char ifname[IFNAMSIZ] = "tun%d";
    int fd = tun_alloc(ifname);

    if (fd < 0)
    {
        fprintf(stderr, "tunnel interface allocation failed\n");
        exit(1);
    }

    const char* ip = argv[1]; // ip
    int port = atoi(argv[2]); // 端口
    int count = atoi(argv[3]); // 数量
    printf("allocted tunnel interface %s\n", ifname);
    printf("press enter key to start connecting %s:%d\n", ip, port);
    getchar();

    connect_many(fd, ip, port, count); // 发起连接

    for (;;)
    {
        union
        {
            unsigned char buf[ETH_FRAME_LEN];
            struct iphdr iphdr;
        };

        const int iphdr_size = sizeof iphdr;

        int nread = read(fd, buf, sizeof(buf));
        if (nread < 0)
        {
            perror("read");
            close(fd);
            exit(1);
        }
        printf("read %d bytes from tunnel interface %s.\n", nread, ifname);

        const int iphdr_len = iphdr.ihl*4;
        if (nread >= iphdr_size
                && iphdr.version == 4
                && iphdr_len >= iphdr_size
                && iphdr_len <= nread
                && iphdr.tot_len == htons(nread)
                && in_checksum(buf, iphdr_len) == 0)
        {
            const void* payload = buf + iphdr_len;
            if (iphdr.protocol == IPPROTO_ICMP)
            {
                icmp_input(fd, buf, payload, nread);
            }
            else if (iphdr.protocol == IPPROTO_TCP)
            {
                tcp_input(fd, buf, payload, nread, true); // 注意到这里的true,表示是被动接收连接
            }
        }
        else
        {
            printf("bad packet\n");
            for (int i = 0; i < nread; ++i)
            {
                if (i % 4 == 0) printf("\n");
                printf("%02x ", buf[i]);
            }
            printf("\n");
        }
    }
    return 0;
}

这一步的运行方法与前面不同,打开4个命令行窗口。

在第1个窗口运行sudo ./connectmany 192.168.0.1 2007 1000,表示将向192.168.0.1:2007 发起1000个并发连接。程序显示:
allocated tunnel interface tun0
press enter key to start connecting 192.168.0.1 2007

在第二个窗口运行
$ sudo ifconfig tun0 192.168.0.1/24

$ sudo tcpdump -i tun0

在第3个窗口运行一个能接收并发TCP 连接的服务程序,可以是httpd, 也可以是muduo 的echo 或 discard 示例,程序应listen 2007端口。
回到第1个窗口敲回车,然后在第4个窗口中用netstat -tpn来观察并发连接。
文中代码目录连接:https://github.com/chenshuo/recipes/tree/master/faketcp

发布了7 篇原创文章 · 获赞 78 · 访问量 10万+
展开阅读全文

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 大白 设计师: CSDN官方博客

分享到微信朋友圈

×

扫一扫,手机浏览