2012-03-31 135 views
1

所以,我有一些copypasta設置原始套接字,它發送數據,該部分工作正常。但是,我將如何設置通過套接字發送的數據?我正在尋找一個DNS請求,如果有幫助的話。下面的代碼。UDP原始套接字設置消息

int main(int argc, char *argv[]) 
{ 
    if (!argv[1]) 
    { 
     printf("Target not specified!\nUsage: "); 
     printf(argv[0]); 
     printf(" <target>\n"); 
     exit(1); 
    } 

    struct ip ip; 
    struct udphdr udp; 
    int sd; 
    const int on = 1; 
    struct sockaddr_in sin; 
    //char msg[] = "\x03\xF0\x01\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x01"; 
    u_char *packet; 
    packet = (u_char *)malloc(120); 


    ip.ip_hl = 0x5; 
    ip.ip_v = 0x4; 
    ip.ip_tos = 0x0; 
    ip.ip_len = 60; 
    ip.ip_id = htons(12830); 
    ip.ip_off = 0x0; 
    ip.ip_ttl = 64; 
    ip.ip_p = IPPROTO_UDP; 
    ip.ip_sum = 0x0; 
    ip.ip_src.s_addr = inet_addr(argv[1]); 
    ip.ip_dst.s_addr = inet_addr("67.228.44.4"); 
    ip.ip_sum = in_cksum((unsigned short *)&ip, sizeof(ip)); 
    memcpy(packet, &ip, sizeof(ip)); 

    udp.source = htons(80); 
    udp.dest = htons(53); 
    udp.len = htons(22); 
    udp.check = 0; 
    udp.check = in_cksum_udp(ip.ip_src.s_addr, ip.ip_dst.s_addr, (unsigned short *)&udp, sizeof(udp)); 
    memcpy(packet + 20, &udp, sizeof(udp)); 

    if ((sd = socket(AF_INET, SOCK_RAW, IPPROTO_RAW)) < 0) { 
     perror("raw socket"); 
     exit(1); 
    } 

    if (setsockopt(sd, IPPROTO_IP, IP_HDRINCL, &on, sizeof(on)) < 0) { 
     perror("setsockopt"); 
     exit(1); 
    } 
    memset(&sin, 0, sizeof(sin)); 
    sin.sin_family = AF_INET; 
    sin.sin_addr.s_addr = ip.ip_dst.s_addr; 

    if (sendto(sd, packet, 120, 0, (struct sockaddr *)&sin, sizeof(struct sockaddr)) < 0) 
    { 
     perror("sendto"); 
     exit(1); 
    } 
} 
+1

不應該'sendto'的最後一個參數是'sizeof(struct sockaddr_in)'? – 2012-03-31 16:28:17

+0

雖然這可能是真的,但它不是我的問題的真正相關性,並且它的工作方式很好,我只是想知道如何設置消息。 – W00t 2012-03-31 17:59:51

+0

你爲什麼使用RAW套接字而不是UDP套接字?我不清楚你在問什麼......你說發送部分正在工作,那麼「設置信息」是什麼意思? – EdwardH 2012-03-31 20:38:43

回答

1

嗯......我想你想知道如何在消息中設置負載?基本上,你想偏移IP和UDP頭,並開始寫你的有效載荷數據。

一個匆匆扔在一起例子是:

int offset = packet + sizeof(struct ip) + sizeof(struct udphdr); 

然後,你可以寫你的有效載荷如下:

strcpy(offset, "1234"); 

這裏的一些工作的ICMP代碼,有效地在RAW寫出來的數據IP套接字:

struct icmphdr *icmp_hdr; 
char *datapart; 

icmp_hdr = (struct icmphdr *) icmp_data; 
icmp_hdr->i_type = ICMP_ECHO; 
icmp_hdr->i_code = 0; 
icmp_hdr->i_id = (unsigned short) getpid(); 
icmp_hdr->i_cksum = 0; 
icmp_hdr->i_seq = 0; 
datapart = icmp_data + sizeof(struct icmphdr); 
memset(datapart, 'E', datasize - sizeof(struct icmphdr)); 
+0

是的,這就是我一直在尋找的東西。謝謝! – W00t 2012-04-04 04:26:41