2012-11-28 21 views
5

मुझे पता है कि यह अजीब लगता है। printf कुछ भी नहीं बदला जाना चाहिए, लेकिन इसके बिना भेजने के लिए विफल रहा। कार्यक्रम सी कोड से कॉपी किया गया था और ubuntux86 पर cpp संकलक द्वारा संकलित किया गया था। मेरे पास एआरपी अनुरोध भेजने के लिए एक कार्यक्रम है। इस printf के बिना भेजने में विफल रहा। अजीब बात यह है कि मैंने एक न्यूनतम प्रोग्राम किया है जो प्रिंटफ चर में उपयोग नहीं करता है (केवल printf में) और printf के साथ यह काम करता है और printf के बिना यह काम नहीं करता है (sendTo पर अमान्य तर्क त्रुटि प्राप्त करें) ।प्रिंटफ परिवर्तन जोड़ना सफलता की विफलता से परिणाम

देते समस्या सिर्फ दिखाने के लिए एक न्यूनतम संस्करण है:

यह SendTo विफल रहा:

struct ifreq ifr;//There is no use except in the printf 
printf("MAC address: is %02x:%02x:%02x:%02x:%02x:%02x \n", 
     ifr.ifr_hwaddr.sa_data[0]&0xFF, 
     ifr.ifr_hwaddr.sa_data[1]&0xFF, 
     ifr.ifr_hwaddr.sa_data[2]&0xFF, 
     ifr.ifr_hwaddr.sa_data[3]&0xFF, 
     ifr.ifr_hwaddr.sa_data[5]&0xFF, 
     ifr.ifr_hwaddr.sa_data[4]&0xFF); 

पूरा कार्यक्रम:

int retVal = sendto(arp_fd, &pkt, sizeof(pkt), 0, (struct sockaddr *) &sa,sizeof(sa)); 
if (retVal < 0) { 
    perror("sendto"); 
    close(arp_fd); 
    exit(1); 
} 

जब यह यह काम जोड़ने

#include "sys/socket.h" 
#include "sys/types.h" 
#include "stdio.h" 
#include "unistd.h" 
#include "string.h" 
#include "net/if.h" 
#include "stdlib.h" 
#include "arpa/inet.h" 
#include "netinet/in.h" 
#include "sys/ioctl.h" 
#include "netpacket/packet.h" 
#include "net/ethernet.h" 

#define ETHER_TYPE_FOR_ARP 0x0806 
#define HW_TYPE_FOR_ETHER 0x0001 
#define OP_CODE_FOR_ARP_REQ 0x0001 
#define HW_LEN_FOR_ETHER 0x06 
#define HW_LEN_FOR_IP 0x04 
#define PROTO_TYPE_FOR_IP 0x0800 

typedef unsigned char byte1; 
typedef unsigned short int byte2; 
typedef unsigned long int byte4; 

// For Proper memory allocation in the structure 
#pragma pack(1) 
typedef struct arp_packet { 
// ETH Header 
    byte1 dest_mac[6]; 
    byte1 src_mac[6]; 
    byte2 ether_type; 
// ARP Header 
    byte2 hw_type; 
    byte2 proto_type; 
    byte1 hw_size; 
    byte1 proto_size; 
    byte2 arp_opcode; 
    byte1 sender_mac[6]; 
    byte4 sender_ip; 
    byte1 target_mac[6]; 
    byte4 target_ip; 
// Paddign 
    char padding[18]; 
} ARP_PKT; 

int main() { 
    struct sockaddr_ll sa; 

// Ethernet Header 
    ARP_PKT pkt; 
    memset(pkt.dest_mac, 0xFF, (6 * sizeof(byte1))); 
    memset(pkt.src_mac, 0x1, (6 * sizeof(byte1))); 
    pkt.ether_type = htons(ETHER_TYPE_FOR_ARP); 
// ARP Header 
    pkt.hw_type = htons(HW_TYPE_FOR_ETHER); 
    pkt.proto_type = htons(PROTO_TYPE_FOR_IP); 
    pkt.hw_size = HW_LEN_FOR_ETHER; 
    pkt.proto_size = HW_LEN_FOR_IP; 
    pkt.arp_opcode = htons(OP_CODE_FOR_ARP_REQ); 
    memcpy(pkt.sender_mac, pkt.src_mac, (6 * sizeof(byte1))); 
    pkt.sender_ip = 2449647808; 
    memset(pkt.target_mac, 0, (6 * sizeof(byte1))); 
    pkt.target_ip = inet_addr("10.0.0.10"); 
// Padding 
    memset(pkt.padding, 0, 18 * sizeof(byte1)); 

    sa.sll_family = AF_PACKET; 
    sa.sll_ifindex = 3; 
    sa.sll_protocol = htons(ETH_P_ARP); 

    /* Send it! */ 
    int arp_fd = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ARP)); 
    struct ifreq ifr; 
    printf("MAC address: is %02x:%02x:%02x:%02x:%02x:%02x \n", 
      ifr.ifr_hwaddr.sa_data[0]&0xFF, 
      ifr.ifr_hwaddr.sa_data[1]&0xFF, 
      ifr.ifr_hwaddr.sa_data[2]&0xFF, 
      ifr.ifr_hwaddr.sa_data[3]&0xFF, 
      ifr.ifr_hwaddr.sa_data[5]&0xFF, 
      ifr.ifr_hwaddr.sa_data[4]&0xFF); 
    int retVal = sendto(arp_fd, &pkt, sizeof(pkt), 0, (struct sockaddr *) &sa,sizeof(sa)); 
    if (retVal < 0) { 
     perror("sendto"); 
     close(arp_fd); 
     exit(1); 
    } 

    printf("\n=========PACKET=========\n"); 
    return 0; 
} 
+0

यह विफल होने पर क्या हुआ? क्या आपको अपने कोड द्वारा मुद्रित त्रुटि मिली है? – unwind

+0

त्रुटि है: sendto: अमान्य तर्क जो perror द्वारा मुद्रित ("sendto"); –

+0

'printf (" [% d] ", errno) '' perror' के बाद' जोड़ें। क्या आप कोई कोड देखते हैं? –

उत्तर

2

ऐसा इसलिए है क्योंकि आपके पास struct sockaddr_ll sa में अनियमित मान हैं। अधिक विशेष रूप से, printf पर आपका कॉल संकलक को तीन रजिस्टरों को स्टैक पर सहेजने के लिए मजबूर करता है इसलिए sa स्थान और उसके मानों को स्थानांतरित करता है। आप से पहले printf देख सकते हैं ताकि यह आपके printf को जोड़कर गीलेर के आधार पर बदल सके।

अंतर देखने के लिए बस sa.sll_halen = 0; जोड़ें ... और आपके असली कार्यक्रम के लिए संपूर्ण संरचना शुरू करें।

+0

क्या आप अपना उत्तर विस्तारित कर सकते हैं। मुझे कारण समझ में नहीं आया, लेकिन Dammm आप सही हैं और सभी बाइट सामान नहीं :) –