legongju.com
我们一直在努力
2025-01-10 04:05 | 星期五

C++中ICMP协议如何实现

在C++中,要实现ICMP协议,通常需要使用原始套接字(raw sockets)来发送和接收ICMP数据包

#include
#include 
#include 
#include 
#include 
#include 
#include 
#include 

const int ICMP_PACKET_SIZE = 64;

// 构建ICMP请求数据包
void build_icmp_packet(char *buffer, int &length) {
    icmphdr *icmp = (icmphdr *) buffer;
    
    // 设置ICMP类型为回显请求
    icmp->type = ICMP_ECHO;
    icmp->code = 0;
    icmp->checksum = 0;
    icmp->un.echo.id = htons(getpid());
    icmp->un.echo.sequence = htons(1);

    // 计算校验和
    int checksum = 0;
    for (int i = 0; i< length / 2; ++i) {
        checksum += ((unsigned short *) buffer)[i];
    }
    while (checksum >> 16) {
        checksum = (checksum & 0xffff) + (checksum >> 16);
    }
    icmp->checksum = ~checksum;
}

int main() {
    int sockfd;
    struct sockaddr_in dest_addr;
    char buffer[ICMP_PACKET_SIZE];
    int length = ICMP_PACKET_SIZE;

    // 创建原始套接字
    if ((sockfd = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP)) < 0) {
        std::cerr << "Failed to create raw socket"<< std::endl;
        return -1;
    }

    // 设置目标地址
    dest_addr.sin_family = AF_INET;
    dest_addr.sin_addr.s_addr = inet_addr("8.8.8.8");

    // 构建ICMP请求数据包
    build_icmp_packet(buffer, length);

    // 发送ICMP请求数据包
    if (sendto(sockfd, buffer, length, 0, (struct sockaddr *) &dest_addr, sizeof(dest_addr)) < 0) {
        std::cerr << "Failed to send ICMP packet"<< std::endl;
        close(sockfd);
        return -1;
    }

    // 接收ICMP响应数据包
    socklen_t addr_len = sizeof(dest_addr);
    if (recvfrom(sockfd, buffer, length, 0, (struct sockaddr *) &dest_addr, &addr_len) < 0) {
        std::cerr << "Failed to receive ICMP packet"<< std::endl;
        close(sockfd);
        return -1;
    }

    // 处理ICMP响应数据包
    icmphdr *icmp = (icmphdr *) buffer;
    if (icmp->type == ICMP_ECHOREPLY) {
        std::cout << "Received ICMP echo reply from " << inet_ntoa(dest_addr.sin_addr)<< std::endl;
    } else {
        std::cerr << "Received unexpected ICMP packet type: " << icmp->type<< std::endl;
    }

    // 关闭套接字
    close(sockfd);

    return 0;
}

这个示例程序首先创建一个原始套接字,然后构建一个ICMP回显请求数据包,并将其发送到指定的目标地址。接着,程序等待接收ICMP响应数据包,并检查其类型是否为回显响应。如果是,则输出相应的信息;否则,输出错误信息。最后,关闭套接字。

注意:运行此程序可能需要root权限,因为原始套接字需要特殊权限。在Linux系统上,可以使用sudo命令运行程序。

未经允许不得转载 » 本文链接:https://www.legongju.com/article/93916.html

相关推荐

  • table在C++多线程环境下的使用注意事项

    table在C++多线程环境下的使用注意事项

    在C++多线程环境下使用std::table时,需要注意以下几点: 线程安全:std::table本身并不是线程安全的。如果多个线程同时访问和修改std::table,可能会导致数据不...

  • C++ table的性能优化策略有哪些

    C++ table的性能优化策略有哪些

    在C++中,表(通常是指数组或向量)的性能优化可以从多个方面进行。以下是一些建议: 预先分配内存:当你知道表的大小时,预先分配足够的内存空间,以避免在运行...

  • 如何自定义C++的table类

    如何自定义C++的table类

    要自定义一个C++的表格(table)类,你需要先确定你的表格类需要哪些功能。例如,你可能需要添加、删除和修改行和列,以及获取和设置单元格的值。下面是一个简单...

  • C++标准库中有无现成的table实现

    C++标准库中有无现成的table实现

    C++ 标准库没有直接提供一个名为 “table” 的容器类 std::vector:这是一个二维向量,可以用来表示一个动态大小的表格。你可以通过调整外部向量的大小来改变表格...

  • C++中stdin在实际项目中的应用案例

    C++中stdin在实际项目中的应用案例

    在实际项目中,stdin(标准输入)通常用于从控制台或其他输入源读取数据 命令行参数解析:
    当编写一个命令行程序时,你可能需要处理用户提供的参数。这些参...

  • C++中stdin输入的安全性问题分析

    C++中stdin输入的安全性问题分析

    在C++中,使用stdin进行输入时可能会遇到一些安全性问题 缓冲区溢出:当输入的数据超过预期的大小时,可能会导致缓冲区溢出。这种情况下,攻击者可以通过输入大量...

  • C++中如何自定义stdin的输入提示

    C++中如何自定义stdin的输入提示

    在 C++ 中,std::cin 是用于从标准输入(通常是键盘)读取数据的对象。要自定义输入提示,你可以使用 std::cout 先打印提示信息,然后再使用 std::cin 读取用户输...

  • C++中stdin在不同操作系统下的表现差异

    C++中stdin在不同操作系统下的表现差异

    在C++中,stdin、stdout和stderr是预定义好的文件指针,分别用于标准输入、标准输出和标准错误输出 行为差异: Windows和Linux/Unix系统在处理文本文件时,对换行...