使用原始套接字的ping请求失败

用户名

我的程序将icmp请求发送到指定的主机。

#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <netinet/ip_icmp.h>
#include <netdb.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#define ICMP_DATALEN    4
#define IP4_HDRLEN              20
#define ICMP_HDRLEN             8

uint16_t                                checksum (uint16_t *addr, int len)
{
    int nleft = len;
    int sum = 0;
    uint16_t *w = addr;
    uint16_t answer = 0;

    while (nleft > 1) {
            sum += *w++;
            nleft -= sizeof (uint16_t);
    }

    if (nleft == 1) {
            *(uint8_t *) (&answer) = *(uint8_t *) w;
            sum += answer;
    }

    sum = (sum >> 16) + (sum & 0xFFFF);
    sum += (sum >> 16);
    answer = ~sum;
    return (answer);
}

int                                             main(int ac, char **av)
{
    int                              icmp_socket;
    uint8_t                          data[ICMP_DATALEN];
    unsigned char                    datagram[IP_MAXPACKET];
    int                              ip_flags[4];
    struct ip                        *iph = (struct ip *) datagram;
    struct icmp                      *icmph = (struct icmp *) (datagram + IP4_HDRLEN);
    struct sockaddr_in               dst;
    int                              one = 1;
    int                              *val = &one;

    if (ac != 3)
    {
            fprintf(stderr, "usage: %s <source adress> <target adress>\n", av[0]);
            return (1);
    }

    if ((icmp_socket = socket(AF_INET, SOCK_RAW, 1)) == -1)
    {
            perror("socket");
            exit(1);
    }
    if (setsockopt(icmp_socket, IPPROTO_IP, IP_HDRINCL, val, sizeof(one)) < 0)
    {
            perror("setsockopt");
            exit(1);
    }

    memset(datagram, 0, IP_MAXPACKET);

    //prepare ipv4 header data
    memset((void*)&dst, 0, sizeof(dst));
    dst.sin_family = AF_INET;
    inet_pton(AF_INET, av[2], &(dst.sin_addr));
    ip_flags[0] = 0;
    ip_flags[1] = 0;
    ip_flags[2] = 0;
    ip_flags[3] = 0;

    //fill in ipv4 header
    iph->ip_hl = IP4_HDRLEN / sizeof(uint32_t);
    iph->ip_v = 4;
    iph->ip_tos = 0;
    iph->ip_len = (IP4_HDRLEN + ICMP_HDRLEN + ICMP_DATALEN);
    iph->ip_id = 666;
    iph->ip_off = (ip_flags[0] << 15)
                                            + (ip_flags[1] << 14)
                                            + (ip_flags[2] << 13)
                                            + ip_flags[3];
    iph->ip_ttl = 255;
    iph->ip_p = IPPROTO_ICMP;
    iph->ip_src.s_addr = inet_addr(av[1]);
    iph->ip_dst.s_addr = dst.sin_addr.s_addr;
    iph->ip_sum = 0;
    iph->ip_sum = checksum((uint16_t *)datagram, IP4_HDRLEN);

    //prepare icmp data
    data[0] = '4';
    data[1] = '2';
    data[2] = '4';
    data[3] = '2';

    //fill in icmp header
    icmph->icmp_type = ICMP_ECHO;
    icmph->icmp_code = 0;
    icmph->icmp_id = htons(666);
    icmph->icmp_seq = htons(0);
    memcpy(datagram + IP4_HDRLEN + ICMP_HDRLEN, data, ICMP_DATALEN);
    icmph->icmp_cksum = 0;
    icmph->icmp_cksum = checksum((uint16_t *)datagram + IP4_HDRLEN, (ICMP_HDRLEN + ICMP_DATALEN));

    int             i = 0;

    printf("sending packet:\n\n");
    while (i < (IP4_HDRLEN + ICMP_HDRLEN + ICMP_DATALEN))
    {
            printf("%02x", datagram[i++]);
            if (i % 2 == 0)
                    printf(" ");
            if (i % 16 == 0)
                    printf("\n");
    }
    printf("ip header checksump: %hx\n", iph->ip_sum);
    printf("icmp header checksump: %hx\n", icmph->icmp_cksum);

    if (sendto(icmp_socket, datagram, (IP4_HDRLEN + ICMP_HDRLEN + ICMP_DATALEN), 0, (struct sockaddr *)&dst, sizeof(dst)) < 0)
    {
            perror("sendto");
            exit(1);
    }
    close(icmp_socket);
    return (0);
}

我用tcpdump观看了网络,并且发送了数据包,但是我从未像ping命令那样收到答复。我希望你可以帮助我 !

西蒙·里希特

校验和按网络字节顺序排列。

您需要使用读取要添加到校验和的字节ntohs,并使用来写回校验和htons

本文收集自互联网,转载请注明来源。

如有侵权,请联系[email protected] 删除。

编辑于
0

我来说两句

0条评论
登录后参与评论

相关文章

来自分类Dev

使用套接字的HTTP请求

来自分类Dev

Python:在OSX中使用原始套接字

来自分类Dev

在C中使用套接字的HTTP请求

来自分类Dev

Java套接字失败

来自分类Dev

MySQL的套接字失败

来自分类Dev

python原始套接字使用的服务器主机

来自分类Dev

错误:在原始套接字中使用sendto()时,地址错误

来自分类Dev

使用Linux原始套接字捕获PTP数据包

来自分类Dev

哪些临时端口与原始套接字一起使用

来自分类Dev

原始套接字多播

来自分类Dev

python套接字如何使用相同的套接字连接正确重定向http / s请求?

来自分类Dev

使用null检查Java从套接字读取HTTP请求

来自分类Dev

使用sails.js访问套接字请求上的会话

来自分类Dev

如何使用Java套接字HTTP / 1.1请求下载图像?

来自分类Dev

使用套接字从服务器请求文件

来自分类Dev

使用一个TCP套接字的多个请求

来自分类Dev

在HashLink上使用套接字读取HTTP请求

来自分类Dev

使用套接字从服务器请求文件

来自分类Dev

如何使用多线程从套接字读取请求

来自分类Dev

带有原始套接字的python套接字setsockopt

来自分类Dev

从套接字解析ByteString失败

来自分类Dev

错误的请求套接字python

来自分类Dev

Systemd基于套接字的激活服务由于“启动请求重复太快”而失败

来自分类Dev

原始套接字未收到icmp响应

来自分类Dev

Golang的原始套接字嗅探

来自分类Dev

Linux内核中原始套接字的工作

来自分类Dev

在原始套接字中连接功能?

来自分类Dev

蓝牙套接字:读取失败,套接字可能关闭或超时

来自分类Dev

使用套接字的IllegalBlockingModeException