2013-05-06 101 views
0

這是我的代碼。它應該執行254個ARP請求並將ip和mac地址存儲到一個unsigned char數組中,然後將它們存儲到數據庫中。除數據庫部分外,這一切似乎都很完美。它只是正確地存儲第一個值(計數器值)而不是數組。我曾嘗試將它們作爲%u,%X傳遞,但這些格式期望unsigned int,但不包含unsigned char數組。任何幫助將不勝感激。將無符號字符數組插入數據庫的問題

#include <stdio.h> 
#include <string.h> 
#include <sys/ioctl.h> 
#include <net/if.h> 
#include <sys/socket.h> 
#include <netpacket/packet.h> 
#include <net/ethernet.h> 
#include <sys/types.h> 
#include <sys/socket.h> 
#include <arpa/inet.h> 
#include <fcntl.h> 
#include <sys/time.h> 
#include <sys/stat.h> 
#include <mysql.h> 



unsigned char macOrigen[6]; 
unsigned char ipOrigen[4]; 
unsigned char ipDestino[4]; 
unsigned char tramaEnviar[1514]; 
unsigned char tramaRecibir[1514]; 
unsigned char mascaraSubred[4]; 
unsigned char ipDestino[4]; 
unsigned char respuestaArp[2] = {0x00, 0x02}; 
unsigned char protocolo[2] = {0x08, 0x00}; 
unsigned char ethertype[2] = {0x08, 0x06}; 
unsigned char macDestino[6] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; 
unsigned char macBroadcast[6] = {0XFF, 0XFF, 0XFF, 0XFF, 0XFF, 0XFF}; 
unsigned char hardware[1] = {0x01}; 
unsigned char longitudHW[1] = {0x06}; 
unsigned char longitudIP[1] = {0x04}; 
unsigned char codigoOperacionSolicitud[2] = {0x00, 0x01}; 

int ObtenerDatos (int packetSocket){ 
    char nombre[10]; 
    int index; 
    struct ifreq interfaz; 
    printf("Ingrese el nombre de la interfaz:\nPuede consultar las inerfaces disponibles tecleando 'ifconfig' en otra terminal.\n\n"); 
    gets(nombre); 
    strcpy(interfaz.ifr_name, nombre); 
    if(ioctl(packetSocket, SIOCGIFINDEX, &interfaz) == -1){ 
     perror("Error al obener el índice.\n"); 
     }else{ 
      index = interfaz.ifr_ifindex; 
     } 
    if(ioctl(packetSocket, SIOCGIFHWADDR, &interfaz) == -1) 
    { 
     perror("No se ha podido obtener la informacion del dispositivo.\n"); 
    }else{ 
     memcpy(macOrigen, interfaz.ifr_hwaddr.sa_data, 6);  
    } 
    if(ioctl(packetSocket, SIOCGIFADDR, &interfaz) == -1){ 
     perror("No se ha podido obtener la informacion del dispositivo.\n"); 
    }else{ 
     memcpy(ipOrigen, interfaz.ifr_addr.sa_data+2, 4); 
    }if(ioctl(packetSocket, SIOCGIFNETMASK, &interfaz) == -1){ 
     perror("Error al obtener la mascara de subred.\n"); 
    }else{ 
     memcpy(mascaraSubred, interfaz.ifr_netmask.sa_data+2, 4); 
    } 

    return index; 
} 

void estructuraTramaArp(unsigned char *tramaEnviar){ 
    memcpy(tramaEnviar, macBroadcast, 6); 
    memcpy(tramaEnviar+6, macOrigen, 6); 
    memcpy(tramaEnviar+12, ethertype, 2); 
    memcpy(tramaEnviar+14, hardware, 1); 
    memcpy(tramaEnviar+15, protocolo, 2); 
    memcpy(tramaEnviar+17, longitudHW, 1); 
    memcpy(tramaEnviar+18, longitudIP, 1); 
    memcpy(tramaEnviar+19, longitudIP, 1); 
    memcpy(tramaEnviar+20, codigoOperacionSolicitud, 2); 
    memcpy(tramaEnviar+22, macOrigen, 6); 
    memcpy(tramaEnviar+28, ipOrigen, 4); 
    memcpy(tramaEnviar+32, macBroadcast, 6); 
    memcpy(tramaEnviar+38, ipDestino, 4); 
} 

void solicitudArp(int packetSocket, unsigned char *tramaEnviar, int index){ 
    int tam; 
    struct sockaddr_ll nic; 
    memset(&nic, 0x00, sizeof(nic)); 
    nic.sll_family = AF_PACKET; 
    nic.sll_protocol = htons(ETH_P_ALL); 
    nic.sll_ifindex = index; 
    tam = sendto(packetSocket, tramaEnviar, 60, 0, (struct sockaddr *)&nic, sizeof(nic)); 
    if (tam == -1){ 
     perror("Error al enviar la solicitud ARP"); 

    }else("Exito al enviar la trama"); 
} 

void recibeTramasArp(int packetSocket, unsigned char *tramaRecibir, int longitud){ 

    struct timeval tiempoEspera; 
    tiempoEspera.tv_sec = 0; 
    tiempoEspera.tv_usec = 900000L; 
    int tam, res, i, j; 
    fd_set lectura, escritura; 
    while(1) 
    { 
     FD_ZERO(&lectura); 
     FD_SET(packetSocket, &escritura); 
     if((res=select(packetSocket+1, &lectura, NULL, NULL, &tiempoEspera)) > 0) 
     { 
      if(FD_ISSET(packetSocket, &lectura)){ 
       if((tam = recvfrom(packetSocket, tramaRecibir, longitud, 0, NULL, 0)) < 0){ 
        perror("Error al recibir.\n"); 
        break; 
        }else{ 
         if(!memcmp(tramaRecibir+0, macOrigen, 6) && !memcmp(tramaRecibir+12,ethertype, 2) && !memcmp(tramaRecibir+16,protocolo,2) && !memcmp(tramaRecibir+20,respuestaArp,2) && !memcmp(tramaRecibir+28,ipDestino,4) && !memcmp(tramaRecibir+38,ipOrigen,4)){ 
     printf("La dirección IP:\n"); 
     for(i = 28; i <= 31; i++){ 
     printf("%.2d:",tramaRecibir[i]); 
     } 
     printf("\nRespondio con la direccion mac:\n"); 
     for(j = 22; j <=27; j++){ 
      printf("%.2x.",tramaRecibir[j]); 
     } 
     printf("\n"); 
     break; 

} 
    printf("No se recibio respuesta.\n"); 
} 
} 
    break; 
}else{ 
    printf("\nEl tiempo de espera se agoto.\n\n"); 
    break; 
} 

} 

} 

int main(int argc, char **argv){ 
    int indice, i, j; 
    char stmt_buf[100]; 
    char bd [100]; 
    int packetSocket = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_ALL)); 
    MYSQL *con = mysql_init(NULL); 
    if (con == NULL){ 
     fprintf(stderr,"%s\n", mysql_error(con)); 

    } 
    if (mysql_real_connect(con, "localhost", "angel", "xmdapoe", "arp", 0, NULL, 0) == NULL){ 
     fprintf(stderr,"%s\n", mysql_error(con)); 
    } 
    if (mysql_query(con, "DROP TABLE IF EXISTS info")) 
    { 
     fprintf(stderr, "%s\n", mysql_error(con)); 
    } 
    if (mysql_query(con, "CREATE TABLE info(id INT, ip TEXT, mac TEXT)")) 
    { 
     fprintf(stderr, "%s\n", mysql_error(con)); 
    } 
    if (packetSocket == -1) 
    { 
     perror("Error al abrir el socket.\n"); 
    }else{ 
     indice = ObtenerDatos(packetSocket); 
     memcpy(ipDestino, ipOrigen, 4); 
      ipDestino[3] = 1; 
     for(i = 0; i<254 ; i++){ 
      printf("Direccion IP destino:"); 
      printf("\n"); 
      for (j = 0; j<4; j++){ 
       printf("%d.",ipDestino[j]); 
       } 
       estructuraTramaArp(tramaEnviar); 
       solicitudArp(packetSocket, tramaEnviar, indice); 
       recibeTramasArp(packetSocket, tramaRecibir, 1514); 
       sprintf(stmt_buf, "insert into info values ('%d', '%.2x', '%.2x')", i, ipOrigen,macOrigen); 
       if(mysql_query(con, stmt_buf)){ 
        fprintf(stderr, "%s\n",mysql_error(con));    
       } 
       ipDestino[3]++; 
      } 
      } 
    close(packetSocket); 
} 

回答

0

使用%.2x的sprintf,您只需打印ipOrigen's和macOrigen的內存地址。 我想你想可能會達到什麼:

sprintf (buf, "%d%d%d%d", ipOrigen[0], ipOrigen[1], ipOrigen[2], ipOrigen[3]); 
0

看看你的sprintf語句,你實際打印ipOrigen和macOrigen數組的第一個元素的地址。對於ipOrigen你可以這樣做:

printf("%x", htonl(*(uint32_t *)ipOrigen)); 

對於macOrigen你可以使用使用htobe64()從endian.h類似的方法。