- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
问题是关于要使用 Tun/Tap 模块的 Linux 主机的正确配置。
我的目标:
利用现有的路由软件(下文中的 APP1 和 APP2),但拦截和修改它发送和接收的所有消息(由 Mediator 完成)。
我的场景:
Ubuntu 10.04 Machine
+---------------------------------------------+
| |
|APP1 --- tap1 --- Mediator --- tap2 --- APP2 |
| |
+---------------------------------------------+
tap1 和 tap2:分别使用 IFF_TAP 标志和 IP 10.0.0.1/24 和 10.0.0.2/24 设置的 tap 设备。创 build 备的代码如下:
#include <stdlib.h>
#include <stdio.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <fcntl.h>
#include <linux/if.h>
#include <linux/if_tun.h>
#include <string.h>
#include <errno.h>
#include <sys/resource.h>
void createTun(char *, char *, short);
int main(void)
{
const short FLAGS = IFF_TAP;
char *tunName;
char *tunIP;
// Create tap1
tunName = "tap1\0";
tunIP = "10.0.0.1/24\0";
createTun(tunName, tunIP, FLAGS);
printf("Created %s with IP %s\n", tunName, tunIP);
// Create tap2
tunName = "tap2\0";
tunIP = "10.0.0.2/24\0";
createTun(tunName, tunIP, FLAGS);
printf("Created %s with IP %s\n", tunName, tunIP);
return 0;
}
void createTun(char *tunName, char *tunIP, short FLAGS)
{
char *cmd;
char *cloneDev = "/dev/net/tun";
char *cmdIPLinkUpTemplate = "ip link set %s up";
char *cmdIPAddrAddTemplate = "ip addr add %s dev %s";
int cmdIPLinkUpRawLength = strlen(cmdIPLinkUpTemplate) - 2;
int cmdIPAddrAddRawLength = strlen(cmdIPAddrAddTemplate) - 4;
FILE *fp;
int fd, err, owner, group;
struct ifreq ifr;
owner = geteuid();
group = getegid();
// open the clone device
if((fd = open(cloneDev, O_RDWR)) < 0)
{
perror("OPEN CLONEDEV failed.");
exit(EXIT_FAILURE);
}
memset(&ifr, 0, sizeof(struct ifreq));
ifr.ifr_flags = FLAGS;
strncpy(ifr.ifr_name, tunName, strlen(tunName));
// create the device
if(ioctl(fd, TUNSETIFF, (void *) &ifr) < 0)
{
perror("IOCTL SETIFF denied.");
close(fd);
exit(EXIT_FAILURE);
}
// set dev owner
if(owner != -1)
{
if(ioctl(fd, TUNSETOWNER, owner) < 0)
{
perror("IOCTL SETOWNER denied.");
close(fd);
exit(EXIT_FAILURE);
}
}
// set dev group
if(group != -1)
{
if(ioctl(fd, TUNSETGROUP, group) < 0)
{
perror("IOCTL SETGROUP denied.");
close(fd);
exit(EXIT_FAILURE);
}
}
// set dev persistent
if(ioctl(fd, TUNSETPERSIST, 1) < 0)
{
perror("IOCTL SETPERSIST denied.");
close(fd);
exit(EXIT_FAILURE);
}
// Set dev up
cmd = malloc(cmdIPLinkUpRawLength + strlen(tunName) + 1);
sprintf(cmd, cmdIPLinkUpTemplate, ifr.ifr_name);
fp = popen(cmd, "r");
if(fp == NULL)
{
perror("POPEN failed.");
close(fd);
free(cmd);
exit(EXIT_FAILURE);
}
pclose(fp);
free(cmd);
// Assign IP
cmd = malloc(cmdIPAddrAddRawLength + strlen(tunIP) + strlen(tunName) + 1);
sprintf(cmd, cmdIPAddrAddTemplate, tunIP, tunName);
fp = popen(cmd, "r");
if(fp == NULL)
{
perror("POPEN failed.");
close(fd);
free(cmd);
exit(EXIT_FAILURE);
}
pclose(fp);
free(cmd);
return;
}
中介者:自行编写的小型代码,用于简单地在 tap1 和 tap2 之间中继数据。基本结构如下:
#include <unistd.h>
#include <stdio.h>
#include <sys/socket.h>
#include <netinet/ip.h>
#include <sys/ioctl.h>
#include <sys/resource.h>
#include <sys/epoll.h>
#include <errno.h>
#include <fcntl.h>
#include <stdlib.h>
#include <string.h>
#include <linux/if.h>
#include <linux/if_tun.h>
int main(int argc, char *argv[])
{
const int NOF_FD = 2;
const char *TUN1 = "tap1";
const char *TUN2 = "tap2";
const char *CLONEDEV = "/dev/net/tun";
int fd_tun1, fd_tun2, fd_epoll;
struct ifreq ifr_tun1, ifr_tun2;
struct epoll_event ev;
const int MAX_EVENTS = 1;
int ready, s, t;
const int MAX_BUF = 2000;
char buf[MAX_BUF];
struct sockaddr_in to;
const short FLAGS = IFF_TAP;
// Open tap1
if((fd_tun1 = open(CLONEDEV, O_RDWR)) < 0)
{
perror("OPEN CLONEDEV for tun1 failed");
exit(EXIT_FAILURE);
}
memset(&ifr_tun1, 0, sizeof(struct ifreq));
ifr_tun1.ifr_flags = FLAGS;
strcpy(ifr_tun1.ifr_name, TUN1);
if(ioctl(fd_tun1, TUNSETIFF, (void *) &ifr_tun1) < 0)
{
perror("IOCTL SETIFF for tap1 failed");
close(fd_tun1);
exit(EXIT_FAILURE);
}
// Open tap2
if((fd_tun2 = open(CLONEDEV, O_RDWR)) < 0)
{
perror("OPEN CLONEDEV for tap2 failed");
exit(EXIT_FAILURE);
}
memset(&ifr_tun2, 0, sizeof(struct ifreq));
ifr_tun2.ifr_flags = FLAGS;
strcpy(ifr_tun2.ifr_name, TUN2);
if(ioctl(fd_tun2, TUNSETIFF, (void *) &ifr_tun2) < 0)
{
perror("IOCTL SETIFF for tun2 failed");
close(fd_tun1);
close(fd_tun2);
exit(EXIT_FAILURE);
}
// Prepare EPOLL
if((fd_epoll = epoll_create(NOF_FD)) < 0)
{
perror("EPOLL CREATE failed");
close(fd_tun1);
close(fd_tun2);
exit(EXIT_FAILURE);
}
memset(&ev, 0, sizeof(ev));
ev.events = EPOLLIN;
ev.data.fd = fd_tun1;
if(epoll_ctl(fd_epoll, EPOLL_CTL_ADD, fd_tun1, &ev) < 0)
{
perror("EPOLL CTL ADD fd_tun1 failed");
close(fd_tun1);
close(fd_tun2);
close(fd_epoll);
exit(EXIT_FAILURE);
}
memset(&ev, 0, sizeof(ev));
ev.events = EPOLLIN;
ev.data.fd = fd_tun2;
if(epoll_ctl(fd_epoll, EPOLL_CTL_ADD, fd_tun2, &ev) < 0)
{
perror("EPOLL CTL ADD fd_tun2 failed");
close(fd_tun1);
close(fd_tun2);
close(fd_epoll);
exit(EXIT_FAILURE);
}
// Do relay
while(1)
{
if((ready = epoll_wait(fd_epoll, &ev, MAX_EVENTS, -1)) < 0)
{
if(errno == EINTR)
continue;
else
{
perror("EPOLL WAIT failed");
close(fd_tun1);
close(fd_tun2);
close(fd_epoll);
exit(EXIT_FAILURE);
}
}
//printf("EPOLL WAIT SIGNALED\n");
if(ev.events & EPOLLIN)
{
if((s = read(ev.data.fd, buf, MAX_BUF)) < 0)
{
perror("READ failed");
close(fd_tun1);
close(fd_tun2);
close(fd_epoll);
exit(EXIT_FAILURE);
}
printf("Read from %s. Bytes: %d\nData:\n", (ev.data.fd == fd_tun1 ? "tun1" : "tun2"), s);
int k;
for(k = 0; k < s; k++)
{
printf("%c", buf[k]);
}
printf("\n");
t = (ev.data.fd == fd_tun1) ? fd_tun2 : fd_tun1;
if((s = write(t, buf, s)) < 0)
{
perror("WRITE failed");
close(fd_tun1);
close(fd_tun2);
close(fd_epoll);
exit(EXIT_FAILURE);
}
printf("Written to %s. Bytes: %d\n", (t == fd_tun1 ? "tun1" : "tun2"), s);
if(epoll_ctl(fd_epoll, EPOLL_CTL_DEL, ev.data.fd, NULL) < 0)
{
perror("EPOLL CTL DEL failed");
close(fd_tun1);
close(fd_tun2);
close(fd_epoll);
exit(EXIT_FAILURE);
}
if(epoll_ctl(fd_epoll, EPOLL_CTL_ADD, ev.data.fd, &ev) < 0)
{
perror("EPOLL CTL ADD failed");
close(fd_tun1);
close(fd_tun2);
close(fd_epoll);
exit(EXIT_FAILURE);
}
}
printf("\n\n");
}
}
APP1 和 APP2:OSPF 路由守护进程分别通过 tap1 和 tap2 进行通信。守护进程的 strace 显示基本上涉及以下系统调用:
socket(PF_INET, SOCK_RAW, 0X59 /*IPPROTO_??? */) = 8 // Opening a socket for OSPF and tap1
fcntl64(8, F_SETFL, 0_RDONLY | 0_NONBLOCK) = 0
setsockopt(8, SOL_IP, IP_TOS, [192], 4) = 0
setsockopt(8, SOL_SOCKET, SO_PRIORITY, [7], 4) = 0
setsockopt(8, SOL_IP, IP_PKTINFO, [1], 4) = 0
setsockopt(8, SOL_IP, IP_MTU_DISCOVER, [0], 4) = 0
setsockopt(8, SOL_IP, IP_MULTICAST_LOOP, [0], 4) = 0
setsockopt(8, SOL_IP, IP_MULTICAST_TTL, [1], 4) = 0
setsockopt(8, SOL_IP, IP_MUTLICAST_IF, "\0\0\0\0\n\0\0\1\223\0\0\0", 12) = 0
setsockopt(8, SOL_SOCKET, SO_BINDTODEVICE, "tap1\0\0\0\0\0\0\0\0\0\0\0\0\0\315\375\307\250\352\t\t8\207\t\10\0\0\0\0", 32) = 0
setsockopt(8, SOL_IP, IP_ADD_MEMBERSHIP, "340\0\0\5\n\0\0\1\223\0\0\0", 12) = 0
// Then it gets in a cycle like:
select(9, [3, 7, 8], [], NULL, {1, 0}) = 0 (Timeout)
clock_gettime(CLOCK_MONOTONIC, {120893, 360452769}) = 0
time(NULL)
clock_gettime(CLOCK_MONOTONIC, {120893, 360504525}) = 0
select(9, [3, 7, 8], [], NULL, {1, 0}) = 0 (Timeout)
clock_gettime(CLOCK_MONOTONIC, {120894, 363022746}) = 0
time(NULL)
...
我的用法:
我的问题:
即使连接到 tap1 的 wireshark 看到来自 tap1 和 tap2 的消息,APP2 也没有收到 APP1 发送的消息,APP2 也没有收到来自 APP1 的消息。在上面显示的 strace 提取中,select() 调用永远不会返回文件描述符 8,它实际上是连接到 tap1 的套接字。
我的问题:
为什么 APP1 没有收到 APP2 发送的消息,即使这些消息是由 APP2 发送、由 Mediator 中继并被连接到 tap1 的 wireshark 看到的?
我是否必须在我的 Linux 主机上添加任何类型/种类的额外路由?
我是否在设置 tun/tap 设备时犯了错误?
我的 Mediator 代码是否无法正常工作?
最佳答案
我没有尝试过你的代码(你能够从用户空间而不是 using a multiqueue flag 打开 TAP 设备两次有点奇怪,但我们假设这是正确的),但是你在处理方式上有一个概念错误TAP 设备。
TUN/TAP 本质上只是一个管道,这个管道的一侧在内核(tapX 接口(interface))中,另一侧在一些用户空间应用程序中。这个应用程序写入管道的任何内容都会作为传入流量到达内核接口(interface)(您可以使用 wireshark 看到它)。无论内核发送到该管道(传出到 tapX),最终都会进入应用程序(您可以在应用程序中读取的数据)。
您的代码当前正在做的是打开同一管道的另一个用户空间部分,而这不是您想要的。你想在管道的另一边获得流量。从技术上讲,您当前正在做的事情可以通过一个简单的桥接接口(interface)来完成,将两个分路器作为端口添加到其中。当然,如果您不仅要桥接,而且要以某种方式修改流量,事情会变得有点复杂。
解决这个问题的一种方法是添加另一对 TAP 接口(interface)。您将 tap1 与 tap3 和 tap2 与 tap4 桥接(如在内核桥中),现在您在“中介”中打开 tap3 和 tap4 以及它们之间的代理帧。这是非常低效的,但可能是您问题的解决方案。
关于c - Linux TUN/TAP : Unable to read data back from TAP devices,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21001713/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!