繁体   English   中英

试图在Linux上的C语言中制作一个简单的分组路由器。 select()无限期地闲置,我不知道为什么

[英]Trying to make a simple packet router in C on Linux. select() idles indefinitely and I don't know why

就像标题说的那样,我正在尝试制作一个分组路由器,但是它卡在了select()上。 我正在使用具有4个主机(h1,h2,h3,r0),3个交换机(s1,s2,s3)和3个接口(r0-eth1,r0-eth2,r0-eth3)的Mininet环境。 我正在编译C代码并在r0上运行可执行文件。 这是代码:

int main(int argc, char *argv[])
{
int sock1, sock2, sock3;
int status = 0;
fd_set readfds;
int buffer[256];



// Create 3 sockets
if ((sock1 = socket(AF_PACKET, SOCK_DGRAM, htons(ETH_P_IP))) < 0){
    perror("socket() failed");
    exit(1);
}
if ((sock2 = socket(AF_PACKET, SOCK_DGRAM, htons(ETH_P_IP))) < 0){
    perror("socket() failed");
    exit(1);
}
if ((sock3 = socket(AF_PACKET, SOCK_DGRAM, htons(ETH_P_IP))) < 0){
    perror("socket() failed");
    exit(1);
}

// Bind sockets to interfaces
struct ifreq ifr;
memset(&ifr, 0, sizeof(ifr));
snprintf(ifr.ifr_name, sizeof(ifr.ifr_name), "r0-eth1");
if (setsockopt(sock1, SOL_SOCKET, SO_BINDTODEVICE, (void *)&ifr, sizeof(ifr)) < 0) {
perror("setsockopt() inf config failed");
exit(1);
}
memset(&ifr, 0, sizeof(ifr));
snprintf(ifr.ifr_name, sizeof(ifr.ifr_name), "r0-eth2");
if (setsockopt(sock2, SOL_SOCKET, SO_BINDTODEVICE, (void *)&ifr, sizeof(ifr)) < 0) {
perror("setsockopt() inf config failed");
exit(1);
}
memset(&ifr, 0, sizeof(ifr));
snprintf(ifr.ifr_name, sizeof(ifr.ifr_name), "r0-eth3");
if (setsockopt(sock3, SOL_SOCKET, SO_BINDTODEVICE, (void *)&ifr, sizeof(ifr)) < 0) {
perror("setsockopt() inf config failed");
exit(1);
}


//Setup select
FD_ZERO(&readfds);
FD_SET(sock1, &readfds);
FD_SET(sock2, &readfds);
FD_SET(sock3, &readfds);

status = select( sock3 + 1 , &readfds , NULL , NULL , NULL);

while(1)
{
    // Check select
    if (status > 0) {
        // Read IP datagram
        if (FD_ISSET(sock1, &readfds)) {
            struct sockaddr_in sockAddr;
            socklen_t sockLen = sizeof(sockAddr);
            memset(&sockAddr, 0, sockLen);
            sockAddr.sin_family = AF_INET;
            sockAddr.sin_addr.s_addr = htonl(INADDR_ANY);
            int bufLen = recvfrom(sock1, buffer, 65535, 0, (struct sockaddr *) &sockAddr, (socklen_t *)&sockLen);
        }
    }
}

我使用以下命令运行它:

    r0 ./router &

然后我做一个

    pingall

查看主机是否可以互相访问。 我的可执行文件卡在select()上,并且永远不会继续运行。 在pingall期间,h1,h2和h3都可以达到r0,但所有其他数据包均被丢弃。 我知道我还没有完成路由部分,但是直到解决该问题,我才能真正继续前进。

感谢帮助。

在我看来,您的问题非常简单。 请注意以下代码段:

     status = select( sock3 + 1 , &readfds , NULL , NULL , NULL);

while(1)
{
    // Check select
    if (status > 0) {
        // Read IP datagram
        if (FD_ISSET(sock1, &readfds)) {
            struct sockaddr_in sockAddr;
            socklen_t sockLen = sizeof(sockAddr);
            memset(&sockAddr, 0, sockLen);
            sockAddr.sin_family = AF_INET;
            sockAddr.sin_addr.s_addr = htonl(INADDR_ANY);
            int bufLen = recvfrom(sock1, buffer, 65535, 0, (struct sockaddr *) &sockAddr, (socklen_t *)&sockLen);
        }
    }
}

您设置为使用select() 然后,您进入while(1)循环并检查status ... ...但是您永远不会再次调用select,永远不会更新status的值,也永远不会让while循环结束。

此外,当使用select()FD_SET ,您将需要在每个信号之后重置FD_SET

我猜您需要检查数据包是否正在发送到主机。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM