使用C++在Linux中使用select读取ICMP回复

3
我正在使用C++和原始套接字向路由器发送ICMP请求,之后我想读取ICMP回复。我的问题是,select()没有接收到回复并且一直超时。我没有得到任何错误(errno返回成功)。路由器正在发送ICMP回复,因为我可以使用Wireshark看到响应。
用于测试我的程序的是运行在VirtualBox 4.2.6上的Ubuntu 12.10以及GN3用于虚拟网络。
我的源代码:
char buffer[IP_MAXPACKET]; // for the received ICMP reply
struct iphdr *ipRec; // ICMP header
timeval tv; // timeout
fd_set mySet; // descriptor set
...
tv.tv_sec = 3; // default time-out 3s
tv.tv_usec = 0;

int retval; // select
...
do {
        FD_ZERO(&mySet);
        FD_SET(mysocket, &mySet);

        retval = select(mysocket+1, &mySet, NULL, NULL, &tv);

        cout << "Errno after select:" << strerror(errno) << endl;

        if(retval == -1) {
            cerr << "select error" << endl;
            break;
        }
        else if (retval) {
            if((length = recvfrom(mysocket, buffer, MAX, 0, result->ai_addr, &(result->ai_addrlen))) == -1) {
                cerr << "Error: while receiving data." << endl;
            }
            else {
                cout << "good" << endl;

                ipRec = (struct iphdr*) buffer;
                icmpRec = (struct icmphdr*) (buffer + ipRec->ihl * 4);

                cout << "the packet." << " PID: " << ntohs(icmpRec->un.echo.id) << " Seq: " << ntohs(icmpRec->un.echo.sequence) << endl;

                if ((icmpRec->type == ICMP_ECHOREPLY) && (ntohs(icmpRec->un.echo.id) == pid) && (ntohs(icmpRec->un.echo.sequence) == (seq - 1))) {
                    minBuff = lengthBuff;
                }
            }
        } else {
            // getting here all the time = select times-out and reads no data

            cout << "mysocket:" << mysocket << endl;
            cout << "retval:" << retval << endl;
            maxBuff = lengthBuff;
            break;
        }
    } while (!((icmpRec->type == ICMP_ECHOREPLY) && (ntohs(icmpRec->un.echo.id) == pid) && (ntohs(icmpRec->un.echo.sequence) == (seq - 1))));

    if (packet)
        delete(packet);
    ...

感谢任何帮助。


请注意,<< 运算符不允许字符串作为其操作数。也许你正在使用 C++? - wildplasser
是的,我的错,显然是C++ :) - Matej Koleják
recvfrom中的MAX是什么?这不应该是IP_MAXPACKET吗? - Icarus3
问题已经解决。在不同的线程中找到了解决方案:https://dev59.com/suo6XIcBkEYKwwoYQiTQ?rq=1。将IPPROTO_RAW更改为IPPROTO_ICMP即可解决。现在可以选择读取ICMP回复数据包。 - Matej Koleják
MAX被定义为IP_MAXPACKET。 - Matej Koleják
1个回答

1

问题已解决。在另一个线程中找到了解决方案:ICMP数据包未发送C。将IPPROTO_RAW更改为IPPROTO_ICMP即可解决。现在可以选择读取ICMP回复数据包。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接