使用 IP_HDRINCL 选项创建 ip header 不做任何事情

creating ip header using IP_HDRINCL option not doing anything

我想通过写我自己的 IP header 发送一个 icmp 数据包所以我使用了 IP_HDRINCL 选项编译的代码和 运行s 没有任何错误但是没有数据包当我 运行 程序发送并使用 wireshark 捕获数据包时,程序也没有错误消息 这是代码:

#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/ip_icmp.h>
#include <netinet/ip.h>
#include <netdb.h>
#include <arpa/inet.h>

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

#define BUFFSIZE 2048

char sendbuf[BUFFSIZE];
int seqno=0;

unsigned short chksum(void *data,int bytes){
    unsigned int sum=0;
    unsigned short *octate=data,result;

    while(bytes>1){
        sum+=*octate++;
        bytes-=2;
    }
    if(bytes==1){
        sum+=*(unsigned char *)octate;
    }
    while(sum>>16){
        sum=(sum>>16)+(sum&0xffff);
    }
    result=~sum;
    return result;
}

struct addrinfo * getaddr(char *name){                               //convert user input host to address structure
    int e;
    struct addrinfo hints,*res;

    memset(&hints,0,sizeof(hints));
    hints.ai_flags=AI_CANONNAME;
    hints.ai_family=AF_INET;
    if(e=getaddrinfo(name,0,&hints,&res)!=0){
        printf("getaddrinfo error: %s\n",gai_strerror(e));
    }
    return res;
}

void create_packet(struct sockaddr *d){

    /*      Creating IP Packet           */

    struct ip *ip;
    ip=(struct ip*)sendbuf;
    ip->ip_v=4;                                                   
    ip->ip_hl=5;                                                 
    ip->ip_tos=0;                            
    ip->ip_len=20+8;                                             
    ip->ip_id=8848;                          
    ip->ip_off=IP_DF;                        
    ip->ip_ttl=7;                                             
    ip->ip_p=IPPROTO_ICMP;
    
    char srcip[]="192.168.1.69";
    struct addrinfo *source = getaddr(srcip);

    struct sockaddr_in *dest=(struct sockaddr_in *)d;
    struct sockaddr_in *src=(struct sockaddr_in *)source->ai_addr;
    ip->ip_src=src->sin_addr;
    ip->ip_dst=dest->sin_addr;

    ip->ip_sum=0;
    ip->ip_sum=chksum(ip,sizeof(*ip));

    
    /*      Creating ICMP Packet           */
    struct icmp *icmp;
    
    icmp=(struct icmp *)(sendbuf+20);
    icmp->icmp_type=ICMP_ECHO;
    icmp->icmp_code=0;
    icmp->icmp_id=getpid();
    icmp->icmp_seq=++seqno;
    icmp->icmp_cksum=0;
    icmp->icmp_cksum=chksum(icmp,8);
}


void main(int argc,char **argv){
    int sock;
    int on=1;
    struct addrinfo *addr=getaddr(argv[1]);
    if((sock=socket(AF_INET,SOCK_RAW,IPPROTO_ICMP))==-1){
        perror("socket error: ");
        return;
    }
    if(setsockopt(sock,IPPROTO_IP,IP_HDRINCL,&on,sizeof(on))==-1){
        perror("setsockopt error");
        return;
    } 
    create_packet(addr->ai_addr);

    if(sendto(sock,sendbuf,28,0,addr->ai_addr,addr->ai_addrlen)==-1){
        perror("sendto error");
        return;
    }

}

虽然仅根据您的代码我无法确定,但我怀疑您 运行 这个程序是在小端处理器(基于 x86)上运行的,并且 IP 期望它的数字很大-endian 顺序。这意味着任何多字节字段(ip_len、ip_off)都以错误的顺序写入。我怀疑如果您更改以下两行:

ip->ip_len=20+8;
ip->ip_off=IP_DF;

对此:

ip->ip_len=htons(20+8);
ip->ip_off=htons(IP_DF);

数据包将被正确发送。

顺便说一下,ip->ip_id 字段也是大端顺序,但除非您发送多个片段,否则这无关紧要。